Files
pgcat/src/main.rs

124 lines
3.3 KiB
Rust
Raw Normal View History

extern crate async_trait;
extern crate bb8;
2022-02-03 13:35:40 -08:00
extern crate bytes;
2022-02-03 15:17:04 -08:00
extern crate md5;
2022-02-03 13:35:40 -08:00
extern crate tokio;
use tokio::net::TcpListener;
2022-02-03 13:35:40 -08:00
2022-02-05 13:15:53 -08:00
use bb8::Pool;
2022-02-04 09:28:52 -08:00
use std::collections::HashMap;
use std::sync::{Arc, Mutex};
2022-02-03 15:17:04 -08:00
mod client;
2022-02-05 10:02:13 -08:00
mod config;
2022-02-03 13:35:40 -08:00
mod errors;
mod messages;
2022-02-03 16:25:05 -08:00
mod pool;
mod server;
2022-02-03 13:35:40 -08:00
2022-02-05 10:02:13 -08:00
// Support for query cancellation: this maps our process_ids and
// secret keys to the backend's.
use config::{Address, User};
2022-02-05 13:15:53 -08:00
use pool::{ClientServerMap, ReplicaPool, ServerPool};
2022-02-04 09:28:52 -08:00
2022-02-05 13:15:53 -08:00
//
// Poor man's config
//
const POOL_SIZE: u32 = 15;
/// Main!
2022-02-03 13:35:40 -08:00
#[tokio::main]
async fn main() {
2022-02-05 10:02:13 -08:00
println!("> Welcome to PgCat! Meow.");
2022-02-03 13:35:40 -08:00
2022-02-05 10:02:13 -08:00
let addr = "0.0.0.0:5433";
let listener = match TcpListener::bind(addr).await {
2022-02-03 13:35:40 -08:00
Ok(sock) => sock,
Err(err) => {
println!("> Error: {:?}", err);
return;
}
};
2022-02-05 10:02:13 -08:00
println!("> Running on {}", addr);
2022-02-04 16:01:35 -08:00
let client_server_map: ClientServerMap = Arc::new(Mutex::new(HashMap::new()));
2022-02-05 10:02:13 -08:00
2022-02-05 13:15:53 -08:00
// Replica pool.
2022-02-05 10:02:13 -08:00
let addresses = vec![
Address {
host: "127.0.0.1".to_string(),
port: "5432".to_string(),
},
Address {
host: "localhost".to_string(),
port: "5432".to_string(),
},
];
2022-02-05 13:15:53 -08:00
let num_addresses = addresses.len() as u32;
2022-02-05 10:02:13 -08:00
let user = User {
name: "lev".to_string(),
password: "lev".to_string(),
};
2022-02-05 13:15:53 -08:00
let database = "lev";
let replica_pool = ReplicaPool::new(addresses).await;
let manager = ServerPool::new(replica_pool, user, database, client_server_map.clone());
// We are round-robining, so ideally the replicas will be equally loaded.
// Therefore, we are allocating number of replicas * pool size of connections.
// However, if a replica dies, the remaining replicas will share the burden,
// also equally.
//
// Note that failover in this case could bring down the remaining replicas, so
// in certain situations, e.g. when replicas are running hot already, failover
// is not at all desirable!!
let pool = Pool::builder()
.max_size(POOL_SIZE * num_addresses)
.build(manager)
.await
.unwrap();
2022-02-03 16:25:05 -08:00
2022-02-03 13:35:40 -08:00
loop {
2022-02-05 13:15:53 -08:00
let pool = pool.clone();
2022-02-04 16:01:35 -08:00
let client_server_map = client_server_map.clone();
2022-02-03 16:25:05 -08:00
2022-02-03 13:54:07 -08:00
let (socket, addr) = match listener.accept().await {
Ok((socket, addr)) => (socket, addr),
2022-02-03 13:35:40 -08:00
Err(err) => {
println!("> Listener: {:?}", err);
continue;
}
};
// Client goes to another thread, bye.
tokio::task::spawn(async move {
2022-02-03 15:17:04 -08:00
println!(">> Client {:?} connected.", addr);
2022-02-04 16:01:35 -08:00
match client::Client::startup(socket, client_server_map).await {
2022-02-03 13:54:07 -08:00
Ok(mut client) => {
2022-02-03 15:17:04 -08:00
println!(">> Client {:?} authenticated successfully!", addr);
2022-02-05 13:15:53 -08:00
match client.handle(pool).await {
2022-02-03 15:17:04 -08:00
Ok(()) => {
println!(">> Client {:?} disconnected.", addr);
}
Err(err) => {
println!(">> Client disconnected with error: {:?}", err);
2022-02-04 16:01:35 -08:00
client.release();
2022-02-03 15:17:04 -08:00
}
}
}
2022-02-03 13:35:40 -08:00
Err(err) => {
println!(">> Error: {:?}", err);
}
};
});
}
}