123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589 |
- const express = require('express')
- const router = express.Router()
- const pageHelper = require('../helpers/page')
- const _ = require('lodash')
- const CleanCSS = require('clean-css')
- const moment = require('moment')
- /* global WIKI */
- const tmplCreateRegex = /^[0-9]+(,[0-9]+)?$/
- /**
- * Robots.txt
- */
- router.get('/robots.txt', (req, res, next) => {
- res.type('text/plain')
- if (_.includes(WIKI.config.seo.robots, 'noindex')) {
- res.send('User-agent: *\nDisallow: /')
- } else {
- res.status(200).end()
- }
- })
- /**
- * Health Endpoint
- */
- router.get('/healthz', (req, res, next) => {
- if (WIKI.models.knex.client.pool.numFree() < 1 && WIKI.models.knex.client.pool.numUsed() < 1) {
- res.status(503).json({ ok: false }).end()
- } else {
- res.status(200).json({ ok: true }).end()
- }
- })
- /**
- * Administration
- */
- router.get(['/a', '/a/*'], (req, res, next) => {
- if (!WIKI.auth.checkAccess(req.user, [
- 'manage:system',
- 'write:users',
- 'manage:users',
- 'write:groups',
- 'manage:groups',
- 'manage:navigation',
- 'manage:theme',
- 'manage:api'
- ])) {
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.status(403).render('unauthorized', { action: 'view' })
- }
- _.set(res.locals, 'pageMeta.title', 'Admin')
- res.render('admin')
- })
- /**
- * Download Page / Version
- */
- router.get(['/d', '/d/*'], async (req, res, next) => {
- const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
- const versionId = (req.query.v) ? _.toSafeInteger(req.query.v) : 0
- const page = await WIKI.models.pages.getPageFromDb({
- path: pageArgs.path,
- locale: pageArgs.locale,
- userId: req.user.id,
- isPrivate: false
- })
- pageArgs.tags = _.get(page, 'tags', [])
- if (versionId > 0) {
- if (!WIKI.auth.checkAccess(req.user, ['read:history'], pageArgs)) {
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.render('unauthorized', { action: 'downloadVersion' })
- }
- } else {
- if (!WIKI.auth.checkAccess(req.user, ['read:source'], pageArgs)) {
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.render('unauthorized', { action: 'download' })
- }
- }
- if (page) {
- const fileName = _.last(page.path.split('/')) + '.' + pageHelper.getFileExtension(page.contentType)
- res.attachment(fileName)
- if (versionId > 0) {
- const pageVersion = await WIKI.models.pageHistory.getVersion({ pageId: page.id, versionId })
- res.send(pageHelper.injectPageMetadata(pageVersion))
- } else {
- res.send(pageHelper.injectPageMetadata(page))
- }
- } else {
- res.status(404).end()
- }
- })
- /**
- * Create/Edit document
- */
- router.get(['/e', '/e/*'], async (req, res, next) => {
- const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
- if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
- return res.redirect(`/e/${pageArgs.locale}/${pageArgs.path}`)
- }
- req.i18n.changeLanguage(pageArgs.locale)
- // -> Set Editor Lang
- _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
- _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
- // -> Check for reserved path
- if (pageHelper.isReservedPath(pageArgs.path)) {
- return next(new Error('Cannot create this page because it starts with a system reserved path.'))
- }
- // -> Get page data from DB
- let page = await WIKI.models.pages.getPageFromDb({
- path: pageArgs.path,
- locale: pageArgs.locale,
- userId: req.user.id,
- isPrivate: false
- })
- pageArgs.tags = _.get(page, 'tags', [])
- // -> Effective Permissions
- const effectivePermissions = WIKI.auth.getEffectivePermissions(req, pageArgs)
- const injectCode = {
- css: WIKI.config.theming.injectCSS,
- head: WIKI.config.theming.injectHead,
- body: WIKI.config.theming.injectBody
- }
- if (page) {
- // -> EDIT MODE
- if (!(effectivePermissions.pages.write || effectivePermissions.pages.manage)) {
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.render('unauthorized', { action: 'edit' })
- }
- // -> Get page tags
- await page.$relatedQuery('tags')
- page.tags = _.map(page.tags, 'tag')
- // Handle missing extra field
- page.extra = page.extra || { css: '', js: '' }
- // -> Beautify Script CSS
- if (!_.isEmpty(page.extra.css)) {
- page.extra.css = new CleanCSS({ format: 'beautify' }).minify(page.extra.css).styles
- }
- _.set(res.locals, 'pageMeta.title', `Edit ${page.title}`)
- _.set(res.locals, 'pageMeta.description', page.description)
- page.mode = 'update'
- page.isPublished = (page.isPublished === true || page.isPublished === 1) ? 'true' : 'false'
- page.content = Buffer.from(page.content).toString('base64')
- } else {
- // -> CREATE MODE
- if (!effectivePermissions.pages.write) {
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.render('unauthorized', { action: 'create' })
- }
- _.set(res.locals, 'pageMeta.title', `New Page`)
- page = {
- path: pageArgs.path,
- localeCode: pageArgs.locale,
- editorKey: null,
- mode: 'create',
- content: null,
- title: null,
- description: null,
- updatedAt: new Date().toISOString(),
- tocOptions: {
- min: 1,
- max: 2,
- useDefault: true
- },
- extra: {
- css: '',
- js: ''
- }
- }
- // -> From Template
- if (req.query.from && tmplCreateRegex.test(req.query.from)) {
- let tmplPageId = 0
- let tmplVersionId = 0
- if (req.query.from.indexOf(',')) {
- const q = req.query.from.split(',')
- tmplPageId = _.toSafeInteger(q[0])
- tmplVersionId = _.toSafeInteger(q[1])
- } else {
- tmplPageId = _.toSafeInteger(req.query.from)
- }
- if (tmplVersionId > 0) {
- // -> From Page Version
- const pageVersion = await WIKI.models.pageHistory.getVersion({ pageId: tmplPageId, versionId: tmplVersionId })
- if (!pageVersion) {
- _.set(res.locals, 'pageMeta.title', 'Page Not Found')
- return res.status(404).render('notfound', { action: 'template' })
- }
- if (!WIKI.auth.checkAccess(req.user, ['read:history'], { path: pageVersion.path, locale: pageVersion.locale })) {
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.render('unauthorized', { action: 'sourceVersion' })
- }
- page.content = Buffer.from(pageVersion.content).toString('base64')
- page.editorKey = pageVersion.editor
- page.title = pageVersion.title
- page.description = pageVersion.description
- } else {
- // -> From Page Live
- const pageOriginal = await WIKI.models.pages.query().findById(tmplPageId)
- if (!pageOriginal) {
- _.set(res.locals, 'pageMeta.title', 'Page Not Found')
- return res.status(404).render('notfound', { action: 'template' })
- }
- if (!WIKI.auth.checkAccess(req.user, ['read:source'], { path: pageOriginal.path, locale: pageOriginal.locale })) {
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.render('unauthorized', { action: 'source' })
- }
- page.content = Buffer.from(pageOriginal.content).toString('base64')
- page.editorKey = pageOriginal.editorKey
- page.title = pageOriginal.title
- page.description = pageOriginal.description
- }
- }
- }
- res.render('editor', { page, injectCode, effectivePermissions })
- })
- /**
- * History
- */
- router.get(['/h', '/h/*'], async (req, res, next) => {
- const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
- if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
- return res.redirect(`/h/${pageArgs.locale}/${pageArgs.path}`)
- }
- req.i18n.changeLanguage(pageArgs.locale)
- _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
- _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
- const page = await WIKI.models.pages.getPageFromDb({
- path: pageArgs.path,
- locale: pageArgs.locale,
- userId: req.user.id,
- isPrivate: false
- })
- if (!page) {
- _.set(res.locals, 'pageMeta.title', 'Page Not Found')
- return res.status(404).render('notfound', { action: 'history' })
- }
- pageArgs.tags = _.get(page, 'tags', [])
- const effectivePermissions = WIKI.auth.getEffectivePermissions(req, pageArgs)
- if (!effectivePermissions.history.read) {
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.render('unauthorized', { action: 'history' })
- }
- if (page) {
- _.set(res.locals, 'pageMeta.title', page.title)
- _.set(res.locals, 'pageMeta.description', page.description)
- res.render('history', { page, effectivePermissions })
- } else {
- res.redirect(`/${pageArgs.path}`)
- }
- })
- /**
- * Page ID redirection
- */
- router.get(['/i', '/i/:id'], async (req, res, next) => {
- const pageId = _.toSafeInteger(req.params.id)
- if (pageId <= 0) {
- return res.redirect('/')
- }
- const page = await WIKI.models.pages.query().column(['path', 'localeCode', 'isPrivate', 'privateNS']).findById(pageId)
- if (!page) {
- _.set(res.locals, 'pageMeta.title', 'Page Not Found')
- return res.status(404).render('notfound', { action: 'view' })
- }
- if (!WIKI.auth.checkAccess(req.user, ['read:pages'], {
- locale: page.localeCode,
- path: page.path,
- private: page.isPrivate,
- privateNS: page.privateNS,
- explicitLocale: false,
- tags: page.tags
- })) {
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.render('unauthorized', { action: 'view' })
- }
- if (WIKI.config.lang.namespacing) {
- return res.redirect(`/${page.localeCode}/${page.path}`)
- } else {
- return res.redirect(`/${page.path}`)
- }
- })
- /**
- * Profile
- */
- router.get(['/p', '/p/*'], (req, res, next) => {
- if (!req.user || req.user.id < 1 || req.user.id === 2) {
- return res.render('unauthorized', { action: 'view' })
- }
- _.set(res.locals, 'pageMeta.title', 'User Profile')
- res.render('profile')
- })
- /**
- * Source
- */
- router.get(['/s', '/s/*'], async (req, res, next) => {
- const pageArgs = pageHelper.parsePath(req.path, { stripExt: true })
- const versionId = (req.query.v) ? _.toSafeInteger(req.query.v) : 0
- const page = await WIKI.models.pages.getPageFromDb({
- path: pageArgs.path,
- locale: pageArgs.locale,
- userId: req.user.id,
- isPrivate: false
- })
- pageArgs.tags = _.get(page, 'tags', [])
- if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
- return res.redirect(`/s/${pageArgs.locale}/${pageArgs.path}`)
- }
- // -> Effective Permissions
- const effectivePermissions = WIKI.auth.getEffectivePermissions(req, pageArgs)
- _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
- _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
- if (versionId > 0) {
- if (!effectivePermissions.history.read) {
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.render('unauthorized', { action: 'sourceVersion' })
- }
- } else {
- if (!effectivePermissions.source.read) {
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.render('unauthorized', { action: 'source' })
- }
- }
- if (page) {
- if (versionId > 0) {
- const pageVersion = await WIKI.models.pageHistory.getVersion({ pageId: page.id, versionId })
- _.set(res.locals, 'pageMeta.title', pageVersion.title)
- _.set(res.locals, 'pageMeta.description', pageVersion.description)
- res.render('source', {
- page: {
- ...page,
- ...pageVersion
- },
- effectivePermissions
- })
- } else {
- _.set(res.locals, 'pageMeta.title', page.title)
- _.set(res.locals, 'pageMeta.description', page.description)
- res.render('source', { page, effectivePermissions })
- }
- } else {
- res.redirect(`/${pageArgs.path}`)
- }
- })
- /**
- * Tags
- */
- router.get(['/t', '/t/*'], (req, res, next) => {
- _.set(res.locals, 'pageMeta.title', 'Tags')
- res.render('tags')
- })
- /**
- * User Avatar
- */
- router.get('/_userav/:uid', async (req, res, next) => {
- if (!WIKI.auth.checkAccess(req.user, ['read:pages'])) {
- return res.sendStatus(403)
- }
- const av = await WIKI.models.users.getUserAvatarData(req.params.uid)
- if (av) {
- res.set('Content-Type', 'image/jpeg')
- res.send(av)
- }
- return res.sendStatus(404)
- })
- /**
- * View document / asset
- */
- router.get('/*', async (req, res, next) => {
- const stripExt = _.some(WIKI.data.pageExtensions, ext => _.endsWith(req.path, `.${ext}`))
- const pageArgs = pageHelper.parsePath(req.path, { stripExt })
- const isPage = (stripExt || pageArgs.path.indexOf('.') === -1)
- if (isPage) {
- if (WIKI.config.lang.namespacing && !pageArgs.explicitLocale) {
- return res.redirect(`/${pageArgs.locale}/${pageArgs.path}`)
- }
- req.i18n.changeLanguage(pageArgs.locale)
- try {
- // -> Get Page from cache
- const page = await WIKI.models.pages.getPage({
- path: pageArgs.path,
- locale: pageArgs.locale,
- userId: req.user.id,
- isPrivate: false
- })
- pageArgs.tags = _.get(page, 'tags', [])
- // -> Effective Permissions
- const effectivePermissions = WIKI.auth.getEffectivePermissions(req, pageArgs)
- // -> Check User Access
- if (!effectivePermissions.pages.read) {
- if (req.user.id === 2) {
- res.cookie('loginRedirect', req.path, {
- maxAge: 15 * 60 * 1000
- })
- }
- if (pageArgs.path === 'home' && req.user.id === 2) {
- return res.redirect('/login')
- }
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.status(403).render('unauthorized', {
- action: 'view'
- })
- }
- _.set(res, 'locals.siteConfig.lang', pageArgs.locale)
- _.set(res, 'locals.siteConfig.rtl', req.i18n.dir() === 'rtl')
- if (page) {
- _.set(res.locals, 'pageMeta.title', page.title)
- _.set(res.locals, 'pageMeta.description', page.description)
- // -> Check Publishing State
- let pageIsPublished = page.isPublished
- if (pageIsPublished && !_.isEmpty(page.publishStartDate)) {
- pageIsPublished = moment(page.publishStartDate).isSameOrBefore()
- }
- if (pageIsPublished && !_.isEmpty(page.publishEndDate)) {
- pageIsPublished = moment(page.publishEndDate).isSameOrAfter()
- }
- if (!pageIsPublished && !effectivePermissions.pages.write) {
- _.set(res.locals, 'pageMeta.title', 'Unauthorized')
- return res.status(403).render('unauthorized', {
- action: 'view'
- })
- }
- // -> Build sidebar navigation
- let sdi = 1
- const sidebar = (await WIKI.models.navigation.getTree({ cache: true, locale: pageArgs.locale, groups: req.user.groups })).map(n => ({
- i: `sdi-${sdi++}`,
- k: n.kind,
- l: n.label,
- c: n.icon,
- y: n.targetType,
- t: n.target
- }))
- // -> Build theme code injection
- const injectCode = {
- css: WIKI.config.theming.injectCSS,
- head: WIKI.config.theming.injectHead,
- body: WIKI.config.theming.injectBody
- }
- // Handle missing extra field
- page.extra = page.extra || { css: '', js: '' }
- if (!_.isEmpty(page.extra.css)) {
- injectCode.css = `${injectCode.css}\n${page.extra.css}`
- }
- if (!_.isEmpty(page.extra.js)) {
- injectCode.body = `${injectCode.body}\n${page.extra.js}`
- }
- // -> Set TOC display options
- const tocOptions = _.get(page, 'tocOptions.useDefault', true) ? {
- min: page.tocOptions.min,
- max: page.tocOptions.max
- } : WIKI.config.theming.tocDepth
- if (req.query.legacy || req.get('user-agent').indexOf('Trident') >= 0) {
- // -> Convert page TOC
- if (_.isString(page.toc)) {
- page.toc = JSON.parse(page.toc)
- }
- // -> Render legacy view
- res.render('legacy/page', {
- page,
- sidebar,
- injectCode,
- isAuthenticated: req.user && req.user.id !== 2
- })
- } else {
- // -> Convert page TOC
- if (!_.isString(page.toc)) {
- page.toc = JSON.stringify(page.toc)
- }
- // -> Inject comments variables
- const commentTmpl = {
- codeTemplate: WIKI.data.commentProvider.codeTemplate,
- head: WIKI.data.commentProvider.head,
- body: WIKI.data.commentProvider.body,
- main: WIKI.data.commentProvider.main
- }
- if (WIKI.config.features.featurePageComments && WIKI.data.commentProvider.codeTemplate) {
- [
- { key: 'pageUrl', value: `${WIKI.config.host}/i/${page.id}` },
- { key: 'pageId', value: page.id }
- ].forEach((cfg) => {
- commentTmpl.head = _.replace(commentTmpl.head, new RegExp(`{{${cfg.key}}}`, 'g'), cfg.value)
- commentTmpl.body = _.replace(commentTmpl.body, new RegExp(`{{${cfg.key}}}`, 'g'), cfg.value)
- commentTmpl.main = _.replace(commentTmpl.main, new RegExp(`{{${cfg.key}}}`, 'g'), cfg.value)
- })
- }
- // -> Render view
- res.render('page', {
- page,
- sidebar,
- tocOptions,
- injectCode,
- comments: commentTmpl,
- effectivePermissions
- })
- }
- } else if (pageArgs.path === 'home') {
- _.set(res.locals, 'pageMeta.title', 'Welcome')
- res.render('welcome', { locale: pageArgs.locale })
- } else {
- _.set(res.locals, 'pageMeta.title', 'Page Not Found')
- if (effectivePermissions.pages.write) {
- res.status(404).render('new', { path: pageArgs.path, locale: pageArgs.locale })
- } else {
- res.status(404).render('notfound', { action: 'view' })
- }
- }
- } catch (err) {
- next(err)
- }
- } else {
- if (!WIKI.auth.checkAccess(req.user, ['read:assets'], pageArgs)) {
- return res.sendStatus(403)
- }
- await WIKI.models.assets.getAsset(pageArgs.path, res)
- }
- })
- module.exports = router
|