v5.migration.test.js 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493
  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, parent: { $ne: null } });
  258. const page2 = await Page.findOne({
  259. path: '/normalize_10/normalize_11_gA', _id: pageId8, isEmpty: true, parent: { $ne: null },
  260. });
  261. const page3 = await Page.findOne({ path: '/normalize_10/normalize_11_gA', _id: pageId9, parent: null }); // not v5
  262. const page4 = await Page.findOne({ path: '/normalize_10/normalize_11_gA/normalize_11_gB', parent: { $ne: null } });
  263. const page5 = await Page.findOne({ path: '/normalize_10/normalize_12_gC', parent: { $ne: null } });
  264. expectAllToBeTruthy([page1, page2, page3, page4, page5]);
  265. await normalizeParentRecursivelyByPages([page3], testUser1);
  266. // AM => After Migration
  267. const page1AM = await Page.findOne({ path: '/normalize_10' });
  268. const page2AM = await Page.findOne({ path: '/normalize_10/normalize_11_gA', _id: pageId8 });
  269. const page3AM = await Page.findOne({ path: '/normalize_10/normalize_11_gA', _id: pageId9 });
  270. const page4AM = await Page.findOne({ path: '/normalize_10/normalize_11_gA/normalize_11_gB' });
  271. const page5AM = await Page.findOne({ path: '/normalize_10/normalize_12_gC' });
  272. expectAllToBeTruthy([page1AM, page3AM, page4AM, page5AM]);
  273. expect(page2AM).toBeNull();
  274. expect(page1AM.isEmpty).toBeTruthy();
  275. expect(page3AM.parent).toStrictEqual(page1AM._id);
  276. expect(page4AM.parent).toStrictEqual(page3AM._id);
  277. expect(page5AM.parent).toStrictEqual(page1AM._id);
  278. expect(page3AM.isEmpty).toBe(false);
  279. });
  280. });
  281. describe('normalizeAllPublicPages()', () => {
  282. jest.setTimeout(60000);
  283. let createPagePaths;
  284. let allPossiblePagePaths;
  285. beforeAll(async() => {
  286. createPagePaths = [
  287. '/publicA', '/publicA/privateB', '/publicA/privateB/publicC', '/parenthesis/(a)[b]{c}d', '/parenthesis/(a)[b]{c}d/public', '/migratedD',
  288. ];
  289. allPossiblePagePaths = [...createPagePaths, '/parenthesis', '/'];
  290. // initialize pages for test
  291. await Page.insertMany([
  292. {
  293. path: '/publicA',
  294. grant: Page.GRANT_PUBLIC,
  295. creator: testUser1,
  296. lastUpdateUser: testUser1,
  297. },
  298. {
  299. path: '/publicA/privateB',
  300. grant: Page.GRANT_OWNER,
  301. creator: testUser1,
  302. lastUpdateUser: testUser1,
  303. grantedUsers: [testUser1._id],
  304. },
  305. {
  306. path: '/publicA/privateB/publicC',
  307. grant: Page.GRANT_PUBLIC,
  308. creator: testUser1,
  309. lastUpdateUser: testUser1,
  310. },
  311. {
  312. path: '/parenthesis/(a)[b]{c}d',
  313. grant: Page.GRANT_PUBLIC,
  314. creator: testUser1,
  315. lastUpdateUser: testUser1,
  316. },
  317. {
  318. path: '/parenthesis/(a)[b]{c}d/public',
  319. grant: Page.GRANT_PUBLIC,
  320. creator: testUser1,
  321. lastUpdateUser: testUser1,
  322. },
  323. ]);
  324. const parent = await Page.find({ path: '/' });
  325. await Page.insertMany([
  326. {
  327. path: '/migratedD',
  328. grant: Page.GRANT_PUBLIC,
  329. creator: testUser1,
  330. lastUpdateUser: testUser1,
  331. parent: parent._id,
  332. },
  333. ]);
  334. // migrate
  335. await crowi.pageService.normalizeAllPublicPages(Page.GRANT_PUBLIC);
  336. jest.setTimeout(30000);
  337. });
  338. test('should migrate all public pages', async() => {
  339. const migratedPages = await Page.find({
  340. path: {
  341. $in: allPossiblePagePaths,
  342. },
  343. parent: { $ne: null },
  344. });
  345. const migratedEmptyPages = await Page.find({
  346. path: {
  347. $in: allPossiblePagePaths,
  348. },
  349. isEmpty: true,
  350. parent: { $ne: null },
  351. });
  352. const nonMigratedPages = await Page.find({
  353. path: {
  354. $in: allPossiblePagePaths,
  355. },
  356. parent: null,
  357. });
  358. const migratedPaths = migratedPages.map(page => page.path).sort();
  359. const migratedEmptyPaths = migratedEmptyPages.map(page => page.path).sort();
  360. const nonMigratedPaths = nonMigratedPages.map(page => page.path).sort();
  361. const expectedMigratedPaths = allPossiblePagePaths.filter(path => path !== '/').sort();
  362. const expectedMigratedEmptyPaths = ['/publicA/privateB', '/parenthesis'].sort();
  363. const expectedNonMigratedPaths = ['/publicA/privateB', '/'].sort();
  364. expect(migratedPaths).toStrictEqual(expectedMigratedPaths);
  365. expect(migratedEmptyPaths).toStrictEqual(expectedMigratedEmptyPaths);
  366. expect(nonMigratedPaths).toStrictEqual(expectedNonMigratedPaths);
  367. });
  368. });
  369. describe('normalizeParentByPageId()', () => {
  370. const normalizeParentByPageId = async(page, user) => {
  371. return crowi.pageService.normalizeParentByPageId(page, user);
  372. };
  373. test('it should normalize not v5 page with usergroup that has parent group', async() => {
  374. const page1 = await Page.findOne({ _id: pageId1, path: '/normalize_1', isEmpty: true });
  375. const page2 = await Page.findOne({ _id: pageId2, path: '/normalize_1/normalize_2', parent: page1._id });
  376. const page3 = await Page.findOne({ _id: pageId3, path: '/normalize_1' }); // NOT v5
  377. expectAllToBeTruthy([page1, page2, page3]);
  378. await normalizeParentByPageId(page3, testUser1);
  379. // AM => After Migration
  380. const page1AM = await Page.findOne({ _id: pageId1, path: '/normalize_1', isEmpty: true });
  381. const page2AM = await Page.findOne({ _id: pageId2, path: '/normalize_1/normalize_2' });
  382. const page3AM = await Page.findOne({ _id: pageId3, path: '/normalize_1' }); // v5 compatible
  383. expectAllToBeTruthy([page2AM, page3AM]);
  384. expect(page1AM).toBeNull();
  385. expect(page2AM.parent).toStrictEqual(page3AM._id);
  386. expect(page3AM.parent).toStrictEqual(rootPage._id);
  387. });
  388. test('should throw error if a page with isolated group becomes the parent of other page with different gourp after normalizing', async() => {
  389. const page4 = await Page.findOne({ _id: pageId4, path: '/normalize_4', isEmpty: true });
  390. const page5 = await Page.findOne({ _id: pageId5, path: '/normalize_4/normalize_5', parent: page4._id });
  391. const page6 = await Page.findOne({ _id: pageId6, path: '/normalize_4' }); // NOT v5
  392. expectAllToBeTruthy([page4, page5, page6]);
  393. let isThrown;
  394. try {
  395. await normalizeParentByPageId(page6, testUser1);
  396. }
  397. catch (err) {
  398. isThrown = true;
  399. }
  400. // AM => After Migration
  401. const page4AM = await Page.findOne({ _id: pageId4, path: '/normalize_4', isEmpty: true });
  402. const page5AM = await Page.findOne({ _id: pageId5, path: '/normalize_4/normalize_5', parent: page4._id });
  403. const page6AM = await Page.findOne({ _id: pageId6, path: '/normalize_4' }); // NOT v5
  404. expectAllToBeTruthy([page4AM, page5AM, page6AM]);
  405. expect(isThrown).toBe(true);
  406. expect(page4AM).toStrictEqual(page4);
  407. expect(page5AM).toStrictEqual(page5);
  408. expect(page6AM).toStrictEqual(page6);
  409. });
  410. });
  411. test('replace private parents with empty pages', async() => {
  412. const replacedPathPages = await Page.find({ path: '/publicA/privateB' }); // ex-private page
  413. const _newEmptyPage = replacedPathPages.filter(page => page.parent != null)[0];
  414. const newEmptyPage = {
  415. path: _newEmptyPage.path,
  416. grant: _newEmptyPage.grant,
  417. isEmpty: _newEmptyPage.isEmpty,
  418. };
  419. const expectedNewEmptyPage = {
  420. path: '/publicA/privateB',
  421. grant: Page.GRANT_PUBLIC,
  422. isEmpty: true,
  423. };
  424. const _privatePage = replacedPathPages.filter(page => page.parent == null)[0];
  425. const privatePage = {
  426. path: _privatePage.path,
  427. grant: _privatePage.grant,
  428. isEmpty: _privatePage.isEmpty,
  429. };
  430. const expectedPrivatePage = {
  431. path: '/publicA/privateB',
  432. grant: Page.GRANT_OWNER,
  433. isEmpty: false,
  434. };
  435. expect(replacedPathPages.length).toBe(2);
  436. expect(newEmptyPage).toStrictEqual(expectedNewEmptyPage);
  437. expect(privatePage).toStrictEqual(expectedPrivatePage);
  438. });
  439. });