page.js 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. const qs = require('querystring')
  2. const _ = require('lodash')
  3. const crypto = require('crypto')
  4. const path = require('path')
  5. const localeSegmentRegex = /^[A-Z]{2}(-[A-Z]{2})?$/i
  6. const localeFolderRegex = /^([a-z]{2}(?:-[a-z]{2})?\/)?(.*)/i
  7. // eslint-disable-next-line no-control-regex
  8. const unsafeCharsRegex = /[\x00-\x1f\x80-\x9f\\"|<>:*?]/
  9. const contentToExt = {
  10. markdown: 'md',
  11. html: 'html'
  12. }
  13. const extToContent = _.invert(contentToExt)
  14. /* global WIKI */
  15. module.exports = {
  16. /**
  17. * Parse raw url path and make it safe
  18. */
  19. parsePath (rawPath, opts = {}) {
  20. let pathObj = {
  21. locale: WIKI.config.lang.code,
  22. path: 'home',
  23. private: false,
  24. privateNS: '',
  25. explicitLocale: false
  26. }
  27. // Clean Path
  28. rawPath = _.trim(qs.unescape(rawPath))
  29. if (_.startsWith(rawPath, '/')) { rawPath = rawPath.substring(1) }
  30. rawPath = rawPath.replace(unsafeCharsRegex, '')
  31. if (rawPath === '') { rawPath = 'home' }
  32. rawPath = rawPath.replace(/\\/g, '').replace(/\/\//g, '').replace(/\.\.+/ig, '')
  33. // Extract Info
  34. let pathParts = _.filter(_.split(rawPath, '/'), p => {
  35. p = _.trim(p)
  36. return !_.isEmpty(p) && p !== '..' && p !== '.'
  37. })
  38. if (pathParts[0].length === 1) {
  39. pathParts.shift()
  40. }
  41. if (localeSegmentRegex.test(pathParts[0])) {
  42. pathObj.locale = pathParts[0]
  43. pathObj.explicitLocale = true
  44. pathParts.shift()
  45. }
  46. // Strip extension
  47. if (opts.stripExt && pathParts.length > 0) {
  48. const lastPart = _.last(pathParts)
  49. if (lastPart.indexOf('.') > 0) {
  50. pathParts.pop()
  51. const lastPartMeta = path.parse(lastPart)
  52. pathParts.push(lastPartMeta.name)
  53. }
  54. }
  55. pathObj.path = _.join(pathParts, '/')
  56. return pathObj
  57. },
  58. /**
  59. * Generate unique hash from page
  60. */
  61. generateHash(opts) {
  62. return crypto.createHash('sha1').update(`${opts.locale}|${opts.path}|${opts.privateNS}`).digest('hex')
  63. },
  64. /**
  65. * Inject Page Metadata
  66. */
  67. injectPageMetadata(page) {
  68. let meta = [
  69. ['title', page.title],
  70. ['description', page.description],
  71. ['published', page.isPublished.toString()],
  72. ['date', page.updatedAt],
  73. ['tags', page.tags ? page.tags.map(t => t.tag).join(', ') : ''],
  74. ['editor', page.editorKey],
  75. ['dateCreated', page.createdAt],
  76. ['doUseTocDefault', page.doUseTocDefault]
  77. ]
  78. if (page.minTocLevel) {
  79. meta.push(['minTocLevel', page.minTocLevel])
  80. }
  81. if (page.tocLevel) {
  82. meta.push(['tocLevel', page.tocLevel])
  83. }
  84. if (page.tocCollapseLevel) {
  85. meta.push(['tocCollapseLevel', page.tocCollapseLevel])
  86. }
  87. switch (page.contentType) {
  88. case 'markdown':
  89. return '---\n' + meta.map(mt => `${mt[0]}: ${mt[1]}`).join('\n') + '\n---\n\n' + page.content
  90. case 'html':
  91. return '<!--\n' + meta.map(mt => `${mt[0]}: ${mt[1]}`).join('\n') + '\n-->\n\n' + page.content
  92. case 'json':
  93. return {
  94. ...page.content,
  95. _meta: _.fromPairs(meta)
  96. }
  97. default:
  98. return page.content
  99. }
  100. },
  101. /**
  102. * Check if path is a reserved path
  103. */
  104. isReservedPath(rawPath) {
  105. const firstSection = _.head(rawPath.split('/'))
  106. if (firstSection.length <= 1) {
  107. return true
  108. } else if (localeSegmentRegex.test(firstSection)) {
  109. return true
  110. } else if (
  111. _.some(WIKI.data.reservedPaths, p => {
  112. return p === firstSection
  113. })) {
  114. return true
  115. } else {
  116. return false
  117. }
  118. },
  119. /**
  120. * Get file extension from content type
  121. */
  122. getFileExtension(contentType) {
  123. return _.get(contentToExt, contentType, 'txt')
  124. },
  125. /**
  126. * Get content type from file extension
  127. */
  128. getContentType (filePath) {
  129. const ext = _.last(filePath.split('.'))
  130. return _.get(extToContent, ext, false)
  131. },
  132. /**
  133. * Get Page Meta object from disk path
  134. */
  135. getPagePath (filePath) {
  136. let fpath = filePath
  137. if (process.platform === 'win32') {
  138. fpath = filePath.replace(/\\/g, '/')
  139. }
  140. let meta = {
  141. locale: WIKI.config.lang.code,
  142. path: _.initial(fpath.split('.')).join('')
  143. }
  144. const result = localeFolderRegex.exec(meta.path)
  145. if (result[1]) {
  146. meta = {
  147. locale: result[1].replace('/', ''),
  148. path: result[2]
  149. }
  150. }
  151. return meta
  152. }
  153. }