fix(deps)!: mime-types@^3.0.0 (#5882)
[express.git] / lib / utils.js
blobf66760a17c033041e49948b17fb033f206c0a8c0
1 /*!
2 * express
3 * Copyright(c) 2009-2013 TJ Holowaychuk
4 * Copyright(c) 2014-2015 Douglas Christopher Wilson
5 * MIT Licensed
6 */
8 'use strict';
10 /**
11 * Module dependencies.
12 * @api private
15 var Buffer = require('safe-buffer').Buffer
16 var contentType = require('content-type');
17 var etag = require('etag');
18 var mime = require('mime-types')
19 var proxyaddr = require('proxy-addr');
20 var qs = require('qs');
21 var querystring = require('querystring');
23 /**
24 * Return strong ETag for `body`.
26 * @param {String|Buffer} body
27 * @param {String} [encoding]
28 * @return {String}
29 * @api private
32 exports.etag = createETagGenerator({ weak: false })
34 /**
35 * Return weak ETag for `body`.
37 * @param {String|Buffer} body
38 * @param {String} [encoding]
39 * @return {String}
40 * @api private
43 exports.wetag = createETagGenerator({ weak: true })
45 /**
46 * Normalize the given `type`, for example "html" becomes "text/html".
48 * @param {String} type
49 * @return {Object}
50 * @api private
53 exports.normalizeType = function(type){
54 return ~type.indexOf('/')
55 ? acceptParams(type)
56 : { value: (mime.lookup(type) || 'application/octet-stream'), params: {} }
59 /**
60 * Normalize `types`, for example "html" becomes "text/html".
62 * @param {Array} types
63 * @return {Array}
64 * @api private
67 exports.normalizeTypes = function(types){
68 var ret = [];
70 for (var i = 0; i < types.length; ++i) {
71 ret.push(exports.normalizeType(types[i]));
74 return ret;
77 /**
78 * Parse accept params `str` returning an
79 * object with `.value`, `.quality` and `.params`.
81 * @param {String} str
82 * @return {Object}
83 * @api private
86 function acceptParams (str) {
87 var parts = str.split(/ *; */);
88 var ret = { value: parts[0], quality: 1, params: {} }
90 for (var i = 1; i < parts.length; ++i) {
91 var pms = parts[i].split(/ *= */);
92 if ('q' === pms[0]) {
93 ret.quality = parseFloat(pms[1]);
94 } else {
95 ret.params[pms[0]] = pms[1];
99 return ret;
103 * Compile "etag" value to function.
105 * @param {Boolean|String|Function} val
106 * @return {Function}
107 * @api private
110 exports.compileETag = function(val) {
111 var fn;
113 if (typeof val === 'function') {
114 return val;
117 switch (val) {
118 case true:
119 case 'weak':
120 fn = exports.wetag;
121 break;
122 case false:
123 break;
124 case 'strong':
125 fn = exports.etag;
126 break;
127 default:
128 throw new TypeError('unknown value for etag function: ' + val);
131 return fn;
135 * Compile "query parser" value to function.
137 * @param {String|Function} val
138 * @return {Function}
139 * @api private
142 exports.compileQueryParser = function compileQueryParser(val) {
143 var fn;
145 if (typeof val === 'function') {
146 return val;
149 switch (val) {
150 case true:
151 case 'simple':
152 fn = querystring.parse;
153 break;
154 case false:
155 break;
156 case 'extended':
157 fn = parseExtendedQueryString;
158 break;
159 default:
160 throw new TypeError('unknown value for query parser function: ' + val);
163 return fn;
167 * Compile "proxy trust" value to function.
169 * @param {Boolean|String|Number|Array|Function} val
170 * @return {Function}
171 * @api private
174 exports.compileTrust = function(val) {
175 if (typeof val === 'function') return val;
177 if (val === true) {
178 // Support plain true/false
179 return function(){ return true };
182 if (typeof val === 'number') {
183 // Support trusting hop count
184 return function(a, i){ return i < val };
187 if (typeof val === 'string') {
188 // Support comma-separated values
189 val = val.split(',')
190 .map(function (v) { return v.trim() })
193 return proxyaddr.compile(val || []);
197 * Set the charset in a given Content-Type string.
199 * @param {String} type
200 * @param {String} charset
201 * @return {String}
202 * @api private
205 exports.setCharset = function setCharset(type, charset) {
206 if (!type || !charset) {
207 return type;
210 // parse type
211 var parsed = contentType.parse(type);
213 // set charset
214 parsed.parameters.charset = charset;
216 // format type
217 return contentType.format(parsed);
221 * Create an ETag generator function, generating ETags with
222 * the given options.
224 * @param {object} options
225 * @return {function}
226 * @private
229 function createETagGenerator (options) {
230 return function generateETag (body, encoding) {
231 var buf = !Buffer.isBuffer(body)
232 ? Buffer.from(body, encoding)
233 : body
235 return etag(buf, options)
240 * Parse an extended query string with qs.
242 * @param {String} str
243 * @return {Object}
244 * @private
247 function parseExtendedQueryString(str) {
248 return qs.parse(str, {
249 allowPrototypes: true