Spaces:
Sleeping
Sleeping
| import { handler } from './handler.js'; | |
| import { env } from './env.js'; | |
| import http from 'http'; | |
| import * as qs from 'querystring'; | |
| /** | |
| * @param {string|RegExp} input The route pattern | |
| * @param {boolean} [loose] Allow open-ended matching. Ignored with `RegExp` input. | |
| */ | |
| function parse$1(input, loose) { | |
| if (input instanceof RegExp) return { keys:false, pattern:input }; | |
| var c, o, tmp, ext, keys=[], pattern='', arr = input.split('/'); | |
| arr[0] || arr.shift(); | |
| while (tmp = arr.shift()) { | |
| c = tmp[0]; | |
| if (c === '*') { | |
| keys.push(c); | |
| pattern += tmp[1] === '?' ? '(?:/(.*))?' : '/(.*)'; | |
| } else if (c === ':') { | |
| o = tmp.indexOf('?', 1); | |
| ext = tmp.indexOf('.', 1); | |
| keys.push( tmp.substring(1, !!~o ? o : !!~ext ? ext : tmp.length) ); | |
| pattern += !!~o && !~ext ? '(?:/([^/]+?))?' : '/([^/]+?)'; | |
| if (!!~ext) pattern += (!!~o ? '?' : '') + '\\' + tmp.substring(ext); | |
| } else { | |
| pattern += '/' + tmp; | |
| } | |
| } | |
| return { | |
| keys: keys, | |
| pattern: new RegExp('^' + pattern + (loose ? '(?=$|\/)' : '\/?$'), 'i') | |
| }; | |
| } | |
| const MAP = { | |
| "": 0, | |
| GET: 1, | |
| HEAD: 2, | |
| PATCH: 3, | |
| OPTIONS: 4, | |
| CONNECT: 5, | |
| DELETE: 6, | |
| TRACE: 7, | |
| POST: 8, | |
| PUT: 9, | |
| }; | |
| class Trouter { | |
| constructor() { | |
| this.routes = []; | |
| this.all = this.add.bind(this, ''); | |
| this.get = this.add.bind(this, 'GET'); | |
| this.head = this.add.bind(this, 'HEAD'); | |
| this.patch = this.add.bind(this, 'PATCH'); | |
| this.options = this.add.bind(this, 'OPTIONS'); | |
| this.connect = this.add.bind(this, 'CONNECT'); | |
| this.delete = this.add.bind(this, 'DELETE'); | |
| this.trace = this.add.bind(this, 'TRACE'); | |
| this.post = this.add.bind(this, 'POST'); | |
| this.put = this.add.bind(this, 'PUT'); | |
| } | |
| use(route, ...fns) { | |
| let handlers = [].concat.apply([], fns); | |
| let { keys, pattern } = parse$1(route, true); | |
| this.routes.push({ keys, pattern, method: '', handlers, midx: MAP[''] }); | |
| return this; | |
| } | |
| add(method, route, ...fns) { | |
| let { keys, pattern } = parse$1(route); | |
| let handlers = [].concat.apply([], fns); | |
| this.routes.push({ keys, pattern, method, handlers, midx: MAP[method] }); | |
| return this; | |
| } | |
| find(method, url) { | |
| let midx = MAP[method]; | |
| let isHEAD = (midx === 2); | |
| let i=0, j=0, k, tmp, arr=this.routes; | |
| let matches=[], params={}, handlers=[]; | |
| for (; i < arr.length; i++) { | |
| tmp = arr[i]; | |
| if (tmp.midx === midx || tmp.midx === 0 || (isHEAD && tmp.midx===1) ) { | |
| if (tmp.keys === false) { | |
| matches = tmp.pattern.exec(url); | |
| if (matches === null) continue; | |
| if (matches.groups !== void 0) for (k in matches.groups) params[k]=matches.groups[k]; | |
| tmp.handlers.length > 1 ? (handlers=handlers.concat(tmp.handlers)) : handlers.push(tmp.handlers[0]); | |
| } else if (tmp.keys.length > 0) { | |
| matches = tmp.pattern.exec(url); | |
| if (matches === null) continue; | |
| for (j=0; j < tmp.keys.length;) params[tmp.keys[j]]=matches[++j]; | |
| tmp.handlers.length > 1 ? (handlers=handlers.concat(tmp.handlers)) : handlers.push(tmp.handlers[0]); | |
| } else if (tmp.pattern.test(url)) { | |
| tmp.handlers.length > 1 ? (handlers=handlers.concat(tmp.handlers)) : handlers.push(tmp.handlers[0]); | |
| } | |
| } // else not a match | |
| } | |
| return { params, handlers }; | |
| } | |
| } | |
| /** | |
| * @typedef ParsedURL | |
| * @type {import('.').ParsedURL} | |
| */ | |
| /** | |
| * @typedef Request | |
| * @property {string} url | |
| * @property {ParsedURL} _parsedUrl | |
| */ | |
| /** | |
| * @param {Request} req | |
| * @returns {ParsedURL|void} | |
| */ | |
| function parse(req) { | |
| let raw = req.url; | |
| if (raw == null) return; | |
| let prev = req._parsedUrl; | |
| if (prev && prev.raw === raw) return prev; | |
| let pathname=raw, search='', query; | |
| if (raw.length > 1) { | |
| let idx = raw.indexOf('?', 1); | |
| if (idx !== -1) { | |
| search = raw.substring(idx); | |
| pathname = raw.substring(0, idx); | |
| if (search.length > 1) { | |
| query = qs.parse(search.substring(1)); | |
| } | |
| } | |
| } | |
| return req._parsedUrl = { pathname, search, query, raw }; | |
| } | |
| function onError(err, req, res) { | |
| let code = typeof err.status === 'number' && err.status; | |
| code = res.statusCode = (code && code >= 100 ? code : 500); | |
| if (typeof err === 'string' || Buffer.isBuffer(err)) res.end(err); | |
| else res.end(err.message || http.STATUS_CODES[code]); | |
| } | |
| const mount = fn => fn instanceof Polka ? fn.attach : fn; | |
| class Polka extends Trouter { | |
| constructor(opts={}) { | |
| super(); | |
| this.parse = parse; | |
| this.server = opts.server; | |
| this.handler = this.handler.bind(this); | |
| this.onError = opts.onError || onError; // catch-all handler | |
| this.onNoMatch = opts.onNoMatch || this.onError.bind(null, { status: 404 }); | |
| this.attach = (req, res) => setImmediate(this.handler, req, res); | |
| } | |
| use(base, ...fns) { | |
| if (base === '/') { | |
| super.use(base, fns.map(mount)); | |
| } else if (typeof base === 'function' || base instanceof Polka) { | |
| super.use('/', [base, ...fns].map(mount)); | |
| } else { | |
| super.use(base, | |
| (req, _, next) => { | |
| if (typeof base === 'string') { | |
| let len = base.length; | |
| base.startsWith('/') || len++; | |
| req.url = req.url.substring(len) || '/'; | |
| req.path = req.path.substring(len) || '/'; | |
| } else { | |
| req.url = req.url.replace(base, '') || '/'; | |
| req.path = req.path.replace(base, '') || '/'; | |
| } | |
| if (req.url.charAt(0) !== '/') { | |
| req.url = '/' + req.url; | |
| } | |
| next(); | |
| }, | |
| fns.map(mount), | |
| (req, _, next) => { | |
| req.path = req._parsedUrl.pathname; | |
| req.url = req.path + req._parsedUrl.search; | |
| next(); | |
| } | |
| ); | |
| } | |
| return this; // chainable | |
| } | |
| listen() { | |
| (this.server = this.server || http.createServer()).on('request', this.attach); | |
| this.server.listen.apply(this.server, arguments); | |
| return this; | |
| } | |
| handler(req, res, next) { | |
| let info = this.parse(req), path = info.pathname; | |
| let obj = this.find(req.method, req.path=path); | |
| req.url = path + info.search; | |
| req.originalUrl = req.originalUrl || req.url; | |
| req.query = info.query || {}; | |
| req.search = info.search; | |
| req.params = obj.params; | |
| if (path.length > 1 && path.indexOf('%', 1) !== -1) { | |
| for (let k in req.params) { | |
| try { req.params[k] = decodeURIComponent(req.params[k]); } | |
| catch (e) { /* malform uri segment */ } | |
| } | |
| } | |
| let i=0, arr=obj.handlers.concat(this.onNoMatch), len=arr.length; | |
| let loop = async () => res.finished || (i < len) && arr[i++](req, res, next); | |
| (next = next || (err => err ? this.onError(err, req, res, next) : loop().catch(next)))(); // init | |
| } | |
| } | |
| function polka (opts) { | |
| return new Polka(opts); | |
| } | |
| const path = env('SOCKET_PATH', false); | |
| const host = env('HOST', '0.0.0.0'); | |
| const port = env('PORT', !path && '3000'); | |
| const server = polka().use(handler); | |
| server.listen({ path, host, port }, () => { | |
| console.log(`Listening on ${path ? path : host + ':' + port}`); | |
| }); | |
| export { host, path, port, server }; | |