folder_pad.rs 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796
  1. use crate::util::cal_diff;
  2. use crate::{
  3. client_folder::builder::FolderPadBuilder,
  4. entities::{
  5. folder_info::FolderDelta,
  6. revision::{md5, Revision},
  7. },
  8. errors::{CollaborateError, CollaborateResult},
  9. };
  10. use flowy_folder_data_model::entities::{app::App, trash::Trash, view::View, workspace::Workspace};
  11. use lib_ot::core::*;
  12. use serde::{Deserialize, Serialize};
  13. use std::sync::Arc;
  14. #[derive(Debug, Deserialize, Serialize, Clone, Eq, PartialEq)]
  15. pub struct FolderPad {
  16. pub(crate) workspaces: Vec<Arc<Workspace>>,
  17. pub(crate) trash: Vec<Arc<Trash>>,
  18. #[serde(skip)]
  19. pub(crate) delta: FolderDelta,
  20. }
  21. impl FolderPad {
  22. pub fn new(workspaces: Vec<Workspace>, trash: Vec<Trash>) -> CollaborateResult<Self> {
  23. FolderPadBuilder::new()
  24. .with_workspace(workspaces)
  25. .with_trash(trash)
  26. .build()
  27. }
  28. pub fn from_revisions(revisions: Vec<Revision>) -> CollaborateResult<Self> {
  29. FolderPadBuilder::new().build_with_revisions(revisions)
  30. }
  31. pub fn from_delta(delta: FolderDelta) -> CollaborateResult<Self> {
  32. FolderPadBuilder::new().build_with_delta(delta)
  33. }
  34. pub fn delta(&self) -> &FolderDelta {
  35. &self.delta
  36. }
  37. pub fn reset_folder(&mut self, delta: FolderDelta) -> CollaborateResult<String> {
  38. let folder = FolderPad::from_delta(delta)?;
  39. self.workspaces = folder.workspaces;
  40. self.trash = folder.trash;
  41. self.delta = folder.delta;
  42. Ok(self.md5())
  43. }
  44. pub fn compose_remote_delta(&mut self, delta: FolderDelta) -> CollaborateResult<String> {
  45. let composed_delta = self.delta.compose(&delta)?;
  46. self.reset_folder(composed_delta)
  47. }
  48. pub fn is_empty(&self) -> bool {
  49. self.workspaces.is_empty() && self.trash.is_empty()
  50. }
  51. #[tracing::instrument(level = "trace", skip(self, workspace), fields(workspace_name=%workspace.name), err)]
  52. pub fn create_workspace(&mut self, workspace: Workspace) -> CollaborateResult<Option<FolderChange>> {
  53. let workspace = Arc::new(workspace);
  54. if self.workspaces.contains(&workspace) {
  55. tracing::warn!("[RootFolder]: Duplicate workspace");
  56. return Ok(None);
  57. }
  58. self.modify_workspaces(move |workspaces| {
  59. workspaces.push(workspace);
  60. Ok(Some(()))
  61. })
  62. }
  63. pub fn update_workspace(
  64. &mut self,
  65. workspace_id: &str,
  66. name: Option<String>,
  67. desc: Option<String>,
  68. ) -> CollaborateResult<Option<FolderChange>> {
  69. self.with_workspace(workspace_id, |workspace| {
  70. if let Some(name) = name {
  71. workspace.name = name;
  72. }
  73. if let Some(desc) = desc {
  74. workspace.desc = desc;
  75. }
  76. Ok(Some(()))
  77. })
  78. }
  79. pub fn read_workspaces(&self, workspace_id: Option<String>) -> CollaborateResult<Vec<Workspace>> {
  80. match workspace_id {
  81. None => {
  82. let workspaces = self
  83. .workspaces
  84. .iter()
  85. .map(|workspace| workspace.as_ref().clone())
  86. .collect::<Vec<Workspace>>();
  87. Ok(workspaces)
  88. }
  89. Some(workspace_id) => {
  90. if let Some(workspace) = self.workspaces.iter().find(|workspace| workspace.id == workspace_id) {
  91. Ok(vec![workspace.as_ref().clone()])
  92. } else {
  93. Err(CollaborateError::record_not_found()
  94. .context(format!("Can't find workspace with id {}", workspace_id)))
  95. }
  96. }
  97. }
  98. }
  99. #[tracing::instrument(level = "trace", skip(self), err)]
  100. pub fn delete_workspace(&mut self, workspace_id: &str) -> CollaborateResult<Option<FolderChange>> {
  101. self.modify_workspaces(|workspaces| {
  102. workspaces.retain(|w| w.id != workspace_id);
  103. Ok(Some(()))
  104. })
  105. }
  106. #[tracing::instrument(level = "trace", skip(self), fields(app_name=%app.name), err)]
  107. pub fn create_app(&mut self, app: App) -> CollaborateResult<Option<FolderChange>> {
  108. let workspace_id = app.workspace_id.clone();
  109. self.with_workspace(&workspace_id, move |workspace| {
  110. if workspace.apps.contains(&app) {
  111. tracing::warn!("[RootFolder]: Duplicate app");
  112. return Ok(None);
  113. }
  114. workspace.apps.push(app);
  115. Ok(Some(()))
  116. })
  117. }
  118. pub fn read_app(&self, app_id: &str) -> CollaborateResult<App> {
  119. for workspace in &self.workspaces {
  120. if let Some(app) = workspace.apps.iter().find(|app| app.id == app_id) {
  121. return Ok(app.clone());
  122. }
  123. }
  124. Err(CollaborateError::record_not_found().context(format!("Can't find app with id {}", app_id)))
  125. }
  126. pub fn update_app(
  127. &mut self,
  128. app_id: &str,
  129. name: Option<String>,
  130. desc: Option<String>,
  131. ) -> CollaborateResult<Option<FolderChange>> {
  132. self.with_app(app_id, move |app| {
  133. if let Some(name) = name {
  134. app.name = name;
  135. }
  136. if let Some(desc) = desc {
  137. app.desc = desc;
  138. }
  139. Ok(Some(()))
  140. })
  141. }
  142. #[tracing::instrument(level = "trace", skip(self), err)]
  143. pub fn delete_app(&mut self, app_id: &str) -> CollaborateResult<Option<FolderChange>> {
  144. let app = self.read_app(app_id)?;
  145. self.with_workspace(&app.workspace_id, |workspace| {
  146. workspace.apps.retain(|app| app.id != app_id);
  147. Ok(Some(()))
  148. })
  149. }
  150. #[tracing::instrument(level = "trace", skip(self), fields(view_name=%view.name), err)]
  151. pub fn create_view(&mut self, view: View) -> CollaborateResult<Option<FolderChange>> {
  152. let app_id = view.belong_to_id.clone();
  153. self.with_app(&app_id, move |app| {
  154. if app.belongings.contains(&view) {
  155. tracing::warn!("[RootFolder]: Duplicate view");
  156. return Ok(None);
  157. }
  158. app.belongings.push(view);
  159. Ok(Some(()))
  160. })
  161. }
  162. pub fn read_view(&self, view_id: &str) -> CollaborateResult<View> {
  163. for workspace in &self.workspaces {
  164. for app in &(*workspace.apps) {
  165. if let Some(view) = app.belongings.iter().find(|b| b.id == view_id) {
  166. return Ok(view.clone());
  167. }
  168. }
  169. }
  170. Err(CollaborateError::record_not_found().context(format!("Can't find view with id {}", view_id)))
  171. }
  172. pub fn read_views(&self, belong_to_id: &str) -> CollaborateResult<Vec<View>> {
  173. for workspace in &self.workspaces {
  174. for app in &(*workspace.apps) {
  175. if app.id == belong_to_id {
  176. return Ok(app.belongings.clone().take_items());
  177. }
  178. }
  179. }
  180. Ok(vec![])
  181. }
  182. pub fn update_view(
  183. &mut self,
  184. view_id: &str,
  185. name: Option<String>,
  186. desc: Option<String>,
  187. modified_time: i64,
  188. ) -> CollaborateResult<Option<FolderChange>> {
  189. let view = self.read_view(view_id)?;
  190. self.with_view(&view.belong_to_id, view_id, |view| {
  191. if let Some(name) = name {
  192. view.name = name;
  193. }
  194. if let Some(desc) = desc {
  195. view.desc = desc;
  196. }
  197. view.modified_time = modified_time;
  198. Ok(Some(()))
  199. })
  200. }
  201. #[tracing::instrument(level = "trace", skip(self), err)]
  202. pub fn delete_view(&mut self, view_id: &str) -> CollaborateResult<Option<FolderChange>> {
  203. let view = self.read_view(view_id)?;
  204. self.with_app(&view.belong_to_id, |app| {
  205. app.belongings.retain(|view| view.id != view_id);
  206. Ok(Some(()))
  207. })
  208. }
  209. pub fn create_trash(&mut self, trash: Vec<Trash>) -> CollaborateResult<Option<FolderChange>> {
  210. self.with_trash(|t| {
  211. let mut new_trash = trash.into_iter().map(Arc::new).collect::<Vec<Arc<Trash>>>();
  212. t.append(&mut new_trash);
  213. Ok(Some(()))
  214. })
  215. }
  216. pub fn read_trash(&self, trash_id: Option<String>) -> CollaborateResult<Vec<Trash>> {
  217. match trash_id {
  218. None => Ok(self.trash.iter().map(|t| t.as_ref().clone()).collect::<Vec<Trash>>()),
  219. Some(trash_id) => match self.trash.iter().find(|t| t.id == trash_id) {
  220. Some(trash) => Ok(vec![trash.as_ref().clone()]),
  221. None => Ok(vec![]),
  222. },
  223. }
  224. }
  225. pub fn delete_trash(&mut self, trash_ids: Option<Vec<String>>) -> CollaborateResult<Option<FolderChange>> {
  226. match trash_ids {
  227. None => self.with_trash(|trash| {
  228. trash.clear();
  229. Ok(Some(()))
  230. }),
  231. Some(trash_ids) => self.with_trash(|trash| {
  232. trash.retain(|t| !trash_ids.contains(&t.id));
  233. Ok(Some(()))
  234. }),
  235. }
  236. }
  237. pub fn md5(&self) -> String {
  238. md5(&self.delta.to_bytes())
  239. }
  240. pub fn to_json(&self) -> CollaborateResult<String> {
  241. serde_json::to_string(self)
  242. .map_err(|e| CollaborateError::internal().context(format!("serial trash to json failed: {}", e)))
  243. }
  244. }
  245. impl FolderPad {
  246. fn modify_workspaces<F>(&mut self, f: F) -> CollaborateResult<Option<FolderChange>>
  247. where
  248. F: FnOnce(&mut Vec<Arc<Workspace>>) -> CollaborateResult<Option<()>>,
  249. {
  250. let cloned_self = self.clone();
  251. match f(&mut self.workspaces)? {
  252. None => Ok(None),
  253. Some(_) => {
  254. let old = cloned_self.to_json()?;
  255. let new = self.to_json()?;
  256. match cal_diff::<PlainTextAttributes>(old, new) {
  257. None => Ok(None),
  258. Some(delta) => {
  259. self.delta = self.delta.compose(&delta)?;
  260. Ok(Some(FolderChange { delta, md5: self.md5() }))
  261. }
  262. }
  263. }
  264. }
  265. }
  266. fn with_workspace<F>(&mut self, workspace_id: &str, f: F) -> CollaborateResult<Option<FolderChange>>
  267. where
  268. F: FnOnce(&mut Workspace) -> CollaborateResult<Option<()>>,
  269. {
  270. self.modify_workspaces(|workspaces| {
  271. if let Some(workspace) = workspaces.iter_mut().find(|workspace| workspace_id == workspace.id) {
  272. f(Arc::make_mut(workspace))
  273. } else {
  274. tracing::warn!("[FolderPad]: Can't find any workspace with id: {}", workspace_id);
  275. Ok(None)
  276. }
  277. })
  278. }
  279. fn with_trash<F>(&mut self, f: F) -> CollaborateResult<Option<FolderChange>>
  280. where
  281. F: FnOnce(&mut Vec<Arc<Trash>>) -> CollaborateResult<Option<()>>,
  282. {
  283. let cloned_self = self.clone();
  284. match f(&mut self.trash)? {
  285. None => Ok(None),
  286. Some(_) => {
  287. let old = cloned_self.to_json()?;
  288. let new = self.to_json()?;
  289. match cal_diff::<PlainTextAttributes>(old, new) {
  290. None => Ok(None),
  291. Some(delta) => {
  292. self.delta = self.delta.compose(&delta)?;
  293. Ok(Some(FolderChange { delta, md5: self.md5() }))
  294. }
  295. }
  296. }
  297. }
  298. }
  299. fn with_app<F>(&mut self, app_id: &str, f: F) -> CollaborateResult<Option<FolderChange>>
  300. where
  301. F: FnOnce(&mut App) -> CollaborateResult<Option<()>>,
  302. {
  303. let workspace_id = match self
  304. .workspaces
  305. .iter()
  306. .find(|workspace| workspace.apps.iter().any(|app| app.id == app_id))
  307. {
  308. None => {
  309. tracing::warn!("[FolderPad]: Can't find any app with id: {}", app_id);
  310. return Ok(None);
  311. }
  312. Some(workspace) => workspace.id.clone(),
  313. };
  314. self.with_workspace(&workspace_id, |workspace| {
  315. // It's ok to unwrap because we get the workspace from the app_id.
  316. f(workspace.apps.iter_mut().find(|app| app_id == app.id).unwrap())
  317. })
  318. }
  319. fn with_view<F>(&mut self, belong_to_id: &str, view_id: &str, f: F) -> CollaborateResult<Option<FolderChange>>
  320. where
  321. F: FnOnce(&mut View) -> CollaborateResult<Option<()>>,
  322. {
  323. self.with_app(belong_to_id, |app| {
  324. match app.belongings.iter_mut().find(|view| view_id == view.id) {
  325. None => {
  326. tracing::warn!("[FolderPad]: Can't find any view with id: {}", view_id);
  327. Ok(None)
  328. }
  329. Some(view) => f(view),
  330. }
  331. })
  332. }
  333. }
  334. pub fn default_folder_delta() -> FolderDelta {
  335. PlainTextDeltaBuilder::new()
  336. .insert(r#"{"workspaces":[],"trash":[]}"#)
  337. .build()
  338. }
  339. pub fn initial_folder_delta(folder_pad: &FolderPad) -> CollaborateResult<FolderDelta> {
  340. let json = folder_pad.to_json()?;
  341. let delta = PlainTextDeltaBuilder::new().insert(&json).build();
  342. Ok(delta)
  343. }
  344. impl std::default::Default for FolderPad {
  345. fn default() -> Self {
  346. FolderPad {
  347. workspaces: vec![],
  348. trash: vec![],
  349. delta: default_folder_delta(),
  350. }
  351. }
  352. }
  353. pub struct FolderChange {
  354. pub delta: FolderDelta,
  355. /// md5: the md5 of the FolderPad's delta after applying the change.
  356. pub md5: String,
  357. }
  358. #[cfg(test)]
  359. mod tests {
  360. #![allow(clippy::all)]
  361. use crate::{client_folder::folder_pad::FolderPad, entities::folder_info::FolderDelta};
  362. use chrono::Utc;
  363. use flowy_folder_data_model::entities::{app::App, trash::Trash, view::View, workspace::Workspace};
  364. use lib_ot::core::{OperationTransformable, PlainTextDelta, PlainTextDeltaBuilder};
  365. #[test]
  366. fn folder_add_workspace() {
  367. let (mut folder, initial_delta, _) = test_folder();
  368. let _time = Utc::now();
  369. let mut workspace_1 = Workspace::default();
  370. workspace_1.name = "My first workspace".to_owned();
  371. let delta_1 = folder.create_workspace(workspace_1).unwrap().unwrap().delta;
  372. let mut workspace_2 = Workspace::default();
  373. workspace_2.name = "My second workspace".to_owned();
  374. let delta_2 = folder.create_workspace(workspace_2).unwrap().unwrap().delta;
  375. let folder_from_delta = make_folder_from_delta(initial_delta, vec![delta_1, delta_2]);
  376. assert_eq!(folder, folder_from_delta);
  377. }
  378. #[test]
  379. fn folder_update_workspace() {
  380. let (mut folder, initial_delta, workspace) = test_folder();
  381. assert_folder_equal(
  382. &folder,
  383. &make_folder_from_delta(initial_delta.clone(), vec![]),
  384. r#"{"workspaces":[{"id":"1","name":"😁 my first workspace","desc":"","apps":[],"modified_time":0,"create_time":0}],"trash":[]}"#,
  385. );
  386. let delta = folder
  387. .update_workspace(&workspace.id, Some("☺️ rename workspace️".to_string()), None)
  388. .unwrap()
  389. .unwrap()
  390. .delta;
  391. let folder_from_delta = make_folder_from_delta(initial_delta, vec![delta]);
  392. assert_folder_equal(
  393. &folder,
  394. &folder_from_delta,
  395. r#"{"workspaces":[{"id":"1","name":"☺️ rename workspace️","desc":"","apps":[],"modified_time":0,"create_time":0}],"trash":[]}"#,
  396. );
  397. }
  398. #[test]
  399. fn folder_add_app() {
  400. let (folder, initial_delta, _app) = test_app_folder();
  401. let folder_from_delta = make_folder_from_delta(initial_delta, vec![]);
  402. assert_eq!(folder, folder_from_delta);
  403. assert_folder_equal(
  404. &folder,
  405. &folder_from_delta,
  406. r#"{
  407. "workspaces": [
  408. {
  409. "id": "1",
  410. "name": "😁 my first workspace",
  411. "desc": "",
  412. "apps": [
  413. {
  414. "id": "",
  415. "workspace_id": "1",
  416. "name": "😁 my first app",
  417. "desc": "",
  418. "belongings": [],
  419. "version": 0,
  420. "modified_time": 0,
  421. "create_time": 0
  422. }
  423. ],
  424. "modified_time": 0,
  425. "create_time": 0
  426. }
  427. ],
  428. "trash": []
  429. }"#,
  430. );
  431. }
  432. #[test]
  433. fn folder_update_app() {
  434. let (mut folder, initial_delta, app) = test_app_folder();
  435. let delta = folder
  436. .update_app(&app.id, Some("🤪 rename app".to_owned()), None)
  437. .unwrap()
  438. .unwrap()
  439. .delta;
  440. let new_folder = make_folder_from_delta(initial_delta, vec![delta]);
  441. assert_folder_equal(
  442. &folder,
  443. &new_folder,
  444. r#"{
  445. "workspaces": [
  446. {
  447. "id": "1",
  448. "name": "😁 my first workspace",
  449. "desc": "",
  450. "apps": [
  451. {
  452. "id": "",
  453. "workspace_id": "1",
  454. "name": "🤪 rename app",
  455. "desc": "",
  456. "belongings": [],
  457. "version": 0,
  458. "modified_time": 0,
  459. "create_time": 0
  460. }
  461. ],
  462. "modified_time": 0,
  463. "create_time": 0
  464. }
  465. ],
  466. "trash": []
  467. }"#,
  468. );
  469. }
  470. #[test]
  471. fn folder_delete_app() {
  472. let (mut folder, initial_delta, app) = test_app_folder();
  473. let delta = folder.delete_app(&app.id).unwrap().unwrap().delta;
  474. let new_folder = make_folder_from_delta(initial_delta, vec![delta]);
  475. assert_folder_equal(
  476. &folder,
  477. &new_folder,
  478. r#"{
  479. "workspaces": [
  480. {
  481. "id": "1",
  482. "name": "😁 my first workspace",
  483. "desc": "",
  484. "apps": [],
  485. "modified_time": 0,
  486. "create_time": 0
  487. }
  488. ],
  489. "trash": []
  490. }"#,
  491. );
  492. }
  493. #[test]
  494. fn folder_add_view() {
  495. let (folder, initial_delta, _view) = test_view_folder();
  496. assert_folder_equal(
  497. &folder,
  498. &make_folder_from_delta(initial_delta, vec![]),
  499. r#"
  500. {
  501. "workspaces": [
  502. {
  503. "id": "1",
  504. "name": "😁 my first workspace",
  505. "desc": "",
  506. "apps": [
  507. {
  508. "id": "",
  509. "workspace_id": "1",
  510. "name": "😁 my first app",
  511. "desc": "",
  512. "belongings": [
  513. {
  514. "id": "",
  515. "belong_to_id": "",
  516. "name": "🎃 my first view",
  517. "desc": "",
  518. "view_type": "Blank",
  519. "version": 0,
  520. "belongings": [],
  521. "modified_time": 0,
  522. "create_time": 0
  523. }
  524. ],
  525. "version": 0,
  526. "modified_time": 0,
  527. "create_time": 0
  528. }
  529. ],
  530. "modified_time": 0,
  531. "create_time": 0
  532. }
  533. ],
  534. "trash": []
  535. }"#,
  536. );
  537. }
  538. #[test]
  539. fn folder_update_view() {
  540. let (mut folder, initial_delta, view) = test_view_folder();
  541. let delta = folder
  542. .update_view(&view.id, Some("😦 rename view".to_owned()), None, 123)
  543. .unwrap()
  544. .unwrap()
  545. .delta;
  546. let new_folder = make_folder_from_delta(initial_delta, vec![delta]);
  547. assert_folder_equal(
  548. &folder,
  549. &new_folder,
  550. r#"{
  551. "workspaces": [
  552. {
  553. "id": "1",
  554. "name": "😁 my first workspace",
  555. "desc": "",
  556. "apps": [
  557. {
  558. "id": "",
  559. "workspace_id": "1",
  560. "name": "😁 my first app",
  561. "desc": "",
  562. "belongings": [
  563. {
  564. "id": "",
  565. "belong_to_id": "",
  566. "name": "😦 rename view",
  567. "desc": "",
  568. "view_type": "Blank",
  569. "version": 0,
  570. "belongings": [],
  571. "modified_time": 123,
  572. "create_time": 0
  573. }
  574. ],
  575. "version": 0,
  576. "modified_time": 0,
  577. "create_time": 0
  578. }
  579. ],
  580. "modified_time": 0,
  581. "create_time": 0
  582. }
  583. ],
  584. "trash": []
  585. }"#,
  586. );
  587. }
  588. #[test]
  589. fn folder_delete_view() {
  590. let (mut folder, initial_delta, view) = test_view_folder();
  591. let delta = folder.delete_view(&view.id).unwrap().unwrap().delta;
  592. let new_folder = make_folder_from_delta(initial_delta, vec![delta]);
  593. assert_folder_equal(
  594. &folder,
  595. &new_folder,
  596. r#"{
  597. "workspaces": [
  598. {
  599. "id": "1",
  600. "name": "😁 my first workspace",
  601. "desc": "",
  602. "apps": [
  603. {
  604. "id": "",
  605. "workspace_id": "1",
  606. "name": "😁 my first app",
  607. "desc": "",
  608. "belongings": [],
  609. "version": 0,
  610. "modified_time": 0,
  611. "create_time": 0
  612. }
  613. ],
  614. "modified_time": 0,
  615. "create_time": 0
  616. }
  617. ],
  618. "trash": []
  619. }"#,
  620. );
  621. }
  622. #[test]
  623. fn folder_add_trash() {
  624. let (folder, initial_delta, _trash) = test_trash();
  625. assert_folder_equal(
  626. &folder,
  627. &make_folder_from_delta(initial_delta, vec![]),
  628. r#"{
  629. "workspaces": [],
  630. "trash": [
  631. {
  632. "id": "1",
  633. "name": "🚽 my first trash",
  634. "modified_time": 0,
  635. "create_time": 0,
  636. "ty": "Unknown"
  637. }
  638. ]
  639. }
  640. "#,
  641. );
  642. }
  643. #[test]
  644. fn folder_delete_trash() {
  645. let (mut folder, initial_delta, trash) = test_trash();
  646. let delta = folder.delete_trash(Some(vec![trash.id])).unwrap().unwrap().delta;
  647. assert_folder_equal(
  648. &folder,
  649. &make_folder_from_delta(initial_delta, vec![delta]),
  650. r#"{
  651. "workspaces": [],
  652. "trash": []
  653. }
  654. "#,
  655. );
  656. }
  657. fn test_folder() -> (FolderPad, FolderDelta, Workspace) {
  658. let mut folder = FolderPad::default();
  659. let folder_json = serde_json::to_string(&folder).unwrap();
  660. let mut delta = PlainTextDeltaBuilder::new().insert(&folder_json).build();
  661. let mut workspace = Workspace::default();
  662. workspace.name = "😁 my first workspace".to_owned();
  663. workspace.id = "1".to_owned();
  664. delta = delta
  665. .compose(&folder.create_workspace(workspace.clone()).unwrap().unwrap().delta)
  666. .unwrap();
  667. (folder, delta, workspace)
  668. }
  669. fn test_app_folder() -> (FolderPad, FolderDelta, App) {
  670. let (mut folder, mut initial_delta, workspace) = test_folder();
  671. let mut app = App::default();
  672. app.workspace_id = workspace.id;
  673. app.name = "😁 my first app".to_owned();
  674. initial_delta = initial_delta
  675. .compose(&folder.create_app(app.clone()).unwrap().unwrap().delta)
  676. .unwrap();
  677. (folder, initial_delta, app)
  678. }
  679. fn test_view_folder() -> (FolderPad, FolderDelta, View) {
  680. let (mut folder, mut initial_delta, app) = test_app_folder();
  681. let mut view = View::default();
  682. view.belong_to_id = app.id.clone();
  683. view.name = "🎃 my first view".to_owned();
  684. initial_delta = initial_delta
  685. .compose(&folder.create_view(view.clone()).unwrap().unwrap().delta)
  686. .unwrap();
  687. (folder, initial_delta, view)
  688. }
  689. fn test_trash() -> (FolderPad, FolderDelta, Trash) {
  690. let mut folder = FolderPad::default();
  691. let folder_json = serde_json::to_string(&folder).unwrap();
  692. let mut delta = PlainTextDeltaBuilder::new().insert(&folder_json).build();
  693. let mut trash = Trash::default();
  694. trash.name = "🚽 my first trash".to_owned();
  695. trash.id = "1".to_owned();
  696. delta = delta
  697. .compose(&folder.create_trash(vec![trash.clone()]).unwrap().unwrap().delta)
  698. .unwrap();
  699. (folder, delta, trash)
  700. }
  701. fn make_folder_from_delta(mut initial_delta: FolderDelta, deltas: Vec<PlainTextDelta>) -> FolderPad {
  702. for delta in deltas {
  703. initial_delta = initial_delta.compose(&delta).unwrap();
  704. }
  705. FolderPad::from_delta(initial_delta).unwrap()
  706. }
  707. fn assert_folder_equal(old: &FolderPad, new: &FolderPad, expected: &str) {
  708. assert_eq!(old, new);
  709. let json1 = old.to_json().unwrap();
  710. let json2 = new.to_json().unwrap();
  711. let expect_folder: FolderPad = serde_json::from_str(expected).unwrap();
  712. assert_eq!(json1, expect_folder.to_json().unwrap());
  713. assert_eq!(json1, json2);
  714. }
  715. }