| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184 | /** * negotiator * Copyright(c) 2012 Isaac Z. Schlueter * Copyright(c) 2014 Federico Romero * Copyright(c) 2014-2015 Douglas Christopher Wilson * MIT Licensed */'use strict';/** * Module exports. * @public */module.exports = preferredEncodings;module.exports.preferredEncodings = preferredEncodings;/** * Module variables. * @private */var simpleEncodingRegExp = /^\s*([^\s;]+)\s*(?:;(.*))?$/;/** * Parse the Accept-Encoding header. * @private */function parseAcceptEncoding(accept) {  var accepts = accept.split(',');  var hasIdentity = false;  var minQuality = 1;  for (var i = 0, j = 0; i < accepts.length; i++) {    var encoding = parseEncoding(accepts[i].trim(), i);    if (encoding) {      accepts[j++] = encoding;      hasIdentity = hasIdentity || specify('identity', encoding);      minQuality = Math.min(minQuality, encoding.q || 1);    }  }  if (!hasIdentity) {    /*     * If identity doesn't explicitly appear in the accept-encoding header,     * it's added to the list of acceptable encoding with the lowest q     */    accepts[j++] = {      encoding: 'identity',      q: minQuality,      i: i    };  }  // trim accepts  accepts.length = j;  return accepts;}/** * Parse an encoding from the Accept-Encoding header. * @private */function parseEncoding(str, i) {  var match = simpleEncodingRegExp.exec(str);  if (!match) return null;  var encoding = match[1];  var q = 1;  if (match[2]) {    var params = match[2].split(';');    for (var j = 0; j < params.length; j++) {      var p = params[j].trim().split('=');      if (p[0] === 'q') {        q = parseFloat(p[1]);        break;      }    }  }  return {    encoding: encoding,    q: q,    i: i  };}/** * Get the priority of an encoding. * @private */function getEncodingPriority(encoding, accepted, index) {  var priority = {o: -1, q: 0, s: 0};  for (var i = 0; i < accepted.length; i++) {    var spec = specify(encoding, accepted[i], index);    if (spec && (priority.s - spec.s || priority.q - spec.q || priority.o - spec.o) < 0) {      priority = spec;    }  }  return priority;}/** * Get the specificity of the encoding. * @private */function specify(encoding, spec, index) {  var s = 0;  if(spec.encoding.toLowerCase() === encoding.toLowerCase()){    s |= 1;  } else if (spec.encoding !== '*' ) {    return null  }  return {    i: index,    o: spec.i,    q: spec.q,    s: s  }};/** * Get the preferred encodings from an Accept-Encoding header. * @public */function preferredEncodings(accept, provided) {  var accepts = parseAcceptEncoding(accept || '');  if (!provided) {    // sorted list of all encodings    return accepts      .filter(isQuality)      .sort(compareSpecs)      .map(getFullEncoding);  }  var priorities = provided.map(function getPriority(type, index) {    return getEncodingPriority(type, accepts, index);  });  // sorted list of accepted encodings  return priorities.filter(isQuality).sort(compareSpecs).map(function getEncoding(priority) {    return provided[priorities.indexOf(priority)];  });}/** * Compare two specs. * @private */function compareSpecs(a, b) {  return (b.q - a.q) || (b.s - a.s) || (a.o - b.o) || (a.i - b.i) || 0;}/** * Get full encoding string. * @private */function getFullEncoding(spec) {  return spec.encoding;}/** * Check if a spec has any quality. * @private */function isQuality(spec) {  return spec.q > 0;}
 |