2022-01-14 20:52:03 +08:00
|
|
|
use bytes::Bytes;
|
2022-01-27 20:39:54 +08:00
|
|
|
use flowy_database::ConnectionPool;
|
2022-10-22 21:57:44 +08:00
|
|
|
|
2022-10-13 23:29:37 +08:00
|
|
|
use flowy_document::DocumentManager;
|
2022-10-22 21:57:44 +08:00
|
|
|
use flowy_folder::entities::{ViewDataFormatPB, ViewLayoutTypePB, ViewPB};
|
2022-03-06 09:03:02 +08:00
|
|
|
use flowy_folder::manager::{ViewDataProcessor, ViewDataProcessorMap};
|
2022-01-27 20:39:54 +08:00
|
|
|
use flowy_folder::{
|
2022-01-14 20:52:03 +08:00
|
|
|
errors::{internal_error, FlowyError},
|
2022-01-30 10:33:21 +08:00
|
|
|
event_map::{FolderCouldServiceV1, WorkspaceDatabase, WorkspaceUser},
|
2022-03-05 21:15:10 +08:00
|
|
|
manager::FolderManager,
|
2022-01-10 23:45:59 +08:00
|
|
|
};
|
2022-09-23 11:23:35 +08:00
|
|
|
use flowy_grid::entities::GridLayout;
|
2022-03-15 19:00:28 +08:00
|
|
|
use flowy_grid::manager::{make_grid_view_data, GridManager};
|
2022-08-10 17:59:28 +08:00
|
|
|
use flowy_grid::util::{make_default_board, make_default_grid};
|
2022-06-15 15:13:50 +08:00
|
|
|
use flowy_grid_data_model::revision::BuildGridContext;
|
2022-02-07 10:37:01 +08:00
|
|
|
use flowy_net::ClientServerConfiguration;
|
2022-01-14 20:52:03 +08:00
|
|
|
use flowy_net::{
|
2022-02-07 14:40:45 +08:00
|
|
|
http_server::folder::FolderHttpCloudService, local_server::LocalServer, ws::connection::FlowyWebSocketConnect,
|
2022-01-14 20:52:03 +08:00
|
|
|
};
|
2022-03-19 16:52:28 +08:00
|
|
|
use flowy_revision::{RevisionWebSocket, WSStateReceiver};
|
2022-11-02 10:21:10 +08:00
|
|
|
use flowy_sync::entities::revision::Revision;
|
2022-06-15 15:13:50 +08:00
|
|
|
use flowy_sync::entities::ws_data::ClientRevisionWSData;
|
2022-01-11 13:34:45 +08:00
|
|
|
use flowy_user::services::UserSession;
|
2022-01-24 16:27:40 +08:00
|
|
|
use futures_core::future::BoxFuture;
|
2022-03-06 09:03:02 +08:00
|
|
|
use lib_infra::future::{BoxResultFuture, FutureResult};
|
2022-01-22 18:48:43 +08:00
|
|
|
use lib_ws::{WSChannel, WSMessageReceiver, WebSocketRawMessage};
|
2022-03-05 22:30:42 +08:00
|
|
|
use std::collections::HashMap;
|
2022-03-15 19:00:28 +08:00
|
|
|
use std::convert::TryFrom;
|
2022-01-14 20:52:03 +08:00
|
|
|
use std::{convert::TryInto, sync::Arc};
|
2022-01-10 23:45:59 +08:00
|
|
|
|
2022-01-20 23:51:11 +08:00
|
|
|
pub struct FolderDepsResolver();
|
|
|
|
impl FolderDepsResolver {
|
2022-01-23 22:33:47 +08:00
|
|
|
pub async fn resolve(
|
2022-01-13 10:53:30 +08:00
|
|
|
local_server: Option<Arc<LocalServer>>,
|
2022-01-10 23:45:59 +08:00
|
|
|
user_session: Arc<UserSession>,
|
|
|
|
server_config: &ClientServerConfiguration,
|
2022-03-05 22:30:42 +08:00
|
|
|
ws_conn: &Arc<FlowyWebSocketConnect>,
|
2022-10-13 23:29:37 +08:00
|
|
|
text_block_manager: &Arc<DocumentManager>,
|
2022-03-05 22:30:42 +08:00
|
|
|
grid_manager: &Arc<GridManager>,
|
2022-01-14 20:52:03 +08:00
|
|
|
) -> Arc<FolderManager> {
|
2022-01-10 23:45:59 +08:00
|
|
|
let user: Arc<dyn WorkspaceUser> = Arc::new(WorkspaceUserImpl(user_session.clone()));
|
|
|
|
let database: Arc<dyn WorkspaceDatabase> = Arc::new(WorkspaceDatabaseImpl(user_session));
|
2022-10-13 23:29:37 +08:00
|
|
|
let web_socket = Arc::new(FolderRevisionWebSocket(ws_conn.clone()));
|
2022-01-17 11:55:36 +08:00
|
|
|
let cloud_service: Arc<dyn FolderCouldServiceV1> = match local_server {
|
2022-02-07 10:37:01 +08:00
|
|
|
None => Arc::new(FolderHttpCloudService::new(server_config.clone())),
|
2022-01-13 10:53:30 +08:00
|
|
|
Some(local_server) => local_server,
|
|
|
|
};
|
2022-01-14 20:52:03 +08:00
|
|
|
|
2022-03-10 17:14:10 +08:00
|
|
|
let view_data_processor = make_view_data_processor(text_block_manager.clone(), grid_manager.clone());
|
2022-03-06 09:03:02 +08:00
|
|
|
let folder_manager =
|
|
|
|
Arc::new(FolderManager::new(user.clone(), cloud_service, database, view_data_processor, web_socket).await);
|
2022-01-24 16:27:40 +08:00
|
|
|
|
|
|
|
if let (Ok(user_id), Ok(token)) = (user.user_id(), user.token()) {
|
|
|
|
match folder_manager.initialize(&user_id, &token).await {
|
2022-01-24 17:35:58 +08:00
|
|
|
Ok(_) => {}
|
2022-01-24 16:27:40 +08:00
|
|
|
Err(e) => tracing::error!("Initialize folder manager failed: {}", e),
|
|
|
|
}
|
|
|
|
}
|
2022-01-20 23:51:11 +08:00
|
|
|
|
2022-01-14 20:52:03 +08:00
|
|
|
let receiver = Arc::new(FolderWSMessageReceiverImpl(folder_manager.clone()));
|
|
|
|
ws_conn.add_ws_message_receiver(receiver).unwrap();
|
|
|
|
folder_manager
|
2022-01-10 23:45:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-10 17:14:10 +08:00
|
|
|
fn make_view_data_processor(
|
2022-10-22 21:57:44 +08:00
|
|
|
document_manager: Arc<DocumentManager>,
|
2022-03-10 17:14:10 +08:00
|
|
|
grid_manager: Arc<GridManager>,
|
|
|
|
) -> ViewDataProcessorMap {
|
2022-10-22 21:57:44 +08:00
|
|
|
let mut map: HashMap<ViewDataFormatPB, Arc<dyn ViewDataProcessor + Send + Sync>> = HashMap::new();
|
2022-03-05 22:30:42 +08:00
|
|
|
|
2022-10-22 21:57:44 +08:00
|
|
|
let document_processor = Arc::new(DocumentViewDataProcessor(document_manager));
|
|
|
|
document_processor.data_types().into_iter().for_each(|data_type| {
|
|
|
|
map.insert(data_type, document_processor.clone());
|
|
|
|
});
|
2022-03-05 22:30:42 +08:00
|
|
|
|
2022-10-22 21:57:44 +08:00
|
|
|
let grid_data_impl = Arc::new(GridViewDataProcessor(grid_manager));
|
|
|
|
grid_data_impl.data_types().into_iter().for_each(|data_type| {
|
|
|
|
map.insert(data_type, grid_data_impl.clone());
|
|
|
|
});
|
2022-03-05 22:30:42 +08:00
|
|
|
|
|
|
|
Arc::new(map)
|
|
|
|
}
|
|
|
|
|
2022-01-10 23:45:59 +08:00
|
|
|
struct WorkspaceDatabaseImpl(Arc<UserSession>);
|
|
|
|
impl WorkspaceDatabase for WorkspaceDatabaseImpl {
|
|
|
|
fn db_pool(&self) -> Result<Arc<ConnectionPool>, FlowyError> {
|
|
|
|
self.0.db_pool().map_err(|e| FlowyError::internal().context(e))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct WorkspaceUserImpl(Arc<UserSession>);
|
|
|
|
impl WorkspaceUser for WorkspaceUserImpl {
|
2022-01-24 17:35:58 +08:00
|
|
|
fn user_id(&self) -> Result<String, FlowyError> {
|
|
|
|
self.0.user_id().map_err(|e| FlowyError::internal().context(e))
|
|
|
|
}
|
2022-01-10 23:45:59 +08:00
|
|
|
|
2022-01-24 17:35:58 +08:00
|
|
|
fn token(&self) -> Result<String, FlowyError> {
|
|
|
|
self.0.token().map_err(|e| FlowyError::internal().context(e))
|
|
|
|
}
|
2022-01-10 23:45:59 +08:00
|
|
|
}
|
2022-01-14 20:52:03 +08:00
|
|
|
|
2022-10-13 23:29:37 +08:00
|
|
|
struct FolderRevisionWebSocket(Arc<FlowyWebSocketConnect>);
|
|
|
|
impl RevisionWebSocket for FolderRevisionWebSocket {
|
2022-01-24 16:27:40 +08:00
|
|
|
fn send(&self, data: ClientRevisionWSData) -> BoxResultFuture<(), FlowyError> {
|
2022-01-14 20:52:03 +08:00
|
|
|
let bytes: Bytes = data.try_into().unwrap();
|
|
|
|
let msg = WebSocketRawMessage {
|
2022-01-22 18:48:43 +08:00
|
|
|
channel: WSChannel::Folder,
|
2022-01-14 20:52:03 +08:00
|
|
|
data: bytes.to_vec(),
|
|
|
|
};
|
2022-01-24 16:27:40 +08:00
|
|
|
|
|
|
|
let ws_conn = self.0.clone();
|
|
|
|
Box::pin(async move {
|
|
|
|
match ws_conn.web_socket().await? {
|
2022-01-24 17:35:58 +08:00
|
|
|
None => {}
|
2022-01-24 16:27:40 +08:00
|
|
|
Some(sender) => {
|
|
|
|
sender.send(msg).map_err(internal_error)?;
|
2022-01-24 17:35:58 +08:00
|
|
|
}
|
2022-01-24 16:27:40 +08:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
})
|
2022-01-14 20:52:03 +08:00
|
|
|
}
|
|
|
|
|
2022-01-24 16:27:40 +08:00
|
|
|
fn subscribe_state_changed(&self) -> BoxFuture<WSStateReceiver> {
|
|
|
|
let ws_conn = self.0.clone();
|
|
|
|
Box::pin(async move { ws_conn.subscribe_websocket_state().await })
|
|
|
|
}
|
2022-01-14 20:52:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
struct FolderWSMessageReceiverImpl(Arc<FolderManager>);
|
|
|
|
impl WSMessageReceiver for FolderWSMessageReceiverImpl {
|
2022-01-24 17:35:58 +08:00
|
|
|
fn source(&self) -> WSChannel {
|
|
|
|
WSChannel::Folder
|
|
|
|
}
|
2022-01-14 20:52:03 +08:00
|
|
|
fn receive_message(&self, msg: WebSocketRawMessage) {
|
|
|
|
let handler = self.0.clone();
|
|
|
|
tokio::spawn(async move {
|
|
|
|
handler.did_receive_ws_data(Bytes::from(msg.data)).await;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2022-03-05 22:30:42 +08:00
|
|
|
|
2022-10-13 23:29:37 +08:00
|
|
|
struct DocumentViewDataProcessor(Arc<DocumentManager>);
|
|
|
|
impl ViewDataProcessor for DocumentViewDataProcessor {
|
2022-10-22 21:57:44 +08:00
|
|
|
fn create_view(
|
2022-09-23 11:23:35 +08:00
|
|
|
&self,
|
2022-11-06 09:59:53 +08:00
|
|
|
_user_id: &str,
|
2022-09-23 11:23:35 +08:00
|
|
|
view_id: &str,
|
|
|
|
layout: ViewLayoutTypePB,
|
2022-10-22 21:57:44 +08:00
|
|
|
view_data: Bytes,
|
2022-09-23 11:23:35 +08:00
|
|
|
) -> FutureResult<(), FlowyError> {
|
|
|
|
// Only accept Document type
|
|
|
|
debug_assert_eq!(layout, ViewLayoutTypePB::Document);
|
2022-11-02 17:15:27 +08:00
|
|
|
let revision = Revision::initial_revision(view_id, view_data);
|
2022-03-06 09:03:02 +08:00
|
|
|
let view_id = view_id.to_string();
|
2022-03-12 09:30:13 +08:00
|
|
|
let manager = self.0.clone();
|
2022-10-22 21:57:44 +08:00
|
|
|
|
2022-03-06 09:03:02 +08:00
|
|
|
FutureResult::new(async move {
|
2022-11-02 10:21:10 +08:00
|
|
|
let _ = manager.create_document(view_id, vec![revision]).await?;
|
2022-03-06 09:03:02 +08:00
|
|
|
Ok(())
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-10-22 21:57:44 +08:00
|
|
|
fn close_view(&self, view_id: &str) -> FutureResult<(), FlowyError> {
|
2022-03-10 17:14:10 +08:00
|
|
|
let manager = self.0.clone();
|
2022-03-06 09:03:02 +08:00
|
|
|
let view_id = view_id.to_string();
|
|
|
|
FutureResult::new(async move {
|
2022-10-13 23:29:37 +08:00
|
|
|
let _ = manager.close_document_editor(view_id)?;
|
2022-03-06 09:03:02 +08:00
|
|
|
Ok(())
|
|
|
|
})
|
2022-03-05 22:30:42 +08:00
|
|
|
}
|
|
|
|
|
2022-10-22 21:57:44 +08:00
|
|
|
fn get_view_data(&self, view: &ViewPB) -> FutureResult<Bytes, FlowyError> {
|
|
|
|
let view_id = view.id.clone();
|
2022-03-10 17:14:10 +08:00
|
|
|
let manager = self.0.clone();
|
2022-03-06 09:03:02 +08:00
|
|
|
FutureResult::new(async move {
|
2022-10-13 23:29:37 +08:00
|
|
|
let editor = manager.open_document_editor(view_id).await?;
|
2022-10-22 21:57:44 +08:00
|
|
|
let document_data = Bytes::from(editor.duplicate().await?);
|
|
|
|
Ok(document_data)
|
2022-03-06 09:03:02 +08:00
|
|
|
})
|
2022-03-05 22:30:42 +08:00
|
|
|
}
|
|
|
|
|
2022-08-10 17:59:28 +08:00
|
|
|
fn create_default_view(
|
|
|
|
&self,
|
|
|
|
user_id: &str,
|
|
|
|
view_id: &str,
|
2022-08-18 21:52:47 +08:00
|
|
|
layout: ViewLayoutTypePB,
|
2022-10-22 21:57:44 +08:00
|
|
|
_data_format: ViewDataFormatPB,
|
2022-08-10 17:59:28 +08:00
|
|
|
) -> FutureResult<Bytes, FlowyError> {
|
2022-08-18 23:12:26 +08:00
|
|
|
debug_assert_eq!(layout, ViewLayoutTypePB::Document);
|
2022-11-06 09:59:53 +08:00
|
|
|
let _user_id = user_id.to_string();
|
2022-03-12 09:30:13 +08:00
|
|
|
let view_id = view_id.to_string();
|
|
|
|
let manager = self.0.clone();
|
2022-10-22 21:57:44 +08:00
|
|
|
let document_content = self.0.initial_document_content();
|
2022-03-12 09:30:13 +08:00
|
|
|
FutureResult::new(async move {
|
2022-10-22 21:57:44 +08:00
|
|
|
let delta_data = Bytes::from(document_content);
|
2022-11-02 17:15:27 +08:00
|
|
|
let revision = Revision::initial_revision(&view_id, delta_data.clone());
|
2022-11-02 10:21:10 +08:00
|
|
|
let _ = manager.create_document(view_id, vec![revision]).await?;
|
2022-03-15 19:00:28 +08:00
|
|
|
Ok(delta_data)
|
2022-03-12 09:30:13 +08:00
|
|
|
})
|
2022-03-05 22:30:42 +08:00
|
|
|
}
|
|
|
|
|
2022-06-15 11:43:24 +08:00
|
|
|
fn create_view_from_delta_data(
|
2022-03-15 19:00:28 +08:00
|
|
|
&self,
|
|
|
|
_user_id: &str,
|
|
|
|
_view_id: &str,
|
|
|
|
data: Vec<u8>,
|
2022-09-23 11:23:35 +08:00
|
|
|
layout: ViewLayoutTypePB,
|
2022-03-15 19:00:28 +08:00
|
|
|
) -> FutureResult<Bytes, FlowyError> {
|
2022-09-23 11:23:35 +08:00
|
|
|
debug_assert_eq!(layout, ViewLayoutTypePB::Document);
|
2022-03-15 19:00:28 +08:00
|
|
|
FutureResult::new(async move { Ok(Bytes::from(data)) })
|
|
|
|
}
|
|
|
|
|
2022-10-22 21:57:44 +08:00
|
|
|
fn data_types(&self) -> Vec<ViewDataFormatPB> {
|
|
|
|
vec![ViewDataFormatPB::DeltaFormat, ViewDataFormatPB::TreeFormat]
|
2022-03-05 22:30:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-11 21:36:00 +08:00
|
|
|
struct GridViewDataProcessor(Arc<GridManager>);
|
|
|
|
impl ViewDataProcessor for GridViewDataProcessor {
|
2022-10-22 21:57:44 +08:00
|
|
|
fn create_view(
|
2022-09-23 11:23:35 +08:00
|
|
|
&self,
|
2022-11-06 09:59:53 +08:00
|
|
|
_user_id: &str,
|
2022-09-23 11:23:35 +08:00
|
|
|
view_id: &str,
|
|
|
|
_layout: ViewLayoutTypePB,
|
|
|
|
delta_data: Bytes,
|
|
|
|
) -> FutureResult<(), FlowyError> {
|
2022-11-02 17:15:27 +08:00
|
|
|
let revision = Revision::initial_revision(view_id, delta_data);
|
2022-03-06 09:03:02 +08:00
|
|
|
let view_id = view_id.to_string();
|
2022-03-12 09:30:13 +08:00
|
|
|
let grid_manager = self.0.clone();
|
2022-03-06 09:03:02 +08:00
|
|
|
FutureResult::new(async move {
|
2022-11-02 10:21:10 +08:00
|
|
|
let _ = grid_manager.create_grid(view_id, vec![revision]).await?;
|
2022-03-06 09:03:02 +08:00
|
|
|
Ok(())
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-10-22 21:57:44 +08:00
|
|
|
fn close_view(&self, view_id: &str) -> FutureResult<(), FlowyError> {
|
2022-03-06 09:03:02 +08:00
|
|
|
let grid_manager = self.0.clone();
|
|
|
|
let view_id = view_id.to_string();
|
|
|
|
FutureResult::new(async move {
|
2022-06-29 16:55:52 +08:00
|
|
|
let _ = grid_manager.close_grid(view_id).await?;
|
2022-03-06 09:03:02 +08:00
|
|
|
Ok(())
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-10-22 21:57:44 +08:00
|
|
|
fn get_view_data(&self, view: &ViewPB) -> FutureResult<Bytes, FlowyError> {
|
2022-03-06 09:03:02 +08:00
|
|
|
let grid_manager = self.0.clone();
|
2022-10-22 21:57:44 +08:00
|
|
|
let view_id = view.id.clone();
|
2022-03-06 09:03:02 +08:00
|
|
|
FutureResult::new(async move {
|
|
|
|
let editor = grid_manager.open_grid(view_id).await?;
|
2022-06-06 20:06:08 +08:00
|
|
|
let delta_bytes = editor.duplicate_grid().await?;
|
|
|
|
Ok(delta_bytes.into())
|
2022-03-06 09:03:02 +08:00
|
|
|
})
|
2022-03-05 22:30:42 +08:00
|
|
|
}
|
|
|
|
|
2022-08-10 17:59:28 +08:00
|
|
|
fn create_default_view(
|
|
|
|
&self,
|
|
|
|
user_id: &str,
|
|
|
|
view_id: &str,
|
2022-08-18 19:32:08 +08:00
|
|
|
layout: ViewLayoutTypePB,
|
2022-10-22 21:57:44 +08:00
|
|
|
data_format: ViewDataFormatPB,
|
2022-08-10 17:59:28 +08:00
|
|
|
) -> FutureResult<Bytes, FlowyError> {
|
2022-10-22 21:57:44 +08:00
|
|
|
debug_assert_eq!(data_format, ViewDataFormatPB::DatabaseFormat);
|
2022-09-23 11:23:35 +08:00
|
|
|
let (build_context, layout) = match layout {
|
|
|
|
ViewLayoutTypePB::Grid => (make_default_grid(), GridLayout::Table),
|
|
|
|
ViewLayoutTypePB::Board => (make_default_board(), GridLayout::Board),
|
2022-08-18 19:32:08 +08:00
|
|
|
ViewLayoutTypePB::Document => {
|
|
|
|
return FutureResult::new(async move {
|
|
|
|
Err(FlowyError::internal().context(format!("Can't handle {:?} layout type", layout)))
|
|
|
|
});
|
|
|
|
}
|
2022-08-10 17:59:28 +08:00
|
|
|
};
|
2022-09-23 11:23:35 +08:00
|
|
|
|
2022-03-12 09:30:13 +08:00
|
|
|
let user_id = user_id.to_string();
|
|
|
|
let view_id = view_id.to_string();
|
|
|
|
let grid_manager = self.0.clone();
|
2022-09-23 11:23:35 +08:00
|
|
|
FutureResult::new(
|
|
|
|
async move { make_grid_view_data(&user_id, &view_id, layout, grid_manager, build_context).await },
|
|
|
|
)
|
2022-03-15 19:00:28 +08:00
|
|
|
}
|
2022-03-12 09:30:13 +08:00
|
|
|
|
2022-06-15 11:43:24 +08:00
|
|
|
fn create_view_from_delta_data(
|
|
|
|
&self,
|
|
|
|
user_id: &str,
|
|
|
|
view_id: &str,
|
|
|
|
data: Vec<u8>,
|
2022-09-23 11:23:35 +08:00
|
|
|
layout: ViewLayoutTypePB,
|
2022-06-15 11:43:24 +08:00
|
|
|
) -> FutureResult<Bytes, FlowyError> {
|
2022-03-15 19:00:28 +08:00
|
|
|
let user_id = user_id.to_string();
|
|
|
|
let view_id = view_id.to_string();
|
|
|
|
let grid_manager = self.0.clone();
|
2022-03-12 09:30:13 +08:00
|
|
|
|
2022-09-23 11:23:35 +08:00
|
|
|
let layout = match layout {
|
|
|
|
ViewLayoutTypePB::Grid => GridLayout::Table,
|
|
|
|
ViewLayoutTypePB::Board => GridLayout::Board,
|
|
|
|
ViewLayoutTypePB::Document => {
|
|
|
|
return FutureResult::new(async move {
|
|
|
|
Err(FlowyError::internal().context(format!("Can't handle {:?} layout type", layout)))
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-03-15 19:00:28 +08:00
|
|
|
FutureResult::new(async move {
|
|
|
|
let bytes = Bytes::from(data);
|
|
|
|
let build_context = BuildGridContext::try_from(bytes)?;
|
2022-09-23 11:23:35 +08:00
|
|
|
make_grid_view_data(&user_id, &view_id, layout, grid_manager, build_context).await
|
2022-03-12 09:30:13 +08:00
|
|
|
})
|
2022-03-05 22:30:42 +08:00
|
|
|
}
|
|
|
|
|
2022-10-22 21:57:44 +08:00
|
|
|
fn data_types(&self) -> Vec<ViewDataFormatPB> {
|
|
|
|
vec![ViewDataFormatPB::DatabaseFormat]
|
2022-03-05 22:30:42 +08:00
|
|
|
}
|
|
|
|
}
|