v5.page.test.ts 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741
  1. import { addSeconds } from 'date-fns';
  2. import mongoose from 'mongoose';
  3. import { PageActionStage, PageActionType } from '../../../src/interfaces/page-operation';
  4. import { getInstance } from '../setup-crowi';
  5. describe('Test page service methods', () => {
  6. let crowi;
  7. let Page;
  8. let Revision;
  9. let User;
  10. let Tag;
  11. let PageTagRelation;
  12. let Bookmark;
  13. let Comment;
  14. let ShareLink;
  15. let PageRedirect;
  16. let PageOperation;
  17. let xssSpy;
  18. let rootPage;
  19. let dummyUser1;
  20. let dummyUser2;
  21. let globalGroupUser1;
  22. let globalGroupUser2;
  23. let globalGroupUser3;
  24. let pageOpId1;
  25. let pageOpId2;
  26. let pageOpId3;
  27. let pageOpId4;
  28. let pageOpId5;
  29. let pageOpId6;
  30. beforeAll(async() => {
  31. crowi = await getInstance();
  32. await crowi.configManager.updateConfigsInTheSameNamespace('crowi', { 'app:isV5Compatible': true });
  33. User = mongoose.model('User');
  34. Page = mongoose.model('Page');
  35. Revision = mongoose.model('Revision');
  36. Tag = mongoose.model('Tag');
  37. PageTagRelation = mongoose.model('PageTagRelation');
  38. Bookmark = mongoose.model('Bookmark');
  39. Comment = mongoose.model('Comment');
  40. ShareLink = mongoose.model('ShareLink');
  41. PageRedirect = mongoose.model('PageRedirect');
  42. PageOperation = mongoose.model('PageOperation');
  43. /*
  44. * Common
  45. */
  46. xssSpy = jest.spyOn(crowi.xss, 'process').mockImplementation(path => path);
  47. // ***********************************************************************************************************
  48. // * Do NOT change properties of globally used documents. Otherwise, it might cause some errors in other tests
  49. // ***********************************************************************************************************
  50. // users
  51. dummyUser1 = await User.findOne({ username: 'v5DummyUser1' });
  52. dummyUser2 = await User.findOne({ username: 'v5DummyUser2' });
  53. globalGroupUser1 = await User.findOne({ username: 'gGroupUser1' });
  54. globalGroupUser2 = await User.findOne({ username: 'gGroupUser2' });
  55. globalGroupUser3 = await User.findOne({ username: 'gGroupUser3' });
  56. // page
  57. rootPage = await Page.findOne({ path: '/' });
  58. /**
  59. * pages
  60. */
  61. const pageId0 = new mongoose.Types.ObjectId();
  62. const pageId1 = new mongoose.Types.ObjectId();
  63. const pageId2 = new mongoose.Types.ObjectId();
  64. const pageId3 = new mongoose.Types.ObjectId();
  65. const pageId4 = new mongoose.Types.ObjectId();
  66. const pageId5 = new mongoose.Types.ObjectId();
  67. const pageId6 = new mongoose.Types.ObjectId();
  68. const pageId7 = new mongoose.Types.ObjectId();
  69. const pageId8 = new mongoose.Types.ObjectId();
  70. const pageId9 = new mongoose.Types.ObjectId();
  71. const pageId10 = new mongoose.Types.ObjectId();
  72. const pageId11 = new mongoose.Types.ObjectId();
  73. const pageId12 = new mongoose.Types.ObjectId();
  74. const pageId13 = new mongoose.Types.ObjectId();
  75. const pageId14 = new mongoose.Types.ObjectId();
  76. const pageId15 = new mongoose.Types.ObjectId();
  77. const pageId16 = new mongoose.Types.ObjectId();
  78. const pageId17 = new mongoose.Types.ObjectId();
  79. const pageId18 = new mongoose.Types.ObjectId();
  80. const pageId19 = new mongoose.Types.ObjectId();
  81. const pageId20 = new mongoose.Types.ObjectId();
  82. await Page.insertMany([
  83. {
  84. _id: pageId0,
  85. path: '/resume_rename_0',
  86. parent: rootPage._id,
  87. grant: Page.GRANT_PUBLIC,
  88. creator: dummyUser1,
  89. lastUpdateUser: dummyUser1._id,
  90. descendantCount: 1,
  91. isEmpty: false,
  92. },
  93. {
  94. _id: pageId1,
  95. path: '/resume_rename_0/resume_rename_1',
  96. parent: pageId0,
  97. grant: Page.GRANT_PUBLIC,
  98. creator: dummyUser1,
  99. lastUpdateUser: dummyUser1._id,
  100. descendantCount: 2,
  101. isEmpty: false,
  102. },
  103. {
  104. _id: pageId2,
  105. path: '/resume_rename_1/resume_rename_2',
  106. parent: pageId1,
  107. grant: Page.GRANT_PUBLIC,
  108. creator: dummyUser1,
  109. lastUpdateUser: dummyUser1._id,
  110. descendantCount: 1,
  111. isEmpty: false,
  112. },
  113. {
  114. _id: pageId3,
  115. path: '/resume_rename_1/resume_rename_2/resume_rename_3',
  116. parent: pageId2,
  117. grant: Page.GRANT_PUBLIC,
  118. creator: dummyUser1,
  119. lastUpdateUser: dummyUser1._id,
  120. descendantCount: 0,
  121. isEmpty: false,
  122. },
  123. {
  124. _id: pageId4,
  125. path: '/resume_rename_4',
  126. parent: rootPage._id,
  127. grant: Page.GRANT_PUBLIC,
  128. creator: dummyUser1,
  129. lastUpdateUser: dummyUser1._id,
  130. descendantCount: 0,
  131. isEmpty: false,
  132. },
  133. {
  134. _id: pageId5,
  135. path: '/resume_rename_4/resume_rename_5',
  136. parent: pageId4,
  137. grant: Page.GRANT_PUBLIC,
  138. creator: dummyUser1,
  139. lastUpdateUser: dummyUser1._id,
  140. descendantCount: 1,
  141. isEmpty: false,
  142. },
  143. {
  144. _id: pageId6,
  145. path: '/resume_rename_5/resume_rename_6',
  146. parent: pageId5,
  147. grant: Page.GRANT_PUBLIC,
  148. creator: dummyUser1,
  149. lastUpdateUser: dummyUser1._id,
  150. descendantCount: 0,
  151. isEmpty: false,
  152. },
  153. {
  154. _id: pageId7,
  155. path: '/resume_rename_7',
  156. parent: rootPage._id,
  157. grant: Page.GRANT_PUBLIC,
  158. creator: dummyUser1,
  159. lastUpdateUser: dummyUser1._id,
  160. descendantCount: 0,
  161. isEmpty: false,
  162. },
  163. {
  164. _id: pageId8,
  165. path: '/resume_rename_8',
  166. parent: rootPage._id,
  167. grant: Page.GRANT_PUBLIC,
  168. creator: dummyUser1,
  169. lastUpdateUser: dummyUser1._id,
  170. descendantCount: 1,
  171. isEmpty: false,
  172. },
  173. {
  174. _id: pageId9,
  175. path: '/resume_rename_8/resume_rename_9',
  176. parent: pageId8,
  177. grant: Page.GRANT_PUBLIC,
  178. creator: dummyUser1,
  179. lastUpdateUser: dummyUser1._id,
  180. descendantCount: 1,
  181. isEmpty: false,
  182. },
  183. {
  184. path: '/resume_rename_9/resume_rename_10',
  185. parent: pageId9,
  186. grant: Page.GRANT_PUBLIC,
  187. creator: dummyUser1,
  188. lastUpdateUser: dummyUser1._id,
  189. descendantCount: 0,
  190. isEmpty: false,
  191. },
  192. {
  193. _id: pageId10,
  194. path: '/resume_rename_11',
  195. parent: rootPage._id,
  196. grant: Page.GRANT_PUBLIC,
  197. creator: dummyUser1,
  198. lastUpdateUser: dummyUser1._id,
  199. descendantCount: 3,
  200. isEmpty: false,
  201. },
  202. {
  203. _id: pageId11,
  204. path: '/resume_rename_11/resume_rename_12',
  205. parent: pageId10,
  206. grant: Page.GRANT_PUBLIC,
  207. creator: dummyUser1,
  208. lastUpdateUser: dummyUser1._id,
  209. descendantCount: 2,
  210. isEmpty: false,
  211. },
  212. {
  213. _id: pageId12,
  214. path: '/resume_rename_11/resume_rename_12/resume_rename_13',
  215. parent: pageId11,
  216. grant: Page.GRANT_PUBLIC,
  217. creator: dummyUser1,
  218. lastUpdateUser: dummyUser1._id,
  219. descendantCount: 1,
  220. isEmpty: false,
  221. },
  222. {
  223. path: '/resume_rename_11/resume_rename_12/resume_rename_13/resume_rename_14',
  224. parent: pageId12,
  225. grant: Page.GRANT_PUBLIC,
  226. creator: dummyUser1,
  227. lastUpdateUser: dummyUser1._id,
  228. descendantCount: 0,
  229. isEmpty: false,
  230. },
  231. {
  232. _id: pageId13,
  233. path: '/resume_rename_15',
  234. parent: rootPage._id,
  235. grant: Page.GRANT_PUBLIC,
  236. creator: dummyUser1,
  237. lastUpdateUser: dummyUser1._id,
  238. descendantCount: 2,
  239. isEmpty: false,
  240. },
  241. {
  242. _id: pageId14,
  243. path: '/resume_rename_15/resume_rename_16',
  244. parent: pageId13,
  245. grant: Page.GRANT_PUBLIC,
  246. creator: dummyUser1,
  247. lastUpdateUser: dummyUser1._id,
  248. descendantCount: 0,
  249. isEmpty: false,
  250. },
  251. {
  252. _id: pageId15,
  253. path: '/resume_rename_15/resume_rename_17',
  254. parent: pageId13,
  255. grant: Page.GRANT_PUBLIC,
  256. creator: dummyUser1,
  257. lastUpdateUser: dummyUser1._id,
  258. descendantCount: 1,
  259. isEmpty: false,
  260. },
  261. {
  262. _id: pageId16,
  263. path: '/resume_rename_15/resume_rename_17/resume_rename_18',
  264. parent: pageId15,
  265. grant: Page.GRANT_PUBLIC,
  266. creator: dummyUser1,
  267. lastUpdateUser: dummyUser1._id,
  268. descendantCount: 1,
  269. isEmpty: false,
  270. },
  271. {
  272. _id: pageId17,
  273. path: '/resume_rename_15/resume_rename_17/resume_rename_18/resume_rename_19',
  274. parent: pageId16,
  275. grant: Page.GRANT_PUBLIC,
  276. creator: dummyUser1,
  277. lastUpdateUser: dummyUser1._id,
  278. descendantCount: 0,
  279. isEmpty: false,
  280. },
  281. {
  282. _id: pageId18,
  283. path: '/fix_descendantCount_1',
  284. parent: rootPage._id,
  285. grant: Page.GRANT_PUBLIC,
  286. creator: dummyUser1,
  287. lastUpdateUser: dummyUser1._id,
  288. descendantCount: 100, // broken
  289. isEmpty: false,
  290. },
  291. {
  292. _id: pageId19,
  293. path: '/fix_descendantCount_1/fix_descendantCount_2',
  294. parent: pageId18,
  295. grant: Page.GRANT_PUBLIC,
  296. creator: dummyUser1,
  297. lastUpdateUser: dummyUser1._id,
  298. descendantCount: 100, // broken
  299. isEmpty: true,
  300. },
  301. {
  302. path: '/fix_descendantCount_1/fix_descendantCount_2/fix_descendantCount_3',
  303. parent: pageId19,
  304. grant: Page.GRANT_PUBLIC,
  305. creator: dummyUser1,
  306. lastUpdateUser: dummyUser1._id,
  307. descendantCount: 100, // broken
  308. isEmpty: false,
  309. },
  310. {
  311. _id: pageId20,
  312. path: '/fix_descendantCount_4',
  313. parent: rootPage._id,
  314. grant: Page.GRANT_PUBLIC,
  315. creator: dummyUser1,
  316. lastUpdateUser: dummyUser1._id,
  317. descendantCount: 100, // broken
  318. isEmpty: false,
  319. },
  320. {
  321. path: '/fix_descendantCount_4/fix_descendantCount_5',
  322. parent: pageId20,
  323. grant: Page.GRANT_PUBLIC,
  324. creator: dummyUser1,
  325. lastUpdateUser: dummyUser1._id,
  326. descendantCount: 100, // broken
  327. isEmpty: false,
  328. },
  329. ]);
  330. /**
  331. * PageOperation
  332. */
  333. pageOpId1 = new mongoose.Types.ObjectId();
  334. pageOpId2 = new mongoose.Types.ObjectId();
  335. pageOpId3 = new mongoose.Types.ObjectId();
  336. pageOpId4 = new mongoose.Types.ObjectId();
  337. const pageOpRevisionId1 = new mongoose.Types.ObjectId();
  338. const pageOpRevisionId2 = new mongoose.Types.ObjectId();
  339. const pageOpRevisionId3 = new mongoose.Types.ObjectId();
  340. const pageOpRevisionId4 = new mongoose.Types.ObjectId();
  341. await PageOperation.insertMany([
  342. {
  343. _id: pageOpId1,
  344. actionType: 'Rename',
  345. actionStage: 'Sub',
  346. fromPath: '/resume_rename_1',
  347. toPath: '/resume_rename_0/resume_rename_1',
  348. page: {
  349. _id: pageId1,
  350. parent: rootPage._id,
  351. descendantCount: 2,
  352. isEmpty: false,
  353. path: '/resume_rename_1',
  354. revision: pageOpRevisionId1,
  355. status: 'published',
  356. grant: 1,
  357. grantedUsers: [],
  358. grantedGroups: null,
  359. creator: dummyUser1._id,
  360. lastUpdateUser: dummyUser1._id,
  361. },
  362. user: {
  363. _id: dummyUser1._id,
  364. },
  365. options: {
  366. createRedirectPage: false,
  367. updateMetadata: true,
  368. },
  369. unprocessableExpiryDate: null,
  370. },
  371. {
  372. _id: pageOpId2,
  373. actionType: 'Rename',
  374. actionStage: 'Sub',
  375. fromPath: '/resume_rename_5',
  376. toPath: '/resume_rename_4/resume_rename_5',
  377. page: {
  378. _id: pageId5,
  379. parent: rootPage._id,
  380. descendantCount: 2,
  381. isEmpty: false,
  382. path: '/resume_rename_5',
  383. revision: pageOpRevisionId2,
  384. status: 'published',
  385. grant: 1,
  386. grantedUsers: [],
  387. grantedGroups: null,
  388. creator: dummyUser1._id,
  389. lastUpdateUser: dummyUser1._id,
  390. },
  391. user: {
  392. _id: dummyUser1._id,
  393. },
  394. options: {
  395. createRedirectPage: false,
  396. updateMetadata: true,
  397. },
  398. unprocessableExpiryDate: new Date(),
  399. },
  400. {
  401. _id: pageOpId3,
  402. actionType: 'Rename',
  403. actionStage: 'Sub',
  404. fromPath: '/resume_rename_7',
  405. // toPath NOT exist
  406. page: {
  407. _id: pageId7,
  408. parent: rootPage._id,
  409. descendantCount: 2,
  410. isEmpty: false,
  411. path: '/resume_rename_7',
  412. revision: pageOpRevisionId3,
  413. status: 'published',
  414. grant: 1,
  415. grantedUsers: [],
  416. grantedGroups: null,
  417. creator: dummyUser1._id,
  418. lastUpdateUser: dummyUser1._id,
  419. },
  420. user: {
  421. _id: dummyUser1._id,
  422. },
  423. options: {
  424. createRedirectPage: false,
  425. updateMetadata: true,
  426. },
  427. unprocessableExpiryDate: new Date(),
  428. },
  429. {
  430. _id: pageOpId4,
  431. actionType: 'Rename',
  432. actionStage: 'Sub',
  433. fromPath: '/resume_rename_9',
  434. toPath: '/resume_rename_8/resume_rename_9',
  435. page: {
  436. _id: pageId9,
  437. parent: rootPage._id,
  438. descendantCount: 1,
  439. isEmpty: false,
  440. path: '/resume_rename_9',
  441. revision: pageOpRevisionId4,
  442. status: 'published',
  443. grant: Page.GRANT_PUBLIC,
  444. grantedUsers: [],
  445. grantedGroups: null,
  446. creator: dummyUser1._id,
  447. lastUpdateUser: dummyUser1._id,
  448. },
  449. user: {
  450. _id: dummyUser1._id,
  451. },
  452. options: {
  453. createRedirectPage: false,
  454. updateMetadata: true,
  455. },
  456. unprocessableExpiryDate: null,
  457. },
  458. ]);
  459. });
  460. describe('restart renameOperation', () => {
  461. const resumeRenameSubOperation = async(renamePage, pageOp, activity?) => {
  462. const mockedPathsAndDescendantCountOfAncestors = jest.spyOn(crowi.pageService, 'fixPathsAndDescendantCountOfAncestors').mockReturnValue(null);
  463. await crowi.pageService.resumeRenameSubOperation(renamePage, pageOp, activity);
  464. const argsForRenameSubOperation = mockedPathsAndDescendantCountOfAncestors.mock.calls[0];
  465. mockedPathsAndDescendantCountOfAncestors.mockRestore();
  466. await crowi.pageService.fixPathsAndDescendantCountOfAncestors(...argsForRenameSubOperation);
  467. };
  468. test('it should successfully restart rename operation', async() => {
  469. // paths before renaming
  470. const _path0 = '/resume_rename_0'; // out of renaming scope
  471. const _path1 = '/resume_rename_0/resume_rename_1'; // renamed already
  472. const _path2 = '/resume_rename_1/resume_rename_2'; // not renamed yet
  473. const _path3 = '/resume_rename_1/resume_rename_2/resume_rename_3'; // not renamed yet
  474. // paths after renaming
  475. const path0 = '/resume_rename_0';
  476. const path1 = '/resume_rename_0/resume_rename_1';
  477. const path2 = '/resume_rename_0/resume_rename_1/resume_rename_2';
  478. const path3 = '/resume_rename_0/resume_rename_1/resume_rename_2/resume_rename_3';
  479. // activity options
  480. const activity = 'randomActivityId';
  481. // page
  482. const _page0 = await Page.findOne({ path: _path0 });
  483. const _page1 = await Page.findOne({ path: _path1 });
  484. const _page2 = await Page.findOne({ path: _path2 });
  485. const _page3 = await Page.findOne({ path: _path3 });
  486. expect(_page0).toBeTruthy();
  487. expect(_page1).toBeTruthy();
  488. expect(_page2).toBeTruthy();
  489. expect(_page3).toBeTruthy();
  490. expect(_page0.descendantCount).toBe(1);
  491. expect(_page1.descendantCount).toBe(2);
  492. expect(_page2.descendantCount).toBe(1);
  493. expect(_page3.descendantCount).toBe(0);
  494. // page operation
  495. const fromPath = '/resume_rename_1';
  496. const toPath = '/resume_rename_0/resume_rename_1';
  497. const _pageOperation = await PageOperation.findOne({
  498. _id: pageOpId1, fromPath, toPath, 'page._id': _page1._id, actionType: PageActionType.Rename, actionStage: PageActionStage.Sub,
  499. });
  500. expect(_pageOperation).toBeTruthy();
  501. // rename
  502. await resumeRenameSubOperation(_page1, _pageOperation, activity);
  503. // page
  504. const page0 = await Page.findById(_page0._id);
  505. const page1 = await Page.findById(_page1._id);
  506. const page2 = await Page.findById(_page2._id);
  507. const page3 = await Page.findById(_page3._id);
  508. expect(page0).toBeTruthy();
  509. expect(page1).toBeTruthy();
  510. expect(page2).toBeTruthy();
  511. expect(page3).toBeTruthy();
  512. // check paths after renaming
  513. expect(page0.path).toBe(path0);
  514. expect(page1.path).toBe(path1);
  515. expect(page2.path).toBe(path2);
  516. expect(page3.path).toBe(path3);
  517. // page operation
  518. const pageOperation = await PageOperation.findById(_pageOperation._id);
  519. expect(pageOperation).toBeNull(); // should not exist
  520. expect(page0.descendantCount).toBe(3);
  521. expect(page1.descendantCount).toBe(2);
  522. expect(page2.descendantCount).toBe(1);
  523. expect(page3.descendantCount).toBe(0);
  524. });
  525. test('it should successfully restart rename operation when unprocessableExpiryDate is null', async() => {
  526. // paths before renaming
  527. const _path0 = '/resume_rename_8'; // out of renaming scope
  528. const _path1 = '/resume_rename_8/resume_rename_9'; // renamed already
  529. const _path2 = '/resume_rename_9/resume_rename_10'; // not renamed yet
  530. // paths after renaming
  531. const path0 = '/resume_rename_8';
  532. const path1 = '/resume_rename_8/resume_rename_9';
  533. const path2 = '/resume_rename_8/resume_rename_9/resume_rename_10';
  534. // activity options
  535. const activityParameters = {
  536. ip: '::ffff:127.0.0.1',
  537. endpoint: '/_api/v3/pages/rename',
  538. activityId: '62e291bc10e0ab61bd691794',
  539. };
  540. // page
  541. const _page0 = await Page.findOne({ path: _path0 });
  542. const _page1 = await Page.findOne({ path: _path1 });
  543. const _page2 = await Page.findOne({ path: _path2 });
  544. expect(_page0).toBeTruthy();
  545. expect(_page1).toBeTruthy();
  546. expect(_page2).toBeTruthy();
  547. expect(_page0.descendantCount).toBe(1);
  548. expect(_page1.descendantCount).toBe(1);
  549. expect(_page2.descendantCount).toBe(0);
  550. // page operation
  551. const fromPath = '/resume_rename_9';
  552. const toPath = '/resume_rename_8/resume_rename_9';
  553. const _pageOperation = await PageOperation.findOne({
  554. _id: pageOpId4, fromPath, toPath, 'page._id': _page1._id, actionType: PageActionType.Rename, actionStage: PageActionStage.Sub,
  555. });
  556. expect(_pageOperation).toBeTruthy();
  557. // rename
  558. await resumeRenameSubOperation(_page1, _pageOperation, activityParameters);
  559. // page
  560. const page0 = await Page.findById(_page0._id);
  561. const page1 = await Page.findById(_page1._id);
  562. const page2 = await Page.findById(_page2._id);
  563. expect(page0).toBeTruthy();
  564. expect(page1).toBeTruthy();
  565. expect(page2).toBeTruthy();
  566. // check paths after renaming
  567. expect(page0.path).toBe(path0);
  568. expect(page1.path).toBe(path1);
  569. expect(page2.path).toBe(path2);
  570. // page operation
  571. const pageOperation = await PageOperation.findById(_pageOperation._id);
  572. expect(pageOperation).toBeNull(); // should not exist
  573. // others
  574. expect(page1.parent).toStrictEqual(page0._id);
  575. expect(page2.parent).toStrictEqual(page1._id);
  576. expect(page0.descendantCount).toBe(2);
  577. expect(page1.descendantCount).toBe(1);
  578. expect(page2.descendantCount).toBe(0);
  579. });
  580. test('it should fail and throw error if the current time is behind unprocessableExpiryDate', async() => {
  581. // path before renaming
  582. const _path0 = '/resume_rename_4'; // out of renaming scope
  583. const _path1 = '/resume_rename_4/resume_rename_5'; // renamed already
  584. const _path2 = '/resume_rename_5/resume_rename_6'; // not renamed yet
  585. // page
  586. const _page0 = await Page.findOne({ path: _path0 });
  587. const _page1 = await Page.findOne({ path: _path1 });
  588. const _page2 = await Page.findOne({ path: _path2 });
  589. expect(_page0).toBeTruthy();
  590. expect(_page1).toBeTruthy();
  591. expect(_page2).toBeTruthy();
  592. // page operation
  593. const fromPath = '/resume_rename_5';
  594. const toPath = '/resume_rename_4/resume_rename_5';
  595. const _pageOperation = await PageOperation.findOne({
  596. _id: pageOpId2, fromPath, toPath, 'page._id': _page1._id, actionType: PageActionType.Rename, actionStage: PageActionStage.Sub,
  597. });
  598. expect(_pageOperation).toBeTruthy();
  599. // Make `unprocessableExpiryDate` 15 seconds ahead of current time.
  600. // The number 15 seconds has no meaning other than placing time in the furue.
  601. const pageOperation = await PageOperation.findByIdAndUpdate(_pageOperation._id, { unprocessableExpiryDate: addSeconds(new Date(), 15) }, { new: true });
  602. expect(pageOperation).toBeTruthy();
  603. await expect(resumeRenameSubOperation(_page1, pageOperation)).rejects.toThrow(new Error('This page operation is currently being processed'));
  604. // cleanup
  605. await PageOperation.findByIdAndDelete(pageOperation._id);
  606. });
  607. test('Missing property(toPath) for PageOperation should throw error', async() => {
  608. // page
  609. const _path1 = '/resume_rename_7';
  610. const _page1 = await Page.findOne({ path: _path1 });
  611. expect(_page1).toBeTruthy();
  612. // page operation
  613. const pageOperation = await PageOperation.findOne({
  614. _id: pageOpId3, 'page._id': _page1._id, actionType: PageActionType.Rename, actionStage: PageActionStage.Sub,
  615. });
  616. expect(pageOperation).toBeTruthy();
  617. const promise = resumeRenameSubOperation(_page1, pageOperation);
  618. await expect(promise).rejects.toThrow(new Error(`Property toPath is missing which is needed to resume rename operation(${pageOperation._id})`));
  619. // cleanup
  620. await PageOperation.findByIdAndDelete(pageOperation._id);
  621. });
  622. });
  623. describe('updateDescendantCountOfPagesWithPaths', () => {
  624. test('should fix descendantCount of pages with one of the given paths', async() => {
  625. // path
  626. const _path1 = '/fix_descendantCount_1';
  627. const _path2 = '/fix_descendantCount_1/fix_descendantCount_2'; // empty
  628. const _path3 = '/fix_descendantCount_1/fix_descendantCount_2/fix_descendantCount_3';
  629. const _path4 = '/fix_descendantCount_4';
  630. const _path5 = '/fix_descendantCount_4/fix_descendantCount_5';
  631. // page
  632. const _page1 = await Page.findOne({ path: _path1 });
  633. const _page2 = await Page.findOne({ path: _path2 });
  634. const _page3 = await Page.findOne({ path: _path3 });
  635. const _page4 = await Page.findOne({ path: _path4 });
  636. const _page5 = await Page.findOne({ path: _path5 });
  637. // check existance
  638. expect(_page1).toBeTruthy();
  639. expect(_page2).toBeTruthy();
  640. expect(_page3).toBeTruthy();
  641. expect(_page4).toBeTruthy();
  642. expect(_page5).toBeTruthy();
  643. // check descendantCount (all broken)
  644. expect(_page1.descendantCount).toBe(100);
  645. expect(_page2.descendantCount).toBe(100);
  646. expect(_page3.descendantCount).toBe(100);
  647. expect(_page4.descendantCount).toBe(100);
  648. expect(_page5.descendantCount).toBe(100);
  649. // check isEmpty
  650. expect(_page1.isEmpty).toBe(false);
  651. expect(_page2.isEmpty).toBe(true);
  652. expect(_page3.isEmpty).toBe(false);
  653. expect(_page4.isEmpty).toBe(false);
  654. expect(_page5.isEmpty).toBe(false);
  655. // check parent
  656. expect(_page1.parent).toStrictEqual(rootPage._id);
  657. expect(_page2.parent).toStrictEqual(_page1._id);
  658. expect(_page3.parent).toStrictEqual(_page2._id);
  659. expect(_page4.parent).toStrictEqual(rootPage._id);
  660. expect(_page5.parent).toStrictEqual(_page4._id);
  661. await crowi.pageService.updateDescendantCountOfPagesWithPaths([_path1, _path2, _path3, _path4, _path5]);
  662. // page
  663. const page1 = await Page.findById(_page1._id);
  664. const page2 = await Page.findById(_page2._id);
  665. const page3 = await Page.findById(_page3._id);
  666. const page4 = await Page.findById(_page4._id);
  667. const page5 = await Page.findById(_page5._id);
  668. // check existance
  669. expect(page1).toBeTruthy();
  670. expect(page2).toBeTruthy();
  671. expect(page3).toBeTruthy();
  672. expect(page4).toBeTruthy();
  673. expect(page5).toBeTruthy();
  674. // check descendantCount (all fixed)
  675. expect(page1.descendantCount).toBe(1);
  676. expect(page2.descendantCount).toBe(1);
  677. expect(page3.descendantCount).toBe(0);
  678. expect(page4.descendantCount).toBe(1);
  679. expect(page5.descendantCount).toBe(0);
  680. // check isEmpty
  681. expect(page1.isEmpty).toBe(false);
  682. expect(page2.isEmpty).toBe(true);
  683. expect(page3.isEmpty).toBe(false);
  684. expect(page4.isEmpty).toBe(false);
  685. expect(page5.isEmpty).toBe(false);
  686. // check parent
  687. expect(page1.parent).toStrictEqual(rootPage._id);
  688. expect(page2.parent).toStrictEqual(page1._id);
  689. expect(page3.parent).toStrictEqual(page2._id);
  690. expect(page4.parent).toStrictEqual(rootPage._id);
  691. expect(page5.parent).toStrictEqual(page4._id);
  692. });
  693. });
  694. });