page.test.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398
  1. const mongoose = require('mongoose');
  2. const { getInstance } = require('../setup-crowi');
  3. let testUser0;
  4. let testUser1;
  5. let testUser2;
  6. let testGroup0;
  7. describe('Page', () => {
  8. // eslint-disable-next-line no-unused-vars
  9. let crowi;
  10. let Page;
  11. let PageQueryBuilder;
  12. let User;
  13. let UserGroup;
  14. let UserGroupRelation;
  15. beforeAll(async(done) => {
  16. crowi = await getInstance();
  17. User = mongoose.model('User');
  18. UserGroup = mongoose.model('UserGroup');
  19. UserGroupRelation = mongoose.model('UserGroupRelation');
  20. Page = mongoose.model('Page');
  21. PageQueryBuilder = Page.PageQueryBuilder;
  22. await User.insertMany([
  23. { name: 'Anon 0', username: 'anonymous0', email: 'anonymous0@example.com' },
  24. { name: 'Anon 1', username: 'anonymous1', email: 'anonymous1@example.com' },
  25. { name: 'Anon 2', username: 'anonymous2', email: 'anonymous2@example.com' },
  26. ]);
  27. await UserGroup.insertMany([
  28. { name: 'TestGroup0' },
  29. { name: 'TestGroup1' },
  30. ]);
  31. testUser0 = await User.findOne({ username: 'anonymous0' });
  32. testUser1 = await User.findOne({ username: 'anonymous1' });
  33. testUser2 = await User.findOne({ username: 'anonymous2' });
  34. testGroup0 = await UserGroup.findOne({ name: 'TestGroup0' });
  35. await UserGroupRelation.insertMany([
  36. {
  37. relatedGroup: testGroup0,
  38. relatedUser: testUser0,
  39. },
  40. {
  41. relatedGroup: testGroup0,
  42. relatedUser: testUser1,
  43. },
  44. ]);
  45. await Page.insertMany([
  46. {
  47. path: '/user/anonymous0/memo',
  48. grant: Page.GRANT_RESTRICTED,
  49. grantedUsers: [testUser0],
  50. creator: testUser0,
  51. },
  52. {
  53. path: '/grant/public',
  54. grant: Page.GRANT_PUBLIC,
  55. grantedUsers: [testUser0],
  56. creator: testUser0,
  57. },
  58. {
  59. path: '/grant/restricted',
  60. grant: Page.GRANT_RESTRICTED,
  61. grantedUsers: [testUser0],
  62. creator: testUser0,
  63. },
  64. {
  65. path: '/grant/specified',
  66. grant: Page.GRANT_SPECIFIED,
  67. grantedUsers: [testUser0],
  68. creator: testUser0,
  69. },
  70. {
  71. path: '/grant/owner',
  72. grant: Page.GRANT_OWNER,
  73. grantedUsers: [testUser0],
  74. creator: testUser0,
  75. },
  76. {
  77. path: '/page/for/extended',
  78. grant: Page.GRANT_PUBLIC,
  79. creator: testUser0,
  80. extended: { hoge: 1 },
  81. },
  82. {
  83. path: '/grant/groupacl',
  84. grant: Page.GRANT_USER_GROUP,
  85. grantedUsers: [],
  86. grantedGroup: testGroup0,
  87. creator: testUser1,
  88. },
  89. {
  90. path: '/page1',
  91. grant: Page.GRANT_PUBLIC,
  92. creator: testUser0,
  93. },
  94. {
  95. path: '/page1/child1',
  96. grant: Page.GRANT_PUBLIC,
  97. creator: testUser0,
  98. },
  99. {
  100. path: '/page2',
  101. grant: Page.GRANT_PUBLIC,
  102. creator: testUser0,
  103. },
  104. ]);
  105. done();
  106. });
  107. describe('.isPublic', () => {
  108. describe('with a public page', () => {
  109. test('should return true', async() => {
  110. const page = await Page.findOne({ path: '/grant/public' });
  111. expect(page.isPublic()).toEqual(true);
  112. });
  113. });
  114. ['restricted', 'specified', 'owner'].forEach((grant) => {
  115. describe(`with a ${grant} page`, () => {
  116. test('should return false', async() => {
  117. const page = await Page.findOne({ path: `/grant/${grant}` });
  118. expect(page.isPublic()).toEqual(false);
  119. });
  120. });
  121. });
  122. });
  123. describe('.getDeletedPageName', () => {
  124. test('should return trash page name', () => {
  125. expect(Page.getDeletedPageName('/hoge')).toEqual('/trash/hoge');
  126. expect(Page.getDeletedPageName('hoge')).toEqual('/trash/hoge');
  127. });
  128. });
  129. describe('.getRevertDeletedPageName', () => {
  130. test('should return reverted trash page name', () => {
  131. expect(Page.getRevertDeletedPageName('/hoge')).toEqual('/hoge');
  132. expect(Page.getRevertDeletedPageName('/trash/hoge')).toEqual('/hoge');
  133. expect(Page.getRevertDeletedPageName('/trash/hoge/trash')).toEqual('/hoge/trash');
  134. });
  135. });
  136. describe('.isDeletableName', () => {
  137. test('should decide deletable or not', () => {
  138. expect(Page.isDeletableName('/hoge')).toBeTruthy();
  139. expect(Page.isDeletableName('/user/xxx')).toBeFalsy();
  140. expect(Page.isDeletableName('/user/xxx123')).toBeFalsy();
  141. expect(Page.isDeletableName('/user/xxx/')).toBeTruthy();
  142. expect(Page.isDeletableName('/user/xxx/hoge')).toBeTruthy();
  143. });
  144. });
  145. describe('.isCreatableName', () => {
  146. test('should decide creatable or not', () => {
  147. expect(Page.isCreatableName('/hoge')).toBeTruthy();
  148. // edge cases
  149. expect(Page.isCreatableName('/me')).toBeFalsy();
  150. expect(Page.isCreatableName('/me/')).toBeFalsy();
  151. expect(Page.isCreatableName('/me/x')).toBeFalsy();
  152. expect(Page.isCreatableName('/meeting')).toBeTruthy();
  153. expect(Page.isCreatableName('/meeting/x')).toBeTruthy();
  154. // end with "edit"
  155. expect(Page.isCreatableName('/meeting/edit')).toBeFalsy();
  156. // under score
  157. expect(Page.isCreatableName('/_')).toBeTruthy();
  158. expect(Page.isCreatableName('/_template')).toBeTruthy();
  159. expect(Page.isCreatableName('/__template')).toBeTruthy();
  160. expect(Page.isCreatableName('/_r/x')).toBeFalsy();
  161. expect(Page.isCreatableName('/_api')).toBeFalsy();
  162. expect(Page.isCreatableName('/_apix')).toBeFalsy();
  163. expect(Page.isCreatableName('/_api/x')).toBeFalsy();
  164. expect(Page.isCreatableName('/hoge/xx.md')).toBeFalsy();
  165. // start with https?
  166. expect(Page.isCreatableName('/http://demo.growi.org/hoge')).toBeFalsy();
  167. expect(Page.isCreatableName('/https://demo.growi.org/hoge')).toBeFalsy();
  168. expect(Page.isCreatableName('http://demo.growi.org/hoge')).toBeFalsy();
  169. expect(Page.isCreatableName('https://demo.growi.org/hoge')).toBeFalsy();
  170. expect(Page.isCreatableName('/ the / path / with / space')).toBeFalsy();
  171. const forbidden = ['installer', 'register', 'login', 'logout',
  172. 'admin', 'files', 'trash', 'paste', 'comments'];
  173. for (let i = 0; i < forbidden.length; i++) {
  174. const pn = forbidden[i];
  175. expect(Page.isCreatableName(`/${pn}`)).toBeFalsy();
  176. expect(Page.isCreatableName(`/${pn}/`)).toBeFalsy();
  177. expect(Page.isCreatableName(`/${pn}/abc`)).toBeFalsy();
  178. }
  179. });
  180. });
  181. describe('.isAccessiblePageByViewer', () => {
  182. describe('with a granted page', () => {
  183. test('should return true with granted user', async() => {
  184. const user = await User.findOne({ email: 'anonymous0@example.com' });
  185. const page = await Page.findOne({ path: '/user/anonymous0/memo' });
  186. const bool = await Page.isAccessiblePageByViewer(page.id, user);
  187. expect(bool).toEqual(true);
  188. });
  189. test('should return false without user', async() => {
  190. const user = null;
  191. const page = await Page.findOne({ path: '/user/anonymous0/memo' });
  192. const bool = await Page.isAccessiblePageByViewer(page.id, user);
  193. expect(bool).toEqual(true);
  194. });
  195. });
  196. describe('with a public page', () => {
  197. test('should return true with user', async() => {
  198. const user = await User.findOne({ email: 'anonymous1@example.com' });
  199. const page = await Page.findOne({ path: '/grant/public' });
  200. const bool = await Page.isAccessiblePageByViewer(page.id, user);
  201. expect(bool).toEqual(true);
  202. });
  203. test('should return true with out', async() => {
  204. const user = null;
  205. const page = await Page.findOne({ path: '/grant/public' });
  206. const bool = await Page.isAccessiblePageByViewer(page.id, user);
  207. expect(bool).toEqual(true);
  208. });
  209. });
  210. describe('with a restricted page', () => {
  211. test('should return false with user who has no grant', async() => {
  212. const user = await User.findOne({ email: 'anonymous1@example.com' });
  213. const page = await Page.findOne({ path: '/grant/owner' });
  214. const bool = await Page.isAccessiblePageByViewer(page.id, user);
  215. expect(bool).toEqual(false);
  216. });
  217. test('should return false without user', async() => {
  218. const user = null;
  219. const page = await Page.findOne({ path: '/grant/owner' });
  220. const bool = await Page.isAccessiblePageByViewer(page.id, user);
  221. expect(bool).toEqual(false);
  222. });
  223. });
  224. });
  225. describe('Extended field', () => {
  226. describe('Slack Channel.', () => {
  227. test('should be empty', async() => {
  228. const page = await Page.findOne({ path: '/page/for/extended' });
  229. expect(page.extended.hoge).toEqual(1);
  230. expect(page.getSlackChannel()).toEqual('');
  231. });
  232. test('set slack channel and should get it and should keep hoge ', async() => {
  233. let page = await Page.findOne({ path: '/page/for/extended' });
  234. await page.updateSlackChannel('slack-channel1');
  235. page = await Page.findOne({ path: '/page/for/extended' });
  236. expect(page.extended.hoge).toEqual(1);
  237. expect(page.getSlackChannel()).toEqual('slack-channel1');
  238. });
  239. });
  240. });
  241. describe('.findPage', () => {
  242. describe('findByIdAndViewer', () => {
  243. test('should find page (public)', async() => {
  244. const expectedPage = await Page.findOne({ path: '/grant/public' });
  245. const page = await Page.findByIdAndViewer(expectedPage.id, testUser0);
  246. expect(page).not.toBeNull();
  247. expect(page.path).toEqual(expectedPage.path);
  248. });
  249. test('should find page (anyone knows link)', async() => {
  250. const expectedPage = await Page.findOne({ path: '/grant/restricted' });
  251. const page = await Page.findByIdAndViewer(expectedPage.id, testUser1);
  252. expect(page).not.toBeNull();
  253. expect(page.path).toEqual(expectedPage.path);
  254. });
  255. test('should find page (only me)', async() => {
  256. const expectedPage = await Page.findOne({ path: '/grant/owner' });
  257. const page = await Page.findByIdAndViewer(expectedPage.id, testUser0);
  258. expect(page).not.toBeNull();
  259. expect(page.path).toEqual(expectedPage.path);
  260. });
  261. test('should not be found by grant (only me)', async() => {
  262. const expectedPage = await Page.findOne({ path: '/grant/owner' });
  263. const page = await Page.findByIdAndViewer(expectedPage.id, testUser1);
  264. expect(page).toBeNull();
  265. });
  266. });
  267. describe('findByIdAndViewer granted userGroup', () => {
  268. test('should find page', async() => {
  269. const expectedPage = await Page.findOne({ path: '/grant/groupacl' });
  270. const page = await Page.findByIdAndViewer(expectedPage.id, testUser0);
  271. expect(page).not.toBeNull();
  272. expect(page.path).toEqual(expectedPage.path);
  273. });
  274. test('should not be found by grant', async() => {
  275. const expectedPage = await Page.findOne({ path: '/grant/groupacl' });
  276. const page = await Page.findByIdAndViewer(expectedPage.id, testUser2);
  277. expect(page).toBeNull();
  278. });
  279. });
  280. });
  281. describe('PageQueryBuilder.addConditionToListWithDescendants', () => {
  282. test('can retrieve descendants of /page', async() => {
  283. const builder = new PageQueryBuilder(Page.find());
  284. builder.addConditionToListWithDescendants('/page');
  285. const result = await builder.query.exec();
  286. // assert totalCount
  287. expect(result.length).toEqual(1);
  288. // assert paths
  289. const pagePaths = result.map((page) => { return page.path });
  290. expect(pagePaths).toContainEqual('/page/for/extended');
  291. });
  292. test('can retrieve descendants of /page1', async() => {
  293. const builder = new PageQueryBuilder(Page.find());
  294. builder.addConditionToListWithDescendants('/page1/');
  295. const result = await builder.query.exec();
  296. // assert totalCount
  297. expect(result.length).toEqual(2);
  298. // assert paths
  299. const pagePaths = result.map((page) => { return page.path });
  300. expect(pagePaths).toContainEqual('/page1');
  301. expect(pagePaths).toContainEqual('/page1/child1');
  302. });
  303. });
  304. describe('PageQueryBuilder.addConditionToListOnlyDescendants', () => {
  305. test('can retrieve only descendants of /page', async() => {
  306. const builder = new PageQueryBuilder(Page.find());
  307. builder.addConditionToListOnlyDescendants('/page');
  308. const result = await builder.query.exec();
  309. // assert totalCount
  310. expect(result.length).toEqual(1);
  311. // assert paths
  312. const pagePaths = result.map((page) => { return page.path });
  313. expect(pagePaths).toContainEqual('/page/for/extended');
  314. });
  315. test('can retrieve only descendants of /page1', async() => {
  316. const builder = new PageQueryBuilder(Page.find());
  317. builder.addConditionToListOnlyDescendants('/page1');
  318. const result = await builder.query.exec();
  319. // assert totalCount
  320. expect(result.length).toEqual(1);
  321. // assert paths
  322. const pagePaths = result.map((page) => { return page.path });
  323. expect(pagePaths).toContainEqual('/page1/child1');
  324. });
  325. });
  326. describe('PageQueryBuilder.addConditionToListByStartWith', () => {
  327. test('can retrieve pages which starts with /page', async() => {
  328. const builder = new PageQueryBuilder(Page.find());
  329. builder.addConditionToListByStartWith('/page');
  330. const result = await builder.query.exec();
  331. // assert totalCount
  332. expect(result.length).toEqual(4);
  333. // assert paths
  334. const pagePaths = result.map((page) => { return page.path });
  335. expect(pagePaths).toContainEqual('/page/for/extended');
  336. expect(pagePaths).toContainEqual('/page1');
  337. expect(pagePaths).toContainEqual('/page1/child1');
  338. expect(pagePaths).toContainEqual('/page2');
  339. });
  340. });
  341. });