v5.migration.test.js 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492
  1. const mongoose = require('mongoose');
  2. const { getInstance } = require('../setup-crowi');
  3. describe('V5 page migration', () => {
  4. let crowi;
  5. let Page;
  6. let User;
  7. let UserGroup;
  8. let UserGroupRelation;
  9. let testUser1;
  10. let rootPage;
  11. const groupIdIsolate = new mongoose.Types.ObjectId();
  12. const groupIdA = new mongoose.Types.ObjectId();
  13. const groupIdB = new mongoose.Types.ObjectId();
  14. const groupIdC = new mongoose.Types.ObjectId();
  15. const pageId1 = new mongoose.Types.ObjectId();
  16. const pageId2 = new mongoose.Types.ObjectId();
  17. const pageId3 = new mongoose.Types.ObjectId();
  18. const pageId4 = new mongoose.Types.ObjectId();
  19. const pageId5 = new mongoose.Types.ObjectId();
  20. const pageId6 = new mongoose.Types.ObjectId();
  21. const pageId7 = new mongoose.Types.ObjectId();
  22. const pageId8 = new mongoose.Types.ObjectId();
  23. const pageId9 = new mongoose.Types.ObjectId();
  24. const pageId10 = new mongoose.Types.ObjectId();
  25. const pageId11 = new mongoose.Types.ObjectId();
  26. beforeAll(async() => {
  27. jest.restoreAllMocks();
  28. crowi = await getInstance();
  29. Page = mongoose.model('Page');
  30. User = mongoose.model('User');
  31. UserGroup = mongoose.model('UserGroup');
  32. UserGroupRelation = mongoose.model('UserGroupRelation');
  33. await crowi.configManager.updateConfigsInTheSameNamespace('crowi', { 'app:isV5Compatible': true });
  34. await User.insertMany([{ name: 'testUser1', username: 'testUser1', email: 'testUser1@example.com' }]);
  35. testUser1 = await User.findOne({ username: 'testUser1' });
  36. rootPage = await Page.findOne({ path: '/' });
  37. await UserGroup.insertMany([
  38. {
  39. _id: groupIdIsolate,
  40. name: 'groupIsolate',
  41. },
  42. {
  43. _id: groupIdA,
  44. name: 'groupA',
  45. },
  46. {
  47. _id: groupIdB,
  48. name: 'groupB',
  49. parent: groupIdA,
  50. },
  51. {
  52. _id: groupIdC,
  53. name: 'groupC',
  54. parent: groupIdB,
  55. },
  56. ]);
  57. await UserGroupRelation.insertMany([
  58. {
  59. relatedGroup: groupIdIsolate,
  60. relatedUser: testUser1._id,
  61. },
  62. {
  63. relatedGroup: groupIdA,
  64. relatedUser: testUser1._id,
  65. },
  66. {
  67. relatedGroup: groupIdB,
  68. relatedUser: testUser1._id,
  69. },
  70. {
  71. relatedGroup: groupIdC,
  72. relatedUser: testUser1._id,
  73. },
  74. ]);
  75. await Page.insertMany([
  76. {
  77. path: '/private1',
  78. grant: Page.GRANT_OWNER,
  79. creator: testUser1,
  80. lastUpdateUser: testUser1,
  81. grantedUsers: [testUser1._id],
  82. },
  83. {
  84. path: '/dummyParent/private1',
  85. grant: Page.GRANT_OWNER,
  86. creator: testUser1,
  87. lastUpdateUser: testUser1,
  88. grantedUsers: [testUser1._id],
  89. },
  90. {
  91. path: '/dummyParent/private1/private2',
  92. grant: Page.GRANT_OWNER,
  93. creator: testUser1,
  94. lastUpdateUser: testUser1,
  95. grantedUsers: [testUser1._id],
  96. },
  97. {
  98. path: '/dummyParent/private1/private3',
  99. grant: Page.GRANT_OWNER,
  100. creator: testUser1,
  101. lastUpdateUser: testUser1,
  102. grantedUsers: [testUser1._id],
  103. },
  104. {
  105. _id: pageId1,
  106. path: '/normalize_1',
  107. parent: rootPage._id,
  108. grant: Page.GRANT_PUBLIC,
  109. isEmpty: true,
  110. },
  111. {
  112. _id: pageId2,
  113. path: '/normalize_1/normalize_2',
  114. parent: pageId1,
  115. grant: Page.GRANT_USER_GROUP,
  116. grantedGroup: groupIdB,
  117. grantedUsers: [testUser1._id],
  118. },
  119. {
  120. _id: pageId3,
  121. path: '/normalize_1',
  122. grant: Page.GRANT_USER_GROUP,
  123. grantedGroup: groupIdA,
  124. grantedUsers: [testUser1._id],
  125. },
  126. {
  127. _id: pageId4,
  128. path: '/normalize_4',
  129. parent: rootPage._id,
  130. grant: Page.GRANT_PUBLIC,
  131. isEmpty: true,
  132. },
  133. {
  134. _id: pageId5,
  135. path: '/normalize_4/normalize_5',
  136. parent: pageId4,
  137. grant: Page.GRANT_USER_GROUP,
  138. grantedGroup: groupIdA,
  139. grantedUsers: [testUser1._id],
  140. },
  141. {
  142. _id: pageId6,
  143. path: '/normalize_4',
  144. grant: Page.GRANT_USER_GROUP,
  145. grantedGroup: groupIdIsolate,
  146. grantedUsers: [testUser1._id],
  147. },
  148. {
  149. path: '/normalize_7/normalize_8_gA',
  150. grant: Page.GRANT_USER_GROUP,
  151. creator: testUser1,
  152. grantedGroup: groupIdA,
  153. grantedUsers: [testUser1._id],
  154. },
  155. {
  156. path: '/normalize_7/normalize_8_gA/normalize_9_gB',
  157. grant: Page.GRANT_USER_GROUP,
  158. creator: testUser1,
  159. grantedGroup: groupIdB,
  160. grantedUsers: [testUser1._id],
  161. },
  162. {
  163. path: '/normalize_7/normalize_8_gC',
  164. grant: Page.GRANT_USER_GROUP,
  165. grantedGroup: groupIdC,
  166. grantedUsers: [testUser1._id],
  167. },
  168. {
  169. _id: pageId7,
  170. path: '/normalize_10',
  171. grant: Page.GRANT_PUBLIC,
  172. isEmpty: true,
  173. parent: rootPage._id,
  174. descendantCount: 3,
  175. },
  176. {
  177. _id: pageId8,
  178. path: '/normalize_10/normalize_11_gA',
  179. isEmpty: true,
  180. parent: pageId7,
  181. descendantCount: 1,
  182. },
  183. {
  184. _id: pageId9, // not v5
  185. path: '/normalize_10/normalize_11_gA',
  186. grant: Page.GRANT_USER_GROUP,
  187. grantedGroup: groupIdA,
  188. grantedUsers: [testUser1._id],
  189. },
  190. {
  191. _id: pageId10,
  192. path: '/normalize_10/normalize_11_gA/normalize_11_gB',
  193. grant: Page.GRANT_USER_GROUP,
  194. grantedGroup: groupIdB,
  195. grantedUsers: [testUser1._id],
  196. parent: pageId8,
  197. descendantCount: 0,
  198. },
  199. {
  200. _id: pageId11,
  201. path: '/normalize_10/normalize_12_gC',
  202. grant: Page.GRANT_USER_GROUP,
  203. grantedGroup: groupIdC,
  204. grantedUsers: [testUser1._id],
  205. parent: pageId7,
  206. descendantCount: 0,
  207. },
  208. ]);
  209. });
  210. // https://github.com/jest-community/eslint-plugin-jest/blob/v24.3.5/docs/rules/expect-expect.md#assertfunctionnames
  211. // pass unless the data is one of [false, 0, '', null, undefined, NaN]
  212. const expectAllToBeTruthy = (dataList) => {
  213. dataList.forEach((data, i) => {
  214. if (data == null) { console.log(`index: ${i}`) }
  215. expect(data).toBeTruthy();
  216. });
  217. };
  218. describe('normalizeParentRecursivelyByPages()', () => {
  219. const normalizeParentRecursivelyByPages = async(pages, user) => {
  220. return crowi.pageService.normalizeParentRecursivelyByPages(pages, user);
  221. };
  222. test('should migrate all pages specified by pageIds', async() => {
  223. jest.restoreAllMocks();
  224. const pagesToRun = await Page.find({ path: { $in: ['/private1', '/dummyParent/private1'] } });
  225. // migrate
  226. await normalizeParentRecursivelyByPages(pagesToRun, testUser1);
  227. const migratedPages = await Page.find({
  228. path: {
  229. $in: ['/private1', '/dummyParent', '/dummyParent/private1', '/dummyParent/private1/private2', '/dummyParent/private1/private3'],
  230. },
  231. });
  232. const migratedPagePaths = migratedPages.filter(doc => doc.parent != null).map(doc => doc.path);
  233. const expected = ['/private1', '/dummyParent', '/dummyParent/private1', '/dummyParent/private1/private2', '/dummyParent/private1/private3'];
  234. expect(migratedPagePaths.sort()).toStrictEqual(expected.sort());
  235. });
  236. test('should change all v4 pages with usergroup to v5 compatible and create new parent page', async() => {
  237. const page8 = await Page.findOne({ path: '/normalize_7/normalize_8_gA' });
  238. const page9 = await Page.findOne({ path: '/normalize_7/normalize_8_gA/normalize_9_gB' });
  239. const page10 = await Page.findOne({ path: '/normalize_7/normalize_8_gC' });
  240. const page11 = await Page.findOne({ path: '/normalize_7' });
  241. expectAllToBeTruthy([page8, page9, page10]);
  242. expect(page11).toBeNull();
  243. await normalizeParentRecursivelyByPages([page8, page9, page10], testUser1);
  244. // AM => After Migration
  245. const page7 = await Page.findOne({ path: '/normalize_7' });
  246. const page8AM = await Page.findOne({ path: '/normalize_7/normalize_8_gA' });
  247. const page9AM = await Page.findOne({ path: '/normalize_7/normalize_8_gA/normalize_9_gB' });
  248. const page10AM = await Page.findOne({ path: '/normalize_7/normalize_8_gC' });
  249. expectAllToBeTruthy([page7, page8AM, page9AM, page10AM]);
  250. expect(page7.isEmpty).toBe(true);
  251. expect(page7.parent).toStrictEqual(rootPage._id);
  252. expect(page8AM.parent).toStrictEqual(page7._id);
  253. expect(page9AM.parent).toStrictEqual(page8AM._id);
  254. expect(page10AM.parent).toStrictEqual(page7._id);
  255. });
  256. test("should replace empty page with same path with new non-empty page and update all related children's parent", async() => {
  257. const page1 = await Page.findOne({ path: '/normalize_10', isEmpty: true });
  258. const page2 = await Page.findOne({ path: '/normalize_10/normalize_11_gA', _id: pageId8, isEmpty: true });
  259. const page3 = await Page.findOne({ path: '/normalize_10/normalize_11_gA', _id: pageId9 }); // not v5
  260. const page4 = await Page.findOne({ path: '/normalize_10/normalize_11_gA/normalize_11_gB' });
  261. const page5 = await Page.findOne({ path: '/normalize_10/normalize_12_gC' });
  262. expectAllToBeTruthy([page1, page2, page3, page4, page5]);
  263. await normalizeParentRecursivelyByPages([page3], testUser1);
  264. // AM => After Migration
  265. const page1AM = await Page.findOne({ path: '/normalize_10' });
  266. const page2AM = await Page.findOne({ path: '/normalize_10/normalize_11_gA', _id: pageId8 });
  267. const page3AM = await Page.findOne({ path: '/normalize_10/normalize_11_gA', _id: pageId9 });
  268. const page4AM = await Page.findOne({ path: '/normalize_10/normalize_11_gA/normalize_11_gB' });
  269. const page5AM = await Page.findOne({ path: '/normalize_10/normalize_12_gC' });
  270. expectAllToBeTruthy([page1AM, page3AM, page4AM, page5AM]);
  271. expect(page2AM).toBeNull();
  272. expect(page1AM.isEmpty).toBeTruthy();
  273. expect(page3AM.parent).toStrictEqual(page1AM._id);
  274. // Todo: enable the code below after this is solved: https://redmine.weseek.co.jp/issues/90060
  275. // expect(page4AM.parent).toStrictEqual(page3AF._id);
  276. expect(page5AM.parent).toStrictEqual(page1AM._id);
  277. expect(page3AM.isEmpty).toBeFalsy();
  278. });
  279. });
  280. describe('normalizeAllPublicPages()', () => {
  281. jest.setTimeout(60000);
  282. let createPagePaths;
  283. let allPossiblePagePaths;
  284. beforeAll(async() => {
  285. createPagePaths = [
  286. '/publicA', '/publicA/privateB', '/publicA/privateB/publicC', '/parenthesis/(a)[b]{c}d', '/parenthesis/(a)[b]{c}d/public', '/migratedD',
  287. ];
  288. allPossiblePagePaths = [...createPagePaths, '/parenthesis', '/'];
  289. // initialize pages for test
  290. await Page.insertMany([
  291. {
  292. path: '/publicA',
  293. grant: Page.GRANT_PUBLIC,
  294. creator: testUser1,
  295. lastUpdateUser: testUser1,
  296. },
  297. {
  298. path: '/publicA/privateB',
  299. grant: Page.GRANT_OWNER,
  300. creator: testUser1,
  301. lastUpdateUser: testUser1,
  302. grantedUsers: [testUser1._id],
  303. },
  304. {
  305. path: '/publicA/privateB/publicC',
  306. grant: Page.GRANT_PUBLIC,
  307. creator: testUser1,
  308. lastUpdateUser: testUser1,
  309. },
  310. {
  311. path: '/parenthesis/(a)[b]{c}d',
  312. grant: Page.GRANT_PUBLIC,
  313. creator: testUser1,
  314. lastUpdateUser: testUser1,
  315. },
  316. {
  317. path: '/parenthesis/(a)[b]{c}d/public',
  318. grant: Page.GRANT_PUBLIC,
  319. creator: testUser1,
  320. lastUpdateUser: testUser1,
  321. },
  322. ]);
  323. const parent = await Page.find({ path: '/' });
  324. await Page.insertMany([
  325. {
  326. path: '/migratedD',
  327. grant: Page.GRANT_PUBLIC,
  328. creator: testUser1,
  329. lastUpdateUser: testUser1,
  330. parent: parent._id,
  331. },
  332. ]);
  333. // migrate
  334. await crowi.pageService.normalizeAllPublicPages(Page.GRANT_PUBLIC);
  335. jest.setTimeout(30000);
  336. });
  337. test('should migrate all public pages', async() => {
  338. const migratedPages = await Page.find({
  339. path: {
  340. $in: allPossiblePagePaths,
  341. },
  342. parent: { $ne: null },
  343. });
  344. const migratedEmptyPages = await Page.find({
  345. path: {
  346. $in: allPossiblePagePaths,
  347. },
  348. isEmpty: true,
  349. parent: { $ne: null },
  350. });
  351. const nonMigratedPages = await Page.find({
  352. path: {
  353. $in: allPossiblePagePaths,
  354. },
  355. parent: null,
  356. });
  357. const migratedPaths = migratedPages.map(page => page.path).sort();
  358. const migratedEmptyPaths = migratedEmptyPages.map(page => page.path).sort();
  359. const nonMigratedPaths = nonMigratedPages.map(page => page.path).sort();
  360. const expectedMigratedPaths = allPossiblePagePaths.filter(path => path !== '/').sort();
  361. const expectedMigratedEmptyPaths = ['/publicA/privateB', '/parenthesis'].sort();
  362. const expectedNonMigratedPaths = ['/publicA/privateB', '/'].sort();
  363. expect(migratedPaths).toStrictEqual(expectedMigratedPaths);
  364. expect(migratedEmptyPaths).toStrictEqual(expectedMigratedEmptyPaths);
  365. expect(nonMigratedPaths).toStrictEqual(expectedNonMigratedPaths);
  366. });
  367. });
  368. describe('normalizeParentByPageId()', () => {
  369. const normalizeParentByPageId = async(page, user) => {
  370. return crowi.pageService.normalizeParentByPageId(page, user);
  371. };
  372. test('it should normalize not v5 page with usergroup that has parent group', async() => {
  373. const page1 = await Page.findOne({ _id: pageId1, path: '/normalize_1', isEmpty: true });
  374. const page2 = await Page.findOne({ _id: pageId2, path: '/normalize_1/normalize_2', parent: page1._id });
  375. const page3 = await Page.findOne({ _id: pageId3, path: '/normalize_1' }); // NOT v5
  376. expectAllToBeTruthy([page1, page2, page3]);
  377. await normalizeParentByPageId(page3, testUser1);
  378. // AM => After Migration
  379. const page1AM = await Page.findOne({ _id: pageId1, path: '/normalize_1', isEmpty: true });
  380. const page2AM = await Page.findOne({ _id: pageId2, path: '/normalize_1/normalize_2' });
  381. const page3AM = await Page.findOne({ _id: pageId3, path: '/normalize_1' }); // v5 compatible
  382. expectAllToBeTruthy([page2AM, page3AM]);
  383. expect(page1AM).toBeNull();
  384. expect(page2AM.parent).toStrictEqual(page3AM._id);
  385. expect(page3AM.parent).toStrictEqual(rootPage._id);
  386. });
  387. test('should throw error if a page with isolated group becomes the parent of other page with different gourp after normalizing', async() => {
  388. const page4 = await Page.findOne({ _id: pageId4, path: '/normalize_4', isEmpty: true });
  389. const page5 = await Page.findOne({ _id: pageId5, path: '/normalize_4/normalize_5', parent: page4._id });
  390. const page6 = await Page.findOne({ _id: pageId6, path: '/normalize_4' }); // NOT v5
  391. expectAllToBeTruthy([page4, page5, page6]);
  392. let isThrown;
  393. try {
  394. await normalizeParentByPageId(page6, testUser1);
  395. }
  396. catch (err) {
  397. isThrown = true;
  398. }
  399. // AM => After Migration
  400. const page4AM = await Page.findOne({ _id: pageId4, path: '/normalize_4', isEmpty: true });
  401. const page5AM = await Page.findOne({ _id: pageId5, path: '/normalize_4/normalize_5', parent: page4._id });
  402. const page6AM = await Page.findOne({ _id: pageId6, path: '/normalize_4' }); // NOT v5
  403. expectAllToBeTruthy([page4AM, page5AM, page6AM]);
  404. expect(isThrown).toBe(true);
  405. expect(page4AM).toStrictEqual(page4);
  406. expect(page5AM).toStrictEqual(page5);
  407. expect(page6AM).toStrictEqual(page6);
  408. });
  409. });
  410. test('replace private parents with empty pages', async() => {
  411. const replacedPathPages = await Page.find({ path: '/publicA/privateB' }); // ex-private page
  412. const _newEmptyPage = replacedPathPages.filter(page => page.parent != null)[0];
  413. const newEmptyPage = {
  414. path: _newEmptyPage.path,
  415. grant: _newEmptyPage.grant,
  416. isEmpty: _newEmptyPage.isEmpty,
  417. };
  418. const expectedNewEmptyPage = {
  419. path: '/publicA/privateB',
  420. grant: Page.GRANT_PUBLIC,
  421. isEmpty: true,
  422. };
  423. const _privatePage = replacedPathPages.filter(page => page.parent == null)[0];
  424. const privatePage = {
  425. path: _privatePage.path,
  426. grant: _privatePage.grant,
  427. isEmpty: _privatePage.isEmpty,
  428. };
  429. const expectedPrivatePage = {
  430. path: '/publicA/privateB',
  431. grant: Page.GRANT_OWNER,
  432. isEmpty: false,
  433. };
  434. expect(replacedPathPages.length).toBe(2);
  435. expect(newEmptyPage).toStrictEqual(expectedNewEmptyPage);
  436. expect(privatePage).toStrictEqual(expectedPrivatePage);
  437. });
  438. });