common.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431
  1. const express = require('express')
  2. const router = express.Router()
  3. const pageHelper = require('../helpers/page')
  4. const _ = require('lodash')
  5. /* global WIKI */
  6. const tmplCreateRegex = /^[0-9]+(,[0-9]+)?$/
  7. /**
  8. * Robots.txt
  9. */
  10. router.get('/robots.txt', (req, res, next) => {
  11. res.type('text/plain')
  12. if (_.includes(WIKI.config.seo.robots, 'noindex')) {
  13. res.send('User-agent: *\nDisallow: /')
  14. } else {
  15. res.status(200).end()
  16. }
  17. })
  18. /**
  19. * Health Endpoint
  20. */
  21. router.get('/healthz', (req, res, next) => {
  22. if (WIKI.models.knex.client.pool.numFree() < 1 && WIKI.models.knex.client.pool.numUsed() < 1) {
  23. res.status(503).json({ ok: false }).end()
  24. } else {
  25. res.status(200).json({ ok: true }).end()
  26. }
  27. })
  28. /**
  29. * Administration
  30. */
  31. router.get(['/a', '/a/*'], (req, res, next) => {
  32. _.set(res.locals, 'pageMeta.title', 'Admin')
  33. res.render('admin')
  34. })
  35. /**
  36. * Download Page / Version
  37. */
  38. router.get(['/d', '/d/*'], async (req, res, next) => {
  39. const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
  40. const versionId = (req.query.v) ? _.toSafeInteger(req.query.v) : 0
  41. const page = await WIKI.models.pages.getPageFromDb({
  42. path: pageArgs.path,
  43. locale: pageArgs.locale,
  44. userId: req.user.id,
  45. isPrivate: false
  46. })
  47. pageArgs.tags = _.get(page, 'tags', [])
  48. if (versionId > 0) {
  49. if (!WIKI.auth.checkAccess(req.user, ['read:history'], pageArgs)) {
  50. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  51. return res.render('unauthorized', { action: 'downloadVersion' })
  52. }
  53. } else {
  54. if (!WIKI.auth.checkAccess(req.user, ['read:source'], pageArgs)) {
  55. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  56. return res.render('unauthorized', { action: 'download' })
  57. }
  58. }
  59. if (page) {
  60. const fileName = _.last(page.path.split('/')) + '.' + pageHelper.getFileExtension(page.contentType)
  61. res.attachment(fileName)
  62. if (versionId > 0) {
  63. const pageVersion = await WIKI.models.pageHistory.getVersion({ pageId: page.id, versionId })
  64. res.send(pageHelper.injectPageMetadata(pageVersion))
  65. } else {
  66. res.send(pageHelper.injectPageMetadata(page))
  67. }
  68. } else {
  69. res.status(404).end()
  70. }
  71. })
  72. /**
  73. * Create/Edit document
  74. */
  75. router.get(['/e', '/e/*'], async (req, res, next) => {
  76. const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
  77. if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
  78. return res.redirect(`/e/${pageArgs.locale}/${pageArgs.path}`)
  79. }
  80. // -> Set Editor Lang
  81. _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
  82. _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
  83. // -> Check for reserved path
  84. if (pageHelper.isReservedPath(pageArgs.path)) {
  85. return next(new Error('Cannot create this page because it starts with a system reserved path.'))
  86. }
  87. // -> Get page data from DB
  88. let page = await WIKI.models.pages.getPageFromDb({
  89. path: pageArgs.path,
  90. locale: pageArgs.locale,
  91. userId: req.user.id,
  92. isPrivate: false
  93. })
  94. pageArgs.tags = _.get(page, 'tags', [])
  95. const injectCode = {
  96. css: WIKI.config.theming.injectCSS,
  97. head: WIKI.config.theming.injectHead,
  98. body: WIKI.config.theming.injectBody
  99. }
  100. if (page) {
  101. // -> EDIT MODE
  102. if (!WIKI.auth.checkAccess(req.user, ['write:pages', 'manage:pages'], pageArgs)) {
  103. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  104. return res.render('unauthorized', { action: 'edit' })
  105. }
  106. // -> Get page tags
  107. await page.$relatedQuery('tags')
  108. page.tags = _.map(page.tags, 'tag')
  109. _.set(res.locals, 'pageMeta.title', `Edit ${page.title}`)
  110. _.set(res.locals, 'pageMeta.description', page.description)
  111. page.mode = 'update'
  112. page.isPublished = (page.isPublished === true || page.isPublished === 1) ? 'true' : 'false'
  113. page.content = Buffer.from(page.content).toString('base64')
  114. } else {
  115. // -> CREATE MODE
  116. if (!WIKI.auth.checkAccess(req.user, ['write:pages'], pageArgs)) {
  117. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  118. return res.render('unauthorized', { action: 'create' })
  119. }
  120. _.set(res.locals, 'pageMeta.title', `New Page`)
  121. page = {
  122. path: pageArgs.path,
  123. localeCode: pageArgs.locale,
  124. editorKey: null,
  125. mode: 'create',
  126. content: null,
  127. title: null,
  128. description: null
  129. }
  130. // -> From Template
  131. if (req.query.from && tmplCreateRegex.test(req.query.from)) {
  132. let tmplPageId = 0
  133. let tmplVersionId = 0
  134. if (req.query.from.indexOf(',')) {
  135. const q = req.query.from.split(',')
  136. tmplPageId = _.toSafeInteger(q[0])
  137. tmplVersionId = _.toSafeInteger(q[1])
  138. } else {
  139. tmplPageId = _.toSafeInteger(req.query.from)
  140. }
  141. if (tmplVersionId > 0) {
  142. // -> From Page Version
  143. const pageVersion = await WIKI.models.pageHistory.getVersion({ pageId: tmplPageId, versionId: tmplVersionId })
  144. if (!pageVersion) {
  145. _.set(res.locals, 'pageMeta.title', 'Page Not Found')
  146. return res.status(404).render('notfound', { action: 'template' })
  147. }
  148. if (!WIKI.auth.checkAccess(req.user, ['read:history'], { path: pageVersion.path, locale: pageVersion.locale })) {
  149. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  150. return res.render('unauthorized', { action: 'sourceVersion' })
  151. }
  152. page.content = Buffer.from(pageVersion.content).toString('base64')
  153. page.editorKey = pageVersion.editor
  154. page.title = pageVersion.title
  155. page.description = pageVersion.description
  156. } else {
  157. // -> From Page Live
  158. const pageOriginal = await WIKI.models.pages.query().findById(tmplPageId)
  159. if (!pageOriginal) {
  160. _.set(res.locals, 'pageMeta.title', 'Page Not Found')
  161. return res.status(404).render('notfound', { action: 'template' })
  162. }
  163. if (!WIKI.auth.checkAccess(req.user, ['read:source'], { path: pageOriginal.path, locale: pageOriginal.locale })) {
  164. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  165. return res.render('unauthorized', { action: 'source' })
  166. }
  167. page.content = Buffer.from(pageOriginal.content).toString('base64')
  168. page.editorKey = pageOriginal.editorKey
  169. page.title = pageOriginal.title
  170. page.description = pageOriginal.description
  171. }
  172. }
  173. }
  174. res.render('editor', { page, injectCode })
  175. })
  176. /**
  177. * History
  178. */
  179. router.get(['/h', '/h/*'], async (req, res, next) => {
  180. const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
  181. if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
  182. return res.redirect(`/h/${pageArgs.locale}/${pageArgs.path}`)
  183. }
  184. _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
  185. _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
  186. const page = await WIKI.models.pages.getPageFromDb({
  187. path: pageArgs.path,
  188. locale: pageArgs.locale,
  189. userId: req.user.id,
  190. isPrivate: false
  191. })
  192. if (!page) {
  193. _.set(res.locals, 'pageMeta.title', 'Page Not Found')
  194. return res.status(404).render('notfound', { action: 'history' })
  195. }
  196. pageArgs.tags = _.get(page, 'tags', [])
  197. if (!WIKI.auth.checkAccess(req.user, ['read:history'], pageArgs)) {
  198. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  199. return res.render('unauthorized', { action: 'history' })
  200. }
  201. if (page) {
  202. _.set(res.locals, 'pageMeta.title', page.title)
  203. _.set(res.locals, 'pageMeta.description', page.description)
  204. res.render('history', { page })
  205. } else {
  206. res.redirect(`/${pageArgs.path}`)
  207. }
  208. })
  209. /**
  210. * Page ID redirection
  211. */
  212. router.get(['/i', '/i/:id'], async (req, res, next) => {
  213. const pageId = _.toSafeInteger(req.params.id)
  214. if (pageId <= 0) {
  215. return res.redirect('/')
  216. }
  217. const page = await WIKI.models.pages.query().column(['path', 'localeCode', 'isPrivate', 'privateNS']).findById(pageId)
  218. if (!page) {
  219. _.set(res.locals, 'pageMeta.title', 'Page Not Found')
  220. return res.status(404).render('notfound', { action: 'view' })
  221. }
  222. if (!WIKI.auth.checkAccess(req.user, ['read:pages'], {
  223. locale: page.localeCode,
  224. path: page.path,
  225. private: page.isPrivate,
  226. privateNS: page.privateNS,
  227. explicitLocale: false,
  228. tags: page.tags
  229. })) {
  230. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  231. return res.render('unauthorized', { action: 'view' })
  232. }
  233. if (WIKI.config.lang.namespacing) {
  234. return res.redirect(`/${page.localeCode}/${page.path}`)
  235. } else {
  236. return res.redirect(`/${page.path}`)
  237. }
  238. })
  239. /**
  240. * Profile
  241. */
  242. router.get(['/p', '/p/*'], (req, res, next) => {
  243. _.set(res.locals, 'pageMeta.title', 'User Profile')
  244. res.render('profile')
  245. })
  246. /**
  247. * Source
  248. */
  249. router.get(['/s', '/s/*'], async (req, res, next) => {
  250. const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
  251. const versionId = (req.query.v) ? _.toSafeInteger(req.query.v) : 0
  252. const page = await WIKI.models.pages.getPageFromDb({
  253. path: pageArgs.path,
  254. locale: pageArgs.locale,
  255. userId: req.user.id,
  256. isPrivate: false
  257. })
  258. pageArgs.tags = _.get(page, 'tags', [])
  259. if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
  260. return res.redirect(`/s/${pageArgs.locale}/${pageArgs.path}`)
  261. }
  262. _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
  263. _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
  264. if (versionId > 0) {
  265. if (!WIKI.auth.checkAccess(req.user, ['read:history'], pageArgs)) {
  266. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  267. return res.render('unauthorized', { action: 'sourceVersion' })
  268. }
  269. } else {
  270. if (!WIKI.auth.checkAccess(req.user, ['read:source'], pageArgs)) {
  271. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  272. return res.render('unauthorized', { action: 'source' })
  273. }
  274. }
  275. if (page) {
  276. if (versionId > 0) {
  277. const pageVersion = await WIKI.models.pageHistory.getVersion({ pageId: page.id, versionId })
  278. _.set(res.locals, 'pageMeta.title', pageVersion.title)
  279. _.set(res.locals, 'pageMeta.description', pageVersion.description)
  280. res.render('source', {
  281. page: {
  282. ...page,
  283. ...pageVersion
  284. }
  285. })
  286. } else {
  287. _.set(res.locals, 'pageMeta.title', page.title)
  288. _.set(res.locals, 'pageMeta.description', page.description)
  289. res.render('source', { page })
  290. }
  291. } else {
  292. res.redirect(`/${pageArgs.path}`)
  293. }
  294. })
  295. /**
  296. * Tags
  297. */
  298. router.get(['/t', '/t/*'], (req, res, next) => {
  299. _.set(res.locals, 'pageMeta.title', 'Tags')
  300. res.render('tags')
  301. })
  302. /**
  303. * View document / asset
  304. */
  305. router.get('/*', async (req, res, next) => {
  306. const stripExt = _.some(WIKI.data.pageExtensions, ext => _.endsWith(req.path, `.${ext}`))
  307. const pageArgs = pageHelper.parsePath(req.path, { stripExt })
  308. const isPage = (stripExt || pageArgs.path.indexOf('.') === -1)
  309. if (isPage) {
  310. if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
  311. return res.redirect(`/${pageArgs.locale}/${pageArgs.path}`)
  312. }
  313. req.i18n.changeLanguage(pageArgs.locale)
  314. try {
  315. const page = await WIKI.models.pages.getPage({
  316. path: pageArgs.path,
  317. locale: pageArgs.locale,
  318. userId: req.user.id,
  319. isPrivate: false
  320. })
  321. pageArgs.tags = _.get(page, 'tags', [])
  322. if (!WIKI.auth.checkAccess(req.user, ['read:pages'], pageArgs)) {
  323. if (req.user.id === 2) {
  324. res.cookie('loginRedirect', req.path, {
  325. maxAge: 15 * 60 * 1000
  326. })
  327. }
  328. if (pageArgs.path === 'home' && req.user.id === 2) {
  329. return res.redirect('/login')
  330. }
  331. _.set(res.locals, 'pageMeta.title', 'Unauthorized')
  332. return res.status(403).render('unauthorized', {
  333. action: 'view'
  334. })
  335. }
  336. _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
  337. _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
  338. if (page) {
  339. _.set(res.locals, 'pageMeta.title', page.title)
  340. _.set(res.locals, 'pageMeta.description', page.description)
  341. const sidebar = await WIKI.models.navigation.getTree({ cache: true, locale: pageArgs.locale })
  342. const injectCode = {
  343. css: WIKI.config.theming.injectCSS,
  344. head: WIKI.config.theming.injectHead,
  345. body: WIKI.config.theming.injectBody
  346. }
  347. if (req.query.legacy || req.get('user-agent').indexOf('Trident') >= 0) {
  348. if (_.isString(page.toc)) {
  349. page.toc = JSON.parse(page.toc)
  350. }
  351. res.render('legacy/page', { page, sidebar, injectCode, isAuthenticated: req.user && req.user.id !== 2 })
  352. } else {
  353. res.render('page', { page, sidebar, injectCode })
  354. }
  355. } else if (pageArgs.path === 'home') {
  356. _.set(res.locals, 'pageMeta.title', 'Welcome')
  357. res.render('welcome', { locale: pageArgs.locale })
  358. } else {
  359. _.set(res.locals, 'pageMeta.title', 'Page Not Found')
  360. if (WIKI.auth.checkAccess(req.user, ['write:pages'], pageArgs)) {
  361. res.status(404).render('new', { path: pageArgs.path, locale: pageArgs.locale })
  362. } else {
  363. res.status(404).render('notfound', { action: 'view' })
  364. }
  365. }
  366. } catch (err) {
  367. next(err)
  368. }
  369. } else {
  370. if (!WIKI.auth.checkAccess(req.user, ['read:assets'], pageArgs)) {
  371. return res.sendStatus(403)
  372. }
  373. await WIKI.models.assets.getAsset(pageArgs.path, res)
  374. }
  375. })
  376. module.exports = router