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.

301 lines
8.6 KiB

  1. const express = require('express')
  2. const router = express.Router()
  3. const pageHelper = require('../helpers/page')
  4. const _ = require('lodash')
  5. /* global WIKI */
  6. /**
  7. * Robots.txt
  8. */
  9. router.get('/robots.txt', (req, res, next) => {
  10. res.type('text/plain')
  11. if (_.includes(WIKI.config.seo.robots, 'noindex')) {
  12. res.send('User-agent: *\nDisallow: /')
  13. } else {
  14. res.status(200).end()
  15. }
  16. })
  17. /**
  18. * Health Endpoint
  19. */
  20. router.get('/healthz', (req, res, next) => {
  21. if (WIKI.models.knex.client.pool.numFree() < 1 && WIKI.models.knex.client.pool.numUsed() < 1) {
  22. res.status(503).json({ ok: false }).end()
  23. } else {
  24. res.status(200).json({ ok: true }).end()
  25. }
  26. })
  27. /**
  28. * Administration
  29. */
  30. router.get(['/a', '/a/*'], (req, res, next) => {
  31. _.set(res.locals, 'pageMeta.title', 'Admin')
  32. res.render('admin')
  33. })
  34. /**
  35. * Create/Edit document
  36. */
  37. router.get(['/e', '/e/*'], async (req, res, next) => {
  38. const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
  39. if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
  40. return res.redirect(`/e/${pageArgs.locale}/${pageArgs.path}`)
  41. }
  42. _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
  43. _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
  44. if (pageHelper.isReservedPath(pageArgs.path)) {
  45. return next(new Error('Cannot create this page because it starts with a system reserved path.'))
  46. }
  47. let page = await WIKI.models.pages.getPageFromDb({
  48. path: pageArgs.path,
  49. locale: pageArgs.locale,
  50. userId: req.user.id,
  51. isPrivate: false
  52. })
  53. pageArgs.tags = _.get(page, 'tags', [])
  54. const injectCode = {
  55. css: WIKI.config.theming.injectCSS,
  56. head: WIKI.config.theming.injectHead,
  57. body: WIKI.config.theming.injectBody
  58. }
  59. if (page) {
  60. if (!WIKI.auth.checkAccess(req.user, ['write:pages', 'manage:pages'], pageArgs)) {
  61. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  62. return res.render('unauthorized', { action: 'edit' })
  63. }
  64. await page.$relatedQuery('tags')
  65. page.tags = _.map(page.tags, 'tag')
  66. _.set(res.locals, 'pageMeta.title', `Edit ${page.title}`)
  67. _.set(res.locals, 'pageMeta.description', page.description)
  68. page.mode = 'update'
  69. page.isPublished = (page.isPublished === true || page.isPublished === 1) ? 'true' : 'false'
  70. page.content = Buffer.from(page.content).toString('base64')
  71. } else {
  72. if (!WIKI.auth.checkAccess(req.user, ['write:pages'], pageArgs)) {
  73. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  74. return res.render('unauthorized', { action: 'create' })
  75. }
  76. _.set(res.locals, 'pageMeta.title', `New Page`)
  77. page = {
  78. path: pageArgs.path,
  79. localeCode: pageArgs.locale,
  80. editorKey: null,
  81. mode: 'create',
  82. content: null
  83. }
  84. }
  85. res.render('editor', { page, injectCode })
  86. })
  87. /**
  88. * History
  89. */
  90. router.get(['/h', '/h/*'], async (req, res, next) => {
  91. const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
  92. if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
  93. return res.redirect(`/h/${pageArgs.locale}/${pageArgs.path}`)
  94. }
  95. _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
  96. _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
  97. const page = await WIKI.models.pages.getPageFromDb({
  98. path: pageArgs.path,
  99. locale: pageArgs.locale,
  100. userId: req.user.id,
  101. isPrivate: false
  102. })
  103. pageArgs.tags = _.get(page, 'tags', [])
  104. if (!WIKI.auth.checkAccess(req.user, ['read:history'], pageArgs)) {
  105. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  106. return res.render('unauthorized', { action: 'history' })
  107. }
  108. if (page) {
  109. _.set(res.locals, 'pageMeta.title', page.title)
  110. _.set(res.locals, 'pageMeta.description', page.description)
  111. res.render('history', { page })
  112. } else {
  113. res.redirect(`/${pageArgs.path}`)
  114. }
  115. })
  116. /**
  117. * Page ID redirection
  118. */
  119. router.get(['/i', '/i/:id'], async (req, res, next) => {
  120. const pageId = _.toSafeInteger(req.params.id)
  121. if (pageId <= 0) {
  122. return res.redirect('/')
  123. }
  124. const page = await WIKI.models.pages.query().column(['path', 'localeCode', 'isPrivate', 'privateNS']).findById(pageId)
  125. if (!page) {
  126. _.set(res.locals, 'pageMeta.title', 'Page Not Found')
  127. return res.status(404).render('notfound', { action: 'view' })
  128. }
  129. if (!WIKI.auth.checkAccess(req.user, ['read:pages'], {
  130. locale: page.localeCode,
  131. path: page.path,
  132. private: page.isPrivate,
  133. privateNS: page.privateNS,
  134. explicitLocale: false,
  135. tags: page.tags
  136. })) {
  137. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  138. return res.render('unauthorized', { action: 'view' })
  139. }
  140. if (WIKI.config.lang.namespacing) {
  141. return res.redirect(`/${page.localeCode}/${page.path}`)
  142. } else {
  143. return res.redirect(`/${page.path}`)
  144. }
  145. })
  146. /**
  147. * Profile
  148. */
  149. router.get(['/p', '/p/*'], (req, res, next) => {
  150. _.set(res.locals, 'pageMeta.title', 'User Profile')
  151. res.render('profile')
  152. })
  153. /**
  154. * Source
  155. */
  156. router.get(['/s', '/s/*'], async (req, res, next) => {
  157. const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
  158. const page = await WIKI.models.pages.getPageFromDb({
  159. path: pageArgs.path,
  160. locale: pageArgs.locale,
  161. userId: req.user.id,
  162. isPrivate: false
  163. })
  164. pageArgs.tags = _.get(page, 'tags', [])
  165. if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
  166. return res.redirect(`/s/${pageArgs.locale}/${pageArgs.path}`)
  167. }
  168. _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
  169. _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
  170. if (!WIKI.auth.checkAccess(req.user, ['read:source'], pageArgs)) {
  171. return res.render('unauthorized', { action: 'source' })
  172. }
  173. if (page) {
  174. _.set(res.locals, 'pageMeta.title', page.title)
  175. _.set(res.locals, 'pageMeta.description', page.description)
  176. res.render('source', { page })
  177. } else {
  178. res.redirect(`/${pageArgs.path}`)
  179. }
  180. })
  181. /**
  182. * Tags
  183. */
  184. router.get(['/t', '/t/*'], (req, res, next) => {
  185. _.set(res.locals, 'pageMeta.title', 'Tags')
  186. res.render('tags')
  187. })
  188. /**
  189. * View document / asset
  190. */
  191. router.get('/*', async (req, res, next) => {
  192. const stripExt = _.some(WIKI.data.pageExtensions, ext => _.endsWith(req.path, `.${ext}`))
  193. const pageArgs = pageHelper.parsePath(req.path, { stripExt })
  194. const isPage = (stripExt || pageArgs.path.indexOf('.') === -1)
  195. if (isPage) {
  196. if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
  197. return res.redirect(`/${pageArgs.locale}/${pageArgs.path}`)
  198. }
  199. req.i18n.changeLanguage(pageArgs.locale)
  200. try {
  201. const page = await WIKI.models.pages.getPage({
  202. path: pageArgs.path,
  203. locale: pageArgs.locale,
  204. userId: req.user.id,
  205. isPrivate: false
  206. })
  207. pageArgs.tags = _.get(page, 'tags', [])
  208. if (!WIKI.auth.checkAccess(req.user, ['read:pages'], pageArgs)) {
  209. if (pageArgs.path === 'home') {
  210. return res.redirect('/login')
  211. }
  212. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  213. return res.status(403).render('unauthorized', {
  214. action: 'view'
  215. })
  216. }
  217. _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
  218. _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
  219. if (page) {
  220. _.set(res.locals, 'pageMeta.title', page.title)
  221. _.set(res.locals, 'pageMeta.description', page.description)
  222. const sidebar = await WIKI.models.navigation.getTree({ cache: true, locale: pageArgs.locale })
  223. const injectCode = {
  224. css: WIKI.config.theming.injectCSS,
  225. head: WIKI.config.theming.injectHead,
  226. body: WIKI.config.theming.injectBody
  227. }
  228. if (req.query.legacy || req.get('user-agent').indexOf('Trident') >= 0) {
  229. if (_.isString(page.toc)) {
  230. page.toc = JSON.parse(page.toc)
  231. }
  232. res.render('legacy/page', { page, sidebar, injectCode, isAuthenticated: req.user && req.user.id !== 2 })
  233. } else {
  234. res.render('page', { page, sidebar, injectCode })
  235. }
  236. } else if (pageArgs.path === 'home') {
  237. _.set(res.locals, 'pageMeta.title', 'Welcome')
  238. res.render('welcome', { locale: pageArgs.locale })
  239. } else {
  240. _.set(res.locals, 'pageMeta.title', 'Page Not Found')
  241. if (WIKI.auth.checkAccess(req.user, ['write:pages'], pageArgs)) {
  242. res.status(404).render('new', { path: pageArgs.path, locale: pageArgs.locale })
  243. } else {
  244. res.status(404).render('notfound', { action: 'view' })
  245. }
  246. }
  247. } catch (err) {
  248. next(err)
  249. }
  250. } else {
  251. if (!WIKI.auth.checkAccess(req.user, ['read:assets'], pageArgs)) {
  252. return res.sendStatus(403)
  253. }
  254. await WIKI.models.assets.getAsset(pageArgs.path, res)
  255. }
  256. })
  257. module.exports = router