script.rs 14 KB

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