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

214 lines
6.8 KiB
Rust
Raw Normal View History

use crate::{
entities::{SignInParams, SignUpParams, UpdateUserParams, UpdateUserRequest, UserDetail},
errors::{ErrorBuilder, UserErrCode, UserError},
event::UserEvent::*,
services::{
user::{construct_server, database::UserDB, UserServer},
workspace::WorkspaceAction,
},
sql_tables::{UserTable, UserTableChangeset},
};
2021-08-20 14:00:03 +00:00
use bytes::Bytes;
use flowy_database::{
query_dsl::*,
schema::{user_table, user_table::dsl},
DBConnection,
ExpressionMethods,
2021-07-13 15:08:20 +00:00
UserDatabaseConnection,
};
use flowy_dispatch::prelude::{EventDispatch, ModuleRequest, ToBytes};
use flowy_infra::kv::KVStore;
2021-07-14 15:00:58 +00:00
use std::sync::{Arc, RwLock};
const DEFAULT_WORKSPACE_NAME: &'static str = "My workspace";
const DEFAULT_WORKSPACE_DESC: &'static str = "This is your first workspace";
const DEFAULT_WORKSPACE: &'static str = "Default_Workspace";
pub struct UserSessionConfig {
root_dir: String,
}
impl UserSessionConfig {
pub fn new(root_dir: &str) -> Self {
Self {
root_dir: root_dir.to_owned(),
}
}
}
pub struct UserSession {
database: UserDB,
config: UserSessionConfig,
workspace: Arc<dyn WorkspaceAction + Send + Sync>,
2021-07-14 15:00:58 +00:00
server: Arc<dyn UserServer + Send + Sync>,
user_id: RwLock<Option<String>>,
}
impl UserSession {
pub fn new<R>(config: UserSessionConfig, workspace: Arc<R>) -> Self
where
R: 'static + WorkspaceAction + Send + Sync,
{
let db = UserDB::new(&config.root_dir);
let server = construct_server();
Self {
database: db,
config,
workspace,
2021-07-14 15:00:58 +00:00
server,
user_id: RwLock::new(None),
}
}
2021-07-16 15:18:12 +00:00
pub fn get_db_connection(&self) -> Result<DBConnection, UserError> {
let user_id = self.get_user_id()?;
2021-07-16 15:18:12 +00:00
self.database.get_connection(&user_id)
}
pub async fn sign_in(&self, params: SignInParams) -> Result<UserTable, UserError> {
let resp = self.server.sign_in(params).await?;
let _ = self.set_user_id(Some(resp.uid.clone()))?;
let user_table = self.save_user(resp.into()).await?;
Ok(user_table)
}
pub async fn sign_up(&self, params: SignUpParams) -> Result<UserTable, UserError> {
let resp = self.server.sign_up(params).await?;
let _ = self.set_user_id(Some(resp.uid.clone()))?;
let user_table = self.save_user(resp.into()).await?;
Ok(user_table)
}
2021-07-16 15:18:12 +00:00
pub fn sign_out(&self) -> Result<(), UserError> {
let user_id = self.get_user_id()?;
2021-07-11 09:38:03 +00:00
let conn = self.get_db_connection()?;
let _ = diesel::delete(dsl::user_table.filter(dsl::id.eq(&user_id))).execute(&*conn)?;
let _ = self.server.sign_out(&user_id);
2021-07-18 01:03:21 +00:00
let _ = self.database.close_user_db(&user_id)?;
let _ = self.set_user_id(None)?;
2021-07-11 09:38:03 +00:00
Ok(())
}
async fn save_user(&self, mut user: UserTable) -> Result<UserTable, UserError> {
if user.workspace.is_empty() {
log::info!("Try to create user default workspace");
let workspace_id = self.create_default_workspace_if_need(&user.id).await?;
user.workspace = workspace_id;
}
2021-07-16 15:18:12 +00:00
let conn = self.get_db_connection()?;
let _ = diesel::insert_into(user_table::table)
.values(user.clone())
.execute(&*conn)?;
Ok(user)
}
pub fn update_user(&self, params: UpdateUserParams) -> Result<UserDetail, UserError> {
let changeset = UserTableChangeset::new(params);
let conn = self.get_db_connection()?;
diesel_update_table!(user_table, changeset, conn);
2021-07-16 15:18:12 +00:00
let user_detail = self.user_detail()?;
Ok(user_detail)
}
2021-07-16 15:18:12 +00:00
pub fn user_detail(&self) -> Result<UserDetail, UserError> {
let user_id = self.get_user_id()?;
let user = dsl::user_table
2021-07-11 09:38:03 +00:00
.filter(user_table::id.eq(&user_id))
.first::<UserTable>(&*(self.get_db_connection()?))?;
let _ = self.server.get_user_info(&user_id);
2021-07-11 09:38:03 +00:00
Ok(UserDetail::from(user))
}
pub fn set_user_id(&self, user_id: Option<String>) -> Result<(), UserError> {
log::trace!("Set user id: {:?}", user_id);
KVStore::set_str(USER_ID_CACHE_KEY, user_id.clone().unwrap_or("".to_owned()));
match self.user_id.write() {
Ok(mut write_guard) => {
*write_guard = user_id;
Ok(())
},
2021-07-25 00:13:59 +00:00
Err(e) => Err(ErrorBuilder::new(UserErrCode::WriteCurrentIdFailed)
.error(e)
.build()),
}
}
2021-07-23 09:30:33 +00:00
pub fn get_user_dir(&self) -> Result<String, UserError> {
let user_id = self.get_user_id()?;
Ok(format!("{}/{}", self.config.root_dir, user_id))
}
pub fn get_user_id(&self) -> Result<String, UserError> {
let mut user_id = {
let read_guard = self.user_id.read().map_err(|e| {
2021-07-25 00:13:59 +00:00
ErrorBuilder::new(UserErrCode::ReadCurrentIdFailed)
.error(e)
.build()
})?;
(*read_guard).clone()
};
if user_id.is_none() {
user_id = KVStore::get_str(USER_ID_CACHE_KEY);
let _ = self.set_user_id(user_id.clone())?;
}
match user_id {
2021-07-25 00:13:59 +00:00
None => Err(ErrorBuilder::new(UserErrCode::UserNotLoginYet).build()),
Some(user_id) => Ok(user_id),
}
}
pub async fn set_current_workspace(&self, workspace_id: &str) -> Result<(), UserError> {
let user_id = self.get_user_id()?;
2021-08-20 14:00:03 +00:00
let payload: Bytes = UpdateUserRequest::new(&user_id)
.workspace(workspace_id)
.into_bytes()
.unwrap();
2021-07-16 15:18:12 +00:00
let request = ModuleRequest::new(UpdateUser).payload(payload);
let _ = EventDispatch::async_send(request)
2021-07-16 15:18:12 +00:00
.await
.parse::<UserDetail, UserError>()
.unwrap()?;
2021-07-16 15:18:12 +00:00
Ok(())
}
async fn create_default_workspace_if_need(&self, user_id: &str) -> Result<String, UserError> {
let key = format!("{}{}", user_id, DEFAULT_WORKSPACE);
if KVStore::get_bool(&key).unwrap_or(false) {
2021-07-25 00:13:59 +00:00
return Err(ErrorBuilder::new(UserErrCode::DefaultWorkspaceAlreadyExist).build());
}
KVStore::set_bool(&key, true);
log::debug!("Create user:{} default workspace", user_id);
let workspace_id = self
.workspace
.create_workspace(DEFAULT_WORKSPACE_NAME, DEFAULT_WORKSPACE_DESC, user_id)
.await?;
Ok(workspace_id)
}
2021-07-16 15:18:12 +00:00
}
2021-07-11 09:38:03 +00:00
2021-07-17 02:26:05 +00:00
pub fn current_user_id() -> Result<String, UserError> {
match KVStore::get_str(USER_ID_CACHE_KEY) {
2021-07-25 00:13:59 +00:00
None => Err(ErrorBuilder::new(UserErrCode::UserNotLoginYet).build()),
2021-07-16 15:18:12 +00:00
Some(user_id) => Ok(user_id),
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.get_db_connection().map_err(|e| format!("{:?}", e))
}
}
const USER_ID_CACHE_KEY: &str = "user_id";