UNPKG

4.65 kBJavaScriptView Raw
1/**
2 * Module dependencies.
3 */
4
5var mime = require('send').mime;
6var crc32 = require('buffer-crc32');
7var basename = require('path').basename;
8var deprecate = require('util').deprecate;
9var proxyaddr = require('proxy-addr');
10
11/**
12 * Simple detection of charset parameter in content-type
13 */
14var charsetRegExp = /;\s*charset\s*=/;
15
16/**
17 * Deprecate function, like core `util.deprecate`,
18 * but with NODE_ENV and color support.
19 *
20 * @param {Function} fn
21 * @param {String} msg
22 * @return {Function}
23 * @api private
24 */
25
26exports.deprecate = function(fn, msg){
27 if (process.env.NODE_ENV === 'test') return fn;
28
29 // prepend module name
30 msg = 'express: ' + msg;
31
32 if (process.stderr.isTTY) {
33 // colorize
34 msg = '\x1b[31;1m' + msg + '\x1b[0m';
35 }
36
37 return deprecate(fn, msg);
38};
39
40/**
41 * Return ETag for `body`.
42 *
43 * @param {String|Buffer} body
44 * @return {String}
45 * @api private
46 */
47
48exports.etag = function(body){
49 return '"' + crc32.signed(body) + '"';
50};
51
52/**
53 * Check if `path` looks absolute.
54 *
55 * @param {String} path
56 * @return {Boolean}
57 * @api private
58 */
59
60exports.isAbsolute = function(path){
61 if ('/' == path[0]) return true;
62 if (':' == path[1] && '\\' == path[2]) return true;
63 if ('\\\\' == path.substring(0, 2)) return true; // Microsoft Azure absolute path
64};
65
66/**
67 * Flatten the given `arr`.
68 *
69 * @param {Array} arr
70 * @return {Array}
71 * @api private
72 */
73
74exports.flatten = function(arr, ret){
75 ret = ret || [];
76 var len = arr.length;
77 for (var i = 0; i < len; ++i) {
78 if (Array.isArray(arr[i])) {
79 exports.flatten(arr[i], ret);
80 } else {
81 ret.push(arr[i]);
82 }
83 }
84 return ret;
85};
86
87/**
88 * Normalize the given `type`, for example "html" becomes "text/html".
89 *
90 * @param {String} type
91 * @return {Object}
92 * @api private
93 */
94
95exports.normalizeType = function(type){
96 return ~type.indexOf('/')
97 ? acceptParams(type)
98 : { value: mime.lookup(type), params: {} };
99};
100
101/**
102 * Normalize `types`, for example "html" becomes "text/html".
103 *
104 * @param {Array} types
105 * @return {Array}
106 * @api private
107 */
108
109exports.normalizeTypes = function(types){
110 var ret = [];
111
112 for (var i = 0; i < types.length; ++i) {
113 ret.push(exports.normalizeType(types[i]));
114 }
115
116 return ret;
117};
118
119/**
120 * Generate Content-Disposition header appropriate for the filename.
121 * non-ascii filenames are urlencoded and a filename* parameter is added
122 *
123 * @param {String} filename
124 * @return {String}
125 * @api private
126 */
127
128exports.contentDisposition = function(filename){
129 var ret = 'attachment';
130 if (filename) {
131 filename = basename(filename);
132 // if filename contains non-ascii characters, add a utf-8 version ala RFC 5987
133 ret = /[^\040-\176]/.test(filename)
134 ? 'attachment; filename=' + encodeURI(filename) + '; filename*=UTF-8\'\'' + encodeURI(filename)
135 : 'attachment; filename="' + filename + '"';
136 }
137
138 return ret;
139};
140
141/**
142 * Parse accept params `str` returning an
143 * object with `.value`, `.quality` and `.params`.
144 * also includes `.originalIndex` for stable sorting
145 *
146 * @param {String} str
147 * @return {Object}
148 * @api private
149 */
150
151function acceptParams(str, index) {
152 var parts = str.split(/ *; */);
153 var ret = { value: parts[0], quality: 1, params: {}, originalIndex: index };
154
155 for (var i = 1; i < parts.length; ++i) {
156 var pms = parts[i].split(/ *= */);
157 if ('q' == pms[0]) {
158 ret.quality = parseFloat(pms[1]);
159 } else {
160 ret.params[pms[0]] = pms[1];
161 }
162 }
163
164 return ret;
165}
166
167/**
168 * Compile "proxy trust" value to function.
169 *
170 * @param {Boolean|String|Number|Array|Function} val
171 * @return {Function}
172 * @api private
173 */
174
175exports.compileTrust = function(val) {
176 if (typeof val === 'function') return val;
177
178 if (val === true) {
179 // Support plain true/false
180 return function(){ return true };
181 }
182
183 if (typeof val === 'number') {
184 // Support trusting hop count
185 return function(a, i){ return i < val };
186 }
187
188 if (typeof val === 'string') {
189 // Support comma-separated values
190 val = val.split(/ *, */);
191 }
192
193 return proxyaddr.compile(val || []);
194}
195
196/**
197 * Set the charset in a given Content-Type string.
198 *
199 * @param {String} type
200 * @param {String} charset
201 * @return {String}
202 * @api private
203 */
204
205exports.setCharset = function(type, charset){
206 if (!type || !charset) return type;
207
208 var exists = charsetRegExp.test(type);
209
210 // removing existing charset
211 if (exists) {
212 var parts = type.split(';');
213
214 for (var i = 1; i < parts.length; i++) {
215 if (charsetRegExp.test(';' + parts[i])) {
216 parts.splice(i, 1);
217 break;
218 }
219 }
220
221 type = parts.join(';');
222 }
223
224 return type + '; charset=' + charset;
225};