├── .editorconfig ├── .github └── FUNDING.yml ├── .gitignore ├── .husky ├── .gitignore └── pre-commit ├── .prettierignore ├── .prettierrc ├── .travis.yml ├── .vscode └── launch.json ├── .yarnrc ├── CHANGELOG.md ├── CONTRIBUTING.md ├── LICENSE ├── README.md ├── assets ├── images │ └── here.jpg └── psd │ └── here.psd ├── constant ├── config.js └── log-prefix.js ├── debug.js ├── index.js ├── lib ├── application.js ├── build-file-list.js ├── config.js ├── fallback-content-type.js ├── get-file-stat.js ├── read-file.js ├── read-folder.js ├── server.js └── watcher.js ├── middleware ├── error.js ├── file-explorer.js ├── gzip.js ├── live-reload.js ├── load-route.js └── log.js ├── package-lock.json ├── package.json ├── renovate.json ├── res ├── list.pug └── reload.js └── test ├── command.js ├── file ├── css │ └── index.css ├── folder │ ├── ddd.html │ ├── icon-search.png │ └── 搜索框.png ├── here.js ├── index.html ├── js │ └── index.js └── mock-server │ ├── ajax │ └── test.json └── test └── commander.js /.editorconfig: -------------------------------------------------------------------------------- 1 | root = true 2 | 3 | [*] 4 | indent_style = space 5 | indent_size = 2 6 | end_of_line = lf 7 | charset = utf-8 8 | trim_trailing_whitespace = true 9 | insert_final_newline = true 10 | curly_bracket_next_line = false 11 | spaces_around_operators = true 12 | indent_brace_style = 1tbs 13 | 14 | [*.js] 15 | quote_type = single 16 | -------------------------------------------------------------------------------- /.github/FUNDING.yml: -------------------------------------------------------------------------------- 1 | patreon: vivaxy 2 | open_collective: vivaxy_personal 3 | custom: ['https://gist.github.com/vivaxy/58eed1803a2eddda05c90aed99430de2'] 4 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .idea 2 | npm-debug.log 3 | node_modules 4 | .DS_Store 5 | yarn.lock 6 | -------------------------------------------------------------------------------- /.husky/.gitignore: -------------------------------------------------------------------------------- 1 | _ -------------------------------------------------------------------------------- /.husky/pre-commit: -------------------------------------------------------------------------------- 1 | #!/bin/sh 2 | . "$(dirname "$0")/_/husky.sh" 3 | 4 | npx lint-staged 5 | -------------------------------------------------------------------------------- /.prettierignore: -------------------------------------------------------------------------------- 1 | CHANGELOG.md 2 | -------------------------------------------------------------------------------- /.prettierrc: -------------------------------------------------------------------------------- 1 | { 2 | "singleQuote": true, 3 | "trailingComma": "es5", 4 | "arrowParens": "always" 5 | } -------------------------------------------------------------------------------- /.travis.yml: -------------------------------------------------------------------------------- 1 | language: node_js 2 | node_js: 3 | - "node" 4 | - "lts/*" 5 | os: osx 6 | -------------------------------------------------------------------------------- /.vscode/launch.json: -------------------------------------------------------------------------------- 1 | { 2 | // Use IntelliSense to learn about possible attributes. 3 | // Hover to view descriptions of existing attributes. 4 | // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 5 | "version": "0.2.0", 6 | "configurations": [ 7 | { 8 | "type": "node", 9 | "request": "launch", 10 | "name": "Launch Program", 11 | "skipFiles": [ 12 | "/**" 13 | ], 14 | "program": "${workspaceFolder}/index.js" 15 | } 16 | ] 17 | } 18 | -------------------------------------------------------------------------------- /.yarnrc: -------------------------------------------------------------------------------- 1 | registry "https://registry.npmjs.org/" 2 | -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | # Changelog 2 | 3 | All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines. 4 | 5 | ## [3.5.0](https://github.com/vivaxy/here/compare/v3.4.1...v3.5.0) (2024-05-23) 6 | 7 | 8 | ### Features 9 | 10 | * **ip:** add ip option ([703d306](https://github.com/vivaxy/here/commit/703d306ef2ea4b36170f6b07e7752d320eb9f1f7)) 11 | 12 | ### [3.4.1](https://github.com/vivaxy/here/compare/v3.4.0...v3.4.1) (2021-02-07) 13 | 14 | 15 | ### Bug Fixes 16 | 17 | * **deps:** update dependency commander to v7 ([342a2bb](https://github.com/vivaxy/here/commit/342a2bb9a3552a692a78479c6c61cfc04acb0ff2)) 18 | * **deps:** update dependency koa-compress to v4 ([a74ddd4](https://github.com/vivaxy/here/commit/a74ddd4b2bad0aa74a4ecf663289cdf3d43633c1)) 19 | * **deps:** update dependency koa-router to v8 ([27ceb5d](https://github.com/vivaxy/here/commit/27ceb5db4103afe28863a5a58e2ac85df8c2b1a3)) 20 | * **deps:** update dependency mime to v2 ([af7298d](https://github.com/vivaxy/here/commit/af7298dcf13bbe2192c56d62ae840d12657af4d4)) 21 | * :bug: fix file mime and commander options ([e419d5f](https://github.com/vivaxy/here/commit/e419d5f1662c55eae0cadc182d890ac0c22c24a6)) 22 | 23 | 24 | # [3.4.0](https://github.com/vivaxy/here/compare/v3.3.0...v3.4.0) (2019-07-10) 25 | 26 | 27 | ### Features 28 | 29 | * **file explorer:** :sparkles: add Access-Control-Allow-Origin * for all files ([787cae9](https://github.com/vivaxy/here/commit/787cae9)) 30 | 31 | 32 | 33 | 34 | # [3.3.0](https://github.com/vivaxy/here/compare/v3.2.2...v3.3.0) (2018-10-12) 35 | 36 | 37 | ### Features 38 | 39 | * **gzip:** :sparkles:Support gzip compression ([1fb8732](https://github.com/vivaxy/here/commit/1fb8732)) 40 | * Drop support for node.js v6 and v7 ([74c277d](https://github.com/vivaxy/here/commit/74c277d)) 41 | 42 | 43 | 44 | 45 | ## [3.2.2](https://github.com/vivaxy/here/compare/v3.2.1...v3.2.2) (2017-12-28) 46 | 47 | 48 | ### Bug Fixes 49 | 50 | * :lock:Fix an issue that users can browse outside the wd ([d50c567](https://github.com/vivaxy/here/commit/d50c567)), closes [#16](https://github.com/vivaxy/here/issues/16) 51 | 52 | 53 | 54 | 55 | ## [3.2.1](https://github.com/vivaxy/here/compare/v3.1.0...v3.2.1) (2017-04-11) 56 | 57 | 58 | 59 | # v3.2.0 60 | 61 | - support restful API, config same as `koa-router` 62 | - upgrade jade to pug 63 | - update node versions in travis ci 64 | - add verbose log for request method, path and cost time 65 | - fix directory config 66 | - change server response log from debug level to verbose 67 | - add ssl test case 68 | - update readme 69 | 70 | # v3.1.0 71 | 72 | - add ssl support 73 | 74 | # v3.0.1 75 | 76 | - remove usage report 77 | 78 | # v3.0.0 79 | 80 | - rewrite in es6 81 | 82 | - use co 83 | -------------------------------------------------------------------------------- /CONTRIBUTING.md: -------------------------------------------------------------------------------- 1 | # Publish 2 | 3 | `npm run release` 4 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Copyright (c) 2015 vivaxy 2 | 3 | Permission is hereby granted, free of charge, to any person 4 | obtaining a copy of this software and associated documentation 5 | files (the "Software"), to deal in the Software without 6 | restriction, including without limitation the rights to use, 7 | copy, modify, merge, publish, distribute, sublicense, and/or sell 8 | copies of the Software, and to permit persons to whom the 9 | Software is furnished to do so, subject to the following 10 | conditions: 11 | 12 | The above copyright notice and this permission notice shall be 13 | included in all copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 16 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES 17 | OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 18 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT 19 | HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, 20 | WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 21 | FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 22 | OTHER DEALINGS IN THE SOFTWARE. 23 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # here 2 | 3 | ![here](./assets/images/here.jpg) 4 | 5 | [![Build Status][travis-image]][travis-url] 6 | [![NPM Version][npm-version-image]][npm-url] 7 | [![NPM Downloads][npm-downloads-image]][npm-url] 8 | [![MIT License][license-image]][license-url] 9 | 10 | Local static server 11 | 12 | Everything start from `here`. 13 | 14 | Node.js >= 8.0 15 | 16 | ## Feature 17 | 18 | - Look up available port automatically, which means multiple instances without specifying port. 19 | 20 | - Custom routes by scripting `here.js`. 21 | 22 | - Live reload. 23 | 24 | - Support https. 25 | 26 | - Add ip address to your server, which makes your server available to other devices. 27 | 28 | - Resolve get, post... every method into local files, for ajax. 29 | 30 | - Respond files without extension as `application/json` for ajax. 31 | 32 | - Open default browser after server launched. 33 | 34 | - When the server is on, press `enter` will open the browser. 35 | 36 | ## Installation 37 | 38 | `[sudo] npm install -g @vivaxy/here` 39 | 40 | ## Usage 41 | 42 | In your local folder, type `here` and it goes\! 43 | 44 | ## Advanced Usage 45 | 46 | #### Specify port to 8888 47 | 48 | `here -p 8888` 49 | 50 | or 51 | 52 | `here --port 8888` 53 | 54 | Default port is `3000`. 55 | 56 | #### Switch protocol to https 57 | 58 | `here -S` 59 | 60 | or 61 | 62 | `here --ssl` 63 | 64 | #### Open Gzip 65 | 66 | `here -G` 67 | 68 | or 69 | 70 | `here --gzip` 71 | 72 | #### Specify server root directory 73 | 74 | `here -d test` 75 | 76 | or 77 | 78 | `here --directory test` 79 | 80 | Default directory is `./`. 81 | 82 | #### Watch file changes, once files changed, reload pages 83 | 84 | `here -w 3` 85 | 86 | or 87 | 88 | `here --watch` 89 | 90 | Default interval is 0 second. 91 | 92 | Recommend to set reload interval to page reload time. 93 | 94 | #### Do not open the browser 95 | 96 | `here -s` 97 | 98 | or 99 | 100 | `here --silent` 101 | 102 | #### Specify open browser IP 103 | 104 | `here --ip localhost` 105 | 106 | or 107 | 108 | `here --ip private` 109 | 110 | or 111 | 112 | `here --ip public` 113 | 114 | Default IP is `public`. 115 | 116 | #### Output log 117 | 118 | `here -l` 119 | 120 | or 121 | 122 | `here --log 0` 123 | 124 | #### Middleware support 125 | 126 | Write `here.js` in server base directory. 127 | 128 | ``` 129 | let db = { 130 | tobi: { 131 | name: 'tobi', 132 | age: 21 133 | }, 134 | loki: { 135 | name: 'loki', 136 | age: 26 137 | }, 138 | jane: { 139 | name: 'jane', 140 | age: 18 141 | } 142 | }; 143 | 144 | module.exports = [ 145 | { 146 | method: 'get', 147 | path: '/pets', 148 | data () { 149 | let names = Object.keys(db); 150 | return names.map((name) => { 151 | return db[name]; 152 | }); 153 | } 154 | }, 155 | { 156 | method: 'get', 157 | path: '/pets/:name', 158 | data () { 159 | let name = this.params.name; 160 | let pet = db[name]; 161 | if (!pet) { 162 | return { 163 | error: `cannot find pet ${name}` 164 | }; 165 | } else { 166 | return pet; 167 | } 168 | } 169 | } 170 | ]; 171 | ``` 172 | 173 | See [koa-router document](https://github.com/alexmingoia/koa-router#module_koa-router--Router+get%7Cput%7Cpost%7Cpatch%7Cdelete) for more detail. 174 | 175 | ## Prior Art 176 | 177 | - [puer](https://www.npmjs.com/package/puer) not support post, respond files without extension as `application/octet-stream` 178 | - [anywhere](https://www.npmjs.com/package/anywhere) not support post, and not support reload 179 | - [browsersync](http://www.browsersync.io/) not support post, respond files without extension as `application/octet-stream` 180 | - [serve](https://github.com/zeit/serve) 181 | 182 | ## Change Log 183 | 184 | [Change Log](CHANGELOG.md) 185 | 186 | ## Contributing 187 | 188 | [Contributing](CONTRIBUTING.md) 189 | 190 | [npm-version-image]: http://img.shields.io/npm/v/@vivaxy/here.svg?style=flat-square 191 | [npm-url]: https://www.npmjs.com/package/@vivaxy/here 192 | [npm-downloads-image]: http://img.shields.io/npm/dt/@vivaxy/here.svg?style=flat-square 193 | [license-image]: http://img.shields.io/npm/l/@vivaxy/here.svg?style=flat-square 194 | [license-url]: LICENSE 195 | [travis-image]: https://img.shields.io/travis/vivaxy/here.svg?style=flat-square 196 | [travis-url]: https://travis-ci.org/vivaxy/here 197 | -------------------------------------------------------------------------------- /assets/images/here.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/vivaxy/here/81a5cf8261c03c1390a25d75960c2382cccc3d27/assets/images/here.jpg -------------------------------------------------------------------------------- /assets/psd/here.psd: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/vivaxy/here/81a5cf8261c03c1390a25d75960c2382cccc3d27/assets/psd/here.psd -------------------------------------------------------------------------------- /constant/config.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2016-08-21 10:25 3 | * @author vivaxy 4 | */ 5 | 6 | exports.IS_DEBUG = 'isDebug'; 7 | exports.LOG_LEVEL = 'logLevel'; 8 | exports.DIRECTORY = 'directory'; 9 | exports.SERVE_HERE_VERSION = 'serveHereVersion'; 10 | exports.PORT = 'port'; 11 | exports.IP = 'ip'; 12 | exports.SSL = 'ssl'; 13 | exports.WATCH = 'watch'; 14 | exports.SILENT = 'silent'; 15 | exports.GZIP = 'gzip'; 16 | -------------------------------------------------------------------------------- /constant/log-prefix.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2016-08-21 10:23 3 | * @author vivaxy 4 | */ 5 | 6 | exports.BROWSER = ' browser:'; 7 | exports.SERVER = ' server:'; 8 | exports.WATCH = ' watch:'; 9 | exports.RESPONSE = 'response:'; 10 | exports.REQUEST = ' request:'; 11 | exports.TIME = ' time:'; 12 | -------------------------------------------------------------------------------- /debug.js: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | /** 4 | * @since 2015-11-06 13:47 5 | * @author vivaxy 6 | */ 7 | 8 | const config = require('./lib/config.js'); 9 | const configKeys = require('./constant/config.js'); 10 | 11 | config.set(configKeys.IS_DEBUG, true); 12 | 13 | require('./lib/application')(); 14 | -------------------------------------------------------------------------------- /index.js: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | /** 4 | * @since 2015-11-06 13:35 5 | * @author vivaxy 6 | */ 7 | 8 | require('./lib/application')(); 9 | -------------------------------------------------------------------------------- /lib/application.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2015-11-20 13:41 3 | * @author vivaxy 4 | */ 5 | 6 | const path = require('path'); 7 | 8 | const log = require('log-util'); 9 | const commander = require('commander'); 10 | 11 | const config = require('./config.js'); 12 | const startServer = require('./server.js'); 13 | const startWatcher = require('./watcher.js'); 14 | const packageJson = require('../package.json'); 15 | const configKeys = require('../constant/config.js'); 16 | 17 | const formatWatch = (value) => { 18 | if (isNaN(Number(value))) { 19 | return 0; 20 | } else if (value < 0) { 21 | return 0; 22 | } 23 | return value; 24 | }; 25 | 26 | const formatDirectory = (value) => { 27 | const absoluteWorkingDirectory = process.cwd(); 28 | if (path.isAbsolute(value)) { 29 | return value; 30 | } 31 | return path.join(absoluteWorkingDirectory, value); 32 | }; 33 | 34 | const prepare = () => { 35 | if (config.get(configKeys.IS_DEBUG)) { 36 | config.set(configKeys.LOG_LEVEL, 0); 37 | } 38 | 39 | config.set(configKeys.DIRECTORY, formatDirectory('.')); 40 | config.set(configKeys.SERVE_HERE_VERSION, packageJson.version); 41 | 42 | commander 43 | .version(config.get(configKeys.SERVE_HERE_VERSION), '-v, --version') 44 | .option( 45 | `-p, --${configKeys.PORT} [port]`, 46 | 'specify port', 47 | config.get(configKeys.PORT) 48 | ) 49 | .option(`-S, --${configKeys.SSL}`, 'switch protocol to https') 50 | .option(`-G, --${configKeys.GZIP}`, 'open gzip') 51 | .option( 52 | `-d, --${configKeys.DIRECTORY} [directory]`, 53 | 'specify root directory', 54 | formatDirectory, 55 | config.get(configKeys.DIRECTORY) 56 | ) 57 | .option( 58 | `-w, --${configKeys.WATCH} [interval]`, 59 | 'will watch files; once changed, reload pages', 60 | formatWatch 61 | ) 62 | .option(`-s, --${configKeys.SILENT}`, 'will not open browser') 63 | .option(`--ip [ip]`, 'open browser with ip', config.get(configKeys.IP)) 64 | .option( 65 | `-l, --${configKeys.LOG_LEVEL} [level]`, 66 | 'output log in some level, lower means more details', 67 | config.get(configKeys.LOG_LEVEL) 68 | ) 69 | .parse(process.argv); 70 | 71 | const commanderOptions = commander.opts(); 72 | 73 | log.setLevel(Number(commanderOptions[configKeys.LOG_LEVEL])); 74 | 75 | if (commanderOptions[configKeys.PORT]) { 76 | config.set(configKeys.PORT, commanderOptions[configKeys.PORT]); 77 | } 78 | if (commanderOptions[configKeys.DIRECTORY]) { 79 | config.set(configKeys.DIRECTORY, commanderOptions[configKeys.DIRECTORY]); 80 | } 81 | 82 | if (commanderOptions[configKeys.SSL]) { 83 | config.set(configKeys.SSL, commanderOptions[configKeys.SSL]); 84 | } 85 | if (commanderOptions[configKeys.GZIP]) { 86 | config.set(configKeys.GZIP, commanderOptions[configKeys.GZIP]); 87 | } 88 | 89 | if (commanderOptions[configKeys.WATCH] !== undefined) { 90 | if (commanderOptions[configKeys.WATCH] === true) { 91 | config.set(configKeys.WATCH, 0); 92 | } else { 93 | config.set(configKeys.WATCH, commanderOptions[configKeys.WATCH]); 94 | } 95 | } 96 | if (commanderOptions[configKeys.SILENT]) { 97 | config.set(configKeys.SILENT, commanderOptions[configKeys.SILENT]); 98 | } 99 | 100 | if (commanderOptions[configKeys.IP]) { 101 | config.set(configKeys.IP, commanderOptions[configKeys.IP]); 102 | } 103 | 104 | if (commanderOptions[configKeys.LOG_LEVEL]) { 105 | config.set(configKeys.LOG_LEVEL, commanderOptions[configKeys.LOG_LEVEL]); 106 | } 107 | }; 108 | 109 | module.exports = () => { 110 | prepare(); 111 | 112 | if (!config.get(configKeys.IS_DEBUG)) { 113 | process.on('uncaughtException', (e) => { 114 | // throw this to preserve default behaviour 115 | // console this instead of throw error to keep the original error trace 116 | log.error(e.stack); 117 | // still exit as uncaught exception 118 | }); 119 | } 120 | 121 | startServer((nativeServer) => { 122 | if (config.get(configKeys.WATCH) !== false) { 123 | startWatcher(nativeServer); 124 | } 125 | }); 126 | }; 127 | -------------------------------------------------------------------------------- /lib/build-file-list.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 150205 09:21 3 | * @author vivaxy 4 | */ 5 | 6 | const fs = require('fs'); 7 | const path = require('path'); 8 | 9 | const pug = require('pug'); 10 | 11 | const join = path.join; 12 | const extname = path.extname; 13 | const isWindows = /^win/.test(process.platform); 14 | 15 | const AHEAD = -1; 16 | const ABACK = 1; 17 | const STILL = 0; 18 | const HTML_EXTENSION = '.html'; 19 | 20 | /** 21 | * 22 | * @param {String} _a fileName 23 | * @param {String} _b fileName 24 | * @returns {number} sort value 25 | */ 26 | const compare = (_a, _b) => { 27 | if (_a < _b) { 28 | return AHEAD; 29 | } else if (_a > _b) { 30 | return ABACK; 31 | } 32 | return STILL; 33 | }; 34 | 35 | /** 36 | * 37 | * @param {Array} files folder file list 38 | * @param {String} pathname folder absolute path 39 | * @param {String} absoluteWorkingDirectory absolute root directory 40 | * @returns {String} html content 41 | */ 42 | module.exports = (files, pathname, absoluteWorkingDirectory) => { 43 | const baseDir = join(absoluteWorkingDirectory, pathname); 44 | 45 | const filesFiltered = files.filter((file) => { 46 | if (file.indexOf('.') === 0) { 47 | return false; 48 | } 49 | // #3 remove inaccessible files from file list 50 | // win下access任何文件都会返回可访问 51 | try { 52 | if (isWindows) { 53 | fs.statSync(join(baseDir, file)); 54 | } else { 55 | fs.accessSync(join(baseDir, file), fs.R_OK); 56 | } 57 | } catch (e) { 58 | return false; 59 | } 60 | return true; 61 | }); 62 | 63 | filesFiltered.sort((a, b) => { 64 | const _isDirectoryA = fs.lstatSync(join(baseDir, a)).isDirectory(); 65 | const _isDirectoryB = fs.lstatSync(join(baseDir, b)).isDirectory(); 66 | // order by priority 67 | if (_isDirectoryA && _isDirectoryB) { 68 | return compare(a, b); 69 | } 70 | if (_isDirectoryA && !_isDirectoryB) { 71 | return AHEAD; 72 | } 73 | if (_isDirectoryB && !_isDirectoryA) { 74 | return ABACK; 75 | } 76 | const _extensionA = extname(a); 77 | const _extensionB = extname(b); 78 | const _isHtmlA = _extensionA === HTML_EXTENSION; 79 | const _isHtmlB = _extensionB === HTML_EXTENSION; 80 | if (_isHtmlA && _isHtmlB) { 81 | return compare(a, b); 82 | } 83 | if (_isHtmlA && !_isHtmlB) { 84 | return AHEAD; 85 | } 86 | if (_isHtmlB && !_isHtmlA) { 87 | return ABACK; 88 | } 89 | if (_extensionA === _extensionB) { 90 | return compare(a, b); 91 | } 92 | return compare(a, b); 93 | }); 94 | 95 | if (pathname !== '/') { 96 | filesFiltered.unshift('..'); 97 | } 98 | 99 | const list = filesFiltered.map((file) => { 100 | const ext = extname(file); 101 | let _ext = 'other'; 102 | if (ext === HTML_EXTENSION) { 103 | _ext = 'html'; 104 | } 105 | if (fs.lstatSync(join(baseDir, file)).isDirectory()) { 106 | _ext = 'dir'; 107 | } 108 | if (file === '..') { 109 | _ext = 'null'; 110 | } 111 | return { 112 | // remove http://ip:port to redirect to correct ip:port 113 | href: join(pathname, file), 114 | className: _ext, 115 | fileName: file, 116 | }; 117 | }); 118 | 119 | return pug.compileFile(join(__dirname, '../res/list.pug'), { 120 | pretty: ' ', 121 | })({ list }); 122 | }; 123 | -------------------------------------------------------------------------------- /lib/config.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2015-11-20 15:40 3 | * @author vivaxy 4 | */ 5 | 6 | const configKeys = require('../constant/config.js'); 7 | 8 | const config = { 9 | [configKeys.IS_DEBUG]: false, 10 | 11 | [configKeys.PORT]: 3000, 12 | [configKeys.SSL]: false, 13 | [configKeys.GZIP]: false, 14 | [configKeys.DIRECTORY]: '.', 15 | [configKeys.WATCH]: false, 16 | [configKeys.SILENT]: false, 17 | [configKeys.IP]: 'public', 18 | [configKeys.LOG_LEVEL]: 2, 19 | 20 | [configKeys.SERVE_HERE_VERSION]: '0.0.0', 21 | }; 22 | 23 | exports.set = (key, value) => { 24 | config[key] = value; 25 | return value; 26 | }; 27 | 28 | exports.get = (key) => { 29 | return config[key]; 30 | }; 31 | -------------------------------------------------------------------------------- /lib/fallback-content-type.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2015-11-20 13:24 3 | * @author vivaxy 4 | */ 5 | 6 | module.exports = 'application/json'; 7 | -------------------------------------------------------------------------------- /lib/get-file-stat.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2015-11-23 10:14 3 | * @author vivaxy 4 | */ 5 | 6 | const fs = require('fs'); 7 | 8 | module.exports = (file) => { 9 | return new Promise((resolve, reject) => { 10 | return fs.lstat(file, (err, stat) => { 11 | if (err) { 12 | reject(err); 13 | } else { 14 | resolve(stat); 15 | } 16 | }); 17 | }); 18 | }; 19 | -------------------------------------------------------------------------------- /lib/read-file.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2015-11-20 20:09 3 | * @author vivaxy 4 | */ 5 | 6 | const fs = require('fs'); 7 | 8 | module.exports = (file) => { 9 | return new Promise((resolve, reject) => { 10 | return fs.readFile(file, (err, content) => { 11 | if (err) { 12 | reject(err); 13 | } else { 14 | resolve(content); 15 | } 16 | }); 17 | }); 18 | }; 19 | -------------------------------------------------------------------------------- /lib/read-folder.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2015-11-23 10:13 3 | * @author vivaxy 4 | */ 5 | 6 | const fs = require('fs'); 7 | 8 | module.exports = (file) => { 9 | return new Promise((resolve, reject) => { 10 | return fs.readdir(file, (err, files) => { 11 | if (err) { 12 | reject(err); 13 | } else { 14 | resolve(files); 15 | } 16 | }); 17 | }); 18 | }; 19 | -------------------------------------------------------------------------------- /lib/server.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2015-11-19 20:15 3 | * @author vivaxy 4 | */ 5 | const path = require('path'); 6 | const https = require('https'); 7 | 8 | const ip = require('ip'); 9 | const Koa = require('koa'); 10 | const pem = require('pem'); 11 | const log = require('log-util'); 12 | const open = require('open'); 13 | 14 | const config = require('./config.js'); 15 | const getFileStat = require('./get-file-stat.js'); 16 | const logPrefix = require('../constant/log-prefix.js'); 17 | const configKeys = require('../constant/config.js'); 18 | 19 | module.exports = (callback) => { 20 | let nativeServer; 21 | const directory = config.get(configKeys.DIRECTORY); 22 | const watch = config.get(configKeys.WATCH); 23 | 24 | const server = new Koa(); 25 | 26 | const serverErrorCallback = (err) => { 27 | if (err.code === 'EADDRINUSE') { 28 | const port = config.get(configKeys.PORT); 29 | log.warn(logPrefix.SERVER, 'port', port, 'in use'); 30 | config.set(configKeys.PORT, port + 1); 31 | listen(); 32 | } 33 | }; 34 | 35 | const openBrowserFunction = (url) => { 36 | open(url) 37 | .then(() => log.debug(logPrefix.BROWSER, url)) 38 | .catch((e) => log.error(logPrefix.BROWSER, e.message)); 39 | }; 40 | 41 | function getIP() { 42 | const ipType = config.get(configKeys.IP); 43 | if (ipType === 'localhost') { 44 | return 'localhost'; 45 | } 46 | if (ipType === 'private') { 47 | return '127.0.0.1'; 48 | } 49 | return ip.address(); 50 | } 51 | 52 | const serverSuccessCallback = () => { 53 | const port = config.get(configKeys.PORT); 54 | const protocol = config.get(configKeys.SSL) ? 'https:' : 'http:'; 55 | const url = `${protocol}//${getIP()}:${port}/`; 56 | 57 | log.success(logPrefix.SERVER, 'listen', url); 58 | 59 | if (!config.get(configKeys.SILENT)) { 60 | openBrowserFunction(url); 61 | } 62 | // hit `space` will open page in browser 63 | const stdIn = process.stdin; 64 | stdIn.setEncoding('utf8'); 65 | stdIn.on('data', openBrowserFunction.bind(null, url)); 66 | callback(nativeServer); 67 | }; 68 | 69 | const listen = () => { 70 | const port = config.get(configKeys.PORT); 71 | if (config.get(configKeys.SSL)) { 72 | pem.createCertificate({ days: 1, selfSigned: true }, (err, keys) => { 73 | if (err) { 74 | log.error(err); 75 | return; 76 | } 77 | nativeServer = https 78 | .createServer( 79 | { key: keys.serviceKey, cert: keys.certificate }, 80 | server.callback() 81 | ) 82 | .on('error', serverErrorCallback) 83 | .listen(port, serverSuccessCallback); 84 | }); 85 | } else { 86 | nativeServer = server 87 | .listen(port, serverSuccessCallback) 88 | .on('error', serverErrorCallback); 89 | } 90 | }; 91 | 92 | const middlewareList = []; 93 | // - log request 94 | // - load route 95 | // - exec route 96 | // - gzip 97 | // - live reload 98 | // - read file 99 | // - handle error 100 | 101 | middlewareList.push(require('../middleware/log.js')); 102 | middlewareList.push(require('../middleware/error.js')); 103 | 104 | const routeFile = path.join(directory, 'here.js'); 105 | getFileStat(routeFile) 106 | .then((stat) => { 107 | if (stat.isFile()) { 108 | middlewareList.push(require('../middleware/load-route.js')(server)); 109 | } 110 | start(); 111 | }) 112 | .catch(() => { 113 | // routeFile not exists 114 | start(); 115 | }); 116 | 117 | function start() { 118 | if (config.get(configKeys.GZIP)) { 119 | middlewareList.push(require('../middleware/gzip.js')); 120 | } 121 | if (watch !== false) { 122 | middlewareList.push(require('../middleware/live-reload.js')); 123 | } 124 | middlewareList.push(require('../middleware/file-explorer.js')); 125 | middlewareList.forEach((middleware) => { 126 | server.use(middleware); 127 | }); 128 | listen(); 129 | } 130 | }; 131 | -------------------------------------------------------------------------------- /lib/watcher.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2015-11-20 21:48 3 | * @author vivaxy 4 | */ 5 | 6 | const path = require('path'); 7 | 8 | const log = require('log-util'); 9 | const chokidar = require('chokidar'); 10 | const debounce = require('debounce'); 11 | const socketServer = require('socket.io'); 12 | 13 | const config = require('./config.js'); 14 | const configKeys = require('../constant/config.js'); 15 | const logPrefix = require('../constant/log-prefix.js'); 16 | 17 | const SPLICE_COUNT = 1; 18 | const NOT_FOUNT_INDEX = -1; 19 | const A_THOUSAND = 1000; 20 | 21 | module.exports = (server) => { 22 | const absoluteWorkingDirectory = config.get(configKeys.DIRECTORY); 23 | const interval = config.get(configKeys.WATCH); 24 | 25 | const socketList = []; 26 | 27 | const io = socketServer(server); 28 | 29 | io.on('connection', (socket) => { 30 | socketList.push(socket); 31 | socket.on('disconnect', () => { 32 | const index = socketList.indexOf(socket); 33 | if (index > NOT_FOUNT_INDEX) { 34 | socketList.splice(index, SPLICE_COUNT); 35 | } 36 | }); 37 | }); 38 | 39 | const reload = debounce(() => { 40 | socketList.forEach((socket) => { 41 | log.debug(logPrefix.WATCH, 'reload'); 42 | socket.emit('reload'); 43 | }); 44 | }, interval * A_THOUSAND); 45 | 46 | const watcher = chokidar.watch(absoluteWorkingDirectory, { 47 | ignored: [ 48 | '**/node_modules', 49 | /[\/\\]\./, 50 | '.git', 51 | '.idea', 52 | '.DS_Store', 53 | ], 54 | }); 55 | 56 | watcher.on('all', (action, filePath) => { 57 | log.debug(logPrefix.WATCH, action, path.relative(absoluteWorkingDirectory, filePath)); 58 | switch (action) { 59 | case 'add': 60 | case 'addDir': 61 | break; 62 | default: 63 | reload(); 64 | break; 65 | } 66 | }); 67 | 68 | watcher.on('ready', () => { 69 | log.success(logPrefix.WATCH, 'ready,', 'reload in', interval, 'seconds'); 70 | }); 71 | }; 72 | -------------------------------------------------------------------------------- /middleware/error.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2015-11-20 13:11 3 | * @author vivaxy 4 | */ 5 | 6 | const FALLBACK_CONTENT_TYPE = require('../lib/fallback-content-type.js'); 7 | 8 | module.exports = async function(ctx, next) { 9 | try { 10 | await next(); 11 | } catch (e) { 12 | ctx.status = 404; 13 | ctx.body = e.stack; 14 | ctx.type = FALLBACK_CONTENT_TYPE; 15 | } 16 | }; 17 | -------------------------------------------------------------------------------- /middleware/file-explorer.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2015-11-20 13:11 3 | * @author vivaxy 4 | */ 5 | const path = require('path'); 6 | 7 | const mime = require('mime'); 8 | const log = require('log-util'); 9 | 10 | const configKeys = require('../constant/config.js'); 11 | const logPrefix = require('../constant/log-prefix.js'); 12 | 13 | const config = require('../lib/config.js'); 14 | const readFile = require('../lib/read-file.js'); 15 | const readFolder = require('../lib/read-folder.js'); 16 | const getFileStat = require('../lib/get-file-stat.js'); 17 | const buildFileBrowser = require('../lib/build-file-list.js'); 18 | const FALLBACK_CONTENT_TYPE = require('../lib/fallback-content-type.js'); 19 | 20 | const NOT_FOUNT_INDEX = -1; 21 | const INDEX_PAGE = 'index.html'; 22 | 23 | module.exports = async function(ctx, next) { 24 | const directory = config.get(configKeys.DIRECTORY); 25 | 26 | // decode for chinese character 27 | const requestPath = decodeURIComponent(ctx.request.path); 28 | const fullRequestPath = path.join(directory, requestPath); 29 | // fix security issue 30 | if (!fullRequestPath.startsWith(directory)) { 31 | return await next(); 32 | } 33 | const stat = await getFileStat(fullRequestPath); 34 | 35 | if (stat.isDirectory()) { 36 | const files = await readFolder(fullRequestPath); 37 | 38 | if (files.indexOf(INDEX_PAGE) !== NOT_FOUNT_INDEX) { 39 | ctx.redirect(path.join(requestPath, INDEX_PAGE), '/'); 40 | } else { 41 | ctx.body = buildFileBrowser(files, requestPath, directory); 42 | ctx.type = mime.getType(INDEX_PAGE); 43 | } 44 | } else if (stat.isFile()) { 45 | ctx.body = await readFile(fullRequestPath); 46 | let type = mime.getType(fullRequestPath); 47 | 48 | if (path.extname(fullRequestPath) === '') { 49 | type = FALLBACK_CONTENT_TYPE; 50 | } 51 | 52 | ctx.type = type; 53 | ctx.set('Access-Control-Allow-Origin', '*'); 54 | log.debug(logPrefix.RESPONSE, ctx.request.method, requestPath, 'as', type); 55 | } 56 | 57 | await next(); 58 | }; 59 | -------------------------------------------------------------------------------- /middleware/gzip.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 20181012 20:12 3 | * @author vivaxy 4 | */ 5 | 6 | const compress = require('koa-compress'); 7 | 8 | module.exports = compress(); 9 | -------------------------------------------------------------------------------- /middleware/live-reload.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2015-11-20 16:03 3 | * @author vivaxy 4 | */ 5 | 6 | const path = require('path'); 7 | 8 | const readFile = require('../lib/read-file.js'); 9 | 10 | const NOT_FOUND_INDEX = 1; 11 | 12 | module.exports = async function(ctx, next) { 13 | await next(); 14 | 15 | if (ctx.type === 'text/html') { 16 | let response = ''; 17 | const body = ctx.body.toString('utf-8'); 18 | const reloadJavascriptContent = await readFile(path.join(__dirname, '../res/reload.js')); 19 | 20 | const reloadJavascript = ``; 21 | 22 | if (body.indexOf('') !== NOT_FOUND_INDEX) { 23 | const section = body.split(''); 24 | 25 | section.splice(1, 0, `${reloadJavascript}`); 26 | response = section.join(''); 27 | } else { 28 | response = body + reloadJavascript; 29 | } 30 | ctx.body = response; 31 | } 32 | }; 33 | -------------------------------------------------------------------------------- /middleware/load-route.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2016-03-02 18:00 3 | * @author vivaxy 4 | */ 5 | 6 | const path = require('path'); 7 | const koaRoute = require('koa-router')(); 8 | 9 | const configKeys = require('../constant/config.js'); 10 | 11 | const config = require('../lib/config.js'); 12 | 13 | const loadRoutes = () => { 14 | const directory = config.get(configKeys.DIRECTORY); 15 | const file = path.join(directory, 'here.js'); 16 | const routeList = require(file); 17 | routeList.forEach((route) => { 18 | koaRoute[route.method](route.path, require('./log.js'), async function(ctx) { 19 | ctx.body = route.data.apply(ctx, arguments); 20 | }); 21 | }); 22 | }; 23 | 24 | module.exports = (server) => { 25 | loadRoutes(); 26 | 27 | server.use(koaRoute.routes()); 28 | server.use(koaRoute.allowedMethods()); 29 | 30 | return async function(ctx, next) { 31 | await next(); 32 | }; 33 | }; 34 | -------------------------------------------------------------------------------- /middleware/log.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2016-08-21 10:13 3 | * @author vivaxy 4 | */ 5 | const log = require('log-util'); 6 | 7 | const logPrefix = require('../constant/log-prefix.js'); 8 | 9 | module.exports = async function(ctx, next) { 10 | const beginTime = new Date().getTime(); 11 | 12 | const request = ctx.request; 13 | log.debug(logPrefix.REQUEST, `${request.method} ${request.path}`); 14 | 15 | await next(); 16 | 17 | const endTime = new Date().getTime(); 18 | 19 | log.debug(logPrefix.TIME, `${endTime - beginTime}ms`); 20 | }; 21 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "@vivaxy/here", 3 | "version": "3.5.0", 4 | "description": "local static server", 5 | "homepage": "https://github.com/vivaxy/here", 6 | "bin": { 7 | "here": "./index.js" 8 | }, 9 | "engines": { 10 | "node": ">=8.0" 11 | }, 12 | "scripts": { 13 | "test": "mocha ./test/command.js --timeout 10000", 14 | "release": "standard-version && git push --follow-tags && npm publish --access=public", 15 | "postinstall": "husky install", 16 | "prepublishOnly": "pinst --disable", 17 | "postpublish": "pinst --enable" 18 | }, 19 | "repository": { 20 | "type": "git", 21 | "url": "https://github.com/vivaxy/here.git" 22 | }, 23 | "bugs": { 24 | "url": "https://github.com/vivaxy/here/issues" 25 | }, 26 | "keywords": [ 27 | "server", 28 | "static server", 29 | "local server", 30 | "vivaxy", 31 | "tool", 32 | "web", 33 | "reload" 34 | ], 35 | "author": "vivaxy", 36 | "contributors": [ 37 | { 38 | "name": "vivaxy", 39 | "email": "xyxuye2007@126.com" 40 | } 41 | ], 42 | "license": "MIT", 43 | "devDependencies": { 44 | "husky": "5", 45 | "lint-staged": "^10.0.0", 46 | "mocha": "^8.0.0", 47 | "pinst": "^2.1.4", 48 | "prettier": "^2.0.0", 49 | "standard-version": "^9.0.0" 50 | }, 51 | "dependencies": { 52 | "chokidar": "^3.3.1", 53 | "commander": "^7.0.0", 54 | "debounce": "^1.0.0", 55 | "ip": "^1.1.0", 56 | "koa": "^2.5.3", 57 | "koa-compress": "^5.0.0", 58 | "koa-router": "^10.0.0", 59 | "log-util": "^2.3.0", 60 | "mime": "^2.0.0", 61 | "open": "^7.4.0", 62 | "pem": "^1.8.0", 63 | "pug": "^3.0.0", 64 | "socket.io": "^3.0.0" 65 | }, 66 | "lint-staged": { 67 | "**/**.{js,json,md}": [ 68 | "prettier --write" 69 | ] 70 | }, 71 | "packageManager": "npm@8.19.4" 72 | } 73 | -------------------------------------------------------------------------------- /renovate.json: -------------------------------------------------------------------------------- 1 | { 2 | "extends": [ 3 | "config:base", 4 | ":preserveSemverRanges" 5 | ] 6 | } 7 | -------------------------------------------------------------------------------- /res/list.pug: -------------------------------------------------------------------------------- 1 | doctype html 2 | html 3 | head 4 | meta(name='viewport', content='width=device-width, initial-scale=1, maximum-scale=1, user-scalable=0') 5 | style. 6 | body { 7 | background: #eee; 8 | width: 100%; 9 | margin: 0; 10 | } 11 | 12 | a { 13 | display: block; 14 | height: 48px; 15 | width: 100%; 16 | text-decoration: none; 17 | background-color: #fff; 18 | border-bottom: 1px solid #ddd; 19 | -webkit-tap-highlight-color: rgba(0, 0, 0, 0.1); 20 | -webkit-user-select: none; 21 | overflow: hidden; 22 | position: relative; 23 | } 24 | 25 | a span { 26 | display: block; 27 | height: 32px; 28 | line-height: 32px; 29 | font-size: 16px; 30 | color: #2980b9; 31 | margin: 8px 0 8px 32px; 32 | overflow: hidden; 33 | text-overflow: ellipsis; 34 | white-space: nowrap; 35 | } 36 | 37 | .html:before { 38 | margin: 4px 8px 0 0; 39 | background-image: url(); 40 | content: " "; 41 | background-size: 100%; 42 | width: 24px; 43 | height: 24px; 44 | float: left; 45 | } 46 | 47 | .dir:before { 48 | margin: 4px 8px 0 0; 49 | background-image: url(); 50 | content: " "; 51 | background-size: 100%; 52 | width: 24px; 53 | height: 24px; 54 | float: left; 55 | } 56 | 57 | .null:before { 58 | margin: 4px 8px 0 0; 59 | content: " "; 60 | width: 24px; 61 | height: 24px; 62 | float: left; 63 | } 64 | 65 | .other:before { 66 | margin: 4px 8px 0 0; 67 | background-image: url(); 68 | content: " "; 69 | background-size: 100%; 70 | width: 24px; 71 | height: 24px; 72 | float: left; 73 | } 74 | body 75 | each val in list 76 | a(href=val.href) 77 | span(class=val.className) #{val.fileName} 78 | -------------------------------------------------------------------------------- /test/command.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2015-09-23 16:38 3 | * @author vivaxy 4 | */ 5 | 6 | const assert = require('assert'); 7 | const childProcess = require('child_process'); 8 | 9 | const packageJson = require('../package.json'); 10 | const logPrefix = require('../constant/log-prefix.js'); 11 | 12 | const spawn = childProcess.spawn; 13 | 14 | const NODE_COMMAND = 'node'; 15 | const HERE_COMMAND = './index.js'; 16 | 17 | describe('test terminal command `here`', () => { 18 | let here; 19 | afterEach(() => { 20 | here.kill(); 21 | }); 22 | it(`\`here\` should output \`${logPrefix.SERVER} listen http://*.*.*.*:*/\``, (done) => { 23 | here = spawn(NODE_COMMAND, [HERE_COMMAND]); 24 | here.stdout.on('data', (data) => { 25 | data = data.toString(); 26 | let regExp = `${logPrefix.SERVER} listen http:\\/\\/\\d+\\.\\d+\\.\\d+\\.\\d+:\\d+\\/\\n`; 27 | assert.equal(true, new RegExp(regExp, 'g').test(data)); 28 | here.kill(); 29 | done(); 30 | }); 31 | }); 32 | it('`here -v` should output `version`', (done) => { 33 | here = spawn(NODE_COMMAND, [HERE_COMMAND, '-v']); 34 | here.stdout.on('data', (data) => { 35 | data = data.toString(); 36 | assert.equal(packageJson.version + '\n', data); 37 | done(); 38 | }); 39 | }); 40 | it('`here --version` should output `version`', (done) => { 41 | here = spawn(NODE_COMMAND, [HERE_COMMAND, '--version']); 42 | here.stdout.on('data', (data) => { 43 | data = data.toString(); 44 | assert.equal(packageJson.version + '\n', data); 45 | done(); 46 | }); 47 | }); 48 | it('`here -h` should output help', (done) => { 49 | here = spawn(NODE_COMMAND, [HERE_COMMAND, '-h']); 50 | here.stdout.on('data', (data) => { 51 | data = data.toString(); 52 | assert.equal(true, !!~data.indexOf('Usage: index [options]') && !!~data.indexOf('Options:')); 53 | done(); 54 | }); 55 | }); 56 | it(`\`here -w\` should output \`${logPrefix.SERVER} listen http://*.*.*.*:*/\` and \`${logPrefix.WATCH} ready, reload in 0 seconds\``, (done) => { 57 | here = spawn(NODE_COMMAND, [HERE_COMMAND, '-w']); 58 | let stdoutCount = 0; 59 | here.stdout.on('data', (data) => { 60 | stdoutCount++; 61 | data = data.toString(); 62 | switch (stdoutCount) { 63 | case 1: 64 | let serverRegExp = `${logPrefix.SERVER} listen http:\\/\\/\\d+\\.\\d+\\.\\d+\\.\\d+:\\d+\\/\\n$`; 65 | assert.equal(true, new RegExp(serverRegExp).test(data)); 66 | break; 67 | case 2: 68 | let watchRegExp = `${logPrefix.WATCH} ready, reload in 0 seconds\\n$`; 69 | assert.equal(true, new RegExp(watchRegExp).test(data)); 70 | here.kill(); 71 | done(); 72 | break; 73 | default: 74 | assert.fail(stdoutCount, 2); 75 | done(); 76 | break; 77 | } 78 | }); 79 | }); 80 | it(`\`here --watch 3\` should output \`${logPrefix.SERVER} listen http://*.*.*.*:*/\` and \`${logPrefix.WATCH} ready, reload in 3 seconds\``, (done) => { 81 | here = spawn(NODE_COMMAND, [HERE_COMMAND, '--watch', '3']); 82 | let stdOutCount = 0; 83 | here.stdout.on('data', (data) => { 84 | stdOutCount++; 85 | data = data.toString(); 86 | switch (stdOutCount) { 87 | case 1: 88 | let serverRegExp = `${logPrefix.SERVER} listen http:\\/\\/\\d+\\.\\d+\\.\\d+\\.\\d+:\\d+\\/\\n$`; 89 | assert.equal(true, new RegExp(serverRegExp).test(data)); 90 | break; 91 | case 2: 92 | let watchRegExp = `${logPrefix.WATCH} ready, reload in 3 seconds\\n$`; 93 | assert.equal(true, new RegExp(watchRegExp).test(data)); 94 | here.kill(); 95 | done(); 96 | break; 97 | default: 98 | assert.fail(stdOutCount, 2); 99 | done(); 100 | break; 101 | } 102 | }); 103 | }); 104 | }); 105 | -------------------------------------------------------------------------------- /test/file/css/index.css: -------------------------------------------------------------------------------- 1 | body { 2 | margin: 0; 3 | width: 100%; 4 | height: 100%; 5 | background: #eee; 6 | } 7 | 8 | .css:after { 9 | content: 'css ok'; 10 | } 11 | -------------------------------------------------------------------------------- /test/file/folder/ddd.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | vivaxy 8 | 9 | 10 | test 11 | 12 | -------------------------------------------------------------------------------- /test/file/folder/icon-search.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/vivaxy/here/81a5cf8261c03c1390a25d75960c2382cccc3d27/test/file/folder/icon-search.png -------------------------------------------------------------------------------- /test/file/folder/搜索框.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/vivaxy/here/81a5cf8261c03c1390a25d75960c2382cccc3d27/test/file/folder/搜索框.png -------------------------------------------------------------------------------- /test/file/here.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2016-03-02 18:05 3 | * @author vivaxy 4 | */ 5 | 6 | 'use strict'; 7 | 8 | let db = { 9 | tobi: { 10 | name: 'tobi', 11 | age: 21 12 | }, 13 | loki: { 14 | name: 'loki', 15 | age: 26 16 | }, 17 | jane: { 18 | name: 'jane', 19 | age: 18 20 | } 21 | }; 22 | 23 | module.exports = [ 24 | { 25 | method: 'get', 26 | path: '/pets', 27 | data () { 28 | let names = Object.keys(db); 29 | return names.map((name) => { 30 | return db[name]; 31 | }); 32 | } 33 | }, 34 | { 35 | method: 'get', 36 | path: '/pets/:name', 37 | data () { 38 | let name = this.params.name; 39 | let pet = db[name]; 40 | if (!pet) { 41 | return { 42 | error: `cannot find pet ${name}` 43 | }; 44 | } else { 45 | return pet; 46 | } 47 | } 48 | } 49 | ]; 50 | -------------------------------------------------------------------------------- /test/file/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | serve here 7 | 8 | 9 | 10 |

html ok

11 |

12 | 13 | 14 | 15 | 16 | 17 | -------------------------------------------------------------------------------- /test/file/js/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 150201 11:50 3 | * @author vivaxy 4 | */ 5 | 6 | 'use strict'; 7 | 8 | document.body.innerHTML += '

js ok

'; 9 | 10 | var ajax = function () { 11 | var req = new XMLHttpRequest(); 12 | req.open('GET', './mock-server/ajax', true); 13 | req.addEventListener('readystatechange', function () { 14 | if (req.readyState === 4 && req.status === 200) { 15 | var response = JSON.parse(req.responseText); 16 | console.log(response); 17 | } 18 | }); 19 | req.send(); 20 | }; 21 | 22 | let image = new Image(); 23 | image.src = 'folder/icon-search.png?_=' + new Date().getTime(); 24 | document.body.appendChild(image); 25 | -------------------------------------------------------------------------------- /test/file/mock-server/ajax: -------------------------------------------------------------------------------- 1 | { 2 | "code": 200, 3 | "msg": "" 4 | } -------------------------------------------------------------------------------- /test/file/mock-server/test.json: -------------------------------------------------------------------------------- 1 | { 2 | "code": 200 3 | } -------------------------------------------------------------------------------- /test/test/commander.js: -------------------------------------------------------------------------------- 1 | /** 2 | * @since 2016-08-20 15:03 3 | * @author vivaxy 4 | */ 5 | 6 | 'use strict'; 7 | 8 | let program = require('commander'); 9 | 10 | function range (val) { 11 | return val.split('..').map(Number); 12 | } 13 | 14 | function list (val) { 15 | return val.split(','); 16 | } 17 | 18 | function collect (val, memo) { 19 | memo.push(val); 20 | return memo; 21 | } 22 | 23 | function increaseVerbosity (v, total) { 24 | return total + 1; 25 | } 26 | 27 | function formatWatch (value) { 28 | if (isNaN(Number(value))) { 29 | value = 0; 30 | } else if (value < 0) { 31 | value = 0; 32 | } 33 | return value; 34 | } 35 | 36 | program 37 | .version('0.0.1') 38 | .usage('[options] ') 39 | .option('-w, --watch [interval]', 'will watch files; once changed, reload pages', formatWatch) 40 | .option('-i, --integer ', 'An integer argument', parseInt) 41 | .option('-f, --float ', 'A float argument', parseFloat) 42 | .option('-r, --range ..', 'A range', range) 43 | .option('-l, --list ', 'A list', list) 44 | .option('-o, --optional [value]', 'An optional value') 45 | .option('-c, --collect [value]', 'A repeatable value', collect, []) 46 | .option('-v, --verbose', 'A value that can be increased', increaseVerbosity, 0) 47 | .parse(process.argv); 48 | 49 | // console.log(' int: %j', program.integer); 50 | // console.log(' float: %j', program.float); 51 | // console.log(' optional: %j', program.optional); 52 | // program.range = program.range || []; 53 | // console.log(' range: %j..%j', program.range[0], program.range[1]); 54 | // console.log(' list: %j', program.list); 55 | // console.log(' collect: %j', program.collect); 56 | console.log(' verbosity: %j', program.verbose); 57 | // console.log(' args: %j', program.args); 58 | // 59 | console.log(' watch', program.watch); 60 | 61 | --------------------------------------------------------------------------------