c794b818ff
- Add etcd-client dependency for distributed configuration storage - Implement EtcdConfig with priority: etcd > environment variables > defaults - Add ServiceRegistry for service registration with lease keep-alive - Integrate etcd-based service discovery for appks gRPC connections - Add service watcher for real-time service instance updates - Migrate Redis configuration from single URL to cluster node list - Update Dockerfile with default IMKS_HOST and IMKS_PORT environment variables - Add etcd bootstrap configuration through environment variables - Implement Redis cluster URL building with optional authentication
190 lines
5.4 KiB
Rust
190 lines
5.4 KiB
Rust
use std::time::{SystemTime, UNIX_EPOCH};
|
|
|
|
use async_trait::async_trait;
|
|
|
|
use crate::socket::message_bus::redis::RedisCommandClient;
|
|
use crate::socket::message_bus::redis::RedisMessageBus;
|
|
use crate::socket::session_store::{SessionError, SessionInfo, SessionStoreTrait};
|
|
|
|
fn now_millis() -> u64 {
|
|
SystemTime::now()
|
|
.duration_since(UNIX_EPOCH)
|
|
.unwrap_or_default()
|
|
.as_millis() as u64
|
|
}
|
|
|
|
const DEFAULT_TTL_SECS: u64 = 60;
|
|
const KEY_PREFIX: &str = "socket.io:session";
|
|
|
|
pub struct RedisSessionStore {
|
|
client: RedisCommandClient,
|
|
ttl_secs: u64,
|
|
}
|
|
|
|
impl RedisSessionStore {
|
|
pub fn new(bus: &RedisMessageBus, ttl_secs: Option<u64>) -> Self {
|
|
Self {
|
|
client: bus.client(),
|
|
ttl_secs: ttl_secs.unwrap_or(DEFAULT_TTL_SECS),
|
|
}
|
|
}
|
|
|
|
fn key(&self, sid: &str) -> String {
|
|
format!("{}:{}", KEY_PREFIX, sid)
|
|
}
|
|
}
|
|
|
|
#[async_trait]
|
|
impl SessionStoreTrait for RedisSessionStore {
|
|
async fn create(
|
|
&self,
|
|
sid: &str,
|
|
transport: &str,
|
|
server_id: &str,
|
|
) -> Result<(), SessionError> {
|
|
let key = self.key(sid);
|
|
let now = now_millis();
|
|
|
|
// Batch all fields in a single HMSET-style call
|
|
self.client
|
|
.query::<()>(
|
|
redis::cmd("HSET")
|
|
.arg(&key)
|
|
.arg("sid")
|
|
.arg(sid)
|
|
.arg("transport")
|
|
.arg(transport)
|
|
.arg("state")
|
|
.arg("connecting")
|
|
.arg("server_id")
|
|
.arg(server_id)
|
|
.arg("created_at")
|
|
.arg(now.to_string())
|
|
.arg("last_ping")
|
|
.arg(now.to_string()),
|
|
)
|
|
.await
|
|
.map_err(|e| SessionError::Redis(e.to_string()))?;
|
|
|
|
self.client
|
|
.query::<()>(redis::cmd("EXPIRE").arg(&key).arg(self.ttl_secs as i64))
|
|
.await
|
|
.map_err(|e| SessionError::Redis(e.to_string()))?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
async fn get(&self, sid: &str) -> Result<Option<SessionInfo>, SessionError> {
|
|
let key = self.key(sid);
|
|
|
|
let values: std::collections::HashMap<String, String> = self
|
|
.client
|
|
.query(redis::cmd("HGETALL").arg(&key))
|
|
.await
|
|
.map_err(|e| SessionError::Redis(e.to_string()))?;
|
|
|
|
if values.is_empty() {
|
|
return Ok(None);
|
|
}
|
|
|
|
let info = SessionInfo {
|
|
sid: values.get("sid").cloned().unwrap_or_default(),
|
|
transport: values.get("transport").cloned().unwrap_or_default(),
|
|
state: values.get("state").cloned().unwrap_or_default(),
|
|
server_id: values.get("server_id").cloned().unwrap_or_default(),
|
|
created_at: values
|
|
.get("created_at")
|
|
.and_then(|v| v.parse::<u64>().ok())
|
|
.unwrap_or(0),
|
|
last_ping: values
|
|
.get("last_ping")
|
|
.and_then(|v| v.parse::<u64>().ok())
|
|
.unwrap_or(0),
|
|
};
|
|
|
|
Ok(Some(info))
|
|
}
|
|
|
|
async fn set_state(&self, sid: &str, state: &str) -> Result<(), SessionError> {
|
|
let key = self.key(sid);
|
|
|
|
self.client
|
|
.query::<()>(redis::cmd("HSET").arg(&key).arg("state").arg(state))
|
|
.await
|
|
.map_err(|e| SessionError::Redis(e.to_string()))?;
|
|
|
|
self.client
|
|
.query::<()>(redis::cmd("EXPIRE").arg(&key).arg(self.ttl_secs as i64))
|
|
.await
|
|
.map_err(|e| SessionError::Redis(e.to_string()))?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
async fn set_transport(&self, sid: &str, transport: &str) -> Result<(), SessionError> {
|
|
let key = self.key(sid);
|
|
|
|
self.client
|
|
.query::<()>(
|
|
redis::cmd("HSET")
|
|
.arg(&key)
|
|
.arg("transport")
|
|
.arg(transport),
|
|
)
|
|
.await
|
|
.map_err(|e| SessionError::Redis(e.to_string()))?;
|
|
|
|
self.client
|
|
.query::<()>(redis::cmd("EXPIRE").arg(&key).arg(self.ttl_secs as i64))
|
|
.await
|
|
.map_err(|e| SessionError::Redis(e.to_string()))?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
async fn update_ping(&self, sid: &str) -> Result<(), SessionError> {
|
|
let key = self.key(sid);
|
|
let now = now_millis();
|
|
|
|
self.client
|
|
.query::<()>(
|
|
redis::cmd("HSET")
|
|
.arg(&key)
|
|
.arg("last_ping")
|
|
.arg(now.to_string()),
|
|
)
|
|
.await
|
|
.map_err(|e| SessionError::Redis(e.to_string()))?;
|
|
|
|
self.client
|
|
.query::<()>(redis::cmd("EXPIRE").arg(&key).arg(self.ttl_secs as i64))
|
|
.await
|
|
.map_err(|e| SessionError::Redis(e.to_string()))?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
async fn remove(&self, sid: &str) -> Result<(), SessionError> {
|
|
let key = self.key(sid);
|
|
|
|
self.client
|
|
.query::<()>(redis::cmd("DEL").arg(&key))
|
|
.await
|
|
.map_err(|e| SessionError::Redis(e.to_string()))?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
async fn exists(&self, sid: &str) -> Result<bool, SessionError> {
|
|
let key = self.key(sid);
|
|
|
|
let exists: bool = self
|
|
.client
|
|
.query(redis::cmd("EXISTS").arg(&key))
|
|
.await
|
|
.map_err(|e| SessionError::Redis(e.to_string()))?;
|
|
|
|
Ok(exists)
|
|
}
|
|
}
|