page.test.js 46 KB

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