common.mjs 18 KB

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