page.js 34 KB

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