|
@@ -1,16 +1,26 @@
|
|
|
use crate::{db::Db, error::Error, value::Value};
|
|
|
use bytes::Bytes;
|
|
|
use parking_lot::RwLock;
|
|
|
-use std::{
|
|
|
- collections::{BTreeMap, HashSet},
|
|
|
- net::SocketAddr,
|
|
|
- sync::Arc,
|
|
|
-};
|
|
|
+use std::{collections::HashSet, net::SocketAddr, sync::Arc};
|
|
|
|
|
|
-#[derive(Debug)]
|
|
|
-pub struct Connections {
|
|
|
- connections: RwLock<BTreeMap<u128, Arc<Connection>>>,
|
|
|
- counter: RwLock<u128>,
|
|
|
+use self::pubsub_server::Pubsub;
|
|
|
+
|
|
|
+pub mod connections;
|
|
|
+pub mod pubsub_connection;
|
|
|
+pub mod pubsub_server;
|
|
|
+
|
|
|
+#[derive(Debug, Clone, Copy, Eq, PartialEq)]
|
|
|
+pub enum ConnectionStatus {
|
|
|
+ Multi,
|
|
|
+ ExecutingTx,
|
|
|
+ Pubsub,
|
|
|
+ Normal,
|
|
|
+}
|
|
|
+
|
|
|
+impl Default for ConnectionStatus {
|
|
|
+ fn default() -> Self {
|
|
|
+ ConnectionStatus::Normal
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
#[derive(Debug)]
|
|
@@ -18,8 +28,7 @@ pub struct ConnectionInfo {
|
|
|
pub name: Option<String>,
|
|
|
pub watch_keys: Vec<(Bytes, u128)>,
|
|
|
pub tx_keys: HashSet<Bytes>,
|
|
|
- pub in_transaction: bool,
|
|
|
- pub in_executing_transaction: bool,
|
|
|
+ pub status: ConnectionStatus,
|
|
|
pub commands: Option<Vec<Vec<Bytes>>>,
|
|
|
}
|
|
|
|
|
@@ -28,50 +37,10 @@ pub struct Connection {
|
|
|
id: u128,
|
|
|
db: Db,
|
|
|
current_db: u32,
|
|
|
- connections: Arc<Connections>,
|
|
|
+ all_connections: Arc<connections::Connections>,
|
|
|
addr: SocketAddr,
|
|
|
info: RwLock<ConnectionInfo>,
|
|
|
-}
|
|
|
-
|
|
|
-impl Connections {
|
|
|
- pub fn new() -> Self {
|
|
|
- Self {
|
|
|
- counter: RwLock::new(0),
|
|
|
- connections: RwLock::new(BTreeMap::new()),
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- pub fn remove(self: Arc<Connections>, conn: Arc<Connection>) {
|
|
|
- let id = conn.id();
|
|
|
- self.connections.write().remove(&id);
|
|
|
- }
|
|
|
-
|
|
|
- pub fn new_connection(
|
|
|
- self: &Arc<Connections>,
|
|
|
- db: Arc<Db>,
|
|
|
- addr: SocketAddr,
|
|
|
- ) -> Arc<Connection> {
|
|
|
- let mut id = self.counter.write();
|
|
|
- *id += 1;
|
|
|
-
|
|
|
- let conn = Arc::new(Connection {
|
|
|
- id: *id,
|
|
|
- db: db.new_db_instance(*id),
|
|
|
- addr,
|
|
|
- connections: self.clone(),
|
|
|
- current_db: 0,
|
|
|
- info: RwLock::new(ConnectionInfo::new()),
|
|
|
- });
|
|
|
-
|
|
|
- self.connections.write().insert(*id, conn.clone());
|
|
|
- conn
|
|
|
- }
|
|
|
-
|
|
|
- pub fn iter(&self, f: &mut dyn FnMut(Arc<Connection>)) {
|
|
|
- for (_, value) in self.connections.read().iter() {
|
|
|
- f(value.clone())
|
|
|
- }
|
|
|
- }
|
|
|
+ pubsub_client: pubsub_connection::PubsubClient,
|
|
|
}
|
|
|
|
|
|
impl ConnectionInfo {
|
|
@@ -81,8 +50,7 @@ impl ConnectionInfo {
|
|
|
watch_keys: vec![],
|
|
|
tx_keys: HashSet::new(),
|
|
|
commands: None,
|
|
|
- in_transaction: false,
|
|
|
- in_executing_transaction: false,
|
|
|
+ status: ConnectionStatus::Normal,
|
|
|
}
|
|
|
}
|
|
|
}
|
|
@@ -92,18 +60,25 @@ impl Connection {
|
|
|
&self.db
|
|
|
}
|
|
|
|
|
|
+ pub fn pubsub(&self) -> Arc<Pubsub> {
|
|
|
+ self.all_connections.pubsub()
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn pubsub_client(&self) -> &pubsub_connection::PubsubClient {
|
|
|
+ &self.pubsub_client
|
|
|
+ }
|
|
|
+
|
|
|
pub fn id(&self) -> u128 {
|
|
|
self.id
|
|
|
}
|
|
|
|
|
|
pub fn stop_transaction(&self) -> Result<Value, Error> {
|
|
|
let info = &mut self.info.write();
|
|
|
- if info.in_transaction {
|
|
|
+ if info.status == ConnectionStatus::Multi {
|
|
|
info.commands = None;
|
|
|
info.watch_keys.clear();
|
|
|
info.tx_keys.clear();
|
|
|
- info.in_transaction = false;
|
|
|
- info.in_executing_transaction = true;
|
|
|
+ info.status = ConnectionStatus::ExecutingTx;
|
|
|
|
|
|
Ok(Value::Ok)
|
|
|
} else {
|
|
@@ -113,31 +88,40 @@ impl Connection {
|
|
|
|
|
|
pub fn start_transaction(&self) -> Result<Value, Error> {
|
|
|
let mut info = self.info.write();
|
|
|
- if !info.in_transaction {
|
|
|
- info.in_transaction = true;
|
|
|
+ if info.status == ConnectionStatus::Normal {
|
|
|
+ info.status = ConnectionStatus::Multi;
|
|
|
Ok(Value::Ok)
|
|
|
} else {
|
|
|
Err(Error::NestedTx)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- /// We are inside a MULTI, most transactions are rather queued for later
|
|
|
- /// execution instead of being executed right away.
|
|
|
- pub fn in_transaction(&self) -> bool {
|
|
|
- self.info.read().in_transaction
|
|
|
+ pub fn start_pubsub(&self) -> Result<Value, Error> {
|
|
|
+ let mut info = self.info.write();
|
|
|
+ match info.status {
|
|
|
+ ConnectionStatus::Normal | ConnectionStatus::Pubsub => {
|
|
|
+ info.status = ConnectionStatus::Pubsub;
|
|
|
+ Ok(Value::Ok)
|
|
|
+ }
|
|
|
+ _ => Err(Error::NestedTx),
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
- /// The commands are being executed inside a transaction (by EXEC). It is
|
|
|
- /// important to keep track of this because some commands change their
|
|
|
- /// behaviour.
|
|
|
- pub fn is_executing_transaction(&self) -> bool {
|
|
|
- self.info.read().in_executing_transaction
|
|
|
+ pub fn reset(&self) {
|
|
|
+ let mut info = self.info.write();
|
|
|
+ info.status = ConnectionStatus::Normal;
|
|
|
+ info.name = None;
|
|
|
+ info.watch_keys = vec![];
|
|
|
+ info.commands = None;
|
|
|
+ info.tx_keys = HashSet::new();
|
|
|
+
|
|
|
+ let pubsub = self.pubsub();
|
|
|
+ pubsub.unsubscribe(&self.pubsub_client.subscriptions(), self);
|
|
|
+ pubsub.punsubscribe(&self.pubsub_client.psubscriptions(), self);
|
|
|
}
|
|
|
|
|
|
- /// EXEC has been called and we need to keep track
|
|
|
- pub fn start_executing_transaction(&self) {
|
|
|
- let info = &mut self.info.write();
|
|
|
- info.in_executing_transaction = true;
|
|
|
+ pub fn status(&self) -> ConnectionStatus {
|
|
|
+ self.info.read().status
|
|
|
}
|
|
|
|
|
|
pub fn watch_key(&self, keys: &[(&Bytes, u128)]) {
|
|
@@ -184,7 +168,7 @@ impl Connection {
|
|
|
pub fn get_queue_commands(&self) -> Option<Vec<Vec<Bytes>>> {
|
|
|
let info = &mut self.info.write();
|
|
|
info.watch_keys = vec![];
|
|
|
- info.in_transaction = false;
|
|
|
+ info.status = ConnectionStatus::ExecutingTx;
|
|
|
info.commands.take()
|
|
|
}
|
|
|
|
|
@@ -199,11 +183,14 @@ impl Connection {
|
|
|
}
|
|
|
|
|
|
pub fn destroy(self: Arc<Connection>) {
|
|
|
- self.connections.clone().remove(self);
|
|
|
+ let pubsub = self.pubsub();
|
|
|
+ pubsub.unsubscribe(&self.pubsub_client.subscriptions(), &self);
|
|
|
+ pubsub.punsubscribe(&self.pubsub_client.psubscriptions(), &self);
|
|
|
+ self.all_connections.clone().remove(self);
|
|
|
}
|
|
|
|
|
|
- pub fn all_connections(&self) -> Arc<Connections> {
|
|
|
- self.connections.clone()
|
|
|
+ pub fn all_connections(&self) -> Arc<connections::Connections> {
|
|
|
+ self.all_connections.clone()
|
|
|
}
|
|
|
|
|
|
pub fn name(&self) -> Option<String> {
|