123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205 |
- /*
- MIT License http://www.opensource.org/licenses/mit-license.php
- Author Tobias Koppers @sokra
- */
- "use strict";
- const globToRegExp = require("./globToRegExp").globToRegExp;
- function parseType(type) {
- const items = type.split("+");
- const t = items.shift();
- return {
- type: t === "*" ? null : t,
- features: items
- };
- }
- function isTypeMatched(baseType, testedType) {
- if (typeof baseType === "string") baseType = parseType(baseType);
- if (typeof testedType === "string") testedType = parseType(testedType);
- if (testedType.type && testedType.type !== baseType.type) return false;
- return testedType.features.every(requiredFeature => {
- return baseType.features.indexOf(requiredFeature) >= 0;
- });
- }
- function isResourceTypeMatched(baseType, testedType) {
- baseType = baseType.split("/");
- testedType = testedType.split("/");
- if (baseType.length !== testedType.length) return false;
- for (let i = 0; i < baseType.length; i++) {
- if (!isTypeMatched(baseType[i], testedType[i])) return false;
- }
- return true;
- }
- function isResourceTypeSupported(context, type) {
- return (
- context.supportedResourceTypes &&
- context.supportedResourceTypes.some(supportedType => {
- return isResourceTypeMatched(supportedType, type);
- })
- );
- }
- function isEnvironment(context, env) {
- return (
- context.environments &&
- context.environments.every(environment => {
- return isTypeMatched(environment, env);
- })
- );
- }
- const globCache = {};
- function getGlobRegExp(glob) {
- const regExp = globCache[glob] || (globCache[glob] = globToRegExp(glob));
- return regExp;
- }
- function matchGlob(glob, relativePath) {
- const regExp = getGlobRegExp(glob);
- return regExp.exec(relativePath);
- }
- function isGlobMatched(glob, relativePath) {
- return !!matchGlob(glob, relativePath);
- }
- function isConditionMatched(context, condition) {
- const items = condition.split("|");
- return items.some(function testFn(item) {
- item = item.trim();
- const inverted = /^!/.test(item);
- if (inverted) return !testFn(item.substr(1));
- if (/^[a-z]+:/.test(item)) {
- // match named condition
- const match = /^([a-z]+):\s*/.exec(item);
- const value = item.substr(match[0].length);
- const name = match[1];
- switch (name) {
- case "referrer":
- return isGlobMatched(value, context.referrer);
- default:
- return false;
- }
- } else if (item.indexOf("/") >= 0) {
- // match supported type
- return isResourceTypeSupported(context, item);
- } else {
- // match environment
- return isEnvironment(context, item);
- }
- });
- }
- function isKeyMatched(context, key) {
- for (;;) {
- const match = /^\[([^\]]+)\]\s*/.exec(key);
- if (!match) return key;
- key = key.substr(match[0].length);
- const condition = match[1];
- if (!isConditionMatched(context, condition)) {
- return false;
- }
- }
- }
- function getField(context, configuration, field) {
- let value;
- Object.keys(configuration).forEach(key => {
- const pureKey = isKeyMatched(context, key);
- if (pureKey === field) {
- value = configuration[key];
- }
- });
- return value;
- }
- function getMain(context, configuration) {
- return getField(context, configuration, "main");
- }
- function getExtensions(context, configuration) {
- return getField(context, configuration, "extensions");
- }
- function matchModule(context, configuration, request) {
- const modulesField = getField(context, configuration, "modules");
- if (!modulesField) return request;
- let newRequest = request;
- const keys = Object.keys(modulesField);
- let iteration = 0;
- let match;
- let index;
- for (let i = 0; i < keys.length; i++) {
- const key = keys[i];
- const pureKey = isKeyMatched(context, key);
- match = matchGlob(pureKey, newRequest);
- if (match) {
- const value = modulesField[key];
- if (typeof value !== "string") {
- return value;
- } else if (/^\(.+\)$/.test(pureKey)) {
- newRequest = newRequest.replace(getGlobRegExp(pureKey), value);
- } else {
- index = 1;
- newRequest = value.replace(/(\/?\*)?\*/g, replaceMatcher);
- }
- i = -1;
- if (iteration++ > keys.length) {
- throw new Error("Request '" + request + "' matches recursively");
- }
- }
- }
- return newRequest;
- function replaceMatcher(find) {
- switch (find) {
- case "/**": {
- const m = match[index++];
- return m ? "/" + m : "";
- }
- case "**":
- case "*":
- return match[index++];
- }
- }
- }
- function matchType(context, configuration, relativePath) {
- const typesField = getField(context, configuration, "types");
- if (!typesField) return undefined;
- let type;
- Object.keys(typesField).forEach(key => {
- const pureKey = isKeyMatched(context, key);
- if (isGlobMatched(pureKey, relativePath)) {
- const value = typesField[key];
- if (!type && /\/\*$/.test(value))
- throw new Error(
- "value ('" +
- value +
- "') of key '" +
- key +
- "' contains '*', but there is no previous value defined"
- );
- type = value.replace(/\/\*$/, "/" + type);
- }
- });
- return type;
- }
- exports.parseType = parseType;
- exports.isTypeMatched = isTypeMatched;
- exports.isResourceTypeSupported = isResourceTypeSupported;
- exports.isEnvironment = isEnvironment;
- exports.isGlobMatched = isGlobMatched;
- exports.isConditionMatched = isConditionMatched;
- exports.isKeyMatched = isKeyMatched;
- exports.getField = getField;
- exports.getMain = getMain;
- exports.getExtensions = getExtensions;
- exports.matchModule = matchModule;
- exports.matchType = matchType;
|