AppFlowy/rust-lib/flowy-user/src/services/user/user_session.rs

397 lines
13 KiB
Rust
Raw Normal View History

use crate::{
2021-09-04 08:53:58 +00:00
entities::{SignInParams, SignUpParams, UpdateUserParams, UserProfile},
2021-09-16 04:35:55 +00:00
errors::{ErrorCode, UserError},
2021-09-10 10:21:35 +00:00
services::user::database::UserDB,
sql_tables::{UserTable, UserTableChangeset},
};
2021-08-29 14:00:42 +00:00
2021-09-10 10:21:35 +00:00
use crate::{
2021-10-05 03:46:56 +00:00
notify::*,
2021-09-10 10:21:35 +00:00
services::server::{construct_user_server, Server},
};
use flowy_database::{
query_dsl::*,
schema::{user_table, user_table::dsl},
DBConnection,
ExpressionMethods,
2021-07-13 15:08:20 +00:00
UserDatabaseConnection,
};
use flowy_infra::kv::KV;
2021-09-27 15:23:23 +00:00
use flowy_net::config::ServerConfig;
2021-08-31 15:01:46 +00:00
use flowy_sqlite::ConnectionPool;
2021-10-05 09:54:11 +00:00
use flowy_ws::{WsController, WsMessageHandler, WsState};
use parking_lot::RwLock;
use serde::{Deserialize, Serialize};
use std::sync::Arc;
2021-11-08 15:15:29 +00:00
use tokio::sync::broadcast;
#[derive(Clone)]
pub enum UserStatus {
Login { token: String },
Logout { token: String },
Expired { token: String },
SignUp { profile: UserProfile },
}
pub struct UserSessionConfig {
root_dir: String,
2021-09-27 15:23:23 +00:00
server_config: ServerConfig,
}
impl UserSessionConfig {
2021-09-27 15:23:23 +00:00
pub fn new(root_dir: &str, server_config: &ServerConfig) -> Self {
Self {
root_dir: root_dir.to_owned(),
2021-09-27 15:23:23 +00:00
server_config: server_config.clone(),
}
}
}
pub struct UserSession {
database: UserDB,
config: UserSessionConfig,
#[allow(dead_code)]
2021-09-01 08:08:32 +00:00
server: Server,
session: RwLock<Option<Session>>,
pub ws_controller: Arc<WsController>,
2021-11-08 15:15:29 +00:00
status_notifier: broadcast::Sender<UserStatus>,
}
impl UserSession {
2021-11-08 15:15:29 +00:00
pub fn new(config: UserSessionConfig) -> Self {
let db = UserDB::new(&config.root_dir);
2021-09-27 15:23:23 +00:00
let server = construct_user_server(&config.server_config);
2021-09-30 09:24:02 +00:00
let ws_controller = Arc::new(WsController::new());
2021-11-08 15:15:29 +00:00
let (status_notifier, _) = broadcast::channel(10);
2021-09-17 11:03:46 +00:00
let user_session = Self {
database: db,
config,
2021-07-14 15:00:58 +00:00
server,
session: RwLock::new(None),
2021-09-18 14:32:00 +00:00
ws_controller,
2021-11-08 15:15:29 +00:00
status_notifier,
2021-09-17 11:03:46 +00:00
};
user_session
}
pub fn init(&self) {
match self.get_session() {
Ok(session) => {
let _ = self.status_notifier.send(UserStatus::Login { token: session.token });
},
Err(_) => {},
}
}
2021-11-08 15:15:29 +00:00
pub fn status_subscribe(&self) -> broadcast::Receiver<UserStatus> { self.status_notifier.subscribe() }
pub fn db_connection(&self) -> Result<DBConnection, UserError> {
let user_id = self.get_session()?.user_id;
2021-07-16 15:18:12 +00:00
self.database.get_connection(&user_id)
}
// The caller will be not 'Sync' before of the return value,
// PooledConnection<ConnectionManager> is not sync. You can use
// db_connection_pool function to require the ConnectionPool that is 'Sync'.
//
// let pool = self.db_connection_pool()?;
// let conn: PooledConnection<ConnectionManager> = pool.get()?;
2021-09-04 09:26:04 +00:00
pub fn db_pool(&self) -> Result<Arc<ConnectionPool>, UserError> {
let user_id = self.get_session()?.user_id;
2021-08-31 15:01:46 +00:00
self.database.get_pool(&user_id)
}
2021-09-06 08:18:34 +00:00
#[tracing::instrument(level = "debug", skip(self))]
2021-09-04 08:53:58 +00:00
pub async fn sign_in(&self, params: SignInParams) -> Result<UserProfile, UserError> {
if self.is_login(&params.email) {
2021-09-04 08:53:58 +00:00
self.user_profile().await
} else {
let resp = self.server.sign_in(params).await?;
2021-09-08 10:25:32 +00:00
let session = Session::new(&resp.user_id, &resp.token, &resp.email);
let _ = self.set_session(Some(session))?;
let user_table = self.save_user(resp.into()).await?;
let user_profile: UserProfile = user_table.into();
2021-11-08 15:15:29 +00:00
let _ = self.status_notifier.send(UserStatus::Login {
2021-09-17 11:03:46 +00:00
token: user_profile.token.clone(),
});
2021-09-04 08:53:58 +00:00
Ok(user_profile)
}
}
2021-09-06 08:18:34 +00:00
#[tracing::instrument(level = "debug", skip(self))]
2021-09-04 08:53:58 +00:00
pub async fn sign_up(&self, params: SignUpParams) -> Result<UserProfile, UserError> {
2021-09-04 01:00:15 +00:00
if self.is_login(&params.email) {
2021-09-04 08:53:58 +00:00
self.user_profile().await
2021-09-04 01:00:15 +00:00
} else {
let resp = self.server.sign_up(params).await?;
let session = Session::new(&resp.user_id, &resp.token, &resp.email);
let _ = self.set_session(Some(session))?;
let user_table = self.save_user(resp.into()).await?;
let user_profile: UserProfile = user_table.into();
2021-11-08 15:15:29 +00:00
let _ = self.status_notifier.send(UserStatus::SignUp {
profile: user_profile.clone(),
2021-09-17 11:03:46 +00:00
});
2021-09-04 08:53:58 +00:00
Ok(user_profile)
2021-09-04 01:00:15 +00:00
}
}
2021-09-06 08:18:34 +00:00
#[tracing::instrument(level = "debug", skip(self))]
2021-08-31 09:25:08 +00:00
pub async fn sign_out(&self) -> Result<(), UserError> {
2021-09-01 08:08:32 +00:00
let session = self.get_session()?;
2021-09-27 15:23:23 +00:00
let _ =
diesel::delete(dsl::user_table.filter(dsl::id.eq(&session.user_id))).execute(&*(self.db_connection()?))?;
2021-09-01 08:08:32 +00:00
let _ = self.database.close_user_db(&session.user_id)?;
let _ = self.set_session(None)?;
let _ = self.status_notifier.send(UserStatus::Logout {
2021-09-17 11:03:46 +00:00
token: session.token.clone(),
});
2021-09-04 09:26:04 +00:00
let _ = self.sign_out_on_server(&session.token).await?;
2021-07-11 09:38:03 +00:00
Ok(())
}
2021-09-06 08:18:34 +00:00
#[tracing::instrument(level = "debug", skip(self))]
2021-08-31 15:01:46 +00:00
pub async fn update_user(&self, params: UpdateUserParams) -> Result<(), UserError> {
2021-09-01 09:57:06 +00:00
let session = self.get_session()?;
2021-09-04 09:26:04 +00:00
let changeset = UserTableChangeset::new(params.clone());
diesel_update_table!(user_table, changeset, &*self.db_connection()?);
2021-09-01 09:57:06 +00:00
2021-09-04 09:26:04 +00:00
let _ = self.update_user_on_server(&session.token, params).await?;
2021-08-31 15:01:46 +00:00
Ok(())
}
pub async fn init_user(&self) -> Result<(), UserError> {
let (_, token) = self.get_session()?.into_part();
2021-09-30 09:24:02 +00:00
let _ = self.start_ws_connection(&token).await?;
2021-11-06 14:35:45 +00:00
Ok(())
}
pub async fn check_user(&self) -> Result<UserProfile, UserError> {
2021-09-17 11:03:46 +00:00
let (user_id, token) = self.get_session()?.into_part();
let user = dsl::user_table
.filter(user_table::id.eq(&user_id))
.first::<UserTable>(&*(self.db_connection()?))?;
let _ = self.read_user_profile_on_server(&token)?;
Ok(user.into())
2021-09-17 11:03:46 +00:00
}
2021-09-04 08:53:58 +00:00
pub async fn user_profile(&self) -> Result<UserProfile, UserError> {
2021-09-07 15:30:43 +00:00
let (user_id, token) = self.get_session()?.into_part();
2021-09-01 08:37:46 +00:00
let user = dsl::user_table
2021-09-07 15:30:43 +00:00
.filter(user_table::id.eq(&user_id))
.first::<UserTable>(&*(self.db_connection()?))?;
2021-09-01 08:37:46 +00:00
2021-09-17 11:03:46 +00:00
let _ = self.read_user_profile_on_server(&token)?;
Ok(user.into())
}
2021-08-29 14:00:42 +00:00
pub fn user_dir(&self) -> Result<String, UserError> {
let session = self.get_session()?;
Ok(format!("{}/{}", self.config.root_dir, session.user_id))
2021-07-23 09:30:33 +00:00
}
2021-09-01 08:08:32 +00:00
pub fn user_id(&self) -> Result<String, UserError> { Ok(self.get_session()?.user_id) }
2021-09-01 14:50:22 +00:00
pub fn token(&self) -> Result<String, UserError> { Ok(self.get_session()?.token) }
2021-09-18 14:32:00 +00:00
2021-09-27 15:23:23 +00:00
pub fn add_ws_handler(&self, handler: Arc<dyn WsMessageHandler>) {
2021-09-30 09:24:02 +00:00
let _ = self.ws_controller.add_handler(handler);
2021-09-18 14:32:00 +00:00
}
2021-09-01 08:08:32 +00:00
}
impl UserSession {
2021-09-17 11:03:46 +00:00
fn read_user_profile_on_server(&self, token: &str) -> Result<(), UserError> {
2021-09-04 09:26:04 +00:00
let server = self.server.clone();
let token = token.to_owned();
2021-09-07 15:30:43 +00:00
tokio::spawn(async move {
2021-09-04 09:26:04 +00:00
match server.get_user(&token).await {
Ok(profile) => {
2021-10-14 06:34:22 +00:00
dart_notify(&token, UserNotification::UserProfileUpdated)
2021-09-27 15:23:23 +00:00
.payload(profile)
.send();
2021-09-04 09:26:04 +00:00
},
Err(e) => {
2021-10-14 06:34:22 +00:00
dart_notify(&token, UserNotification::UserProfileUpdated)
.error(e)
.send();
2021-09-04 09:26:04 +00:00
},
}
});
Ok(())
}
async fn update_user_on_server(&self, token: &str, params: UpdateUserParams) -> Result<(), UserError> {
let server = self.server.clone();
let token = token.to_owned();
2021-09-06 08:18:34 +00:00
let _ = tokio::spawn(async move {
2021-09-04 09:26:04 +00:00
match server.update_user(&token, params).await {
Ok(_) => {},
Err(e) => {
// TODO: retry?
log::error!("update user profile failed: {:?}", e);
},
}
2021-09-05 05:50:23 +00:00
})
.await;
2021-09-04 09:26:04 +00:00
Ok(())
}
async fn sign_out_on_server(&self, token: &str) -> Result<(), UserError> {
let server = self.server.clone();
let token = token.to_owned();
2021-09-06 08:18:34 +00:00
let _ = tokio::spawn(async move {
2021-09-04 09:26:04 +00:00
match server.sign_out(&token).await {
Ok(_) => {},
Err(e) => log::error!("Sign out failed: {:?}", e),
}
2021-09-05 05:50:23 +00:00
})
.await;
2021-09-04 09:26:04 +00:00
Ok(())
}
2021-09-01 08:08:32 +00:00
async fn save_user(&self, user: UserTable) -> Result<UserTable, UserError> {
let conn = self.db_connection()?;
2021-09-27 15:23:23 +00:00
let _ = diesel::insert_into(user_table::table)
.values(user.clone())
.execute(&*conn)?;
2021-09-01 08:08:32 +00:00
Ok(user)
}
fn set_session(&self, session: Option<Session>) -> Result<(), UserError> {
2021-11-03 07:37:38 +00:00
tracing::debug!("Set user session: {:?}", session);
match &session {
None => KV::remove(SESSION_CACHE_KEY).map_err(|e| UserError::new(ErrorCode::InternalError, &e))?,
2021-09-03 04:44:48 +00:00
Some(session) => KV::set_str(SESSION_CACHE_KEY, session.clone().into()),
}
2021-09-03 04:44:48 +00:00
*self.session.write() = session;
Ok(())
}
fn get_session(&self) -> Result<Session, UserError> {
2021-09-03 04:44:48 +00:00
let mut session = { (*self.session.read()).clone() };
if session.is_none() {
2021-09-03 04:44:48 +00:00
match KV::get_str(SESSION_CACHE_KEY) {
None => {},
Some(s) => {
session = Some(Session::from(s));
let _ = self.set_session(session.clone())?;
},
}
}
match session {
2021-09-16 04:35:55 +00:00
None => Err(UserError::unauthorized()),
Some(session) => Ok(session),
}
}
fn is_login(&self, email: &str) -> bool {
match self.get_session() {
Ok(session) => session.email == email,
Err(_) => false,
}
}
2021-09-17 11:03:46 +00:00
2021-10-05 09:54:11 +00:00
#[tracing::instrument(level = "debug", skip(self, token))]
2021-09-30 09:24:02 +00:00
pub async fn start_ws_connection(&self, token: &str) -> Result<(), UserError> {
2021-11-06 14:35:45 +00:00
if cfg!(feature = "http_server") {
let addr = format!("{}/{}", self.server.ws_addr(), token);
self.listen_on_websocket();
let _ = self.ws_controller.start_connect(addr).await?;
}
2021-09-17 11:03:46 +00:00
Ok(())
}
2021-10-05 09:54:11 +00:00
#[tracing::instrument(level = "debug", skip(self))]
fn listen_on_websocket(&self) {
let mut notify = self.ws_controller.state_subscribe();
let ws_controller = self.ws_controller.clone();
let _ = tokio::spawn(async move {
loop {
match notify.recv().await {
Ok(state) => {
2021-11-03 07:37:38 +00:00
tracing::info!("Websocket state changed: {}", state);
2021-10-05 09:54:11 +00:00
match state {
WsState::Init => {},
WsState::Connected(_) => {},
2021-10-14 06:34:22 +00:00
WsState::Disconnected(_) => match ws_controller.retry().await {
Ok(_) => {},
Err(e) => {
log::error!("Retry websocket connect failed: {:?}", e);
},
2021-10-05 09:54:11 +00:00
},
}
},
Err(e) => {
log::error!("Websocket state notify error: {:?}", e);
break;
},
}
}
});
}
2021-08-31 15:01:46 +00:00
}
2021-09-27 15:23:23 +00:00
pub async fn update_user(
_server: Server,
pool: Arc<ConnectionPool>,
params: UpdateUserParams,
) -> Result<(), UserError> {
2021-08-31 15:01:46 +00:00
let changeset = UserTableChangeset::new(params);
let conn = pool.get()?;
2021-09-07 09:12:03 +00:00
diesel_update_table!(user_table, changeset, &*conn);
2021-08-31 15:01:46 +00:00
Ok(())
2021-07-16 15:18:12 +00:00
}
2021-07-11 09:38:03 +00:00
2021-07-13 15:08:20 +00:00
impl UserDatabaseConnection for UserSession {
fn get_connection(&self) -> Result<DBConnection, String> { self.db_connection().map_err(|e| format!("{:?}", e)) }
2021-07-13 15:08:20 +00:00
}
const SESSION_CACHE_KEY: &str = "session_cache_key";
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
struct Session {
user_id: String,
token: String,
email: String,
}
impl Session {
pub fn new(user_id: &str, token: &str, email: &str) -> Self {
Self {
user_id: user_id.to_owned(),
token: token.to_owned(),
email: email.to_owned(),
}
}
2021-09-07 15:30:43 +00:00
pub fn into_part(self) -> (String, String) { (self.user_id, self.token) }
}
impl std::convert::From<String> for Session {
fn from(s: String) -> Self {
match serde_json::from_str(&s) {
Ok(s) => s,
Err(e) => {
2021-09-01 08:08:32 +00:00
log::error!("Deserialize string to Session failed: {:?}", e);
Session::default()
},
}
}
}
impl std::convert::Into<String> for Session {
fn into(self) -> String {
match serde_json::to_string(&self) {
Ok(s) => s,
Err(e) => {
2021-09-01 08:08:32 +00:00
log::error!("Serialize session to string failed: {:?}", e);
"".to_string()
},
}
}
}