page.test.js 40 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021
  1. /* eslint-disable no-unused-vars */
  2. import { advanceTo } from 'jest-date-mock';
  3. import Tag from '~/server/models/tag';
  4. const mongoose = require('mongoose');
  5. const { getInstance } = require('../setup-crowi');
  6. let rootPage;
  7. let dummyUser1;
  8. let testUser1;
  9. let testUser2;
  10. let parentTag;
  11. let childTag;
  12. let parentForRename1;
  13. let parentForRename2;
  14. let parentForRename3;
  15. let parentForRename4;
  16. let parentForRename5;
  17. let parentForRename6;
  18. let parentForRename7;
  19. let parentForRename8;
  20. let parentForRename9;
  21. let irrelevantPage1;
  22. let irrelevantPage2;
  23. let childForRename1;
  24. let childForRename2;
  25. let childForRename3;
  26. let parentForDuplicate;
  27. let parentForDelete1;
  28. let parentForDelete2;
  29. let childForDelete;
  30. let parentForDeleteCompletely;
  31. let parentForRevert1;
  32. let parentForRevert2;
  33. let childForDuplicate;
  34. let childForDeleteCompletely;
  35. let childForRevert;
  36. describe('PageService', () => {
  37. let crowi;
  38. let Page;
  39. let Revision;
  40. let User;
  41. let PageTagRelation;
  42. let Bookmark;
  43. let Comment;
  44. let ShareLink;
  45. let xssSpy;
  46. beforeAll(async() => {
  47. crowi = await getInstance();
  48. await crowi.configManager.updateConfigsInTheSameNamespace('crowi', { 'app:isV5Compatible': null });
  49. User = mongoose.model('User');
  50. Page = mongoose.model('Page');
  51. Revision = mongoose.model('Revision');
  52. PageTagRelation = mongoose.model('PageTagRelation');
  53. Bookmark = mongoose.model('Bookmark');
  54. Comment = mongoose.model('Comment');
  55. ShareLink = mongoose.model('ShareLink');
  56. await User.insertMany([
  57. { name: 'someone1', username: 'someone1', email: 'someone1@example.com' },
  58. { name: 'someone2', username: 'someone2', email: 'someone2@example.com' },
  59. ]);
  60. testUser1 = await User.findOne({ username: 'someone1' });
  61. testUser2 = await User.findOne({ username: 'someone2' });
  62. dummyUser1 = await User.findOne({ username: 'v5DummyUser1' });
  63. rootPage = await Page.findOne({ path: '/' });
  64. await Page.insertMany([
  65. {
  66. path: '/parentForRename1',
  67. grant: Page.GRANT_PUBLIC,
  68. creator: testUser1,
  69. lastUpdateUser: testUser1,
  70. },
  71. {
  72. path: '/parentForRename2',
  73. grant: Page.GRANT_PUBLIC,
  74. creator: testUser1,
  75. lastUpdateUser: testUser1,
  76. },
  77. {
  78. path: '/parentForRename3',
  79. grant: Page.GRANT_PUBLIC,
  80. creator: testUser1,
  81. lastUpdateUser: testUser1,
  82. },
  83. {
  84. path: '/parentForRename4',
  85. grant: Page.GRANT_PUBLIC,
  86. creator: testUser1,
  87. lastUpdateUser: testUser1,
  88. },
  89. {
  90. path: '/parentForRename5',
  91. grant: Page.GRANT_PUBLIC,
  92. creator: testUser1,
  93. lastUpdateUser: testUser1,
  94. },
  95. {
  96. path: '/parentForRename6',
  97. grant: Page.GRANT_PUBLIC,
  98. creator: testUser1,
  99. lastUpdateUser: testUser1,
  100. },
  101. {
  102. path: '/level1/level2',
  103. grant: Page.GRANT_PUBLIC,
  104. creator: testUser1,
  105. lastUpdateUser: testUser1,
  106. },
  107. {
  108. path: '/level1/level2/child',
  109. grant: Page.GRANT_PUBLIC,
  110. creator: testUser1,
  111. lastUpdateUser: testUser1,
  112. },
  113. {
  114. path: '/level1/level2/level2',
  115. grant: Page.GRANT_PUBLIC,
  116. creator: testUser1,
  117. lastUpdateUser: testUser1,
  118. },
  119. {
  120. path: '/parentForRename6-2021H1',
  121. grant: Page.GRANT_PUBLIC,
  122. creator: testUser1,
  123. lastUpdateUser: testUser1,
  124. },
  125. {
  126. path: '/level1-2021H1',
  127. grant: Page.GRANT_PUBLIC,
  128. creator: testUser1,
  129. lastUpdateUser: testUser1,
  130. },
  131. {
  132. path: '/parentForRename1/child',
  133. grant: Page.GRANT_PUBLIC,
  134. creator: testUser1,
  135. lastUpdateUser: testUser1,
  136. },
  137. {
  138. path: '/parentForRename2/child',
  139. grant: Page.GRANT_PUBLIC,
  140. creator: testUser1,
  141. lastUpdateUser: testUser1,
  142. },
  143. {
  144. path: '/parentForRename3/child',
  145. grant: Page.GRANT_PUBLIC,
  146. creator: testUser1,
  147. lastUpdateUser: testUser1,
  148. },
  149. {
  150. path: '/parentForDuplicate',
  151. grant: Page.GRANT_PUBLIC,
  152. creator: testUser1,
  153. lastUpdateUser: testUser1,
  154. revision: '600d395667536503354cbe91',
  155. },
  156. {
  157. path: '/parentForDuplicate/child',
  158. grant: Page.GRANT_PUBLIC,
  159. creator: testUser1,
  160. lastUpdateUser: testUser1,
  161. revision: '600d395667536503354cbe92',
  162. },
  163. {
  164. path: '/parentForDelete1',
  165. grant: Page.GRANT_PUBLIC,
  166. creator: testUser1,
  167. lastUpdateUser: testUser1,
  168. },
  169. {
  170. path: '/parentForDelete2',
  171. grant: Page.GRANT_PUBLIC,
  172. creator: testUser1,
  173. lastUpdateUser: testUser1,
  174. },
  175. {
  176. path: '/parentForDelete/child',
  177. grant: Page.GRANT_PUBLIC,
  178. creator: testUser1,
  179. lastUpdateUser: testUser1,
  180. },
  181. {
  182. path: '/parentForDeleteCompletely',
  183. grant: Page.GRANT_PUBLIC,
  184. creator: testUser1,
  185. lastUpdateUser: testUser1,
  186. },
  187. {
  188. path: '/parentForDeleteCompletely/child',
  189. grant: Page.GRANT_PUBLIC,
  190. creator: testUser1,
  191. lastUpdateUser: testUser1,
  192. },
  193. {
  194. path: '/trash/parentForRevert1',
  195. status: Page.STATUS_DELETED,
  196. grant: Page.GRANT_PUBLIC,
  197. creator: testUser1,
  198. lastUpdateUser: testUser1,
  199. },
  200. {
  201. path: '/trash/parentForRevert2',
  202. status: Page.STATUS_DELETED,
  203. grant: Page.GRANT_PUBLIC,
  204. creator: testUser1,
  205. lastUpdateUser: testUser1,
  206. },
  207. {
  208. path: '/trash/parentForRevert/child',
  209. status: Page.STATUS_DELETED,
  210. grant: Page.GRANT_PUBLIC,
  211. creator: testUser1,
  212. lastUpdateUser: testUser1,
  213. },
  214. ]);
  215. parentForRename1 = await Page.findOne({ path: '/parentForRename1' });
  216. parentForRename2 = await Page.findOne({ path: '/parentForRename2' });
  217. parentForRename3 = await Page.findOne({ path: '/parentForRename3' });
  218. parentForRename4 = await Page.findOne({ path: '/parentForRename4' });
  219. parentForRename5 = await Page.findOne({ path: '/parentForRename5' });
  220. parentForRename6 = await Page.findOne({ path: '/parentForRename6' });
  221. parentForRename7 = await Page.findOne({ path: '/level1/level2' });
  222. parentForRename8 = await Page.findOne({ path: '/level1/level2/child' });
  223. parentForRename9 = await Page.findOne({ path: '/level1/level2/level2' });
  224. irrelevantPage1 = await Page.findOne({ path: '/parentForRename6-2021H1' });
  225. irrelevantPage2 = await Page.findOne({ path: '/level1-2021H1' });
  226. parentForDuplicate = await Page.findOne({ path: '/parentForDuplicate' });
  227. parentForDelete1 = await Page.findOne({ path: '/parentForDelete1' });
  228. parentForDelete2 = await Page.findOne({ path: '/parentForDelete2' });
  229. parentForDeleteCompletely = await Page.findOne({ path: '/parentForDeleteCompletely' });
  230. parentForRevert1 = await Page.findOne({ path: '/trash/parentForRevert1' });
  231. parentForRevert2 = await Page.findOne({ path: '/trash/parentForRevert2' });
  232. childForRename1 = await Page.findOne({ path: '/parentForRename1/child' });
  233. childForRename2 = await Page.findOne({ path: '/parentForRename2/child' });
  234. childForRename3 = await Page.findOne({ path: '/parentForRename3/child' });
  235. childForDuplicate = await Page.findOne({ path: '/parentForDuplicate/child' });
  236. childForDelete = await Page.findOne({ path: '/parentForDelete/child' });
  237. childForDeleteCompletely = await Page.findOne({ path: '/parentForDeleteCompletely/child' });
  238. childForRevert = await Page.findOne({ path: '/trash/parentForRevert/child' });
  239. await Tag.insertMany([
  240. { name: 'Parent' },
  241. { name: 'Child' },
  242. ]);
  243. parentTag = await Tag.findOne({ name: 'Parent' });
  244. childTag = await Tag.findOne({ name: 'Child' });
  245. await PageTagRelation.insertMany([
  246. { relatedPage: parentForDuplicate, relatedTag: parentTag },
  247. { relatedPage: childForDuplicate, relatedTag: childTag },
  248. ]);
  249. await Revision.insertMany([
  250. {
  251. _id: '600d395667536503354cbe91',
  252. pageId: parentForDuplicate._id,
  253. body: 'duplicateBody',
  254. },
  255. {
  256. _id: '600d395667536503354cbe92',
  257. pageId: childForDuplicate._id,
  258. body: 'duplicateChildBody',
  259. },
  260. ]);
  261. xssSpy = jest.spyOn(crowi.xss, 'process').mockImplementation(path => path);
  262. /**
  263. * getParentAndFillAncestors
  264. */
  265. const pageIdPAF1 = new mongoose.Types.ObjectId();
  266. const pageIdPAF2 = new mongoose.Types.ObjectId();
  267. const pageIdPAF3 = new mongoose.Types.ObjectId();
  268. await Page.insertMany([
  269. {
  270. _id: pageIdPAF1,
  271. path: '/PAF1',
  272. grant: Page.GRANT_PUBLIC,
  273. creator: dummyUser1,
  274. lastUpdateUser: dummyUser1._id,
  275. isEmpty: false,
  276. parent: rootPage._id,
  277. descendantCount: 0,
  278. },
  279. {
  280. _id: pageIdPAF2,
  281. path: '/emp_anc3',
  282. grant: Page.GRANT_PUBLIC,
  283. isEmpty: true,
  284. descendantCount: 1,
  285. parent: rootPage._id,
  286. },
  287. {
  288. path: '/emp_anc3/PAF3',
  289. grant: Page.GRANT_PUBLIC,
  290. creator: dummyUser1,
  291. lastUpdateUser: dummyUser1._id,
  292. isEmpty: false,
  293. descendantCount: 0,
  294. parent: pageIdPAF2,
  295. },
  296. {
  297. _id: pageIdPAF3,
  298. path: '/emp_anc4',
  299. grant: Page.GRANT_PUBLIC,
  300. isEmpty: true,
  301. descendantCount: 1,
  302. parent: rootPage._id,
  303. },
  304. {
  305. path: '/emp_anc4/PAF4',
  306. grant: Page.GRANT_PUBLIC,
  307. creator: dummyUser1,
  308. lastUpdateUser: dummyUser1._id,
  309. isEmpty: false,
  310. descendantCount: 0,
  311. parent: pageIdPAF3,
  312. },
  313. {
  314. path: '/emp_anc4',
  315. grant: Page.GRANT_OWNER,
  316. grantedUsers: [dummyUser1._id],
  317. creator: dummyUser1,
  318. lastUpdateUser: dummyUser1._id,
  319. isEmpty: false,
  320. },
  321. {
  322. path: '/get_parent_A',
  323. creator: dummyUser1,
  324. lastUpdateUser: dummyUser1,
  325. parent: null,
  326. },
  327. {
  328. path: '/get_parent_A/get_parent_B',
  329. creator: dummyUser1,
  330. lastUpdateUser: dummyUser1,
  331. parent: null,
  332. },
  333. {
  334. path: '/get_parent_C',
  335. creator: dummyUser1,
  336. lastUpdateUser: dummyUser1,
  337. parent: rootPage._id,
  338. },
  339. {
  340. path: '/get_parent_C/get_parent_D',
  341. creator: dummyUser1,
  342. lastUpdateUser: dummyUser1,
  343. parent: null,
  344. },
  345. ]);
  346. });
  347. describe('rename page without using renameDescendantsWithStreamSpy', () => {
  348. test('rename page with different tree with isRecursively [deeper]', async() => {
  349. const resultPage = await crowi.pageService.renamePage(parentForRename6, '/parentForRename6/renamedChild', testUser1, { isRecursively: true },
  350. { ip: '::ffff:127.0.0.1', endpoint: '/_api/v3/pages/rename' });
  351. const wrongPage = await Page.findOne({ path: '/parentForRename6/renamedChild/renamedChild' });
  352. const expectPage1 = await Page.findOne({ path: '/parentForRename6/renamedChild' });
  353. const expectPage2 = await Page.findOne({ path: '/parentForRename6-2021H1' });
  354. expect(resultPage.path).toEqual(expectPage1.path);
  355. expect(expectPage2.path).not.toBeNull();
  356. // Check that pages that are not to be renamed have not been renamed
  357. expect(wrongPage).toBeNull();
  358. });
  359. test('rename page with different tree with isRecursively [shallower]', async() => {
  360. // setup
  361. expect(await Page.findOne({ path: '/level1' })).toBeNull();
  362. expect(await Page.findOne({ path: '/level1/level2' })).not.toBeNull();
  363. expect(await Page.findOne({ path: '/level1/level2/child' })).not.toBeNull();
  364. expect(await Page.findOne({ path: '/level1/level2/level2' })).not.toBeNull();
  365. expect(await Page.findOne({ path: '/level1-2021H1' })).not.toBeNull();
  366. // when
  367. // rename /level1/level2 --> /level1
  368. await crowi.pageService.renamePage(parentForRename7, '/level1', testUser1, { isRecursively: true },
  369. { ip: '::ffff:127.0.0.1', endpoint: '/_api/v3/pages/rename' });
  370. // then
  371. expect(await Page.findOne({ path: '/level1' })).not.toBeNull();
  372. expect(await Page.findOne({ path: '/level1/child' })).not.toBeNull();
  373. expect(await Page.findOne({ path: '/level1/level2' })).not.toBeNull();
  374. expect(await Page.findOne({ path: '/level1/level2/child' })).toBeNull();
  375. expect(await Page.findOne({ path: '/level1/level2/level2' })).toBeNull();
  376. // Check that pages that are not to be renamed have not been renamed
  377. expect(await Page.findOne({ path: '/level1-2021H1' })).not.toBeNull();
  378. });
  379. });
  380. describe('rename page', () => {
  381. let pageEventSpy;
  382. let renameDescendantsWithStreamSpy;
  383. // mock new Date() and Date.now()
  384. advanceTo(new Date(2000, 1, 1, 0, 0, 0));
  385. const dateToUse = new Date();
  386. beforeEach(async() => {
  387. pageEventSpy = jest.spyOn(crowi.pageService.pageEvent, 'emit').mockImplementation();
  388. renameDescendantsWithStreamSpy = jest.spyOn(crowi.pageService, 'renameDescendantsWithStream').mockImplementation();
  389. });
  390. describe('renamePage()', () => {
  391. test('rename page without options', async() => {
  392. const resultPage = await crowi.pageService.renamePage(parentForRename1,
  393. '/renamed1', testUser2, {}, { ip: '::ffff:127.0.0.1', endpoint: '/_api/v3/pages/rename' });
  394. expect(xssSpy).toHaveBeenCalled();
  395. expect(pageEventSpy).toHaveBeenCalledWith('rename');
  396. expect(resultPage.path).toBe('/renamed1');
  397. expect(resultPage.updatedAt).toEqual(parentForRename1.updatedAt);
  398. expect(resultPage.lastUpdateUser).toEqual(testUser1._id);
  399. });
  400. test('rename page with updateMetadata option', async() => {
  401. const resultPage = await crowi.pageService.renamePage(parentForRename2, '/renamed2', testUser2, { updateMetadata: true },
  402. { ip: '::ffff:127.0.0.1', endpoint: '/_api/v3/pages/rename' });
  403. expect(xssSpy).toHaveBeenCalled();
  404. expect(pageEventSpy).toHaveBeenCalledWith('rename');
  405. expect(resultPage.path).toBe('/renamed2');
  406. expect(resultPage.updatedAt).toEqual(dateToUse);
  407. expect(resultPage.lastUpdateUser).toEqual(testUser2._id);
  408. });
  409. test('rename page with createRedirectPage option', async() => {
  410. const resultPage = await crowi.pageService.renamePage(parentForRename3, '/renamed3', testUser2, { createRedirectPage: true },
  411. { ip: '::ffff:127.0.0.1', endpoint: '/_api/v3/pages/rename' });
  412. expect(xssSpy).toHaveBeenCalled();
  413. expect(pageEventSpy).toHaveBeenCalledWith('rename');
  414. expect(resultPage.path).toBe('/renamed3');
  415. expect(resultPage.updatedAt).toEqual(parentForRename3.updatedAt);
  416. expect(resultPage.lastUpdateUser).toEqual(testUser1._id);
  417. });
  418. test('rename page with isRecursively', async() => {
  419. const resultPage = await crowi.pageService.renamePage(parentForRename4, '/renamed4', testUser2, { isRecursively: true },
  420. { ip: '::ffff:127.0.0.1', endpoint: '/_api/v3/pages/rename' });
  421. expect(xssSpy).toHaveBeenCalled();
  422. expect(renameDescendantsWithStreamSpy).toHaveBeenCalled();
  423. expect(pageEventSpy).toHaveBeenCalledWith('rename');
  424. expect(resultPage.path).toBe('/renamed4');
  425. expect(resultPage.updatedAt).toEqual(parentForRename4.updatedAt);
  426. expect(resultPage.lastUpdateUser).toEqual(testUser1._id);
  427. });
  428. test('rename page with different tree with isRecursively', async() => {
  429. const resultPage = await crowi.pageService.renamePage(parentForRename5, '/parentForRename5/renamedChild', testUser1, { isRecursively: true },
  430. { ip: '::ffff:127.0.0.1', endpoint: '/_api/v3/pages/rename' });
  431. const wrongPage = await Page.findOne({ path: '/parentForRename5/renamedChild/renamedChild' });
  432. const expectPage = await Page.findOne({ path: '/parentForRename5/renamedChild' });
  433. expect(resultPage.path).toEqual(expectPage.path);
  434. expect(wrongPage).toBeNull();
  435. });
  436. });
  437. test('renameDescendants without options', async() => {
  438. const oldPagePathPrefix = new RegExp('^/parentForRename1', 'i');
  439. const newPagePathPrefix = '/renamed1';
  440. await crowi.pageService.renameDescendants([childForRename1], testUser2, {}, oldPagePathPrefix, newPagePathPrefix);
  441. const resultPage = await Page.findOne({ path: '/renamed1/child' });
  442. expect(resultPage).not.toBeNull();
  443. expect(pageEventSpy).toHaveBeenCalledWith('updateMany', [childForRename1], testUser2);
  444. expect(resultPage.path).toBe('/renamed1/child');
  445. expect(resultPage.updatedAt).toEqual(childForRename1.updatedAt);
  446. expect(resultPage.lastUpdateUser).toEqual(testUser1._id);
  447. });
  448. test('renameDescendants with updateMetadata option', async() => {
  449. const oldPagePathPrefix = new RegExp('^/parentForRename2', 'i');
  450. const newPagePathPrefix = '/renamed2';
  451. await crowi.pageService.renameDescendants([childForRename2], testUser2, { updateMetadata: true }, oldPagePathPrefix, newPagePathPrefix);
  452. const resultPage = await Page.findOne({ path: '/renamed2/child' });
  453. expect(resultPage).not.toBeNull();
  454. expect(pageEventSpy).toHaveBeenCalledWith('updateMany', [childForRename2], testUser2);
  455. expect(resultPage.path).toBe('/renamed2/child');
  456. expect(resultPage.updatedAt).toEqual(dateToUse);
  457. expect(resultPage.lastUpdateUser).toEqual(testUser2._id);
  458. });
  459. test('renameDescendants with createRedirectPage option', async() => {
  460. const oldPagePathPrefix = new RegExp('^/parentForRename3', 'i');
  461. const newPagePathPrefix = '/renamed3';
  462. await crowi.pageService.renameDescendants([childForRename3], testUser2, { createRedirectPage: true }, oldPagePathPrefix, newPagePathPrefix);
  463. const resultPage = await Page.findOne({ path: '/renamed3/child' });
  464. expect(resultPage).not.toBeNull();
  465. expect(pageEventSpy).toHaveBeenCalledWith('updateMany', [childForRename3], testUser2);
  466. expect(resultPage.path).toBe('/renamed3/child');
  467. expect(resultPage.updatedAt).toEqual(childForRename3.updatedAt);
  468. expect(resultPage.lastUpdateUser).toEqual(testUser1._id);
  469. });
  470. });
  471. describe('duplicate page', () => {
  472. let duplicateDescendantsWithStreamSpy;
  473. // TODO https://redmine.weseek.co.jp/issues/87537 : activate outer module mockImplementation
  474. // jest.mock('~/server/models/serializers/page-serializer');
  475. // const { serializePageSecurely } = require('~/server/models/serializers/page-serializer');
  476. // serializePageSecurely.mockImplementation(page => page);
  477. beforeEach(async() => {
  478. duplicateDescendantsWithStreamSpy = jest.spyOn(crowi.pageService, 'duplicateDescendantsWithStream').mockImplementation();
  479. });
  480. test('duplicate page (isRecursively: false)', async() => {
  481. const dummyId = '600d395667536503354c9999';
  482. crowi.models.Page.findRelatedTagsById = jest.fn().mockImplementation(() => { return parentTag });
  483. const originTagsMock = jest.spyOn(Page, 'findRelatedTagsById').mockImplementation(() => { return parentTag });
  484. jest.spyOn(PageTagRelation, 'updatePageTags').mockImplementation(() => { return [dummyId, parentTag.name] });
  485. jest.spyOn(PageTagRelation, 'listTagNamesByPage').mockImplementation(() => { return [parentTag.name] });
  486. const resultPage = await crowi.pageService.duplicate(parentForDuplicate, '/newParentDuplicate', testUser2, false);
  487. const duplicatedToPageRevision = await Revision.findOne({ pageId: resultPage._id });
  488. expect(xssSpy).toHaveBeenCalled();
  489. expect(duplicateDescendantsWithStreamSpy).not.toHaveBeenCalled();
  490. // TODO https://redmine.weseek.co.jp/issues/87537 : activate outer module mockImplementation
  491. // expect(serializePageSecurely).toHaveBeenCalled();
  492. expect(resultPage.path).toBe('/newParentDuplicate');
  493. expect(resultPage.lastUpdateUser._id).toEqual(testUser2._id);
  494. expect(duplicatedToPageRevision._id).not.toEqual(parentForDuplicate.revision._id);
  495. expect(resultPage.grant).toEqual(parentForDuplicate.grant);
  496. expect(resultPage.tags).toEqual([originTagsMock().name]);
  497. });
  498. test('duplicate page (isRecursively: true)', async() => {
  499. const dummyId = '600d395667536503354c9999';
  500. crowi.models.Page.findRelatedTagsById = jest.fn().mockImplementation(() => { return parentTag });
  501. const originTagsMock = jest.spyOn(Page, 'findRelatedTagsById').mockImplementation(() => { return parentTag });
  502. jest.spyOn(PageTagRelation, 'updatePageTags').mockImplementation(() => { return [dummyId, parentTag.name] });
  503. jest.spyOn(PageTagRelation, 'listTagNamesByPage').mockImplementation(() => { return [parentTag.name] });
  504. const resultPageRecursivly = await crowi.pageService.duplicate(parentForDuplicate, '/newParentDuplicateRecursively', testUser2, true);
  505. const duplicatedRecursivelyToPageRevision = await Revision.findOne({ pageId: resultPageRecursivly._id });
  506. expect(xssSpy).toHaveBeenCalled();
  507. expect(duplicateDescendantsWithStreamSpy).toHaveBeenCalled();
  508. // TODO https://redmine.weseek.co.jp/issues/87537 : activate outer module mockImplementation
  509. // expect(serializePageSecurely).toHaveBeenCalled();
  510. expect(resultPageRecursivly.path).toBe('/newParentDuplicateRecursively');
  511. expect(resultPageRecursivly.lastUpdateUser._id).toEqual(testUser2._id);
  512. expect(duplicatedRecursivelyToPageRevision._id).not.toEqual(parentForDuplicate.revision._id);
  513. expect(resultPageRecursivly.grant).toEqual(parentForDuplicate.grant);
  514. expect(resultPageRecursivly.tags).toEqual([originTagsMock().name]);
  515. });
  516. test('duplicateDescendants()', async() => {
  517. const duplicateTagsMock = await jest.spyOn(crowi.pageService, 'duplicateTags').mockImplementationOnce();
  518. await crowi.pageService.duplicateDescendants([childForDuplicate], testUser2, parentForDuplicate.path, '/newPathPrefix');
  519. const childForDuplicateRevision = await Revision.findOne({ pageId: childForDuplicate._id });
  520. const insertedPage = await Page.findOne({ path: '/newPathPrefix/child' });
  521. const insertedRevision = await Revision.findOne({ pageId: insertedPage._id });
  522. expect(insertedPage).not.toBeNull();
  523. expect(insertedPage.path).toEqual('/newPathPrefix/child');
  524. expect(insertedPage.lastUpdateUser).toEqual(testUser2._id);
  525. expect([insertedRevision]).not.toBeNull();
  526. expect(insertedRevision.pageId).toEqual(insertedPage._id);
  527. expect(insertedRevision._id).not.toEqual(childForDuplicateRevision._id);
  528. expect(insertedRevision.body).toEqual(childForDuplicateRevision.body);
  529. expect(duplicateTagsMock).toHaveBeenCalled();
  530. });
  531. test('duplicateTags()', async() => {
  532. const pageIdMapping = {
  533. [parentForDuplicate._id]: '60110bdd85339d7dc732dddd',
  534. };
  535. const duplicateTagsReturn = await crowi.pageService.duplicateTags(pageIdMapping);
  536. const parentoForDuplicateTag = await PageTagRelation.findOne({ relatedPage: parentForDuplicate._id });
  537. expect(duplicateTagsReturn).toHaveLength(1);
  538. expect(duplicateTagsReturn[0].relatedTag).toEqual(parentoForDuplicateTag.relatedTag);
  539. });
  540. });
  541. describe('delete page', () => {
  542. let getDeletedPageNameSpy;
  543. let pageEventSpy;
  544. let deleteDescendantsWithStreamSpy;
  545. const dateToUse = new Date('2000-01-01');
  546. beforeEach(async() => {
  547. jest.spyOn(global.Date, 'now').mockImplementation(() => dateToUse);
  548. getDeletedPageNameSpy = jest.spyOn(Page, 'getDeletedPageName');
  549. pageEventSpy = jest.spyOn(crowi.pageService.pageEvent, 'emit');
  550. deleteDescendantsWithStreamSpy = jest.spyOn(crowi.pageService, 'deleteDescendantsWithStream').mockImplementation();
  551. });
  552. test('delete page without options', async() => {
  553. const resultPage = await crowi.pageService.deletePage(parentForDelete1, testUser2, { }, false, {
  554. ip: '::ffff:127.0.0.1',
  555. endpoint: '/_api/v3/pages/delete',
  556. });
  557. expect(getDeletedPageNameSpy).toHaveBeenCalled();
  558. expect(deleteDescendantsWithStreamSpy).not.toHaveBeenCalled();
  559. expect(resultPage.status).toBe(Page.STATUS_DELETED);
  560. expect(resultPage.path).toBe('/trash/parentForDelete1');
  561. expect(resultPage.deleteUser).toEqual(testUser2._id);
  562. expect(resultPage.deletedAt).toEqual(dateToUse);
  563. expect(resultPage.updatedAt).toEqual(parentForDelete1.updatedAt);
  564. expect(resultPage.lastUpdateUser).toEqual(testUser1._id);
  565. expect(pageEventSpy).toHaveBeenCalledWith('delete', parentForDelete1, testUser2);
  566. expect(pageEventSpy).toHaveBeenCalledWith('create', resultPage, testUser2);
  567. });
  568. test('delete page with isRecursively', async() => {
  569. const resultPage = await crowi.pageService.deletePage(parentForDelete2, testUser2, { }, true, {
  570. ip: '::ffff:127.0.0.1',
  571. endpoint: '/_api/v3/pages/delete',
  572. });
  573. expect(getDeletedPageNameSpy).toHaveBeenCalled();
  574. expect(deleteDescendantsWithStreamSpy).toHaveBeenCalled();
  575. expect(resultPage.status).toBe(Page.STATUS_DELETED);
  576. expect(resultPage.path).toBe('/trash/parentForDelete2');
  577. expect(resultPage.deleteUser).toEqual(testUser2._id);
  578. expect(resultPage.deletedAt).toEqual(dateToUse);
  579. expect(resultPage.updatedAt).toEqual(parentForDelete2.updatedAt);
  580. expect(resultPage.lastUpdateUser).toEqual(testUser1._id);
  581. expect(pageEventSpy).toHaveBeenCalledWith('delete', parentForDelete2, testUser2);
  582. expect(pageEventSpy).toHaveBeenCalledWith('create', resultPage, testUser2);
  583. });
  584. test('deleteDescendants', async() => {
  585. await crowi.pageService.deleteDescendants([childForDelete], testUser2);
  586. const resultPage = await Page.findOne({ path: '/trash/parentForDelete/child' });
  587. expect(resultPage.status).toBe(Page.STATUS_DELETED);
  588. expect(resultPage.path).toBe('/trash/parentForDelete/child');
  589. expect(resultPage.deleteUser).toEqual(testUser2._id);
  590. expect(resultPage.deletedAt).toEqual(dateToUse);
  591. expect(resultPage.updatedAt).toEqual(childForDelete.updatedAt);
  592. expect(resultPage.lastUpdateUser).toEqual(testUser1._id);
  593. });
  594. });
  595. describe('delete page completely', () => {
  596. let pageEventSpy;
  597. let deleteCompletelyOperationSpy;
  598. let deleteCompletelyDescendantsWithStreamSpy;
  599. let deleteManyBookmarkSpy;
  600. let deleteManyCommentSpy;
  601. let deleteManyPageTagRelationSpy;
  602. let deleteManyShareLinkSpy;
  603. let deleteManyRevisionSpy;
  604. let deleteManyPageSpy;
  605. let removeAllAttachmentsSpy;
  606. beforeEach(async() => {
  607. pageEventSpy = jest.spyOn(crowi.pageService.pageEvent, 'emit');
  608. deleteCompletelyOperationSpy = jest.spyOn(crowi.pageService, 'deleteCompletelyOperation');
  609. deleteCompletelyDescendantsWithStreamSpy = jest.spyOn(crowi.pageService, 'deleteCompletelyDescendantsWithStream').mockImplementation();
  610. deleteManyBookmarkSpy = jest.spyOn(Bookmark, 'deleteMany').mockImplementation();
  611. deleteManyCommentSpy = jest.spyOn(Comment, 'deleteMany').mockImplementation();
  612. deleteManyPageTagRelationSpy = jest.spyOn(PageTagRelation, 'deleteMany').mockImplementation();
  613. deleteManyShareLinkSpy = jest.spyOn(ShareLink, 'deleteMany').mockImplementation();
  614. deleteManyRevisionSpy = jest.spyOn(Revision, 'deleteMany').mockImplementation();
  615. deleteManyPageSpy = jest.spyOn(Page, 'deleteMany').mockImplementation();
  616. removeAllAttachmentsSpy = jest.spyOn(crowi.attachmentService, 'removeAllAttachments').mockImplementation();
  617. });
  618. test('deleteCompletelyOperation', async() => {
  619. await crowi.pageService.deleteCompletelyOperation([parentForDeleteCompletely._id], [parentForDeleteCompletely.path], { });
  620. expect(deleteManyBookmarkSpy).toHaveBeenCalledWith({ page: { $in: [parentForDeleteCompletely._id] } });
  621. expect(deleteManyCommentSpy).toHaveBeenCalledWith({ page: { $in: [parentForDeleteCompletely._id] } });
  622. expect(deleteManyPageTagRelationSpy).toHaveBeenCalledWith({ relatedPage: { $in: [parentForDeleteCompletely._id] } });
  623. expect(deleteManyShareLinkSpy).toHaveBeenCalledWith({ relatedPage: { $in: [parentForDeleteCompletely._id] } });
  624. expect(deleteManyRevisionSpy).toHaveBeenCalledWith({ pageId: { $in: [parentForDeleteCompletely._id] } });
  625. expect(deleteManyPageSpy).toHaveBeenCalledWith({ _id: { $in: [parentForDeleteCompletely._id] } });
  626. expect(removeAllAttachmentsSpy).toHaveBeenCalled();
  627. });
  628. test('delete completely without options', async() => {
  629. await crowi.pageService.deleteCompletely(parentForDeleteCompletely, testUser2, { }, false, false, {
  630. ip: '::ffff:127.0.0.1',
  631. endpoint: '/_api/v3/pages/deletecompletely',
  632. });
  633. expect(deleteCompletelyOperationSpy).toHaveBeenCalled();
  634. expect(deleteCompletelyDescendantsWithStreamSpy).not.toHaveBeenCalled();
  635. expect(pageEventSpy).toHaveBeenCalledWith('deleteCompletely', parentForDeleteCompletely, testUser2);
  636. });
  637. test('delete completely with isRecursively', async() => {
  638. await crowi.pageService.deleteCompletely(parentForDeleteCompletely, testUser2, { }, true, false, {
  639. ip: '::ffff:127.0.0.1',
  640. endpoint: '/_api/v3/pages/deletecompletely',
  641. });
  642. expect(deleteCompletelyOperationSpy).toHaveBeenCalled();
  643. expect(deleteCompletelyDescendantsWithStreamSpy).toHaveBeenCalled();
  644. expect(pageEventSpy).toHaveBeenCalledWith('deleteCompletely', parentForDeleteCompletely, testUser2);
  645. });
  646. });
  647. describe('revert page', () => {
  648. let getRevertDeletedPageNameSpy;
  649. let findByPathSpy;
  650. let findSpy;
  651. let deleteCompletelySpy;
  652. let revertDeletedDescendantsWithStreamSpy;
  653. beforeEach(async() => {
  654. getRevertDeletedPageNameSpy = jest.spyOn(Page, 'getRevertDeletedPageName');
  655. deleteCompletelySpy = jest.spyOn(crowi.pageService, 'deleteCompletely').mockImplementation();
  656. revertDeletedDescendantsWithStreamSpy = jest.spyOn(crowi.pageService, 'revertDeletedDescendantsWithStream').mockImplementation();
  657. });
  658. test('revert deleted page when the redirect from page exists', async() => {
  659. const resultPage = await crowi.pageService.revertDeletedPage(parentForRevert1, testUser2, {}, false, {
  660. ip: '::ffff:127.0.0.1',
  661. endpoint: '/_api/v3/pages/revert',
  662. });
  663. expect(getRevertDeletedPageNameSpy).toHaveBeenCalledWith(parentForRevert1.path);
  664. expect(revertDeletedDescendantsWithStreamSpy).not.toHaveBeenCalled();
  665. expect(resultPage.path).toBe('/parentForRevert1');
  666. expect(resultPage.lastUpdateUser._id).toEqual(testUser2._id);
  667. expect(resultPage.status).toBe(Page.STATUS_PUBLISHED);
  668. expect(resultPage.deleteUser).toBeNull();
  669. expect(resultPage.deletedAt).toBeNull();
  670. });
  671. test('revert deleted page when the redirect from page does not exist', async() => {
  672. findByPathSpy = jest.spyOn(Page, 'findByPath').mockImplementation(() => {
  673. return null;
  674. });
  675. const resultPage = await crowi.pageService.revertDeletedPage(parentForRevert2, testUser2, {}, true, {
  676. ip: '::ffff:127.0.0.1',
  677. endpoint: '/_api/v3/pages/revert',
  678. });
  679. expect(getRevertDeletedPageNameSpy).toHaveBeenCalledWith(parentForRevert2.path);
  680. expect(findByPathSpy).toHaveBeenCalledWith('/parentForRevert2');
  681. expect(deleteCompletelySpy).not.toHaveBeenCalled();
  682. expect(revertDeletedDescendantsWithStreamSpy).toHaveBeenCalled();
  683. expect(resultPage.path).toBe('/parentForRevert2');
  684. expect(resultPage.lastUpdateUser._id).toEqual(testUser2._id);
  685. expect(resultPage.status).toBe(Page.STATUS_PUBLISHED);
  686. expect(resultPage.deleteUser).toBeNull();
  687. expect(resultPage.deletedAt).toBeNull();
  688. });
  689. test('revert deleted descendants', async() => {
  690. await crowi.pageService.revertDeletedDescendants([childForRevert], testUser2);
  691. const resultPage = await Page.findOne({ path: '/parentForRevert/child' });
  692. const revrtedFromPage = await Page.findOne({ path: '/trash/parentForRevert/child' });
  693. const revrtedFromPageRevision = await Revision.findOne({ pageId: resultPage._id });
  694. expect(getRevertDeletedPageNameSpy).toHaveBeenCalledWith(childForRevert.path);
  695. expect(resultPage.path).toBe('/parentForRevert/child');
  696. expect(resultPage.lastUpdateUser._id).toEqual(testUser2._id);
  697. expect(resultPage.status).toBe(Page.STATUS_PUBLISHED);
  698. expect(resultPage.deleteUser).toBeNull();
  699. expect(resultPage.deletedAt).toBeNull();
  700. expect(revrtedFromPage).toBeNull();
  701. expect(revrtedFromPageRevision).toBeNull();
  702. });
  703. });
  704. describe('getParentAndFillAncestors', () => {
  705. test('return parent if exist', async() => {
  706. const page1 = await Page.findOne({ path: '/PAF1' });
  707. const parent = await crowi.pageService.getParentAndFillAncestorsByUser(dummyUser1, page1.path);
  708. expect(parent).toBeTruthy();
  709. expect(page1.parent).toStrictEqual(parent._id);
  710. });
  711. test('create parent and ancestors when they do not exist, and return the new parent', async() => {
  712. const path1 = '/emp_anc1';
  713. const path2 = '/emp_anc1/emp_anc2';
  714. const path3 = '/emp_anc1/emp_anc2/PAF2';
  715. const _page1 = await Page.findOne({ path: path1 }); // not exist
  716. const _page2 = await Page.findOne({ path: path2 }); // not exist
  717. const _page3 = await Page.findOne({ path: path3 }); // not exist
  718. expect(_page1).toBeNull();
  719. expect(_page2).toBeNull();
  720. expect(_page3).toBeNull();
  721. const parent = await crowi.pageService.getParentAndFillAncestorsByUser(dummyUser1, path3);
  722. const page1 = await Page.findOne({ path: path1 });
  723. const page2 = await Page.findOne({ path: path2 });
  724. const page3 = await Page.findOne({ path: path3 });
  725. expect(parent._id).toStrictEqual(page2._id);
  726. expect(parent.path).toStrictEqual(page2.path);
  727. expect(parent.parent).toStrictEqual(page2.parent);
  728. expect(parent).toBeTruthy();
  729. expect(page1).toBeTruthy();
  730. expect(page2).toBeTruthy();
  731. expect(page3).toBeNull();
  732. expect(page1.parent).toStrictEqual(rootPage._id);
  733. expect(page2.parent).toStrictEqual(page1._id);
  734. });
  735. test('return parent even if the parent page is empty', async() => {
  736. const path1 = '/emp_anc3';
  737. const path2 = '/emp_anc3/PAF3';
  738. const _page1 = await Page.findOne({ path: path1, isEmpty: true });
  739. const _page2 = await Page.findOne({ path: path2, isEmpty: false });
  740. expect(_page1).toBeTruthy();
  741. expect(_page2).toBeTruthy();
  742. const parent = await crowi.pageService.getParentAndFillAncestorsByUser(dummyUser1, _page2.path);
  743. const page1 = await Page.findOne({ path: path1, isEmpty: true }); // parent
  744. const page2 = await Page.findOne({ path: path2, isEmpty: false });
  745. // check for the parent (should be the same as page1)
  746. expect(parent._id).toStrictEqual(page1._id);
  747. expect(parent.path).toStrictEqual(page1.path);
  748. expect(parent.parent).toStrictEqual(page1.parent);
  749. expect(page1.parent).toStrictEqual(rootPage._id);
  750. expect(page2.parent).toStrictEqual(page1._id);
  751. });
  752. test('should find parent while NOT updating private legacy page\'s parent', async() => {
  753. const path1 = '/emp_anc4';
  754. const path2 = '/emp_anc4/PAF4';
  755. const _page1 = await Page.findOne({ path: path1, isEmpty: true, grant: Page.GRANT_PUBLIC });
  756. const _page2 = await Page.findOne({ path: path2, isEmpty: false, grant: Page.GRANT_PUBLIC });
  757. const _page3 = await Page.findOne({ path: path1, isEmpty: false, grant: Page.GRANT_OWNER });
  758. expect(_page1).toBeTruthy();
  759. expect(_page2).toBeTruthy();
  760. expect(_page3).toBeTruthy();
  761. expect(_page3.parent).toBeNull();
  762. const parent = await crowi.pageService.getParentAndFillAncestorsByUser(dummyUser1, _page2.path);
  763. const page1 = await Page.findOne({ path: path1, isEmpty: true, grant: Page.GRANT_PUBLIC });
  764. const page2 = await Page.findOne({ path: path2, isEmpty: false, grant: Page.GRANT_PUBLIC });
  765. const page3 = await Page.findOne({ path: path1, isEmpty: false, grant: Page.GRANT_OWNER });
  766. expect(page1).toBeTruthy();
  767. expect(page2).toBeTruthy();
  768. expect(page3).toBeTruthy();
  769. expect(page3.parent).toBeNull(); // parent property of page in private legacy pages should be null
  770. expect(page1._id).toStrictEqual(parent._id);
  771. expect(page2.parent).toStrictEqual(parent._id);
  772. });
  773. test('should find parent while NOT creating unnecessary empty pages with all v4 public pages', async() => {
  774. // All pages does not have parent (v4 schema)
  775. const _pageA = await Page.findOne({
  776. path: '/get_parent_A',
  777. grant: Page.GRANT_PUBLIC,
  778. isEmpty: false,
  779. parent: null,
  780. });
  781. const _pageAB = await Page.findOne({
  782. path: '/get_parent_A/get_parent_B',
  783. grant: Page.GRANT_PUBLIC,
  784. isEmpty: false,
  785. parent: null,
  786. });
  787. const _emptyA = await Page.findOne({
  788. path: '/get_parent_A',
  789. grant: Page.GRANT_PUBLIC,
  790. isEmpty: true,
  791. });
  792. const _emptyAB = await Page.findOne({
  793. path: '/get_parent_A/get_parent_B',
  794. grant: Page.GRANT_PUBLIC,
  795. isEmpty: true,
  796. });
  797. expect(_pageA).not.toBeNull();
  798. expect(_pageAB).not.toBeNull();
  799. expect(_emptyA).toBeNull();
  800. expect(_emptyAB).toBeNull();
  801. const parent = await crowi.pageService.getParentAndFillAncestorsByUser(dummyUser1, '/get_parent_A/get_parent_B/get_parent_C');
  802. const pageA = await Page.findOne({ path: '/get_parent_A', grant: Page.GRANT_PUBLIC, isEmpty: false });
  803. const pageAB = await Page.findOne({ path: '/get_parent_A/get_parent_B', grant: Page.GRANT_PUBLIC, isEmpty: false });
  804. const emptyA = await Page.findOne({ path: '/get_parent_A', grant: Page.GRANT_PUBLIC, isEmpty: true });
  805. const emptyAB = await Page.findOne({ path: '/get_parent_A/get_parent_B', grant: Page.GRANT_PUBLIC, isEmpty: true });
  806. // -- Check existance
  807. expect(parent).not.toBeNull();
  808. expect(pageA).not.toBeNull();
  809. expect(pageAB).not.toBeNull();
  810. expect(emptyA).toBeNull();
  811. expect(emptyAB).toBeNull();
  812. // -- Check parent
  813. expect(pageA.parent).not.toBeNull();
  814. expect(pageAB.parent).not.toBeNull();
  815. });
  816. test('should find parent while NOT creating unnecessary empty pages with some v5 public pages', async() => {
  817. const _pageC = await Page.findOne({
  818. path: '/get_parent_C',
  819. grant: Page.GRANT_PUBLIC,
  820. isEmpty: false,
  821. parent: { $ne: null },
  822. });
  823. const _pageCD = await Page.findOne({
  824. path: '/get_parent_C/get_parent_D',
  825. grant: Page.GRANT_PUBLIC,
  826. isEmpty: false,
  827. });
  828. const _emptyC = await Page.findOne({
  829. path: '/get_parent_C',
  830. grant: Page.GRANT_PUBLIC,
  831. isEmpty: true,
  832. });
  833. const _emptyCD = await Page.findOne({
  834. path: '/get_parent_C/get_parent_D',
  835. grant: Page.GRANT_PUBLIC,
  836. isEmpty: true,
  837. });
  838. expect(_pageC).not.toBeNull();
  839. expect(_pageCD).not.toBeNull();
  840. expect(_emptyC).toBeNull();
  841. expect(_emptyCD).toBeNull();
  842. const parent = await crowi.pageService.getParentAndFillAncestorsByUser(dummyUser1, '/get_parent_C/get_parent_D/get_parent_E');
  843. const pageC = await Page.findOne({ path: '/get_parent_C', grant: Page.GRANT_PUBLIC, isEmpty: false });
  844. const pageCD = await Page.findOne({ path: '/get_parent_C/get_parent_D', grant: Page.GRANT_PUBLIC, isEmpty: false });
  845. const emptyC = await Page.findOne({ path: '/get_parent_C', grant: Page.GRANT_PUBLIC, isEmpty: true });
  846. const emptyCD = await Page.findOne({ path: '/get_parent_C/get_parent_D', grant: Page.GRANT_PUBLIC, isEmpty: true });
  847. // -- Check existance
  848. expect(parent).not.toBeNull();
  849. expect(pageC).not.toBeNull();
  850. expect(pageCD).not.toBeNull();
  851. expect(emptyC).toBeNull();
  852. expect(emptyCD).toBeNull();
  853. // -- Check parent attribute
  854. expect(pageC.parent).toStrictEqual(rootPage._id);
  855. expect(pageCD.parent).toStrictEqual(pageC._id);
  856. // -- Check the found parent
  857. expect(parent.toObject()).toStrictEqual(pageCD.toObject());
  858. });
  859. });
  860. });