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.
305 lines
5.8 KiB
305 lines
5.8 KiB
6 months ago
|
/*!
|
||
|
* express
|
||
|
* Copyright(c) 2009-2013 TJ Holowaychuk
|
||
|
* Copyright(c) 2014-2015 Douglas Christopher Wilson
|
||
|
* MIT Licensed
|
||
|
*/
|
||
|
|
||
|
'use strict';
|
||
|
|
||
|
/**
|
||
|
* Module dependencies.
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
var Buffer = require('safe-buffer').Buffer
|
||
|
var contentDisposition = require('content-disposition');
|
||
|
var contentType = require('content-type');
|
||
|
var deprecate = require('depd')('express');
|
||
|
var flatten = require('array-flatten');
|
||
|
var mime = require('send').mime;
|
||
|
var etag = require('etag');
|
||
|
var proxyaddr = require('proxy-addr');
|
||
|
var qs = require('qs');
|
||
|
var querystring = require('querystring');
|
||
|
|
||
|
/**
|
||
|
* Return strong ETag for `body`.
|
||
|
*
|
||
|
* @param {String|Buffer} body
|
||
|
* @param {String} [encoding]
|
||
|
* @return {String}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
exports.etag = createETagGenerator({ weak: false })
|
||
|
|
||
|
/**
|
||
|
* Return weak ETag for `body`.
|
||
|
*
|
||
|
* @param {String|Buffer} body
|
||
|
* @param {String} [encoding]
|
||
|
* @return {String}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
exports.wetag = createETagGenerator({ weak: true })
|
||
|
|
||
|
/**
|
||
|
* Check if `path` looks absolute.
|
||
|
*
|
||
|
* @param {String} path
|
||
|
* @return {Boolean}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
exports.isAbsolute = function(path){
|
||
|
if ('/' === path[0]) return true;
|
||
|
if (':' === path[1] && ('\\' === path[2] || '/' === path[2])) return true; // Windows device path
|
||
|
if ('\\\\' === path.substring(0, 2)) return true; // Microsoft Azure absolute path
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Flatten the given `arr`.
|
||
|
*
|
||
|
* @param {Array} arr
|
||
|
* @return {Array}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
exports.flatten = deprecate.function(flatten,
|
||
|
'utils.flatten: use array-flatten npm module instead');
|
||
|
|
||
|
/**
|
||
|
* Normalize the given `type`, for example "html" becomes "text/html".
|
||
|
*
|
||
|
* @param {String} type
|
||
|
* @return {Object}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
exports.normalizeType = function(type){
|
||
|
return ~type.indexOf('/')
|
||
|
? acceptParams(type)
|
||
|
: { value: mime.lookup(type), params: {} };
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Normalize `types`, for example "html" becomes "text/html".
|
||
|
*
|
||
|
* @param {Array} types
|
||
|
* @return {Array}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
exports.normalizeTypes = function(types){
|
||
|
var ret = [];
|
||
|
|
||
|
for (var i = 0; i < types.length; ++i) {
|
||
|
ret.push(exports.normalizeType(types[i]));
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Generate Content-Disposition header appropriate for the filename.
|
||
|
* non-ascii filenames are urlencoded and a filename* parameter is added
|
||
|
*
|
||
|
* @param {String} filename
|
||
|
* @return {String}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
exports.contentDisposition = deprecate.function(contentDisposition,
|
||
|
'utils.contentDisposition: use content-disposition npm module instead');
|
||
|
|
||
|
/**
|
||
|
* Parse accept params `str` returning an
|
||
|
* object with `.value`, `.quality` and `.params`.
|
||
|
* also includes `.originalIndex` for stable sorting
|
||
|
*
|
||
|
* @param {String} str
|
||
|
* @param {Number} index
|
||
|
* @return {Object}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
function acceptParams(str, index) {
|
||
|
var parts = str.split(/ *; */);
|
||
|
var ret = { value: parts[0], quality: 1, params: {}, originalIndex: index };
|
||
|
|
||
|
for (var i = 1; i < parts.length; ++i) {
|
||
|
var pms = parts[i].split(/ *= */);
|
||
|
if ('q' === pms[0]) {
|
||
|
ret.quality = parseFloat(pms[1]);
|
||
|
} else {
|
||
|
ret.params[pms[0]] = pms[1];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Compile "etag" value to function.
|
||
|
*
|
||
|
* @param {Boolean|String|Function} val
|
||
|
* @return {Function}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
exports.compileETag = function(val) {
|
||
|
var fn;
|
||
|
|
||
|
if (typeof val === 'function') {
|
||
|
return val;
|
||
|
}
|
||
|
|
||
|
switch (val) {
|
||
|
case true:
|
||
|
case 'weak':
|
||
|
fn = exports.wetag;
|
||
|
break;
|
||
|
case false:
|
||
|
break;
|
||
|
case 'strong':
|
||
|
fn = exports.etag;
|
||
|
break;
|
||
|
default:
|
||
|
throw new TypeError('unknown value for etag function: ' + val);
|
||
|
}
|
||
|
|
||
|
return fn;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Compile "query parser" value to function.
|
||
|
*
|
||
|
* @param {String|Function} val
|
||
|
* @return {Function}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
exports.compileQueryParser = function compileQueryParser(val) {
|
||
|
var fn;
|
||
|
|
||
|
if (typeof val === 'function') {
|
||
|
return val;
|
||
|
}
|
||
|
|
||
|
switch (val) {
|
||
|
case true:
|
||
|
case 'simple':
|
||
|
fn = querystring.parse;
|
||
|
break;
|
||
|
case false:
|
||
|
fn = newObject;
|
||
|
break;
|
||
|
case 'extended':
|
||
|
fn = parseExtendedQueryString;
|
||
|
break;
|
||
|
default:
|
||
|
throw new TypeError('unknown value for query parser function: ' + val);
|
||
|
}
|
||
|
|
||
|
return fn;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Compile "proxy trust" value to function.
|
||
|
*
|
||
|
* @param {Boolean|String|Number|Array|Function} val
|
||
|
* @return {Function}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
exports.compileTrust = function(val) {
|
||
|
if (typeof val === 'function') return val;
|
||
|
|
||
|
if (val === true) {
|
||
|
// Support plain true/false
|
||
|
return function(){ return true };
|
||
|
}
|
||
|
|
||
|
if (typeof val === 'number') {
|
||
|
// Support trusting hop count
|
||
|
return function(a, i){ return i < val };
|
||
|
}
|
||
|
|
||
|
if (typeof val === 'string') {
|
||
|
// Support comma-separated values
|
||
|
val = val.split(',')
|
||
|
.map(function (v) { return v.trim() })
|
||
|
}
|
||
|
|
||
|
return proxyaddr.compile(val || []);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set the charset in a given Content-Type string.
|
||
|
*
|
||
|
* @param {String} type
|
||
|
* @param {String} charset
|
||
|
* @return {String}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
exports.setCharset = function setCharset(type, charset) {
|
||
|
if (!type || !charset) {
|
||
|
return type;
|
||
|
}
|
||
|
|
||
|
// parse type
|
||
|
var parsed = contentType.parse(type);
|
||
|
|
||
|
// set charset
|
||
|
parsed.parameters.charset = charset;
|
||
|
|
||
|
// format type
|
||
|
return contentType.format(parsed);
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Create an ETag generator function, generating ETags with
|
||
|
* the given options.
|
||
|
*
|
||
|
* @param {object} options
|
||
|
* @return {function}
|
||
|
* @private
|
||
|
*/
|
||
|
|
||
|
function createETagGenerator (options) {
|
||
|
return function generateETag (body, encoding) {
|
||
|
var buf = !Buffer.isBuffer(body)
|
||
|
? Buffer.from(body, encoding)
|
||
|
: body
|
||
|
|
||
|
return etag(buf, options)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Parse an extended query string with qs.
|
||
|
*
|
||
|
* @return {Object}
|
||
|
* @private
|
||
|
*/
|
||
|
|
||
|
function parseExtendedQueryString(str) {
|
||
|
return qs.parse(str, {
|
||
|
allowPrototypes: true
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return new empty object.
|
||
|
*
|
||
|
* @return {Object}
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
function newObject() {
|
||
|
return {};
|
||
|
}
|