script.rs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449
  1. use flowy_folder::event_map::FolderEvent::*;
  2. use flowy_folder::{errors::ErrorCode, services::folder_editor::FolderEditor};
  3. use flowy_folder_data_model::entities::view::{RepeatedViewId, ViewId};
  4. use flowy_folder_data_model::entities::workspace::WorkspaceId;
  5. use flowy_folder_data_model::entities::{
  6. app::{App, RepeatedApp},
  7. trash::Trash,
  8. view::{RepeatedView, View, ViewDataType},
  9. workspace::Workspace,
  10. };
  11. use flowy_folder_data_model::entities::{
  12. app::{AppId, CreateAppPayload, UpdateAppPayload},
  13. trash::{RepeatedTrash, TrashId, TrashType},
  14. view::{CreateViewPayload, UpdateViewPayload},
  15. workspace::{CreateWorkspacePayload, RepeatedWorkspace},
  16. };
  17. use flowy_revision::disk::RevisionState;
  18. use flowy_revision::REVISION_WRITE_INTERVAL_IN_MILLIS;
  19. use flowy_sync::entities::text_block_info::TextBlockInfo;
  20. use flowy_test::{event_builder::*, FlowySDKTest};
  21. use std::{sync::Arc, time::Duration};
  22. use tokio::time::sleep;
  23. pub enum FolderScript {
  24. // Workspace
  25. ReadAllWorkspaces,
  26. CreateWorkspace {
  27. name: String,
  28. desc: String,
  29. },
  30. // AssertWorkspaceRevisionJson(String),
  31. AssertWorkspace(Workspace),
  32. ReadWorkspace(Option<String>),
  33. // App
  34. CreateApp {
  35. name: String,
  36. desc: String,
  37. },
  38. // AssertAppRevisionJson(String),
  39. AssertApp(App),
  40. ReadApp(String),
  41. UpdateApp {
  42. name: Option<String>,
  43. desc: Option<String>,
  44. },
  45. DeleteApp,
  46. // View
  47. CreateView {
  48. name: String,
  49. desc: String,
  50. data_type: ViewDataType,
  51. },
  52. AssertView(View),
  53. ReadView(String),
  54. UpdateView {
  55. name: Option<String>,
  56. desc: Option<String>,
  57. },
  58. DeleteView,
  59. DeleteViews(Vec<String>),
  60. // Trash
  61. RestoreAppFromTrash,
  62. RestoreViewFromTrash,
  63. ReadTrash,
  64. DeleteAllTrash,
  65. // Sync
  66. AssertCurrentRevId(i64),
  67. AssertNextSyncRevId(Option<i64>),
  68. AssertRevisionState {
  69. rev_id: i64,
  70. state: RevisionState,
  71. },
  72. }
  73. pub struct FolderTest {
  74. pub sdk: FlowySDKTest,
  75. pub all_workspace: Vec<Workspace>,
  76. pub workspace: Workspace,
  77. pub app: App,
  78. pub view: View,
  79. pub trash: Vec<Trash>,
  80. // pub folder_editor:
  81. }
  82. impl FolderTest {
  83. pub async fn new() -> Self {
  84. let sdk = FlowySDKTest::default();
  85. let _ = sdk.init_user().await;
  86. let mut workspace = create_workspace(&sdk, "FolderWorkspace", "Folder test workspace").await;
  87. let mut app = create_app(&sdk, &workspace.id, "Folder App", "Folder test app").await;
  88. let view = create_view(
  89. &sdk,
  90. &app.id,
  91. "Folder View",
  92. "Folder test view",
  93. ViewDataType::TextBlock,
  94. )
  95. .await;
  96. app.belongings = RepeatedView {
  97. items: vec![view.clone()],
  98. };
  99. workspace.apps = RepeatedApp {
  100. items: vec![app.clone()],
  101. };
  102. Self {
  103. sdk,
  104. all_workspace: vec![],
  105. workspace,
  106. app,
  107. view,
  108. trash: vec![],
  109. }
  110. }
  111. pub async fn run_scripts(&mut self, scripts: Vec<FolderScript>) {
  112. for script in scripts {
  113. self.run_script(script).await;
  114. }
  115. }
  116. pub async fn run_script(&mut self, script: FolderScript) {
  117. let sdk = &self.sdk;
  118. let folder_editor: Arc<FolderEditor> = sdk.folder_manager.folder_editor().await;
  119. let rev_manager = folder_editor.rev_manager();
  120. let cache = rev_manager.revision_cache().await;
  121. match script {
  122. FolderScript::ReadAllWorkspaces => {
  123. let all_workspace = read_workspace(sdk, None).await;
  124. self.all_workspace = all_workspace;
  125. }
  126. FolderScript::CreateWorkspace { name, desc } => {
  127. let workspace = create_workspace(sdk, &name, &desc).await;
  128. self.workspace = workspace;
  129. }
  130. // FolderScript::AssertWorkspaceRevisionJson(expected_json) => {
  131. // let workspace = read_workspace(sdk, Some(self.workspace.id.clone()))
  132. // .await
  133. // .pop()
  134. // .unwrap();
  135. // let workspace_revision: WorkspaceRevision = workspace.into();
  136. // let json = serde_json::to_string(&workspace_revision).unwrap();
  137. // assert_eq!(json, expected_json);
  138. // }
  139. FolderScript::AssertWorkspace(workspace) => {
  140. assert_eq!(self.workspace, workspace);
  141. }
  142. FolderScript::ReadWorkspace(workspace_id) => {
  143. let workspace = read_workspace(sdk, workspace_id).await.pop().unwrap();
  144. self.workspace = workspace;
  145. }
  146. FolderScript::CreateApp { name, desc } => {
  147. let app = create_app(sdk, &self.workspace.id, &name, &desc).await;
  148. self.app = app;
  149. }
  150. // FolderScript::AssertAppRevisionJson(expected_json) => {
  151. // let app_revision: AppRevision = self.app.clone().into();
  152. // let json = serde_json::to_string(&app_revision).unwrap();
  153. // assert_eq!(json, expected_json);
  154. // }
  155. FolderScript::AssertApp(app) => {
  156. assert_eq!(self.app, app);
  157. }
  158. FolderScript::ReadApp(app_id) => {
  159. let app = read_app(sdk, &app_id).await;
  160. self.app = app;
  161. }
  162. FolderScript::UpdateApp { name, desc } => {
  163. update_app(sdk, &self.app.id, name, desc).await;
  164. }
  165. FolderScript::DeleteApp => {
  166. delete_app(sdk, &self.app.id).await;
  167. }
  168. FolderScript::CreateView { name, desc, data_type } => {
  169. let view = create_view(sdk, &self.app.id, &name, &desc, data_type).await;
  170. self.view = view;
  171. }
  172. FolderScript::AssertView(view) => {
  173. assert_eq!(self.view, view);
  174. }
  175. FolderScript::ReadView(view_id) => {
  176. let view = read_view(sdk, &view_id).await;
  177. self.view = view;
  178. }
  179. FolderScript::UpdateView { name, desc } => {
  180. update_view(sdk, &self.view.id, name, desc).await;
  181. }
  182. FolderScript::DeleteView => {
  183. delete_view(sdk, vec![self.view.id.clone()]).await;
  184. }
  185. FolderScript::DeleteViews(view_ids) => {
  186. delete_view(sdk, view_ids).await;
  187. }
  188. FolderScript::RestoreAppFromTrash => {
  189. restore_app_from_trash(sdk, &self.app.id).await;
  190. }
  191. FolderScript::RestoreViewFromTrash => {
  192. restore_view_from_trash(sdk, &self.view.id).await;
  193. }
  194. FolderScript::ReadTrash => {
  195. let trash = read_trash(sdk).await;
  196. self.trash = trash.into_inner();
  197. }
  198. FolderScript::DeleteAllTrash => {
  199. delete_all_trash(sdk).await;
  200. self.trash = vec![];
  201. }
  202. FolderScript::AssertRevisionState { rev_id, state } => {
  203. let record = cache.get(rev_id).await.unwrap();
  204. assert_eq!(record.state, state);
  205. if let RevisionState::Ack = state {
  206. // There is a defer action that writes the revisions to disk, so we wait here.
  207. // Make sure everything is written.
  208. sleep(Duration::from_millis(2 * REVISION_WRITE_INTERVAL_IN_MILLIS)).await;
  209. }
  210. }
  211. FolderScript::AssertCurrentRevId(rev_id) => {
  212. assert_eq!(rev_manager.rev_id(), rev_id, "Current rev_id is not match");
  213. }
  214. FolderScript::AssertNextSyncRevId(rev_id) => {
  215. let next_revision = rev_manager.next_sync_revision().await.unwrap();
  216. if rev_id.is_none() {
  217. assert!(next_revision.is_none(), "Next revision should be None");
  218. return;
  219. }
  220. let next_revision = next_revision
  221. .unwrap_or_else(|| panic!("Expected Next revision is {}, but receive None", rev_id.unwrap()));
  222. let mut notify = rev_manager.ack_notify();
  223. let _ = notify.recv().await;
  224. assert_eq!(next_revision.rev_id, rev_id.unwrap());
  225. }
  226. }
  227. }
  228. }
  229. pub fn invalid_workspace_name_test_case() -> Vec<(String, ErrorCode)> {
  230. vec![
  231. ("".to_owned(), ErrorCode::WorkspaceNameInvalid),
  232. ("1234".repeat(100), ErrorCode::WorkspaceNameTooLong),
  233. ]
  234. }
  235. pub async fn create_workspace(sdk: &FlowySDKTest, name: &str, desc: &str) -> Workspace {
  236. let request = CreateWorkspacePayload {
  237. name: name.to_owned(),
  238. desc: desc.to_owned(),
  239. };
  240. let workspace = FolderEventBuilder::new(sdk.clone())
  241. .event(CreateWorkspace)
  242. .payload(request)
  243. .async_send()
  244. .await
  245. .parse::<Workspace>();
  246. workspace
  247. }
  248. pub async fn read_workspace(sdk: &FlowySDKTest, workspace_id: Option<String>) -> Vec<Workspace> {
  249. let request = WorkspaceId { value: workspace_id };
  250. let repeated_workspace = FolderEventBuilder::new(sdk.clone())
  251. .event(ReadWorkspaces)
  252. .payload(request.clone())
  253. .async_send()
  254. .await
  255. .parse::<RepeatedWorkspace>();
  256. let workspaces;
  257. if let Some(workspace_id) = &request.value {
  258. workspaces = repeated_workspace
  259. .into_inner()
  260. .into_iter()
  261. .filter(|workspace| &workspace.id == workspace_id)
  262. .collect::<Vec<Workspace>>();
  263. debug_assert_eq!(workspaces.len(), 1);
  264. } else {
  265. workspaces = repeated_workspace.items;
  266. }
  267. workspaces
  268. }
  269. pub async fn create_app(sdk: &FlowySDKTest, workspace_id: &str, name: &str, desc: &str) -> App {
  270. let create_app_request = CreateAppPayload {
  271. workspace_id: workspace_id.to_owned(),
  272. name: name.to_string(),
  273. desc: desc.to_string(),
  274. color_style: Default::default(),
  275. };
  276. let app = FolderEventBuilder::new(sdk.clone())
  277. .event(CreateApp)
  278. .payload(create_app_request)
  279. .async_send()
  280. .await
  281. .parse::<App>();
  282. app
  283. }
  284. pub async fn read_app(sdk: &FlowySDKTest, app_id: &str) -> App {
  285. let request = AppId {
  286. value: app_id.to_owned(),
  287. };
  288. let app = FolderEventBuilder::new(sdk.clone())
  289. .event(ReadApp)
  290. .payload(request)
  291. .async_send()
  292. .await
  293. .parse::<App>();
  294. app
  295. }
  296. pub async fn update_app(sdk: &FlowySDKTest, app_id: &str, name: Option<String>, desc: Option<String>) {
  297. let request = UpdateAppPayload {
  298. app_id: app_id.to_string(),
  299. name,
  300. desc,
  301. color_style: None,
  302. is_trash: None,
  303. };
  304. FolderEventBuilder::new(sdk.clone())
  305. .event(UpdateApp)
  306. .payload(request)
  307. .async_send()
  308. .await;
  309. }
  310. pub async fn delete_app(sdk: &FlowySDKTest, app_id: &str) {
  311. let request = AppId {
  312. value: app_id.to_string(),
  313. };
  314. FolderEventBuilder::new(sdk.clone())
  315. .event(DeleteApp)
  316. .payload(request)
  317. .async_send()
  318. .await;
  319. }
  320. pub async fn create_view(sdk: &FlowySDKTest, app_id: &str, name: &str, desc: &str, data_type: ViewDataType) -> View {
  321. let request = CreateViewPayload {
  322. belong_to_id: app_id.to_string(),
  323. name: name.to_string(),
  324. desc: desc.to_string(),
  325. thumbnail: None,
  326. data_type,
  327. plugin_type: 0,
  328. data: vec![],
  329. };
  330. let view = FolderEventBuilder::new(sdk.clone())
  331. .event(CreateView)
  332. .payload(request)
  333. .async_send()
  334. .await
  335. .parse::<View>();
  336. view
  337. }
  338. pub async fn read_view(sdk: &FlowySDKTest, view_id: &str) -> View {
  339. let view_id: ViewId = view_id.into();
  340. FolderEventBuilder::new(sdk.clone())
  341. .event(ReadView)
  342. .payload(view_id)
  343. .async_send()
  344. .await
  345. .parse::<View>()
  346. }
  347. pub async fn update_view(sdk: &FlowySDKTest, view_id: &str, name: Option<String>, desc: Option<String>) {
  348. let request = UpdateViewPayload {
  349. view_id: view_id.to_string(),
  350. name,
  351. desc,
  352. thumbnail: None,
  353. };
  354. FolderEventBuilder::new(sdk.clone())
  355. .event(UpdateView)
  356. .payload(request)
  357. .async_send()
  358. .await;
  359. }
  360. pub async fn delete_view(sdk: &FlowySDKTest, view_ids: Vec<String>) {
  361. let request = RepeatedViewId { items: view_ids };
  362. FolderEventBuilder::new(sdk.clone())
  363. .event(DeleteView)
  364. .payload(request)
  365. .async_send()
  366. .await;
  367. }
  368. #[allow(dead_code)]
  369. pub async fn set_latest_view(sdk: &FlowySDKTest, view_id: &str) -> TextBlockInfo {
  370. let view_id: ViewId = view_id.into();
  371. FolderEventBuilder::new(sdk.clone())
  372. .event(SetLatestView)
  373. .payload(view_id)
  374. .async_send()
  375. .await
  376. .parse::<TextBlockInfo>()
  377. }
  378. pub async fn read_trash(sdk: &FlowySDKTest) -> RepeatedTrash {
  379. FolderEventBuilder::new(sdk.clone())
  380. .event(ReadTrash)
  381. .async_send()
  382. .await
  383. .parse::<RepeatedTrash>()
  384. }
  385. pub async fn restore_app_from_trash(sdk: &FlowySDKTest, app_id: &str) {
  386. let id = TrashId {
  387. id: app_id.to_owned(),
  388. ty: TrashType::TrashApp,
  389. };
  390. FolderEventBuilder::new(sdk.clone())
  391. .event(PutbackTrash)
  392. .payload(id)
  393. .async_send()
  394. .await;
  395. }
  396. pub async fn restore_view_from_trash(sdk: &FlowySDKTest, view_id: &str) {
  397. let id = TrashId {
  398. id: view_id.to_owned(),
  399. ty: TrashType::TrashView,
  400. };
  401. FolderEventBuilder::new(sdk.clone())
  402. .event(PutbackTrash)
  403. .payload(id)
  404. .async_send()
  405. .await;
  406. }
  407. pub async fn delete_all_trash(sdk: &FlowySDKTest) {
  408. FolderEventBuilder::new(sdk.clone())
  409. .event(DeleteAllTrash)
  410. .async_send()
  411. .await;
  412. }