You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

584 lines
18 KiB

  1. const express = require('express')
  2. const router = express.Router()
  3. const pageHelper = require('../helpers/page')
  4. const _ = require('lodash')
  5. const CleanCSS = require('clean-css')
  6. const moment = require('moment')
  7. const qs = require('querystring')
  8. /* global WIKI */
  9. const tmplCreateRegex = /^[0-9]+(,[0-9]+)?$/
  10. /**
  11. * Robots.txt
  12. */
  13. router.get('/robots.txt', (req, res, next) => {
  14. res.type('text/plain')
  15. if (_.includes(WIKI.config.seo.robots, 'noindex')) {
  16. res.send('User-agent: *\nDisallow: /')
  17. } else {
  18. res.status(200).end()
  19. }
  20. })
  21. /**
  22. * Health Endpoint
  23. */
  24. router.get('/healthz', (req, res, next) => {
  25. if (WIKI.models.knex.client.pool.numFree() < 1 && WIKI.models.knex.client.pool.numUsed() < 1) {
  26. res.status(503).json({ ok: false }).end()
  27. } else {
  28. res.status(200).json({ ok: true }).end()
  29. }
  30. })
  31. /**
  32. * Administration
  33. */
  34. router.get(['/a', '/a/*'], (req, res, next) => {
  35. if (!WIKI.auth.checkAccess(req.user, [
  36. 'manage:system',
  37. 'write:users',
  38. 'manage:users',
  39. 'write:groups',
  40. 'manage:groups',
  41. 'manage:navigation',
  42. 'manage:theme',
  43. 'manage:api'
  44. ])) {
  45. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  46. return res.status(403).render('unauthorized', { action: 'view' })
  47. }
  48. _.set(res.locals, 'pageMeta.title', 'Admin')
  49. res.render('admin')
  50. })
  51. /**
  52. * Download Page / Version
  53. */
  54. router.get(['/d', '/d/*'], async (req, res, next) => {
  55. const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
  56. const versionId = (req.query.v) ? _.toSafeInteger(req.query.v) : 0
  57. const page = await WIKI.models.pages.getPageFromDb({
  58. path: pageArgs.path,
  59. locale: pageArgs.locale,
  60. userId: req.user.id,
  61. isPrivate: false
  62. })
  63. pageArgs.tags = _.get(page, 'tags', [])
  64. if (versionId > 0) {
  65. if (!WIKI.auth.checkAccess(req.user, ['read:history'], pageArgs)) {
  66. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  67. return res.render('unauthorized', { action: 'downloadVersion' })
  68. }
  69. } else {
  70. if (!WIKI.auth.checkAccess(req.user, ['read:source'], pageArgs)) {
  71. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  72. return res.render('unauthorized', { action: 'download' })
  73. }
  74. }
  75. if (page) {
  76. const fileName = _.last(page.path.split('/')) + '.' + pageHelper.getFileExtension(page.contentType)
  77. res.attachment(fileName)
  78. if (versionId > 0) {
  79. const pageVersion = await WIKI.models.pageHistory.getVersion({ pageId: page.id, versionId })
  80. res.send(pageHelper.injectPageMetadata(pageVersion))
  81. } else {
  82. res.send(pageHelper.injectPageMetadata(page))
  83. }
  84. } else {
  85. res.status(404).end()
  86. }
  87. })
  88. /**
  89. * Create/Edit document
  90. */
  91. router.get(['/e', '/e/*'], async (req, res, next) => {
  92. const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
  93. if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
  94. return res.redirect(`/e/${pageArgs.locale}/${pageArgs.path}`)
  95. }
  96. req.i18n.changeLanguage(pageArgs.locale)
  97. // -> Set Editor Lang
  98. _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
  99. _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
  100. // -> Check for reserved path
  101. if (pageHelper.isReservedPath(pageArgs.path)) {
  102. return next(new Error('Cannot create this page because it starts with a system reserved path.'))
  103. }
  104. // -> Get page data from DB
  105. let page = await WIKI.models.pages.getPageFromDb({
  106. path: pageArgs.path,
  107. locale: pageArgs.locale,
  108. userId: req.user.id,
  109. isPrivate: false
  110. })
  111. pageArgs.tags = _.get(page, 'tags', [])
  112. // -> Effective Permissions
  113. const effectivePermissions = WIKI.auth.getEffectivePermissions(req, pageArgs)
  114. const injectCode = {
  115. css: WIKI.config.theming.injectCSS,
  116. head: WIKI.config.theming.injectHead,
  117. body: WIKI.config.theming.injectBody
  118. }
  119. if (page) {
  120. // -> EDIT MODE
  121. if (!(effectivePermissions.pages.write || effectivePermissions.pages.manage)) {
  122. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  123. return res.render('unauthorized', { action: 'edit' })
  124. }
  125. // -> Get page tags
  126. await page.$relatedQuery('tags')
  127. page.tags = _.map(page.tags, 'tag')
  128. // Handle missing extra field
  129. page.extra = page.extra || { css: '', js: '' }
  130. // -> Beautify Script CSS
  131. if (!_.isEmpty(page.extra.css)) {
  132. page.extra.css = new CleanCSS({ format: 'beautify' }).minify(page.extra.css).styles
  133. }
  134. _.set(res.locals, 'pageMeta.title', `Edit ${page.title}`)
  135. _.set(res.locals, 'pageMeta.description', page.description)
  136. page.mode = 'update'
  137. page.isPublished = (page.isPublished === true || page.isPublished === 1) ? 'true' : 'false'
  138. page.content = Buffer.from(page.content).toString('base64')
  139. } else {
  140. // -> CREATE MODE
  141. if (!effectivePermissions.pages.write) {
  142. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  143. return res.render('unauthorized', { action: 'create' })
  144. }
  145. _.set(res.locals, 'pageMeta.title', `New Page`)
  146. page = {
  147. path: pageArgs.path,
  148. localeCode: pageArgs.locale,
  149. editorKey: null,
  150. mode: 'create',
  151. content: null,
  152. title: null,
  153. description: null,
  154. updatedAt: new Date().toISOString(),
  155. extra: {
  156. css: '',
  157. js: ''
  158. }
  159. }
  160. // -> From Template
  161. if (req.query.from && tmplCreateRegex.test(req.query.from)) {
  162. let tmplPageId = 0
  163. let tmplVersionId = 0
  164. if (req.query.from.indexOf(',')) {
  165. const q = req.query.from.split(',')
  166. tmplPageId = _.toSafeInteger(q[0])
  167. tmplVersionId = _.toSafeInteger(q[1])
  168. } else {
  169. tmplPageId = _.toSafeInteger(req.query.from)
  170. }
  171. if (tmplVersionId > 0) {
  172. // -> From Page Version
  173. const pageVersion = await WIKI.models.pageHistory.getVersion({ pageId: tmplPageId, versionId: tmplVersionId })
  174. if (!pageVersion) {
  175. _.set(res.locals, 'pageMeta.title', 'Page Not Found')
  176. return res.status(404).render('notfound', { action: 'template' })
  177. }
  178. if (!WIKI.auth.checkAccess(req.user, ['read:history'], { path: pageVersion.path, locale: pageVersion.locale })) {
  179. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  180. return res.render('unauthorized', { action: 'sourceVersion' })
  181. }
  182. page.content = Buffer.from(pageVersion.content).toString('base64')
  183. page.editorKey = pageVersion.editor
  184. page.title = pageVersion.title
  185. page.description = pageVersion.description
  186. } else {
  187. // -> From Page Live
  188. const pageOriginal = await WIKI.models.pages.query().findById(tmplPageId)
  189. if (!pageOriginal) {
  190. _.set(res.locals, 'pageMeta.title', 'Page Not Found')
  191. return res.status(404).render('notfound', { action: 'template' })
  192. }
  193. if (!WIKI.auth.checkAccess(req.user, ['read:source'], { path: pageOriginal.path, locale: pageOriginal.locale })) {
  194. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  195. return res.render('unauthorized', { action: 'source' })
  196. }
  197. page.content = Buffer.from(pageOriginal.content).toString('base64')
  198. page.editorKey = pageOriginal.editorKey
  199. page.title = pageOriginal.title
  200. page.description = pageOriginal.description
  201. }
  202. }
  203. }
  204. res.render('editor', { page, injectCode, effectivePermissions })
  205. })
  206. /**
  207. * History
  208. */
  209. router.get(['/h', '/h/*'], async (req, res, next) => {
  210. const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
  211. if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
  212. return res.redirect(`/h/${pageArgs.locale}/${pageArgs.path}`)
  213. }
  214. req.i18n.changeLanguage(pageArgs.locale)
  215. _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
  216. _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
  217. const page = await WIKI.models.pages.getPageFromDb({
  218. path: pageArgs.path,
  219. locale: pageArgs.locale,
  220. userId: req.user.id,
  221. isPrivate: false
  222. })
  223. if (!page) {
  224. _.set(res.locals, 'pageMeta.title', 'Page Not Found')
  225. return res.status(404).render('notfound', { action: 'history' })
  226. }
  227. pageArgs.tags = _.get(page, 'tags', [])
  228. const effectivePermissions = WIKI.auth.getEffectivePermissions(req, pageArgs)
  229. if (!effectivePermissions.history.read) {
  230. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  231. return res.render('unauthorized', { action: 'history' })
  232. }
  233. if (page) {
  234. _.set(res.locals, 'pageMeta.title', page.title)
  235. _.set(res.locals, 'pageMeta.description', page.description)
  236. res.render('history', { page, effectivePermissions })
  237. } else {
  238. res.redirect(`/${pageArgs.path}`)
  239. }
  240. })
  241. /**
  242. * Page ID redirection
  243. */
  244. router.get(['/i', '/i/:id'], async (req, res, next) => {
  245. const pageId = _.toSafeInteger(req.params.id)
  246. if (pageId <= 0) {
  247. return res.redirect('/')
  248. }
  249. const page = await WIKI.models.pages.query().column(['path', 'localeCode', 'isPrivate', 'privateNS']).findById(pageId)
  250. if (!page) {
  251. _.set(res.locals, 'pageMeta.title', 'Page Not Found')
  252. return res.status(404).render('notfound', { action: 'view' })
  253. }
  254. if (!WIKI.auth.checkAccess(req.user, ['read:pages'], {
  255. locale: page.localeCode,
  256. path: page.path,
  257. private: page.isPrivate,
  258. privateNS: page.privateNS,
  259. explicitLocale: false,
  260. tags: page.tags
  261. })) {
  262. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  263. return res.render('unauthorized', { action: 'view' })
  264. }
  265. if (WIKI.config.lang.namespacing) {
  266. return res.redirect(`/${page.localeCode}/${page.path}`)
  267. } else {
  268. return res.redirect(`/${page.path}`)
  269. }
  270. })
  271. /**
  272. * Profile
  273. */
  274. router.get(['/p', '/p/*'], (req, res, next) => {
  275. if (!req.user || req.user.id < 1 || req.user.id === 2) {
  276. return res.render('unauthorized', { action: 'view' })
  277. }
  278. _.set(res.locals, 'pageMeta.title', 'User Profile')
  279. res.render('profile')
  280. })
  281. /**
  282. * Source
  283. */
  284. router.get(['/s', '/s/*'], async (req, res, next) => {
  285. const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
  286. const versionId = (req.query.v) ? _.toSafeInteger(req.query.v) : 0
  287. const page = await WIKI.models.pages.getPageFromDb({
  288. path: pageArgs.path,
  289. locale: pageArgs.locale,
  290. userId: req.user.id,
  291. isPrivate: false
  292. })
  293. pageArgs.tags = _.get(page, 'tags', [])
  294. if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
  295. return res.redirect(`/s/${pageArgs.locale}/${pageArgs.path}`)
  296. }
  297. // -> Effective Permissions
  298. const effectivePermissions = WIKI.auth.getEffectivePermissions(req, pageArgs)
  299. _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
  300. _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
  301. if (versionId > 0) {
  302. if (!effectivePermissions.history.read) {
  303. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  304. return res.render('unauthorized', { action: 'sourceVersion' })
  305. }
  306. } else {
  307. if (!effectivePermissions.source.read) {
  308. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  309. return res.render('unauthorized', { action: 'source' })
  310. }
  311. }
  312. if (page) {
  313. if (versionId > 0) {
  314. const pageVersion = await WIKI.models.pageHistory.getVersion({ pageId: page.id, versionId })
  315. _.set(res.locals, 'pageMeta.title', pageVersion.title)
  316. _.set(res.locals, 'pageMeta.description', pageVersion.description)
  317. res.render('source', {
  318. page: {
  319. ...page,
  320. ...pageVersion
  321. },
  322. effectivePermissions
  323. })
  324. } else {
  325. _.set(res.locals, 'pageMeta.title', page.title)
  326. _.set(res.locals, 'pageMeta.description', page.description)
  327. res.render('source', { page, effectivePermissions })
  328. }
  329. } else {
  330. res.redirect(`/${pageArgs.path}`)
  331. }
  332. })
  333. /**
  334. * Tags
  335. */
  336. router.get(['/t', '/t/*'], (req, res, next) => {
  337. _.set(res.locals, 'pageMeta.title', 'Tags')
  338. res.render('tags')
  339. })
  340. /**
  341. * User Avatar
  342. */
  343. router.get('/_userav/:uid', async (req, res, next) => {
  344. if (!WIKI.auth.checkAccess(req.user, ['read:pages'])) {
  345. return res.sendStatus(403)
  346. }
  347. const av = await WIKI.models.users.getUserAvatarData(req.params.uid)
  348. if (av) {
  349. res.set('Content-Type', 'image/jpeg')
  350. res.send(av)
  351. }
  352. return res.sendStatus(404)
  353. })
  354. /**
  355. * View document / asset
  356. */
  357. router.get('/*', async (req, res, next) => {
  358. const stripExt = _.some(WIKI.config.pageExtensions, ext => _.endsWith(req.path, `.${ext}`))
  359. const pageArgs = pageHelper.parsePath(req.path, { stripExt })
  360. const isPage = (stripExt || pageArgs.path.indexOf('.') === -1)
  361. if (isPage) {
  362. if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
  363. const query = !_.isEmpty(req.query) ? `?${qs.stringify(req.query)}` : ''
  364. return res.redirect(`/${pageArgs.locale}/${pageArgs.path}${query}`)
  365. }
  366. req.i18n.changeLanguage(pageArgs.locale)
  367. try {
  368. // -> Get Page from cache
  369. const page = await WIKI.models.pages.getPage({
  370. path: pageArgs.path,
  371. locale: pageArgs.locale,
  372. userId: req.user.id,
  373. isPrivate: false
  374. })
  375. pageArgs.tags = _.get(page, 'tags', [])
  376. // -> Effective Permissions
  377. const effectivePermissions = WIKI.auth.getEffectivePermissions(req, pageArgs)
  378. // -> Check User Access
  379. if (!effectivePermissions.pages.read) {
  380. if (req.user.id === 2) {
  381. res.cookie('loginRedirect', req.path, {
  382. maxAge: 15 * 60 * 1000
  383. })
  384. }
  385. if (pageArgs.path === 'home' && req.user.id === 2) {
  386. return res.redirect('/login')
  387. }
  388. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  389. return res.status(403).render('unauthorized', {
  390. action: 'view'
  391. })
  392. }
  393. _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
  394. _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
  395. if (page) {
  396. _.set(res.locals, 'pageMeta.title', page.title)
  397. _.set(res.locals, 'pageMeta.description', page.description)
  398. // -> Check Publishing State
  399. let pageIsPublished = page.isPublished
  400. if (pageIsPublished && !_.isEmpty(page.publishStartDate)) {
  401. pageIsPublished = moment(page.publishStartDate).isSameOrBefore()
  402. }
  403. if (pageIsPublished && !_.isEmpty(page.publishEndDate)) {
  404. pageIsPublished = moment(page.publishEndDate).isSameOrAfter()
  405. }
  406. if (!pageIsPublished && !effectivePermissions.pages.write) {
  407. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  408. return res.status(403).render('unauthorized', {
  409. action: 'view'
  410. })
  411. }
  412. // -> Build sidebar navigation
  413. let sdi = 1
  414. const sidebar = (await WIKI.models.navigation.getTree({ cache: true, locale: pageArgs.locale, groups: req.user.groups })).map(n => ({
  415. i: `sdi-${sdi++}`,
  416. k: n.kind,
  417. l: n.label,
  418. c: n.icon,
  419. y: n.targetType,
  420. t: n.target
  421. }))
  422. // -> Build theme code injection
  423. const injectCode = {
  424. css: WIKI.config.theming.injectCSS,
  425. head: WIKI.config.theming.injectHead,
  426. body: WIKI.config.theming.injectBody
  427. }
  428. // Handle missing extra field
  429. page.extra = page.extra || { css: '', js: '' }
  430. if (!_.isEmpty(page.extra.css)) {
  431. injectCode.css = `${injectCode.css}\n${page.extra.css}`
  432. }
  433. if (!_.isEmpty(page.extra.js)) {
  434. injectCode.body = `${injectCode.body}\n${page.extra.js}`
  435. }
  436. if (req.query.legacy || req.get('user-agent').indexOf('Trident') >= 0) {
  437. // -> Convert page TOC
  438. if (_.isString(page.toc)) {
  439. page.toc = JSON.parse(page.toc)
  440. }
  441. // -> Render legacy view
  442. res.render('legacy/page', {
  443. page,
  444. sidebar,
  445. injectCode,
  446. isAuthenticated: req.user && req.user.id !== 2
  447. })
  448. } else {
  449. // -> Convert page TOC
  450. if (!_.isString(page.toc)) {
  451. page.toc = JSON.stringify(page.toc)
  452. }
  453. // -> Inject comments variables
  454. const commentTmpl = {
  455. codeTemplate: WIKI.data.commentProvider.codeTemplate,
  456. head: WIKI.data.commentProvider.head,
  457. body: WIKI.data.commentProvider.body,
  458. main: WIKI.data.commentProvider.main
  459. }
  460. if (WIKI.config.features.featurePageComments && WIKI.data.commentProvider.codeTemplate) {
  461. [
  462. { key: 'pageUrl', value: `${WIKI.config.host}/i/${page.id}` },
  463. { key: 'pageId', value: page.id }
  464. ].forEach((cfg) => {
  465. commentTmpl.head = _.replace(commentTmpl.head, new RegExp(`{{${cfg.key}}}`, 'g'), cfg.value)
  466. commentTmpl.body = _.replace(commentTmpl.body, new RegExp(`{{${cfg.key}}}`, 'g'), cfg.value)
  467. commentTmpl.main = _.replace(commentTmpl.main, new RegExp(`{{${cfg.key}}}`, 'g'), cfg.value)
  468. })
  469. }
  470. // -> Page Filename (for edit on external repo button)
  471. let pageFilename = WIKI.config.lang.namespacing ? `${pageArgs.locale}/${page.path}` : page.path
  472. pageFilename += page.contentType === 'markdown' ? '.md' : '.html'
  473. // -> Render view
  474. res.render('page', {
  475. page,
  476. sidebar,
  477. injectCode,
  478. comments: commentTmpl,
  479. effectivePermissions,
  480. pageFilename
  481. })
  482. }
  483. } else if (pageArgs.path === 'home') {
  484. _.set(res.locals, 'pageMeta.title', 'Welcome')
  485. res.render('welcome', { locale: pageArgs.locale })
  486. } else {
  487. _.set(res.locals, 'pageMeta.title', 'Page Not Found')
  488. if (effectivePermissions.pages.write) {
  489. res.status(404).render('new', { path: pageArgs.path, locale: pageArgs.locale })
  490. } else {
  491. res.status(404).render('notfound', { action: 'view' })
  492. }
  493. }
  494. } catch (err) {
  495. next(err)
  496. }
  497. } else {
  498. if (!WIKI.auth.checkAccess(req.user, ['read:assets'], pageArgs)) {
  499. return res.sendStatus(403)
  500. }
  501. await WIKI.models.assets.getAsset(pageArgs.path, res)
  502. }
  503. })
  504. module.exports = router