script.rs 13 KB

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