206 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			206 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
/**
 | 
						|
 * 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 = {encoding: encoding, 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 {
 | 
						|
    encoding: encoding,
 | 
						|
    i: index,
 | 
						|
    o: spec.i,
 | 
						|
    q: spec.q,
 | 
						|
    s: s
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
/**
 | 
						|
 * Get the preferred encodings from an Accept-Encoding header.
 | 
						|
 * @public
 | 
						|
 */
 | 
						|
 | 
						|
function preferredEncodings(accept, provided, preferred) {
 | 
						|
  var accepts = parseAcceptEncoding(accept || '');
 | 
						|
 | 
						|
  var comparator = preferred ? function comparator (a, b) {
 | 
						|
    if (a.q !== b.q) {
 | 
						|
      return b.q - a.q // higher quality first
 | 
						|
    }
 | 
						|
 | 
						|
    var aPreferred = preferred.indexOf(a.encoding)
 | 
						|
    var bPreferred = preferred.indexOf(b.encoding)
 | 
						|
 | 
						|
    if (aPreferred === -1 && bPreferred === -1) {
 | 
						|
      // consider the original specifity/order
 | 
						|
      return (b.s - a.s) || (a.o - b.o) || (a.i - b.i)
 | 
						|
    }
 | 
						|
 | 
						|
    if (aPreferred !== -1 && bPreferred !== -1) {
 | 
						|
      return aPreferred - bPreferred // consider the preferred order
 | 
						|
    }
 | 
						|
 | 
						|
    return aPreferred === -1 ? 1 : -1 // preferred first
 | 
						|
  } : compareSpecs;
 | 
						|
 | 
						|
  if (!provided) {
 | 
						|
    // sorted list of all encodings
 | 
						|
    return accepts
 | 
						|
      .filter(isQuality)
 | 
						|
      .sort(comparator)
 | 
						|
      .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(comparator).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);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * 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;
 | 
						|
}
 |