page.js 29 KB

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