mirror of
https://github.com/postgresml/pgcat.git
synced 2026-03-23 17:36:28 +00:00
124 lines
3.8 KiB
Rust
124 lines
3.8 KiB
Rust
// PgCat, a PostgreSQL pooler with load balancing, failover, and sharding support.
|
|
// Copyright (C) 2022 Lev Kokotov <lev@levthe.dev>
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
extern crate async_trait;
|
|
extern crate bb8;
|
|
extern crate bytes;
|
|
extern crate md5;
|
|
extern crate serde;
|
|
extern crate serde_derive;
|
|
extern crate tokio;
|
|
extern crate toml;
|
|
|
|
use tokio::net::TcpListener;
|
|
|
|
use std::collections::HashMap;
|
|
use std::sync::{Arc, Mutex};
|
|
|
|
mod client;
|
|
mod config;
|
|
mod errors;
|
|
mod messages;
|
|
mod pool;
|
|
mod server;
|
|
mod sharding;
|
|
|
|
// Support for query cancellation: this maps our process_ids and
|
|
// secret keys to the backend's.
|
|
use config::{Address, User};
|
|
use pool::{ClientServerMap, ConnectionPool};
|
|
|
|
/// Main!
|
|
#[tokio::main]
|
|
async fn main() {
|
|
println!("> Welcome to PgCat! Meow.");
|
|
|
|
let config = match config::parse("pgcat.toml").await {
|
|
Ok(config) => config,
|
|
Err(err) => {
|
|
return;
|
|
}
|
|
};
|
|
|
|
let addr = format!("{}:{}", config.general.host, config.general.port);
|
|
let listener = match TcpListener::bind(&addr).await {
|
|
Ok(sock) => sock,
|
|
Err(err) => {
|
|
println!("> Error: {:?}", err);
|
|
return;
|
|
}
|
|
};
|
|
|
|
println!("> Running on {}", addr);
|
|
|
|
// Tracks which client is connected to which server for query cancellation.
|
|
let client_server_map: ClientServerMap = Arc::new(Mutex::new(HashMap::new()));
|
|
|
|
println!("> Pool size: {}", config.general.pool_size);
|
|
println!("> Pool mode: {}", config.general.pool_mode);
|
|
println!("> Ban time: {}s", config.general.ban_time);
|
|
println!(
|
|
"> Healthcheck timeout: {}ms",
|
|
config.general.healthcheck_timeout
|
|
);
|
|
|
|
let pool = ConnectionPool::from_config(config.clone(), client_server_map.clone()).await;
|
|
let transaction_mode = config.general.pool_mode == "transaction";
|
|
|
|
println!("> Waiting for clients...");
|
|
|
|
loop {
|
|
let pool = pool.clone();
|
|
let client_server_map = client_server_map.clone();
|
|
|
|
let (socket, addr) = match listener.accept().await {
|
|
Ok((socket, addr)) => (socket, addr),
|
|
Err(err) => {
|
|
println!("> Listener: {:?}", err);
|
|
continue;
|
|
}
|
|
};
|
|
|
|
// Client goes to another thread, bye.
|
|
tokio::task::spawn(async move {
|
|
println!(
|
|
">> Client {:?} connected, transaction pooling: {}",
|
|
addr, transaction_mode
|
|
);
|
|
|
|
match client::Client::startup(socket, client_server_map, transaction_mode).await {
|
|
Ok(mut client) => {
|
|
println!(">> Client {:?} authenticated successfully!", addr);
|
|
|
|
match client.handle(pool).await {
|
|
Ok(()) => {
|
|
println!(">> Client {:?} disconnected.", addr);
|
|
}
|
|
|
|
Err(err) => {
|
|
println!(">> Client disconnected with error: {:?}", err);
|
|
client.release();
|
|
}
|
|
}
|
|
}
|
|
|
|
Err(err) => {
|
|
println!(">> Error: {:?}", err);
|
|
}
|
|
};
|
|
});
|
|
}
|
|
}
|