page.js 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218
  1. module.exports = function(crowi, app) {
  2. 'use strict';
  3. var debug = require('debug')('crowi:routes:page')
  4. , Page = crowi.model('Page')
  5. , User = crowi.model('User')
  6. , Config = crowi.model('Config')
  7. , config = crowi.getConfig()
  8. , Revision = crowi.model('Revision')
  9. , Bookmark = crowi.model('Bookmark')
  10. , UserGroupRelation = crowi.model('UserGroupRelation')
  11. , ApiResponse = require('../util/apiResponse')
  12. , interceptorManager = crowi.getInterceptorManager()
  13. , actions = {};
  14. // register page events
  15. var pageEvent = crowi.event('page');
  16. pageEvent.on('update', function(page, user) {
  17. crowi.getIo().sockets.emit('page edited', {page, user});
  18. });
  19. function getPathFromRequest(req) {
  20. var path = '/' + (req.params[0] || '');
  21. return path.replace(/\.md$/, '');
  22. }
  23. function isUserPage(path) {
  24. if (path.match(/^\/user\/[^\/]+\/?$/)) {
  25. return true;
  26. }
  27. return false;
  28. }
  29. // TODO: total とかでちゃんと計算する
  30. function generatePager(options) {
  31. var next = null,
  32. prev = null,
  33. offset = parseInt(options.offset, 10),
  34. limit = parseInt(options.limit, 10),
  35. length = options.length || 0;
  36. if (offset > 0) {
  37. prev = offset - limit;
  38. if (prev < 0) {
  39. prev = 0;
  40. }
  41. }
  42. if (length < limit) {
  43. next = null;
  44. } else {
  45. next = offset + limit;
  46. }
  47. return {
  48. prev: prev,
  49. next: next,
  50. offset: offset,
  51. };
  52. }
  53. /**
  54. * switch action by behaviorType
  55. */
  56. actions.pageListShowWrapper = function(req, res) {
  57. const behaviorType = Config.behaviorType(config);
  58. if (!behaviorType || 'crowi' === behaviorType) {
  59. return actions.pageListShow(req, res);
  60. }
  61. else {
  62. return actions.pageListShowForCrowiPlus(req, res);
  63. }
  64. }
  65. /**
  66. * switch action by behaviorType
  67. */
  68. actions.pageShowWrapper = function(req, res) {
  69. const behaviorType = Config.behaviorType(config);
  70. if (!behaviorType || 'crowi' === behaviorType) {
  71. return actions.pageShow(req, res);
  72. }
  73. else {
  74. return actions.pageShowForCrowiPlus(req, res);
  75. }
  76. }
  77. /**
  78. * switch action by behaviorType
  79. */
  80. actions.trashPageListShowWrapper = function(req, res) {
  81. const behaviorType = Config.behaviorType(config);
  82. if (!behaviorType || 'crowi' === behaviorType) {
  83. // Crowi behavior for '/trash/*'
  84. return actions.deletedPageListShow(req, res);
  85. }
  86. else {
  87. // redirect to '/trash'
  88. return res.redirect('/trash');
  89. }
  90. }
  91. /**
  92. * switch action by behaviorType
  93. */
  94. actions.trashPageShowWrapper = function(req, res) {
  95. const behaviorType = Config.behaviorType(config);
  96. if (!behaviorType || 'crowi' === behaviorType) {
  97. // redirect to '/trash/'
  98. return res.redirect('/trash/');
  99. }
  100. else {
  101. // Crowi behavior for '/trash/*'
  102. return actions.deletedPageListShow(req, res);
  103. }
  104. }
  105. /**
  106. * switch action by behaviorType
  107. */
  108. actions.deletedPageListShowWrapper = function(req, res) {
  109. const behaviorType = Config.behaviorType(config);
  110. if (!behaviorType || 'crowi' === behaviorType) {
  111. // Crowi behavior for '/trash/*'
  112. return actions.deletedPageListShow(req, res);
  113. }
  114. else {
  115. const path = '/trash' + getPathFromRequest(req);
  116. return res.redirect(path);
  117. }
  118. }
  119. actions.pageListShow = function(req, res) {
  120. var path = getPathFromRequest(req);
  121. var limit = 50;
  122. var offset = parseInt(req.query.offset) || 0;
  123. var SEENER_THRESHOLD = 10;
  124. // add slash if root
  125. path = path + (path == '/' ? '' : '/');
  126. debug('Page list show', path);
  127. // index page
  128. var pagerOptions = {
  129. offset: offset,
  130. limit : limit
  131. };
  132. var queryOptions = {
  133. offset: offset,
  134. limit : limit + 1,
  135. isPopulateRevisionBody: Config.isEnabledTimeline(config),
  136. };
  137. var renderVars = {
  138. page: null,
  139. path: path,
  140. isPortal: false,
  141. pages: [],
  142. tree: [],
  143. };
  144. Page.hasPortalPage(path, req.user, req.query.revision)
  145. .then(function(portalPage) {
  146. renderVars.page = portalPage;
  147. renderVars.isPortal = (portalPage != null);
  148. if (portalPage) {
  149. renderVars.revision = portalPage.revision;
  150. return Revision.findRevisionList(portalPage.path, {});
  151. } else {
  152. return Promise.resolve([]);
  153. }
  154. }).then(function(tree) {
  155. renderVars.tree = tree;
  156. return Page.findListByStartWith(path, req.user, queryOptions);
  157. }).then(function(pageList) {
  158. if (pageList.length > limit) {
  159. pageList.pop();
  160. }
  161. pagerOptions.length = pageList.length;
  162. renderVars.viewConfig = {
  163. seener_threshold: SEENER_THRESHOLD,
  164. };
  165. renderVars.pager = generatePager(pagerOptions);
  166. renderVars.pages = pageList;
  167. res.render('customlayout-selector/page_list', renderVars);
  168. }).catch(function(err) {
  169. debug('Error on rendering pageListShow', err);
  170. });
  171. };
  172. actions.pageListShowForCrowiPlus = function(req, res) {
  173. var path = getPathFromRequest(req);
  174. // omit the slash of the last
  175. path = path.replace((/\/$/), '');
  176. // redirect
  177. return res.redirect(path);
  178. }
  179. actions.pageShowForCrowiPlus = function(req, res) {
  180. var path = getPathFromRequest(req);
  181. var limit = 50;
  182. var offset = parseInt(req.query.offset) || 0;
  183. var SEENER_THRESHOLD = 10;
  184. // index page
  185. var pagerOptions = {
  186. offset: offset,
  187. limit : limit
  188. };
  189. var queryOptions = {
  190. offset: offset,
  191. limit : limit + 1,
  192. isPopulateRevisionBody: Config.isEnabledTimeline(config),
  193. includeDeletedPage: path.startsWith('/trash/'),
  194. };
  195. var renderVars = {
  196. path: path,
  197. page: null,
  198. revision: {},
  199. author: false,
  200. pages: [],
  201. tree: [],
  202. userRelatedGroups: [],
  203. };
  204. var pageTeamplate = 'customlayout-selector/page';
  205. var isRedirect = false;
  206. Page.findPage(path, req.user, req.query.revision)
  207. .then(function(page) {
  208. debug('Page found', page._id, page.path);
  209. // redirect
  210. if (page.redirectTo) {
  211. debug(`Redirect to '${page.redirectTo}'`);
  212. isRedirect = true;
  213. return res.redirect(encodeURI(page.redirectTo + '?redirectFrom=' + page.path));
  214. }
  215. renderVars.page = page;
  216. if (page) {
  217. renderVars.path = page.path;
  218. renderVars.revision = page.revision;
  219. renderVars.author = page.revision.author;
  220. return Revision.findRevisionList(page.path, {})
  221. .then(function(tree) {
  222. renderVars.tree = tree;
  223. return Promise.resolve();
  224. }).then(function() {
  225. var userPage = isUserPage(page.path);
  226. var userData = null;
  227. if (userPage) {
  228. // change template
  229. pageTeamplate = 'customlayout-selector/user_page';
  230. return User.findUserByUsername(User.getUsernameByPath(page.path))
  231. .then(function(data) {
  232. if (data === null) {
  233. throw new Error('The user not found.');
  234. }
  235. userData = data;
  236. renderVars.pageUser = userData;
  237. return Bookmark.findByUser(userData, {limit: 10, populatePage: true, requestUser: req.user});
  238. }).then(function(bookmarkList) {
  239. renderVars.bookmarkList = bookmarkList;
  240. return Page.findListByCreator(userData, {limit: 10}, req.user);
  241. }).then(function(createdList) {
  242. renderVars.createdList = createdList;
  243. return Promise.resolve();
  244. }).catch(function(err) {
  245. debug('Error on finding user related entities', err);
  246. // pass
  247. });
  248. }
  249. else {
  250. return Promise.resolve();
  251. }
  252. });
  253. } else {
  254. return Promise.resolve();
  255. }
  256. })
  257. // page not exists
  258. .catch(function(err) {
  259. debug('Page not found', path);
  260. // change template
  261. pageTeamplate = 'customlayout-selector/not_found';
  262. })
  263. // get list pages
  264. .then(function() {
  265. if (!isRedirect) {
  266. Page.findListWithDescendants(path, req.user, queryOptions)
  267. .then(function(pageList) {
  268. if (pageList.length > limit) {
  269. pageList.pop();
  270. }
  271. pagerOptions.length = pageList.length;
  272. renderVars.viewConfig = {
  273. seener_threshold: SEENER_THRESHOLD,
  274. };
  275. renderVars.pager = generatePager(pagerOptions);
  276. renderVars.pages = pageList;
  277. return Promise.resolve();
  278. })
  279. .then(function() {
  280. return interceptorManager.process('beforeRenderPage', req, res, renderVars);
  281. })
  282. .then(function() {
  283. res.render(req.query.presentation ? 'page_presentation' : pageTeamplate, renderVars);
  284. })
  285. .catch(function(err) {
  286. console.log(err);
  287. debug('Error on rendering pageListShowForCrowiPlus', err);
  288. });
  289. }
  290. })
  291. .then(function() {
  292. return UserGroupRelation.findAllRelationForUser(req.user);
  293. }).then(function (groupRelations) {
  294. debug('findPage : relatedGroups ', groupRelations);
  295. renderVars.userRelatedGroups = groupRelations.map(relation => relation.relatedGroup);
  296. debug('findPage : groups ', renderVars.userRelatedGroups);
  297. return Promise.resolve();
  298. });
  299. }
  300. actions.deletedPageListShow = function(req, res) {
  301. var path = '/trash' + getPathFromRequest(req);
  302. var limit = 50;
  303. var offset = parseInt(req.query.offset) || 0;
  304. // index page
  305. var pagerOptions = {
  306. offset: offset,
  307. limit : limit
  308. };
  309. var queryOptions = {
  310. offset: offset,
  311. limit : limit + 1,
  312. includeDeletedPage: true,
  313. };
  314. var renderVars = {
  315. page: null,
  316. path: path,
  317. pages: [],
  318. };
  319. Page.findListWithDescendants(path, req.user, queryOptions)
  320. .then(function(pageList) {
  321. if (pageList.length > limit) {
  322. pageList.pop();
  323. }
  324. pagerOptions.length = pageList.length;
  325. renderVars.pager = generatePager(pagerOptions);
  326. renderVars.pages = pageList;
  327. res.render('customlayout-selector/page_list', renderVars);
  328. }).catch(function(err) {
  329. debug('Error on rendering deletedPageListShow', err);
  330. });
  331. };
  332. actions.search = function(req, res) {
  333. // spec: ?q=query&sort=sort_order&author=author_filter
  334. var query = req.query.q;
  335. var search = require('../util/search')(crowi);
  336. search.searchPageByKeyword(query)
  337. .then(function(pages) {
  338. debug('pages', pages);
  339. if (pages.hits.total <= 0) {
  340. return Promise.resolve([]);
  341. }
  342. var ids = pages.hits.hits.map(function(page) {
  343. return page._id;
  344. });
  345. return Page.findListByPageIds(ids);
  346. }).then(function(pages) {
  347. res.render('customlayout-selector/page_list', {
  348. path: '/',
  349. pages: pages,
  350. pager: generatePager({offset: 0, limit: 50})
  351. });
  352. }).catch(function(err) {
  353. debug('search error', err);
  354. });
  355. };
  356. function renderPage(pageData, req, res) {
  357. // create page
  358. if (!pageData) {
  359. return res.render('customlayout-selector/not_found', {
  360. author: {},
  361. page: false,
  362. });
  363. }
  364. if (pageData.redirectTo) {
  365. return res.redirect(encodeURI(pageData.redirectTo + '?redirectFrom=' + pageData.path));
  366. }
  367. var renderVars = {
  368. path: pageData.path,
  369. page: pageData,
  370. revision: pageData.revision || {},
  371. author: pageData.revision.author || false,
  372. };
  373. var userPage = isUserPage(pageData.path);
  374. var userData = null;
  375. Revision.findRevisionList(pageData.path, {})
  376. .then(function(tree) {
  377. renderVars.tree = tree;
  378. return Promise.resolve();
  379. }).then(function() {
  380. if (userPage) {
  381. return User.findUserByUsername(User.getUsernameByPath(pageData.path))
  382. .then(function(data) {
  383. if (data === null) {
  384. throw new Error('The user not found.');
  385. }
  386. userData = data;
  387. renderVars.pageUser = userData;
  388. return Bookmark.findByUser(userData, {limit: 10, populatePage: true, requestUser: req.user});
  389. }).then(function(bookmarkList) {
  390. renderVars.bookmarkList = bookmarkList;
  391. return Page.findListByCreator(userData, {limit: 10}, req.user);
  392. }).then(function(createdList) {
  393. renderVars.createdList = createdList;
  394. return Promise.resolve();
  395. }).catch(function(err) {
  396. debug('Error on finding user related entities', err);
  397. // pass
  398. });
  399. } else {
  400. return Promise.resolve();
  401. }
  402. }).then(function() {
  403. return interceptorManager.process('beforeRenderPage', req, res, renderVars);
  404. }).then(function() {
  405. var defaultPageTeamplate = 'customlayout-selector/page';
  406. if (userData) {
  407. defaultPageTeamplate = 'customlayout-selector/user_page';
  408. }
  409. res.render(req.query.presentation ? 'page_presentation' : defaultPageTeamplate, renderVars);
  410. }).catch(function(err) {
  411. debug('Error: renderPage()', err);
  412. if (err) {
  413. res.redirect('/');
  414. }
  415. });
  416. }
  417. actions.pageShow = function(req, res) {
  418. var path = path || getPathFromRequest(req);
  419. var options = {};
  420. // FIXME: せっかく getPathFromRequest になってるのにここが生 params[0] だとダサイ
  421. var isMarkdown = req.params[0].match(/.+\.md$/) || false;
  422. res.locals.path = path;
  423. Page.findPage(path, req.user, req.query.revision)
  424. .then(function(page) {
  425. debug('Page found', page._id, page.path);
  426. if (isMarkdown) {
  427. res.set('Content-Type', 'text/plain');
  428. return res.send(page.revision.body);
  429. }
  430. return renderPage(page, req, res);
  431. }).catch(function(err) {
  432. const normalizedPath = Page.normalizePath(path);
  433. if (normalizedPath !== path) {
  434. return res.redirect(normalizedPath);
  435. }
  436. // pageShow は /* にマッチしてる最後の砦なので、creatableName でない routing は
  437. // これ以前に定義されているはずなので、こうしてしまって問題ない。
  438. if (!Page.isCreatableName(path)) {
  439. // 削除済みページの場合 /trash 以下に移動しているので creatableName になっていないので、表示を許可
  440. debug('Page is not creatable name.', path);
  441. res.redirect('/');
  442. return ;
  443. }
  444. if (req.query.revision) {
  445. return res.redirect(encodeURI(path));
  446. }
  447. if (isMarkdown) {
  448. return res.redirect('/');
  449. }
  450. Page.hasPortalPage(path + '/', req.user)
  451. .then(function(page) {
  452. if (page) {
  453. return res.redirect(encodeURI(path) + '/');
  454. } else {
  455. var fixed = Page.fixToCreatableName(path)
  456. if (fixed !== path) {
  457. debug('fixed page name', fixed)
  458. res.redirect(encodeURI(fixed));
  459. return ;
  460. }
  461. // if guest user
  462. if (!req.user) {
  463. res.redirect('/');
  464. }
  465. // render editor
  466. debug('Catch pageShow', err);
  467. return renderPage(null, req, res);
  468. }
  469. }).catch(function(err) {
  470. debug('Error on rendering pageShow (redirect to portal)', err);
  471. });
  472. });
  473. };
  474. actions.pageEdit = function(req, res) {
  475. var pageForm = req.body.pageForm;
  476. var body = pageForm.body;
  477. var currentRevision = pageForm.currentRevision;
  478. var grant = pageForm.grant;
  479. var path = pageForm.path;
  480. var grantUserGroupId = pageForm.grantUserGroupId
  481. // TODO: make it pluggable
  482. var notify = pageForm.notify || {};
  483. debug('notify: ', notify);
  484. var redirectPath = encodeURI(path);
  485. var pageData = {};
  486. var updateOrCreate;
  487. var previousRevision = false;
  488. // set to render
  489. res.locals.pageForm = pageForm;
  490. // 削除済みページはここで編集不可判定される
  491. if (!Page.isCreatableName(path)) {
  492. res.redirect(redirectPath);
  493. return ;
  494. }
  495. var ignoreNotFound = true;
  496. Page.findPage(path, req.user, null, ignoreNotFound)
  497. .then(function(data) {
  498. pageData = data;
  499. if (!req.form.isValid) {
  500. debug('Form data not valid');
  501. throw new Error('Form data not valid.');
  502. }
  503. if (data && !data.isUpdatable(currentRevision)) {
  504. debug('Conflict occured');
  505. req.form.errors.push('page_edit.notice.conflict');
  506. throw new Error('Conflict.');
  507. }
  508. if (data) {
  509. previousRevision = data.revision;
  510. return Page.updatePage(data, body, req.user, { grant: grant, grantUserGroupId: grantUserGroupId});
  511. } else {
  512. // new page
  513. updateOrCreate = 'create';
  514. return Page.create(path, body, req.user, { grant: grant, grantUserGroupId: grantUserGroupId});
  515. }
  516. }).then(function(data) {
  517. // data is a saved page data.
  518. pageData = data;
  519. if (!data) {
  520. throw new Error('Data not found');
  521. }
  522. // TODO: move to events
  523. if (notify.slack) {
  524. if (notify.slack.on && notify.slack.channel) {
  525. data.updateSlackChannel(notify.slack.channel).then(function(){}).catch(function(){});
  526. if (crowi.slack) {
  527. notify.slack.channel.split(',').map(function(chan) {
  528. var message = crowi.slack.prepareSlackMessage(pageData, req.user, chan, updateOrCreate, previousRevision);
  529. crowi.slack.post(message.channel, message.text, message).then(function(){}).catch(function(){});
  530. });
  531. }
  532. }
  533. }
  534. return res.redirect(redirectPath);
  535. }).catch(function(err) {
  536. debug('Page create or edit error.', err);
  537. if (pageData && !req.form.isValid) {
  538. return renderPage(pageData, req, res);
  539. }
  540. return res.redirect(redirectPath);
  541. });
  542. };
  543. // app.get( '/users/:username([^/]+)/bookmarks' , loginRequired(crowi, app) , page.userBookmarkList);
  544. actions.userBookmarkList = function(req, res) {
  545. var username = req.params.username;
  546. var limit = 50;
  547. var offset = parseInt(req.query.offset) || 0;
  548. var user;
  549. var renderVars = {};
  550. var pagerOptions = { offset: offset, limit : limit };
  551. var queryOptions = { offset: offset, limit : limit + 1, populatePage: true, requestUser: req.user};
  552. User.findUserByUsername(username)
  553. .then(function(user) {
  554. if (user === null) {
  555. throw new Error('The user not found.');
  556. }
  557. renderVars.pageUser = user;
  558. return Bookmark.findByUser(user, queryOptions);
  559. }).then(function(bookmarks) {
  560. if (bookmarks.length > limit) {
  561. bookmarks.pop();
  562. }
  563. pagerOptions.length = bookmarks.length;
  564. renderVars.pager = generatePager(pagerOptions);
  565. renderVars.bookmarks = bookmarks;
  566. return res.render('user/bookmarks', renderVars);
  567. }).catch(function(err) {
  568. debug('Error on rendereing bookmark', err);
  569. res.redirect('/');
  570. });
  571. };
  572. // app.get( '/users/:username([^/]+)/recent-create' , loginRequired(crowi, app) , page.userRecentCreatedList);
  573. actions.userRecentCreatedList = function(req, res) {
  574. var username = req.params.username;
  575. var limit = 50;
  576. var offset = parseInt(req.query.offset) || 0;
  577. var user;
  578. var renderVars = {};
  579. var pagerOptions = { offset: offset, limit : limit };
  580. var queryOptions = { offset: offset, limit : limit + 1};
  581. User.findUserByUsername(username)
  582. .then(function(user) {
  583. if (user === null) {
  584. throw new Error('The user not found.');
  585. }
  586. renderVars.pageUser = user;
  587. return Page.findListByCreator(user, queryOptions, req.user);
  588. }).then(function(pages) {
  589. if (pages.length > limit) {
  590. pages.pop();
  591. }
  592. pagerOptions.length = pages.length;
  593. renderVars.pager = generatePager(pagerOptions);
  594. renderVars.pages = pages;
  595. return res.render('user/recent-create', renderVars);
  596. }).catch(function(err) {
  597. debug('Error on rendereing recent-created', err);
  598. res.redirect('/');
  599. });
  600. };
  601. var api = actions.api = {};
  602. /**
  603. * redirector
  604. */
  605. api.redirector = function(req, res){
  606. var id = req.params.id;
  607. Page.findPageById(id)
  608. .then(function(pageData) {
  609. if (pageData.grant == Page.GRANT_RESTRICTED && !pageData.isGrantedFor(req.user)) {
  610. return Page.pushToGrantedUsers(pageData, req.user);
  611. }
  612. return Promise.resolve(pageData);
  613. }).then(function(page) {
  614. return res.redirect(encodeURI(page.path));
  615. }).catch(function(err) {
  616. return res.redirect('/');
  617. });
  618. };
  619. /**
  620. * @api {get} /pages.list List pages by user
  621. * @apiName ListPage
  622. * @apiGroup Page
  623. *
  624. * @apiParam {String} path
  625. * @apiParam {String} user
  626. */
  627. api.list = function(req, res) {
  628. var username = req.query.user || null;
  629. var path = req.query.path || null;
  630. var limit = 50;
  631. var offset = parseInt(req.query.offset) || 0;
  632. var pagerOptions = { offset: offset, limit : limit };
  633. var queryOptions = { offset: offset, limit : limit + 1};
  634. // Accepts only one of these
  635. if (username === null && path === null) {
  636. return res.json(ApiResponse.error('Parameter user or path is required.'));
  637. }
  638. if (username !== null && path !== null) {
  639. return res.json(ApiResponse.error('Parameter user or path is required.'));
  640. }
  641. var pageFetcher;
  642. if (path === null) {
  643. pageFetcher = User.findUserByUsername(username)
  644. .then(function(user) {
  645. if (user === null) {
  646. throw new Error('The user not found.');
  647. }
  648. return Page.findListByCreator(user, queryOptions, req.user);
  649. });
  650. } else {
  651. pageFetcher = Page.findListByStartWith(path, req.user, queryOptions);
  652. }
  653. pageFetcher
  654. .then(function(pages) {
  655. if (pages.length > limit) {
  656. pages.pop();
  657. }
  658. pagerOptions.length = pages.length;
  659. var result = {};
  660. result.pages = pages;
  661. return res.json(ApiResponse.success(result));
  662. }).catch(function(err) {
  663. return res.json(ApiResponse.error(err));
  664. });
  665. };
  666. /**
  667. * @api {post} /pages.create Create new page
  668. * @apiName CreatePage
  669. * @apiGroup Page
  670. *
  671. * @apiParam {String} body
  672. * @apiParam {String} path
  673. * @apiParam {String} grant
  674. */
  675. api.create = function(req, res){
  676. var body = req.body.body || null;
  677. var pagePath = req.body.path || null;
  678. var grant = req.body.grant || null;
  679. var grantUserGroupId = req.body.grantUserGroupId || null;
  680. if (body === null || pagePath === null) {
  681. return res.json(ApiResponse.error('Parameters body and path are required.'));
  682. }
  683. var ignoreNotFound = true;
  684. Page.findPage(pagePath, req.user, null, ignoreNotFound)
  685. .then(function(data) {
  686. if (data !== null) {
  687. throw new Error('Page exists');
  688. }
  689. return Page.create(pagePath, body, req.user, { grant: grant, grantUserGroupId: grantUserGroupId});
  690. }).then(function(data) {
  691. if (!data) {
  692. throw new Error('Failed to create page.');
  693. }
  694. var result = { page: data.toObject() };
  695. result.page.lastUpdateUser = User.filterToPublicFields(data.lastUpdateUser);
  696. result.page.creator = User.filterToPublicFields(data.creator);
  697. return res.json(ApiResponse.success(result));
  698. }).catch(function(err) {
  699. return res.json(ApiResponse.error(err));
  700. });;
  701. };
  702. /**
  703. * @api {post} /pages.update Update page
  704. * @apiName UpdatePage
  705. * @apiGroup Page
  706. *
  707. * @apiParam {String} body
  708. * @apiParam {String} page_id
  709. * @apiParam {String} revision_id
  710. * @apiParam {String} grant
  711. *
  712. * In the case of the page exists:
  713. * - If revision_id is specified => update the page,
  714. * - If revision_id is not specified => force update by the new contents.
  715. */
  716. api.update = function(req, res){
  717. var pageBody = req.body.body || null;
  718. var pageId = req.body.page_id || null;
  719. var revisionId = req.body.revision_id || null;
  720. var grant = req.body.grant || null;
  721. var grantUserGroupId = req.body.grantUserGroupId || null;
  722. if (pageId === null || pageBody === null) {
  723. return res.json(ApiResponse.error('page_id and body are required.'));
  724. }
  725. Page.findPageByIdAndGrantedUser(pageId, req.user)
  726. .then(function(pageData) {
  727. if (pageData && revisionId !== null && !pageData.isUpdatable(revisionId)) {
  728. throw new Error('Revision error.');
  729. };
  730. var grantOption = {grant: pageData.grant};
  731. if (grant !== null) {
  732. grantOption.grant = grant;
  733. }
  734. if (grantUserGroupId != null) {
  735. grantOption.grantUserGroupId = grantUserGroupId;
  736. }
  737. return Page.updatePage(pageData, pageBody, req.user, grantOption);
  738. }).then(function(pageData) {
  739. var result = {
  740. page: pageData.toObject(),
  741. };
  742. result.page.lastUpdateUser = User.filterToPublicFields(result.page.lastUpdateUser);
  743. return res.json(ApiResponse.success(result));
  744. }).catch(function(err) {
  745. debug('error on _api/pages.update', err);
  746. return res.json(ApiResponse.error(err));
  747. });
  748. };
  749. /**
  750. * @api {get} /pages.get Get page data
  751. * @apiName GetPage
  752. * @apiGroup Page
  753. *
  754. * @apiParam {String} page_id
  755. * @apiParam {String} path
  756. * @apiParam {String} revision_id
  757. */
  758. api.get = function(req, res){
  759. const pagePath = req.query.path || null;
  760. const pageId = req.query.page_id || null; // TODO: handling
  761. const revisionId = req.query.revision_id || null;
  762. if (!pageId && !pagePath) {
  763. return res.json(ApiResponse.error(new Error('Parameter path or page_id is required.')));
  764. }
  765. let pageFinder;
  766. if (pageId) { // prioritized
  767. pageFinder = Page.findPageByIdAndGrantedUser(pageId, req.user);
  768. } else if (pagePath) {
  769. pageFinder = Page.findPage(pagePath, req.user, revisionId);
  770. }
  771. pageFinder.then(function(pageData) {
  772. var result = {};
  773. result.page = pageData;
  774. return res.json(ApiResponse.success(result));
  775. }).catch(function(err) {
  776. return res.json(ApiResponse.error(err));
  777. });
  778. };
  779. /**
  780. * @api {post} /pages.seen Mark as seen user
  781. * @apiName SeenPage
  782. * @apiGroup Page
  783. *
  784. * @apiParam {String} page_id Page Id.
  785. */
  786. api.seen = function(req, res){
  787. var pageId = req.body.page_id;
  788. if (!pageId) {
  789. return res.json(ApiResponse.error('page_id required'));
  790. }
  791. Page.findPageByIdAndGrantedUser(pageId, req.user)
  792. .then(function(page) {
  793. return page.seen(req.user);
  794. }).then(function(user) {
  795. var result = {};
  796. result.seenUser = user;
  797. return res.json(ApiResponse.success(result));
  798. }).catch(function(err) {
  799. debug('Seen user update error', err);
  800. return res.json(ApiResponse.error(err));
  801. });
  802. };
  803. /**
  804. * @api {post} /likes.add Like page
  805. * @apiName LikePage
  806. * @apiGroup Page
  807. *
  808. * @apiParam {String} page_id Page Id.
  809. */
  810. api.like = function(req, res){
  811. var id = req.body.page_id;
  812. Page.findPageByIdAndGrantedUser(id, req.user)
  813. .then(function(pageData) {
  814. return pageData.like(req.user);
  815. }).then(function(data) {
  816. var result = {page: data};
  817. return res.json(ApiResponse.success(result));
  818. }).catch(function(err) {
  819. debug('Like failed', err);
  820. return res.json(ApiResponse.error({}));
  821. });
  822. };
  823. /**
  824. * @api {post} /likes.remove Unlike page
  825. * @apiName UnlikePage
  826. * @apiGroup Page
  827. *
  828. * @apiParam {String} page_id Page Id.
  829. */
  830. api.unlike = function(req, res){
  831. var id = req.body.page_id;
  832. Page.findPageByIdAndGrantedUser(id, req.user)
  833. .then(function(pageData) {
  834. return pageData.unlike(req.user);
  835. }).then(function(data) {
  836. var result = {page: data};
  837. return res.json(ApiResponse.success(result));
  838. }).catch(function(err) {
  839. debug('Unlike failed', err);
  840. return res.json(ApiResponse.error({}));
  841. });
  842. };
  843. /**
  844. * @api {get} /pages.updatePost
  845. * @apiName Get UpdatePost setting list
  846. * @apiGroup Page
  847. *
  848. * @apiParam {String} path
  849. */
  850. api.getUpdatePost = function(req, res) {
  851. var path = req.query.path;
  852. var UpdatePost = crowi.model('UpdatePost');
  853. if (!path) {
  854. return res.json(ApiResponse.error({}));
  855. }
  856. UpdatePost.findSettingsByPath(path)
  857. .then(function(data) {
  858. data = data.map(function(e) {
  859. return e.channel;
  860. });
  861. debug('Found updatePost data', data);
  862. var result = {updatePost: data};
  863. return res.json(ApiResponse.success(result));
  864. }).catch(function(err) {
  865. debug('Error occured while get setting', err);
  866. return res.json(ApiResponse.error({}));
  867. });
  868. };
  869. /**
  870. * @api {post} /pages.remove Remove page
  871. * @apiName RemovePage
  872. * @apiGroup Page
  873. *
  874. * @apiParam {String} page_id Page Id.
  875. * @apiParam {String} revision_id
  876. */
  877. api.remove = function(req, res){
  878. var pageId = req.body.page_id;
  879. var previousRevision = req.body.revision_id || null;
  880. // get completely flag
  881. const isCompletely = (req.body.completely !== undefined);
  882. // get recursively flag
  883. const isRecursively = (req.body.recursively !== undefined);
  884. Page.findPageByIdAndGrantedUser(pageId, req.user)
  885. .then(function(pageData) {
  886. debug('Delete page', pageData._id, pageData.path);
  887. if (isCompletely) {
  888. if (isRecursively) {
  889. return Page.completelyDeletePageRecursively(pageData, req.user);
  890. }
  891. else {
  892. return Page.completelyDeletePage(pageData, req.user);
  893. }
  894. }
  895. // else
  896. if (!pageData.isUpdatable(previousRevision)) {
  897. throw new Error('Someone could update this page, so couldn\'t delete.');
  898. }
  899. if (isRecursively) {
  900. return Page.deletePageRecursively(pageData, req.user);
  901. }
  902. else {
  903. return Page.deletePage(pageData, req.user);
  904. }
  905. }).then(function(data) {
  906. debug('Page deleted', data.path);
  907. var result = {};
  908. result.page = data;
  909. return res.json(ApiResponse.success(result));
  910. }).catch(function(err) {
  911. debug('Error occured while get setting', err, err.stack);
  912. return res.json(ApiResponse.error('Failed to delete page.'));
  913. });
  914. };
  915. /**
  916. * @api {post} /pages.revertRemove Revert removed page
  917. * @apiName RevertRemovePage
  918. * @apiGroup Page
  919. *
  920. * @apiParam {String} page_id Page Id.
  921. */
  922. api.revertRemove = function(req, res){
  923. var pageId = req.body.page_id;
  924. // get recursively flag
  925. const isRecursively = (req.body.recursively !== undefined);
  926. Page.findPageByIdAndGrantedUser(pageId, req.user)
  927. .then(function(pageData) {
  928. if (isRecursively) {
  929. return Page.revertDeletedPageRecursively(pageData, req.user);
  930. }
  931. else {
  932. return Page.revertDeletedPage(pageData, req.user);
  933. }
  934. }).then(function(data) {
  935. debug('Complete to revert deleted page', data.path);
  936. var result = {};
  937. result.page = data;
  938. return res.json(ApiResponse.success(result));
  939. }).catch(function(err) {
  940. debug('Error occured while get setting', err, err.stack);
  941. return res.json(ApiResponse.error('Failed to revert deleted page.'));
  942. });
  943. };
  944. /**
  945. * @api {post} /pages.rename Rename page
  946. * @apiName RenamePage
  947. * @apiGroup Page
  948. *
  949. * @apiParam {String} page_id Page Id.
  950. * @apiParam {String} path
  951. * @apiParam {String} revision_id
  952. * @apiParam {String} new_path
  953. * @apiParam {Bool} create_redirect
  954. */
  955. api.rename = function(req, res){
  956. var pageId = req.body.page_id;
  957. var previousRevision = req.body.revision_id || null;
  958. var newPagePath = Page.normalizePath(req.body.new_path);
  959. var options = {
  960. createRedirectPage: req.body.create_redirect || 0,
  961. moveUnderTrees: req.body.move_trees || 0,
  962. };
  963. var isRecursiveMove = req.body.move_recursively || 0;
  964. var page = {};
  965. if (!Page.isCreatableName(newPagePath)) {
  966. return res.json(ApiResponse.error(`このページ名は作成できません (${newPagePath})`));
  967. }
  968. Page.findPageByPath(newPagePath)
  969. .then(function(page) {
  970. // if page found, cannot cannot rename to that path
  971. return res.json(ApiResponse.error(`このページ名は作成できません (${newPagePath})。ページが存在します。`));
  972. }).catch(function(err) {
  973. Page.findPageById(pageId)
  974. .then(function(pageData) {
  975. page = pageData;
  976. if (!pageData.isUpdatable(previousRevision)) {
  977. throw new Error('Someone could update this page, so couldn\'t delete.');
  978. }
  979. if (isRecursiveMove) {
  980. return Page.renameRecursively(pageData, newPagePath, req.user, options);
  981. }
  982. else {
  983. return Page.rename(pageData, newPagePath, req.user, options);
  984. }
  985. }).then(function() {
  986. var result = {};
  987. result.page = page;
  988. return res.json(ApiResponse.success(result));
  989. }).catch(function(err) {
  990. return res.json(ApiResponse.error('Failed to update page.'));
  991. });
  992. });
  993. };
  994. /**
  995. * @api {post} /pages.duplicate Duplicate page
  996. * @apiName DuplicatePage
  997. * @apiGroup Page
  998. *
  999. * @apiParam {String} page_id Page Id.
  1000. * @apiParam {String} new_path
  1001. */
  1002. api.duplicate = function (req, res) {
  1003. var pageId = req.body.page_id;
  1004. var newPagePath = Page.normalizePath(req.body.new_path);
  1005. var page = {};
  1006. Page.findPageById(pageId)
  1007. .then(function (pageData) {
  1008. req.body.path = newPagePath;
  1009. req.body.body = pageData.revision.body;
  1010. req.body.grant = pageData.grant;
  1011. return api.create(req, res);
  1012. });
  1013. };
  1014. /**
  1015. * @api {post} /pages.unlink Remove the redirecting page
  1016. * @apiName UnlinkPage
  1017. * @apiGroup Page
  1018. *
  1019. * @apiParam {String} page_id Page Id.
  1020. * @apiParam {String} revision_id
  1021. */
  1022. api.unlink = function(req, res){
  1023. var pageId = req.body.page_id;
  1024. Page.findPageByIdAndGrantedUser(pageId, req.user)
  1025. .then(function(pageData) {
  1026. debug('Unlink page', pageData._id, pageData.path);
  1027. return Page.removeRedirectOriginPageByPath(pageData.path)
  1028. .then(() => pageData);
  1029. }).then(function(data) {
  1030. debug('Redirect Page deleted', data.path);
  1031. var result = {};
  1032. result.page = data;
  1033. return res.json(ApiResponse.success(result));
  1034. }).catch(function(err) {
  1035. debug('Error occured while get setting', err, err.stack);
  1036. return res.json(ApiResponse.error('Failed to delete redirect page.'));
  1037. });
  1038. };
  1039. return actions;
  1040. };