You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

167 lines
4.2 KiB

10 years ago
9 years ago
9 years ago
9 years ago
9 years ago
10 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
9 years ago
  1. #!/usr/bin/env node
  2. 'use strict'
  3. const localWebServer = require('../')
  4. const cliOptions = require('../lib/cli-options')
  5. const commandLineArgs = require('command-line-args')
  6. const commandLineUsage = require('command-line-usage')
  7. const ansi = require('ansi-escape-sequences')
  8. const loadConfig = require('config-master')
  9. const path = require('path')
  10. const os = require('os')
  11. const arrayify = require('array-back')
  12. const t = require('typical')
  13. const flatten = require('reduce-flatten')
  14. const usage = commandLineUsage(cliOptions.usageData)
  15. const stored = loadConfig('local-web-server')
  16. let options
  17. let isHttps = false
  18. try {
  19. options = collectOptions()
  20. } catch (err) {
  21. stop([ `[red]{Error}: ${err.message}`, usage ], 1)
  22. return
  23. }
  24. if (options.misc.help) {
  25. stop(usage, 0)
  26. } else if (options.misc.config) {
  27. stop(JSON.stringify(options.server, null, ' '), 0)
  28. } else {
  29. const valid = validateOptions(options)
  30. if (!valid) {
  31. /* gracefully end the process */
  32. return
  33. }
  34. const convert = require('koa-convert')
  35. const Koa = require('koa')
  36. const app = new Koa()
  37. const _use = app.use
  38. app.use = x => _use.call(app, convert(x))
  39. app.on('error', err => {
  40. if (options.server['log-format']) {
  41. console.error(ansi.format(err.message, 'red'))
  42. }
  43. const ws = localWebServer({
  44. static: {
  45. root: options.server.directory,
  46. options: {
  47. hidden: true
  48. }
  49. },
  50. serveIndex: {
  51. path: options.server.directory,
  52. options: {
  53. icons: true,
  54. hidden: true
  55. }
  56. },
  57. log: {
  58. format: options.server['log-format']
  59. },
  60. cacheControl: options.server.cacheControl,
  61. compress: options.server.compress,
  62. mime: options.server.mime,
  63. forbid: options.server.forbid,
  64. spa: options.server.spa,
  65. 'no-cache': options.server['no-cache'],
  66. rewrite: options.server.rewrite,
  67. verbose: options.server.verbose,
  68. mocks: options.server.mocks
  69. })
  70. app.use(ws)
  71. if (options.server.https) {
  72. options.server.key = path.resolve(__dirname, '..', 'ssl', '127.0.0.1.key')
  73. options.server.cert = path.resolve(__dirname, '..', 'ssl', '127.0.0.1.crt')
  74. }
  75. if (options.server.key && options.server.cert) {
  76. const https = require('https')
  77. const fs = require('fs')
  78. isHttps = true
  79. const serverOptions = {
  80. key: fs.readFileSync(options.server.key),
  81. cert: fs.readFileSync(options.server.cert)
  82. }
  83. const server = https.createServer(serverOptions, app.callback())
  84. server.listen(options.server.port, onServerUp)
  85. } else {
  86. app.listen(options.server.port, onServerUp)
  87. }
  88. }
  89. function stop (msgs, exitCode) {
  90. arrayify(msgs).forEach(msg => console.error(ansi.format(msg)))
  91. process.exitCode = exitCode
  92. }
  93. function onServerUp () {
  94. let ipList = Object.keys(os.networkInterfaces())
  95. .map(key => os.networkInterfaces()[key])
  96. .reduce(flatten, [])
  97. .filter(iface => iface.family === 'IPv4')
  98. ipList.unshift({ address: os.hostname() })
  99. ipList = ipList
  100. .map(iface => `[underline]{${isHttps ? 'https' : 'http'}://${iface.address}:${options.server.port}}`)
  101. .join(', ')
  102. console.error(ansi.format(
  103. path.resolve(options.server.directory) === process.cwd()
  104. ? `serving at ${ipList}`
  105. : `serving [underline]{${options.server.directory}} at ${ipList}`
  106. ))
  107. }
  108. function collectOptions () {
  109. let options = {}
  110. /* parse command line args */
  111. options = commandLineArgs(cliOptions.definitions)
  112. const builtIn = {
  113. port: 8000,
  114. directory: process.cwd(),
  115. forbid: [],
  116. rewrite: []
  117. }
  118. if (options.server.rewrite) {
  119. options.server.rewrite = parseRewriteRules(options.server.rewrite)
  120. }
  121. /* override built-in defaults with stored config and then command line args */
  122. options.server = Object.assign(builtIn, stored, options.server)
  123. return options
  124. }
  125. function parseRewriteRules (rules) {
  126. return rules && rules.map(rule => {
  127. const matches = rule.match(/(\S*)\s*->\s*(\S*)/)
  128. return {
  129. from: matches[1],
  130. to: matches[2]
  131. }
  132. })
  133. }
  134. function validateOptions (options) {
  135. let valid = true
  136. function invalid (msg) {
  137. return `[red underline]{Invalid:} [bold]{${msg}}`
  138. }
  139. if (!t.isNumber(options.server.port)) {
  140. stop([ invalid(`--port must be numeric`), usage ], 1)
  141. valid = false
  142. }
  143. return valid
  144. }