page.js 34 KB

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