123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313 |
- /**
- * Module dependencies.
- */
- var mime = require('connect').mime
- , crc32 = require('buffer-crc32');
- /**
- * toString ref.
- */
- var toString = {}.toString;
- /**
- * Return ETag for `body`.
- *
- * @param {String|Buffer} body
- * @return {String}
- * @api private
- */
- exports.etag = function(body){
- return '"' + crc32.signed(body) + '"';
- };
- /**
- * Make `locals()` bound to the given `obj`.
- *
- * This is used for `app.locals` and `res.locals`.
- *
- * @param {Object} obj
- * @return {Function}
- * @api private
- */
- exports.locals = function(obj){
- function locals(obj){
- for (var key in obj) locals[key] = obj[key];
- return obj;
- };
- return locals;
- };
- /**
- * 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]) return true;
- };
- /**
- * Flatten the given `arr`.
- *
- * @param {Array} arr
- * @return {Array}
- * @api private
- */
- exports.flatten = function(arr, ret){
- var ret = ret || []
- , len = arr.length;
- for (var i = 0; i < len; ++i) {
- if (Array.isArray(arr[i])) {
- exports.flatten(arr[i], ret);
- } else {
- ret.push(arr[i]);
- }
- }
- return ret;
- };
- /**
- * 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;
- };
- /**
- * Return the acceptable type in `types`, if any.
- *
- * @param {Array} types
- * @param {String} str
- * @return {String}
- * @api private
- */
- exports.acceptsArray = function(types, str){
- // accept anything when Accept is not present
- if (!str) return types[0];
- // parse
- var accepted = exports.parseAccept(str)
- , normalized = exports.normalizeTypes(types)
- , len = accepted.length;
- for (var i = 0; i < len; ++i) {
- for (var j = 0, jlen = types.length; j < jlen; ++j) {
- if (exports.accept(normalized[j], accepted[i])) {
- return types[j];
- }
- }
- }
- };
- /**
- * Check if `type(s)` are acceptable based on
- * the given `str`.
- *
- * @param {String|Array} type(s)
- * @param {String} str
- * @return {Boolean|String}
- * @api private
- */
- exports.accepts = function(type, str){
- if ('string' == typeof type) type = type.split(/ *, */);
- return exports.acceptsArray(type, str);
- };
- /**
- * Check if `type` array is acceptable for `other`.
- *
- * @param {Object} type
- * @param {Object} other
- * @return {Boolean}
- * @api private
- */
- exports.accept = function(type, other){
- var t = type.value.split('/');
- return (t[0] == other.type || '*' == other.type)
- && (t[1] == other.subtype || '*' == other.subtype)
- && paramsEqual(type.params, other.params);
- };
- /**
- * Check if accept params are equal.
- *
- * @param {Object} a
- * @param {Object} b
- * @return {Boolean}
- * @api private
- */
- function paramsEqual(a, b){
- return !Object.keys(a).some(function(k) {
- return a[k] != b[k];
- });
- }
- /**
- * Parse accept `str`, returning
- * an array objects containing
- * `.type` and `.subtype` along
- * with the values provided by
- * `parseQuality()`.
- *
- * @param {Type} name
- * @return {Type}
- * @api private
- */
- exports.parseAccept = function(str){
- return exports
- .parseParams(str)
- .map(function(obj){
- var parts = obj.value.split('/');
- obj.type = parts[0];
- obj.subtype = parts[1];
- return obj;
- });
- };
- /**
- * Parse quality `str`, returning an
- * array of objects with `.value`,
- * `.quality` and optional `.params`
- *
- * @param {String} str
- * @return {Array}
- * @api private
- */
- exports.parseParams = function(str){
- return str
- .split(/ *, */)
- .map(acceptParams)
- .filter(function(obj){
- return obj.quality;
- })
- .sort(function(a, b){
- if (a.quality === b.quality) {
- return a.originalIndex - b.originalIndex;
- } else {
- return b.quality - a.quality;
- }
- });
- };
- /**
- * 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;
- }
- /**
- * Escape special characters in the given string of html.
- *
- * @param {String} html
- * @return {String}
- * @api private
- */
- exports.escape = function(html) {
- return String(html)
- .replace(/&/g, '&')
- .replace(/"/g, '"')
- .replace(/</g, '<')
- .replace(/>/g, '>');
- };
- /**
- * Normalize the given path string,
- * returning a regular expression.
- *
- * An empty array should be passed,
- * which will contain the placeholder
- * key names. For example "/user/:id" will
- * then contain ["id"].
- *
- * @param {String|RegExp|Array} path
- * @param {Array} keys
- * @param {Boolean} sensitive
- * @param {Boolean} strict
- * @return {RegExp}
- * @api private
- */
- exports.pathRegexp = function(path, keys, sensitive, strict) {
- if (toString.call(path) == '[object RegExp]') return path;
- if (Array.isArray(path)) path = '(' + path.join('|') + ')';
- path = path
- .concat(strict ? '' : '/?')
- .replace(/\/\(/g, '(?:/')
- .replace(/(\/)?(\.)?:(\w+)(?:(\(.*?\)))?(\?)?(\*)?/g, function(_, slash, format, key, capture, optional, star){
- keys.push({ name: key, optional: !! optional });
- slash = slash || '';
- return ''
- + (optional ? '' : slash)
- + '(?:'
- + (optional ? slash : '')
- + (format || '') + (capture || (format && '([^/.]+?)' || '([^/]+?)')) + ')'
- + (optional || '')
- + (star ? '(/*)?' : '');
- })
- .replace(/([\/.])/g, '\\$1')
- .replace(/\*/g, '(.*)');
- return new RegExp('^' + path + '$', sensitive ? '' : 'i');
- }
|