common.js 8.3 KB

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