cmux/node_modules/micro/lib/index.js
2026-01-29 17:36:26 -08:00

170 lines
4.1 KiB
JavaScript

// Native
const {Stream} = require('stream');
// Packages
const contentType = require('content-type');
const getRawBody = require('raw-body');
// based on is-stream https://github.com/sindresorhus/is-stream/blob/c918e3795ea2451b5265f331a00fb6a8aaa27816/license
function isStream(stream) {
return stream !== null &&
typeof stream === 'object' &&
typeof stream.pipe === 'function';
}
function readable(stream) {
return isStream(stream) &&
stream.readable !== false &&
typeof stream._read === 'function' &&
typeof stream._readableState === 'object';
}
const {NODE_ENV} = process.env;
const DEV = NODE_ENV === 'development';
const serve = fn => (req, res) => exports.run(req, res, fn);
module.exports = serve;
exports = serve;
exports.default = serve;
const createError = (code, message, original) => {
const err = new Error(message);
err.statusCode = code;
err.originalError = original;
return err;
};
const send = (res, code, obj = null) => {
res.statusCode = code;
if (obj === null) {
res.end();
return;
}
if (Buffer.isBuffer(obj)) {
if (!res.getHeader('Content-Type')) {
res.setHeader('Content-Type', 'application/octet-stream');
}
res.setHeader('Content-Length', obj.length);
res.end(obj);
return;
}
if (obj instanceof Stream || readable(obj)) {
if (!res.getHeader('Content-Type')) {
res.setHeader('Content-Type', 'application/octet-stream');
}
obj.pipe(res);
return;
}
let str = obj;
if (typeof obj === 'object' || typeof obj === 'number') {
// We stringify before setting the header
// in case `JSON.stringify` throws and a
// 500 has to be sent instead
// the `JSON.stringify` call is split into
// two cases as `JSON.stringify` is optimized
// in V8 if called with only one argument
if (DEV) {
str = JSON.stringify(obj, null, 2);
} else {
str = JSON.stringify(obj);
}
if (!res.getHeader('Content-Type')) {
res.setHeader('Content-Type', 'application/json; charset=utf-8');
}
}
res.setHeader('Content-Length', Buffer.byteLength(str));
res.end(str);
};
const sendError = (req, res, errorObj) => {
const statusCode = errorObj.statusCode || errorObj.status;
const message = statusCode ? errorObj.message : 'Internal Server Error';
send(res, statusCode || 500, DEV ? errorObj.stack : message);
if (errorObj instanceof Error) {
console.error(errorObj.stack);
} else {
console.warn('thrown error must be an instance Error');
}
};
exports.send = send;
exports.sendError = sendError;
exports.createError = createError;
exports.run = (req, res, fn) =>
new Promise(resolve => resolve(fn(req, res)))
.then(val => {
if (val === null) {
send(res, 204, null);
return;
}
// Send value if it is not undefined, otherwise assume res.end
// will be called later
// eslint-disable-next-line no-undefined
if (val !== undefined) {
send(res, res.statusCode || 200, val);
}
})
.catch(err => sendError(req, res, err));
// Maps requests to buffered raw bodies so that
// multiple calls to `json` work as expected
const rawBodyMap = new WeakMap();
const parseJSON = str => {
try {
return JSON.parse(str);
} catch (err) {
throw createError(400, 'Invalid JSON', err);
}
};
exports.buffer = (req, {limit = '1mb', encoding} = {}) =>
Promise.resolve().then(() => {
const type = req.headers['content-type'] || 'text/plain';
const length = req.headers['content-length'];
// eslint-disable-next-line no-undefined
if (encoding === undefined) {
encoding = contentType.parse(type).parameters.charset;
}
const body = rawBodyMap.get(req);
if (body) {
return body;
}
return getRawBody(req, {limit, length, encoding})
.then(buf => {
rawBodyMap.set(req, buf);
return buf;
})
.catch(err => {
if (err.type === 'entity.too.large') {
throw createError(413, `Body exceeded ${limit} limit`, err);
} else {
throw createError(400, 'Invalid body', err);
}
});
});
exports.text = (req, {limit, encoding} = {}) =>
exports.buffer(req, {limit, encoding}).then(body => body.toString(encoding));
exports.json = (req, opts) =>
exports.text(req, opts).then(body => parseJSON(body));