You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					408 lines
				
				11 KiB
			
		
		
			
		
	
	
					408 lines
				
				11 KiB
			| 
											3 years ago
										 | var feature = require('caniuse-lite/dist/unpacker/feature').default | ||
|  | var region = require('caniuse-lite/dist/unpacker/region').default | ||
|  | var path = require('path') | ||
|  | var fs = require('fs') | ||
|  | 
 | ||
|  | var BrowserslistError = require('./error') | ||
|  | 
 | ||
|  | var IS_SECTION = /^\s*\[(.+)]\s*$/ | ||
|  | var CONFIG_PATTERN = /^browserslist-config-/ | ||
|  | var SCOPED_CONFIG__PATTERN = /@[^/]+\/browserslist-config(-|$|\/)/ | ||
|  | var TIME_TO_UPDATE_CANIUSE = 6 * 30 * 24 * 60 * 60 * 1000 | ||
|  | var FORMAT = | ||
|  |   'Browserslist config should be a string or an array ' + | ||
|  |   'of strings with browser queries' | ||
|  | 
 | ||
|  | var dataTimeChecked = false | ||
|  | var filenessCache = {} | ||
|  | var configCache = {} | ||
|  | function checkExtend(name) { | ||
|  |   var use = ' Use `dangerousExtend` option to disable.' | ||
|  |   if (!CONFIG_PATTERN.test(name) && !SCOPED_CONFIG__PATTERN.test(name)) { | ||
|  |     throw new BrowserslistError( | ||
|  |       'Browserslist config needs `browserslist-config-` prefix. ' + use | ||
|  |     ) | ||
|  |   } | ||
|  |   if (name.replace(/^@[^/]+\//, '').indexOf('.') !== -1) { | ||
|  |     throw new BrowserslistError( | ||
|  |       '`.` not allowed in Browserslist config name. ' + use | ||
|  |     ) | ||
|  |   } | ||
|  |   if (name.indexOf('node_modules') !== -1) { | ||
|  |     throw new BrowserslistError( | ||
|  |       '`node_modules` not allowed in Browserslist config.' + use | ||
|  |     ) | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function isFile(file) { | ||
|  |   if (file in filenessCache) { | ||
|  |     return filenessCache[file] | ||
|  |   } | ||
|  |   var result = fs.existsSync(file) && fs.statSync(file).isFile() | ||
|  |   if (!process.env.BROWSERSLIST_DISABLE_CACHE) { | ||
|  |     filenessCache[file] = result | ||
|  |   } | ||
|  |   return result | ||
|  | } | ||
|  | 
 | ||
|  | function eachParent(file, callback) { | ||
|  |   var dir = isFile(file) ? path.dirname(file) : file | ||
|  |   var loc = path.resolve(dir) | ||
|  |   do { | ||
|  |     var result = callback(loc) | ||
|  |     if (typeof result !== 'undefined') return result | ||
|  |   } while (loc !== (loc = path.dirname(loc))) | ||
|  |   return undefined | ||
|  | } | ||
|  | 
 | ||
|  | function check(section) { | ||
|  |   if (Array.isArray(section)) { | ||
|  |     for (var i = 0; i < section.length; i++) { | ||
|  |       if (typeof section[i] !== 'string') { | ||
|  |         throw new BrowserslistError(FORMAT) | ||
|  |       } | ||
|  |     } | ||
|  |   } else if (typeof section !== 'string') { | ||
|  |     throw new BrowserslistError(FORMAT) | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function pickEnv(config, opts) { | ||
|  |   if (typeof config !== 'object') return config | ||
|  | 
 | ||
|  |   var name | ||
|  |   if (typeof opts.env === 'string') { | ||
|  |     name = opts.env | ||
|  |   } else if (process.env.BROWSERSLIST_ENV) { | ||
|  |     name = process.env.BROWSERSLIST_ENV | ||
|  |   } else if (process.env.NODE_ENV) { | ||
|  |     name = process.env.NODE_ENV | ||
|  |   } else { | ||
|  |     name = 'production' | ||
|  |   } | ||
|  | 
 | ||
|  |   if (opts.throwOnMissing) { | ||
|  |     if (name && name !== 'defaults' && !config[name]) { | ||
|  |       throw new BrowserslistError( | ||
|  |         'Missing config for Browserslist environment `' + name + '`' | ||
|  |       ) | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   return config[name] || config.defaults | ||
|  | } | ||
|  | 
 | ||
|  | function parsePackage(file) { | ||
|  |   var config = JSON.parse( | ||
|  |     fs | ||
|  |       .readFileSync(file) | ||
|  |       .toString() | ||
|  |       .replace(/^\uFEFF/m, '') | ||
|  |   ) | ||
|  |   if (config.browserlist && !config.browserslist) { | ||
|  |     throw new BrowserslistError( | ||
|  |       '`browserlist` key instead of `browserslist` in ' + file | ||
|  |     ) | ||
|  |   } | ||
|  |   var list = config.browserslist | ||
|  |   if (Array.isArray(list) || typeof list === 'string') { | ||
|  |     list = { defaults: list } | ||
|  |   } | ||
|  |   for (var i in list) { | ||
|  |     check(list[i]) | ||
|  |   } | ||
|  | 
 | ||
|  |   return list | ||
|  | } | ||
|  | 
 | ||
|  | function latestReleaseTime(agents) { | ||
|  |   var latest = 0 | ||
|  |   for (var name in agents) { | ||
|  |     var dates = agents[name].releaseDate || {} | ||
|  |     for (var key in dates) { | ||
|  |       if (latest < dates[key]) { | ||
|  |         latest = dates[key] | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  |   return latest * 1000 | ||
|  | } | ||
|  | 
 | ||
|  | function normalizeStats(data, stats) { | ||
|  |   if (!data) { | ||
|  |     data = {} | ||
|  |   } | ||
|  |   if (stats && 'dataByBrowser' in stats) { | ||
|  |     stats = stats.dataByBrowser | ||
|  |   } | ||
|  | 
 | ||
|  |   if (typeof stats !== 'object') return undefined | ||
|  | 
 | ||
|  |   var normalized = {} | ||
|  |   for (var i in stats) { | ||
|  |     var versions = Object.keys(stats[i]) | ||
|  |     if (versions.length === 1 && data[i] && data[i].versions.length === 1) { | ||
|  |       var normal = data[i].versions[0] | ||
|  |       normalized[i] = {} | ||
|  |       normalized[i][normal] = stats[i][versions[0]] | ||
|  |     } else { | ||
|  |       normalized[i] = stats[i] | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   return normalized | ||
|  | } | ||
|  | 
 | ||
|  | function normalizeUsageData(usageData, data) { | ||
|  |   for (var browser in usageData) { | ||
|  |     var browserUsage = usageData[browser] | ||
|  |     // eslint-disable-next-line max-len
 | ||
|  |     // https://github.com/browserslist/browserslist/issues/431#issuecomment-565230615
 | ||
|  |     // caniuse-db returns { 0: "percentage" } for `and_*` regional stats
 | ||
|  |     if ('0' in browserUsage) { | ||
|  |       var versions = data[browser].versions | ||
|  |       browserUsage[versions[versions.length - 1]] = browserUsage[0] | ||
|  |       delete browserUsage[0] | ||
|  |     } | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | module.exports = { | ||
|  |   loadQueries: function loadQueries(ctx, name) { | ||
|  |     if (!ctx.dangerousExtend && !process.env.BROWSERSLIST_DANGEROUS_EXTEND) { | ||
|  |       checkExtend(name) | ||
|  |     } | ||
|  |     var queries = require(require.resolve(name, { paths: ['.', ctx.path] })) | ||
|  |     if (queries) { | ||
|  |       if (Array.isArray(queries)) { | ||
|  |         return queries | ||
|  |       } else if (typeof queries === 'object') { | ||
|  |         if (!queries.defaults) queries.defaults = [] | ||
|  |         return pickEnv(queries, ctx, name) | ||
|  |       } | ||
|  |     } | ||
|  |     throw new BrowserslistError( | ||
|  |       '`' + | ||
|  |         name + | ||
|  |         '` config exports not an array of queries' + | ||
|  |         ' or an object of envs' | ||
|  |     ) | ||
|  |   }, | ||
|  | 
 | ||
|  |   loadStat: function loadStat(ctx, name, data) { | ||
|  |     if (!ctx.dangerousExtend && !process.env.BROWSERSLIST_DANGEROUS_EXTEND) { | ||
|  |       checkExtend(name) | ||
|  |     } | ||
|  |     var stats = require(require.resolve( | ||
|  |       path.join(name, 'browserslist-stats.json'), | ||
|  |       { paths: ['.'] } | ||
|  |     )) | ||
|  |     return normalizeStats(data, stats) | ||
|  |   }, | ||
|  | 
 | ||
|  |   getStat: function getStat(opts, data) { | ||
|  |     var stats | ||
|  |     if (opts.stats) { | ||
|  |       stats = opts.stats | ||
|  |     } else if (process.env.BROWSERSLIST_STATS) { | ||
|  |       stats = process.env.BROWSERSLIST_STATS | ||
|  |     } else if (opts.path && path.resolve && fs.existsSync) { | ||
|  |       stats = eachParent(opts.path, function (dir) { | ||
|  |         var file = path.join(dir, 'browserslist-stats.json') | ||
|  |         return isFile(file) ? file : undefined | ||
|  |       }) | ||
|  |     } | ||
|  |     if (typeof stats === 'string') { | ||
|  |       try { | ||
|  |         stats = JSON.parse(fs.readFileSync(stats)) | ||
|  |       } catch (e) { | ||
|  |         throw new BrowserslistError("Can't read " + stats) | ||
|  |       } | ||
|  |     } | ||
|  |     return normalizeStats(data, stats) | ||
|  |   }, | ||
|  | 
 | ||
|  |   loadConfig: function loadConfig(opts) { | ||
|  |     if (process.env.BROWSERSLIST) { | ||
|  |       return process.env.BROWSERSLIST | ||
|  |     } else if (opts.config || process.env.BROWSERSLIST_CONFIG) { | ||
|  |       var file = opts.config || process.env.BROWSERSLIST_CONFIG | ||
|  |       if (path.basename(file) === 'package.json') { | ||
|  |         return pickEnv(parsePackage(file), opts) | ||
|  |       } else { | ||
|  |         return pickEnv(module.exports.readConfig(file), opts) | ||
|  |       } | ||
|  |     } else if (opts.path) { | ||
|  |       return pickEnv(module.exports.findConfig(opts.path), opts) | ||
|  |     } else { | ||
|  |       return undefined | ||
|  |     } | ||
|  |   }, | ||
|  | 
 | ||
|  |   loadCountry: function loadCountry(usage, country, data) { | ||
|  |     var code = country.replace(/[^\w-]/g, '') | ||
|  |     if (!usage[code]) { | ||
|  |       var compressed | ||
|  |       try { | ||
|  |         compressed = require('caniuse-lite/data/regions/' + code + '.js') | ||
|  |       } catch (e) { | ||
|  |         throw new BrowserslistError("Unknown region name `" + code + "`.") | ||
|  |       } | ||
|  |       var usageData = region(compressed) | ||
|  |       normalizeUsageData(usageData, data) | ||
|  |       usage[country] = {} | ||
|  |       for (var i in usageData) { | ||
|  |         for (var j in usageData[i]) { | ||
|  |           usage[country][i + ' ' + j] = usageData[i][j] | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  | 
 | ||
|  |   loadFeature: function loadFeature(features, name) { | ||
|  |     name = name.replace(/[^\w-]/g, '') | ||
|  |     if (features[name]) return | ||
|  |     var compressed | ||
|  |     try { | ||
|  |       compressed = require('caniuse-lite/data/features/' + name + '.js') | ||
|  |     } catch (e) { | ||
|  |       throw new BrowserslistError("Unknown feature name `" + name + "`.") | ||
|  |     } | ||
|  |     var stats = feature(compressed).stats | ||
|  |     features[name] = {} | ||
|  |     for (var i in stats) { | ||
|  |       for (var j in stats[i]) { | ||
|  |         features[name][i + ' ' + j] = stats[i][j] | ||
|  |       } | ||
|  |     } | ||
|  |   }, | ||
|  | 
 | ||
|  |   parseConfig: function parseConfig(string) { | ||
|  |     var result = { defaults: [] } | ||
|  |     var sections = ['defaults'] | ||
|  | 
 | ||
|  |     string | ||
|  |       .toString() | ||
|  |       .replace(/#[^\n]*/g, '') | ||
|  |       .split(/\n|,/) | ||
|  |       .map(function (line) { | ||
|  |         return line.trim() | ||
|  |       }) | ||
|  |       .filter(function (line) { | ||
|  |         return line !== '' | ||
|  |       }) | ||
|  |       .forEach(function (line) { | ||
|  |         if (IS_SECTION.test(line)) { | ||
|  |           sections = line.match(IS_SECTION)[1].trim().split(' ') | ||
|  |           sections.forEach(function (section) { | ||
|  |             if (result[section]) { | ||
|  |               throw new BrowserslistError( | ||
|  |                 'Duplicate section ' + section + ' in Browserslist config' | ||
|  |               ) | ||
|  |             } | ||
|  |             result[section] = [] | ||
|  |           }) | ||
|  |         } else { | ||
|  |           sections.forEach(function (section) { | ||
|  |             result[section].push(line) | ||
|  |           }) | ||
|  |         } | ||
|  |       }) | ||
|  | 
 | ||
|  |     return result | ||
|  |   }, | ||
|  | 
 | ||
|  |   readConfig: function readConfig(file) { | ||
|  |     if (!isFile(file)) { | ||
|  |       throw new BrowserslistError("Can't read " + file + ' config') | ||
|  |     } | ||
|  |     return module.exports.parseConfig(fs.readFileSync(file)) | ||
|  |   }, | ||
|  | 
 | ||
|  |   findConfig: function findConfig(from) { | ||
|  |     from = path.resolve(from) | ||
|  | 
 | ||
|  |     var passed = [] | ||
|  |     var resolved = eachParent(from, function (dir) { | ||
|  |       if (dir in configCache) { | ||
|  |         return configCache[dir] | ||
|  |       } | ||
|  | 
 | ||
|  |       passed.push(dir) | ||
|  | 
 | ||
|  |       var config = path.join(dir, 'browserslist') | ||
|  |       var pkg = path.join(dir, 'package.json') | ||
|  |       var rc = path.join(dir, '.browserslistrc') | ||
|  | 
 | ||
|  |       var pkgBrowserslist | ||
|  |       if (isFile(pkg)) { | ||
|  |         try { | ||
|  |           pkgBrowserslist = parsePackage(pkg) | ||
|  |         } catch (e) { | ||
|  |           if (e.name === 'BrowserslistError') throw e | ||
|  |           console.warn( | ||
|  |             '[Browserslist] Could not parse ' + pkg + '. Ignoring it.' | ||
|  |           ) | ||
|  |         } | ||
|  |       } | ||
|  | 
 | ||
|  |       if (isFile(config) && pkgBrowserslist) { | ||
|  |         throw new BrowserslistError( | ||
|  |           dir + ' contains both browserslist and package.json with browsers' | ||
|  |         ) | ||
|  |       } else if (isFile(rc) && pkgBrowserslist) { | ||
|  |         throw new BrowserslistError( | ||
|  |           dir + ' contains both .browserslistrc and package.json with browsers' | ||
|  |         ) | ||
|  |       } else if (isFile(config) && isFile(rc)) { | ||
|  |         throw new BrowserslistError( | ||
|  |           dir + ' contains both .browserslistrc and browserslist' | ||
|  |         ) | ||
|  |       } else if (isFile(config)) { | ||
|  |         return module.exports.readConfig(config) | ||
|  |       } else if (isFile(rc)) { | ||
|  |         return module.exports.readConfig(rc) | ||
|  |       } else { | ||
|  |         return pkgBrowserslist | ||
|  |       } | ||
|  |     }) | ||
|  |     if (!process.env.BROWSERSLIST_DISABLE_CACHE) { | ||
|  |       passed.forEach(function (dir) { | ||
|  |         configCache[dir] = resolved | ||
|  |       }) | ||
|  |     } | ||
|  |     return resolved | ||
|  |   }, | ||
|  | 
 | ||
|  |   clearCaches: function clearCaches() { | ||
|  |     dataTimeChecked = false | ||
|  |     filenessCache = {} | ||
|  |     configCache = {} | ||
|  | 
 | ||
|  |     this.cache = {} | ||
|  |   }, | ||
|  | 
 | ||
|  |   oldDataWarning: function oldDataWarning(agentsObj) { | ||
|  |     if (dataTimeChecked) return | ||
|  |     dataTimeChecked = true | ||
|  |     if (process.env.BROWSERSLIST_IGNORE_OLD_DATA) return | ||
|  | 
 | ||
|  |     var latest = latestReleaseTime(agentsObj) | ||
|  |     var halfYearAgo = Date.now() - TIME_TO_UPDATE_CANIUSE | ||
|  | 
 | ||
|  |     if (latest !== 0 && latest < halfYearAgo) { | ||
|  |       console.warn( | ||
|  |         'Browserslist: caniuse-lite is outdated. Please run:\n' + | ||
|  |           '  npx update-browserslist-db@latest\n' + | ||
|  |           '  Why you should do it regularly: ' + | ||
|  |           'https://github.com/browserslist/update-db#readme' | ||
|  |       ) | ||
|  |     } | ||
|  |   }, | ||
|  | 
 | ||
|  |   currentNode: function currentNode() { | ||
|  |     return 'node ' + process.versions.node | ||
|  |   } | ||
|  | } |