page.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597
  1. const _ = require('lodash')
  2. const graphHelper = require('../../helpers/graph')
  3. module.exports = {
  4. Query: {
  5. /**
  6. * PAGE HISTORY
  7. */
  8. async pageHistoryById (obj, args, context, info) {
  9. const page = await WIKI.db.pages.query().select('path', 'localeCode').findById(args.id)
  10. if (WIKI.auth.checkAccess(context.req.user, ['read:history'], {
  11. path: page.path,
  12. locale: page.localeCode
  13. })) {
  14. return WIKI.db.pageHistory.getHistory({
  15. pageId: args.id,
  16. offsetPage: args.offsetPage || 0,
  17. offsetSize: args.offsetSize || 100
  18. })
  19. } else {
  20. throw new WIKI.Error.PageHistoryForbidden()
  21. }
  22. },
  23. /**
  24. * PAGE VERSION
  25. */
  26. async pageVersionById (obj, args, context, info) {
  27. const page = await WIKI.db.pages.query().select('path', 'localeCode').findById(args.pageId)
  28. if (WIKI.auth.checkAccess(context.req.user, ['read:history'], {
  29. path: page.path,
  30. locale: page.localeCode
  31. })) {
  32. return WIKI.db.pageHistory.getVersion({
  33. pageId: args.pageId,
  34. versionId: args.versionId
  35. })
  36. } else {
  37. throw new WIKI.Error.PageHistoryForbidden()
  38. }
  39. },
  40. /**
  41. * SEARCH PAGES
  42. */
  43. async searchPages (obj, args, context) {
  44. if (WIKI.data.searchEngine) {
  45. const resp = await WIKI.data.searchEngine.query(args.query, args)
  46. return {
  47. ...resp,
  48. results: _.filter(resp.results, r => {
  49. return WIKI.auth.checkAccess(context.req.user, ['read:pages'], {
  50. path: r.path,
  51. locale: r.locale,
  52. tags: r.tags // Tags are needed since access permissions can be limited by page tags too
  53. })
  54. })
  55. }
  56. } else {
  57. return {
  58. results: [],
  59. suggestions: [],
  60. totalHits: 0
  61. }
  62. }
  63. },
  64. /**
  65. * LIST PAGES
  66. */
  67. async pages (obj, args, context, info) {
  68. let results = await WIKI.db.pages.query().column([
  69. 'pages.id',
  70. 'path',
  71. { locale: 'localeCode' },
  72. 'title',
  73. 'description',
  74. 'isPublished',
  75. 'isPrivate',
  76. 'privateNS',
  77. 'contentType',
  78. 'createdAt',
  79. 'updatedAt'
  80. ])
  81. .withGraphJoined('tags')
  82. .modifyGraph('tags', builder => {
  83. builder.select('tag')
  84. })
  85. .modify(queryBuilder => {
  86. if (args.limit) {
  87. queryBuilder.limit(args.limit)
  88. }
  89. if (args.locale) {
  90. queryBuilder.where('localeCode', args.locale)
  91. }
  92. if (args.creatorId && args.authorId && args.creatorId > 0 && args.authorId > 0) {
  93. queryBuilder.where(function () {
  94. this.where('creatorId', args.creatorId).orWhere('authorId', args.authorId)
  95. })
  96. } else {
  97. if (args.creatorId && args.creatorId > 0) {
  98. queryBuilder.where('creatorId', args.creatorId)
  99. }
  100. if (args.authorId && args.authorId > 0) {
  101. queryBuilder.where('authorId', args.authorId)
  102. }
  103. }
  104. if (args.tags && args.tags.length > 0) {
  105. queryBuilder.whereIn('tags.tag', args.tags.map(t => _.trim(t).toLowerCase()))
  106. }
  107. const orderDir = args.orderByDirection === 'DESC' ? 'desc' : 'asc'
  108. switch (args.orderBy) {
  109. case 'CREATED':
  110. queryBuilder.orderBy('createdAt', orderDir)
  111. break
  112. case 'PATH':
  113. queryBuilder.orderBy('path', orderDir)
  114. break
  115. case 'TITLE':
  116. queryBuilder.orderBy('title', orderDir)
  117. break
  118. case 'UPDATED':
  119. queryBuilder.orderBy('updatedAt', orderDir)
  120. break
  121. default:
  122. queryBuilder.orderBy('pages.id', orderDir)
  123. break
  124. }
  125. })
  126. results = _.filter(results, r => {
  127. return WIKI.auth.checkAccess(context.req.user, ['read:pages'], {
  128. path: r.path,
  129. locale: r.locale
  130. })
  131. }).map(r => ({
  132. ...r,
  133. tags: _.map(r.tags, 'tag')
  134. }))
  135. if (args.tags && args.tags.length > 0) {
  136. results = _.filter(results, r => _.every(args.tags, t => _.includes(r.tags, t)))
  137. }
  138. return results
  139. },
  140. /**
  141. * FETCH SINGLE PAGE
  142. */
  143. async pageById (obj, args, context, info) {
  144. let page = await WIKI.db.pages.getPageFromDb(args.id)
  145. if (page) {
  146. if (WIKI.auth.checkAccess(context.req.user, ['manage:pages', 'delete:pages'], {
  147. path: page.path,
  148. locale: page.localeCode
  149. })) {
  150. return {
  151. ...page,
  152. locale: page.localeCode,
  153. editor: page.editorKey
  154. }
  155. } else {
  156. throw new WIKI.Error.PageViewForbidden()
  157. }
  158. } else {
  159. throw new WIKI.Error.PageNotFound()
  160. }
  161. },
  162. /**
  163. * FETCH TAGS
  164. */
  165. async tags (obj, args, context, info) {
  166. const pages = await WIKI.db.pages.query()
  167. .column([
  168. 'path',
  169. { locale: 'localeCode' }
  170. ])
  171. .withGraphJoined('tags')
  172. const allTags = _.filter(pages, r => {
  173. return WIKI.auth.checkAccess(context.req.user, ['read:pages'], {
  174. path: r.path,
  175. locale: r.locale
  176. })
  177. }).flatMap(r => r.tags)
  178. return _.orderBy(_.uniqBy(allTags, 'id'), ['tag'], ['asc'])
  179. },
  180. /**
  181. * SEARCH TAGS
  182. */
  183. async searchTags (obj, args, context, info) {
  184. const query = _.trim(args.query)
  185. const pages = await WIKI.db.pages.query()
  186. .column([
  187. 'path',
  188. { locale: 'localeCode' }
  189. ])
  190. .withGraphJoined('tags')
  191. .modifyGraph('tags', builder => {
  192. builder.select('tag')
  193. })
  194. .modify(queryBuilder => {
  195. queryBuilder.andWhere(builderSub => {
  196. if (WIKI.config.db.type === 'postgres') {
  197. builderSub.where('tags.tag', 'ILIKE', `%${query}%`)
  198. } else {
  199. builderSub.where('tags.tag', 'LIKE', `%${query}%`)
  200. }
  201. })
  202. })
  203. const allTags = _.filter(pages, r => {
  204. return WIKI.auth.checkAccess(context.req.user, ['read:pages'], {
  205. path: r.path,
  206. locale: r.locale
  207. })
  208. }).flatMap(r => r.tags).map(t => t.tag)
  209. return _.uniq(allTags).slice(0, 5)
  210. },
  211. /**
  212. * FETCH PAGE TREE
  213. */
  214. async pageTree (obj, args, context, info) {
  215. let curPage = null
  216. if (!args.locale) { args.locale = WIKI.config.lang.code }
  217. if (args.path && !args.parent) {
  218. curPage = await WIKI.db.knex('pageTree').first('parent', 'ancestors').where({
  219. path: args.path,
  220. localeCode: args.locale
  221. })
  222. if (curPage) {
  223. args.parent = curPage.parent || 0
  224. } else {
  225. return []
  226. }
  227. }
  228. const results = await WIKI.db.knex('pageTree').where(builder => {
  229. builder.where('localeCode', args.locale)
  230. switch (args.mode) {
  231. case 'FOLDERS':
  232. builder.andWhere('isFolder', true)
  233. break
  234. case 'PAGES':
  235. builder.andWhereNotNull('pageId')
  236. break
  237. }
  238. if (!args.parent || args.parent < 1) {
  239. builder.whereNull('parent')
  240. } else {
  241. builder.where('parent', args.parent)
  242. if (args.includeAncestors && curPage && curPage.ancestors.length > 0) {
  243. builder.orWhereIn('id', _.isString(curPage.ancestors) ? JSON.parse(curPage.ancestors) : curPage.ancestors)
  244. }
  245. }
  246. }).orderBy([{ column: 'isFolder', order: 'desc' }, 'title'])
  247. return results.filter(r => {
  248. return WIKI.auth.checkAccess(context.req.user, ['read:pages'], {
  249. path: r.path,
  250. locale: r.localeCode
  251. })
  252. }).map(r => ({
  253. ...r,
  254. parent: r.parent || 0,
  255. locale: r.localeCode
  256. }))
  257. },
  258. /**
  259. * FETCH PAGE LINKS
  260. */
  261. async pageLinks (obj, args, context, info) {
  262. let results
  263. if (WIKI.config.db.type === 'mysql' || WIKI.config.db.type === 'mariadb' || WIKI.config.db.type === 'sqlite') {
  264. results = await WIKI.db.knex('pages')
  265. .column({ id: 'pages.id' }, { path: 'pages.path' }, 'title', { link: 'pageLinks.path' }, { locale: 'pageLinks.localeCode' })
  266. .leftJoin('pageLinks', 'pages.id', 'pageLinks.pageId')
  267. .where({
  268. 'pages.localeCode': args.locale
  269. })
  270. .unionAll(
  271. WIKI.db.knex('pageLinks')
  272. .column({ id: 'pages.id' }, { path: 'pages.path' }, 'title', { link: 'pageLinks.path' }, { locale: 'pageLinks.localeCode' })
  273. .leftJoin('pages', 'pageLinks.pageId', 'pages.id')
  274. .where({
  275. 'pages.localeCode': args.locale
  276. })
  277. )
  278. } else {
  279. results = await WIKI.db.knex('pages')
  280. .column({ id: 'pages.id' }, { path: 'pages.path' }, 'title', { link: 'pageLinks.path' }, { locale: 'pageLinks.localeCode' })
  281. .fullOuterJoin('pageLinks', 'pages.id', 'pageLinks.pageId')
  282. .where({
  283. 'pages.localeCode': args.locale
  284. })
  285. }
  286. return _.reduce(results, (result, val) => {
  287. // -> Check if user has access to source and linked page
  288. if (
  289. !WIKI.auth.checkAccess(context.req.user, ['read:pages'], { path: val.path, locale: args.locale }) ||
  290. !WIKI.auth.checkAccess(context.req.user, ['read:pages'], { path: val.link, locale: val.locale })
  291. ) {
  292. return result
  293. }
  294. const existingEntry = _.findIndex(result, ['id', val.id])
  295. if (existingEntry >= 0) {
  296. if (val.link) {
  297. result[existingEntry].links.push(`${val.locale}/${val.link}`)
  298. }
  299. } else {
  300. result.push({
  301. id: val.id,
  302. title: val.title,
  303. path: `${args.locale}/${val.path}`,
  304. links: val.link ? [`${val.locale}/${val.link}`] : []
  305. })
  306. }
  307. return result
  308. }, [])
  309. },
  310. /**
  311. * CHECK FOR EDITING CONFLICT
  312. */
  313. async checkConflicts (obj, args, context, info) {
  314. let page = await WIKI.db.pages.query().select('path', 'localeCode', 'updatedAt').findById(args.id)
  315. if (page) {
  316. if (WIKI.auth.checkAccess(context.req.user, ['write:pages', 'manage:pages'], {
  317. path: page.path,
  318. locale: page.localeCode
  319. })) {
  320. return page.updatedAt > args.checkoutDate
  321. } else {
  322. throw new WIKI.Error.PageUpdateForbidden()
  323. }
  324. } else {
  325. throw new WIKI.Error.PageNotFound()
  326. }
  327. },
  328. /**
  329. * FETCH LATEST VERSION FOR CONFLICT COMPARISON
  330. */
  331. async checkConflictsLatest (obj, args, context, info) {
  332. let page = await WIKI.db.pages.getPageFromDb(args.id)
  333. if (page) {
  334. if (WIKI.auth.checkAccess(context.req.user, ['write:pages', 'manage:pages'], {
  335. path: page.path,
  336. locale: page.localeCode
  337. })) {
  338. return {
  339. ...page,
  340. tags: page.tags.map(t => t.tag),
  341. locale: page.localeCode
  342. }
  343. } else {
  344. throw new WIKI.Error.PageViewForbidden()
  345. }
  346. } else {
  347. throw new WIKI.Error.PageNotFound()
  348. }
  349. }
  350. },
  351. Mutation: {
  352. /**
  353. * CREATE PAGE
  354. */
  355. async createPage(obj, args, context) {
  356. try {
  357. const page = await WIKI.db.pages.createPage({
  358. ...args,
  359. user: context.req.user
  360. })
  361. return {
  362. responseResult: graphHelper.generateSuccess('Page created successfully.'),
  363. page
  364. }
  365. } catch (err) {
  366. return graphHelper.generateError(err)
  367. }
  368. },
  369. /**
  370. * UPDATE PAGE
  371. */
  372. async updatePage(obj, args, context) {
  373. try {
  374. const page = await WIKI.db.pages.updatePage({
  375. ...args,
  376. user: context.req.user
  377. })
  378. return {
  379. responseResult: graphHelper.generateSuccess('Page has been updated.'),
  380. page
  381. }
  382. } catch (err) {
  383. return graphHelper.generateError(err)
  384. }
  385. },
  386. /**
  387. * CONVERT PAGE
  388. */
  389. async convertPage(obj, args, context) {
  390. try {
  391. await WIKI.db.pages.convertPage({
  392. ...args,
  393. user: context.req.user
  394. })
  395. return {
  396. responseResult: graphHelper.generateSuccess('Page has been converted.')
  397. }
  398. } catch (err) {
  399. return graphHelper.generateError(err)
  400. }
  401. },
  402. /**
  403. * RENAME PAGE
  404. */
  405. async renamePage(obj, args, context) {
  406. try {
  407. await WIKI.db.pages.movePage({
  408. ...args,
  409. user: context.req.user
  410. })
  411. return {
  412. responseResult: graphHelper.generateSuccess('Page has been moved.')
  413. }
  414. } catch (err) {
  415. return graphHelper.generateError(err)
  416. }
  417. },
  418. /**
  419. * DELETE PAGE
  420. */
  421. async deletePage(obj, args, context) {
  422. try {
  423. await WIKI.db.pages.deletePage({
  424. ...args,
  425. user: context.req.user
  426. })
  427. return {
  428. responseResult: graphHelper.generateSuccess('Page has been deleted.')
  429. }
  430. } catch (err) {
  431. return graphHelper.generateError(err)
  432. }
  433. },
  434. /**
  435. * DELETE TAG
  436. */
  437. async deleteTag (obj, args, context) {
  438. try {
  439. const tagToDel = await WIKI.db.tags.query().findById(args.id)
  440. if (tagToDel) {
  441. await tagToDel.$relatedQuery('pages').unrelate()
  442. await WIKI.db.tags.query().deleteById(args.id)
  443. } else {
  444. throw new Error('This tag does not exist.')
  445. }
  446. return {
  447. responseResult: graphHelper.generateSuccess('Tag has been deleted.')
  448. }
  449. } catch (err) {
  450. return graphHelper.generateError(err)
  451. }
  452. },
  453. /**
  454. * UPDATE TAG
  455. */
  456. async updateTag (obj, args, context) {
  457. try {
  458. const affectedRows = await WIKI.db.tags.query()
  459. .findById(args.id)
  460. .patch({
  461. tag: _.trim(args.tag).toLowerCase(),
  462. title: _.trim(args.title)
  463. })
  464. if (affectedRows < 1) {
  465. throw new Error('This tag does not exist.')
  466. }
  467. return {
  468. responseResult: graphHelper.generateSuccess('Tag has been updated successfully.')
  469. }
  470. } catch (err) {
  471. return graphHelper.generateError(err)
  472. }
  473. },
  474. /**
  475. * FLUSH PAGE CACHE
  476. */
  477. async flushCache(obj, args, context) {
  478. try {
  479. await WIKI.db.pages.flushCache()
  480. WIKI.events.outbound.emit('flushCache')
  481. return {
  482. responseResult: graphHelper.generateSuccess('Pages Cache has been flushed successfully.')
  483. }
  484. } catch (err) {
  485. return graphHelper.generateError(err)
  486. }
  487. },
  488. /**
  489. * MIGRATE ALL PAGES FROM SOURCE LOCALE TO TARGET LOCALE
  490. */
  491. async migrateToLocale(obj, args, context) {
  492. try {
  493. const count = await WIKI.db.pages.migrateToLocale(args)
  494. return {
  495. responseResult: graphHelper.generateSuccess('Migrated content to target locale successfully.'),
  496. count
  497. }
  498. } catch (err) {
  499. return graphHelper.generateError(err)
  500. }
  501. },
  502. /**
  503. * REBUILD TREE
  504. */
  505. async rebuildPageTree(obj, args, context) {
  506. try {
  507. await WIKI.db.pages.rebuildTree()
  508. return {
  509. responseResult: graphHelper.generateSuccess('Page tree rebuilt successfully.')
  510. }
  511. } catch (err) {
  512. return graphHelper.generateError(err)
  513. }
  514. },
  515. /**
  516. * RENDER PAGE
  517. */
  518. async renderPage (obj, args, context) {
  519. try {
  520. const page = await WIKI.db.pages.query().findById(args.id)
  521. if (!page) {
  522. throw new WIKI.Error.PageNotFound()
  523. }
  524. await WIKI.db.pages.renderPage(page)
  525. return {
  526. responseResult: graphHelper.generateSuccess('Page rendered successfully.')
  527. }
  528. } catch (err) {
  529. return graphHelper.generateError(err)
  530. }
  531. },
  532. /**
  533. * RESTORE PAGE VERSION
  534. */
  535. async restorePage (obj, args, context) {
  536. try {
  537. const page = await WIKI.db.pages.query().select('path', 'localeCode').findById(args.pageId)
  538. if (!page) {
  539. throw new WIKI.Error.PageNotFound()
  540. }
  541. if (!WIKI.auth.checkAccess(context.req.user, ['write:pages'], {
  542. path: page.path,
  543. locale: page.localeCode
  544. })) {
  545. throw new WIKI.Error.PageRestoreForbidden()
  546. }
  547. const targetVersion = await WIKI.db.pageHistory.getVersion({ pageId: args.pageId, versionId: args.versionId })
  548. if (!targetVersion) {
  549. throw new WIKI.Error.PageNotFound()
  550. }
  551. await WIKI.db.pages.updatePage({
  552. ...targetVersion,
  553. id: targetVersion.pageId,
  554. user: context.req.user,
  555. action: 'restored'
  556. })
  557. return {
  558. responseResult: graphHelper.generateSuccess('Page version restored successfully.')
  559. }
  560. } catch (err) {
  561. return graphHelper.generateError(err)
  562. }
  563. },
  564. /**
  565. * Purge history
  566. */
  567. async purgePagesHistory (obj, args, context) {
  568. try {
  569. await WIKI.db.pageHistory.purge(args.olderThan)
  570. return {
  571. responseResult: graphHelper.generateSuccess('Page history purged successfully.')
  572. }
  573. } catch (err) {
  574. return graphHelper.generateError(err)
  575. }
  576. }
  577. },
  578. Page: {
  579. async tags (obj) {
  580. return WIKI.db.pages.relatedQuery('tags').for(obj.id)
  581. }
  582. // comments(pg) {
  583. // return pg.$relatedQuery('comments')
  584. // }
  585. }
  586. }