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.
		
		
		
		
		
			
		
			
				
					
					
						
							306 lines
						
					
					
						
							5.8 KiB
						
					
					
				
			
		
		
	
	
							306 lines
						
					
					
						
							5.8 KiB
						
					
					
				/*! | 
						|
 * 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 | 
						|
 * @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: | 
						|
      fn = exports.wetag; | 
						|
      break; | 
						|
    case false: | 
						|
      break; | 
						|
    case 'strong': | 
						|
      fn = exports.etag; | 
						|
      break; | 
						|
    case 'weak': | 
						|
      fn = exports.wetag; | 
						|
      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: | 
						|
      fn = querystring.parse; | 
						|
      break; | 
						|
    case false: | 
						|
      fn = newObject; | 
						|
      break; | 
						|
    case 'extended': | 
						|
      fn = parseExtendedQueryString; | 
						|
      break; | 
						|
    case 'simple': | 
						|
      fn = querystring.parse; | 
						|
      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(/ *, */); | 
						|
  } | 
						|
 | 
						|
  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 {}; | 
						|
}
 | 
						|
 |