498 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			JavaScript
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			498 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			JavaScript
		
	
	
		
			Executable File
		
	
	
	
	
| var feature = require('caniuse-lite/dist/unpacker/feature').default
 | |
| var region = require('caniuse-lite/dist/unpacker/region').default
 | |
| var fs = require('fs')
 | |
| var path = require('path')
 | |
| 
 | |
| var BrowserslistError = require('./error')
 | |
| 
 | |
| var IS_SECTION = /^\s*\[(.+)]\s*$/
 | |
| var CONFIG_PATTERN = /^browserslist-config-/
 | |
| var SCOPED_CONFIG__PATTERN = /@[^/]+(?:\/[^/]+)?\/browserslist-config(?:-|$|\/)/
 | |
| var FORMAT =
 | |
|   'Browserslist config should be a string or an array ' +
 | |
|   'of strings with browser queries'
 | |
| var PATHTYPE_UNKNOWN = 'unknown'
 | |
| var PATHTYPE_DIR = 'directory'
 | |
| var PATHTYPE_FILE = 'file'
 | |
| 
 | |
| var dataTimeChecked = false
 | |
| var statCache = {}
 | |
| var configPathCache = {}
 | |
| var parseConfigCache = {}
 | |
| 
 | |
| 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 getPathType(filepath) {
 | |
|   var stats
 | |
|   try {
 | |
|     stats = fs.existsSync(filepath) && fs.statSync(filepath)
 | |
|   } catch (err) {
 | |
|     /* c8 ignore start */
 | |
|     if (
 | |
|       err.code !== 'ENOENT' &&
 | |
|       err.code !== 'EACCES' &&
 | |
|       err.code !== 'ERR_ACCESS_DENIED'
 | |
|     ) {
 | |
|       throw err
 | |
|     }
 | |
|     /* c8 ignore end */
 | |
|   }
 | |
| 
 | |
|   if (stats && stats.isDirectory()) return PATHTYPE_DIR
 | |
|   if (stats && stats.isFile()) return PATHTYPE_FILE
 | |
| 
 | |
|   return PATHTYPE_UNKNOWN
 | |
| }
 | |
| 
 | |
| function isFile(file) {
 | |
|   return getPathType(file) === PATHTYPE_FILE
 | |
| }
 | |
| 
 | |
| function isDirectory(dir) {
 | |
|   return getPathType(dir) === PATHTYPE_DIR
 | |
| }
 | |
| 
 | |
| function eachParent(file, callback, cache) {
 | |
|   var loc = path.resolve(file)
 | |
|   var pathsForCacheResult = []
 | |
|   var result
 | |
|   do {
 | |
|     if (!pathInRoot(loc)) {
 | |
|       break
 | |
|     }
 | |
|     if (cache && loc in cache) {
 | |
|       result = cache[loc]
 | |
|       break
 | |
|     }
 | |
|     pathsForCacheResult.push(loc)
 | |
| 
 | |
|     if (!isDirectory(loc)) {
 | |
|       continue
 | |
|     }
 | |
| 
 | |
|     var locResult = callback(loc)
 | |
|     if (typeof locResult !== 'undefined') {
 | |
|       result = locResult
 | |
|       break
 | |
|     }
 | |
|   } while (loc !== (loc = path.dirname(loc)))
 | |
| 
 | |
|   if (cache && !process.env.BROWSERSLIST_DISABLE_CACHE) {
 | |
|     pathsForCacheResult.forEach(function (cachePath) {
 | |
|       cache[cachePath] = result
 | |
|     })
 | |
|   }
 | |
|   return result
 | |
| }
 | |
| 
 | |
| function pathInRoot(p) {
 | |
|   if (!process.env.BROWSERSLIST_ROOT_PATH) return true
 | |
|   var rootPath = path.resolve(process.env.BROWSERSLIST_ROOT_PATH)
 | |
|   if (path.relative(rootPath, p).substring(0, 2) === '..') {
 | |
|     return false
 | |
|   }
 | |
|   return true
 | |
| }
 | |
| 
 | |
| 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 text = fs
 | |
|     .readFileSync(file)
 | |
|     .toString()
 | |
|     .replace(/^\uFEFF/m, '')
 | |
|   var list
 | |
|   if (text.indexOf('"browserslist"') >= 0) {
 | |
|     list = JSON.parse(text).browserslist
 | |
|   } else if (text.indexOf('"browserlist"') >= 0) {
 | |
|     var config = JSON.parse(text)
 | |
|     if (config.browserlist && !config.browserslist) {
 | |
|       throw new BrowserslistError(
 | |
|         '`browserlist` key instead of `browserslist` in ' + file
 | |
|       )
 | |
|     }
 | |
|   }
 | |
|   if (Array.isArray(list) || typeof list === 'string') {
 | |
|     list = { defaults: list }
 | |
|   }
 | |
|   for (var i in list) {
 | |
|     check(list[i])
 | |
|   }
 | |
| 
 | |
|   return list
 | |
| }
 | |
| 
 | |
| function parsePackageOrReadConfig(file) {
 | |
|   if (file in parseConfigCache) {
 | |
|     return parseConfigCache[file]
 | |
|   }
 | |
| 
 | |
|   var isPackage = path.basename(file) === 'package.json'
 | |
|   var result = isPackage ? parsePackage(file) : module.exports.readConfig(file)
 | |
| 
 | |
|   if (!process.env.BROWSERSLIST_DISABLE_CACHE) {
 | |
|     parseConfigCache[file] = result
 | |
|   }
 | |
|   return result
 | |
| }
 | |
| 
 | |
| 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 getMonthsPassed(date) {
 | |
|   var now = new Date()
 | |
|   var past = new Date(date)
 | |
| 
 | |
|   var years = now.getFullYear() - past.getFullYear()
 | |
|   var months = now.getMonth() - past.getMonth()
 | |
| 
 | |
|   return years * 12 + months
 | |
| }
 | |
| 
 | |
| 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]
 | |
|     // 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 (typeof queries === 'object' && queries !== null && queries.__esModule) {
 | |
|       queries = queries.default
 | |
|     }
 | |
|     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(
 | |
|       // Use forward slashes for module paths, also on Windows.
 | |
|       require.resolve(path.posix.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
 | |
|         },
 | |
|         statCache
 | |
|       )
 | |
|     }
 | |
|     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
 | |
|       return pickEnv(parsePackageOrReadConfig(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) {
 | |
|       features[name][i] = {}
 | |
|       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))
 | |
|   },
 | |
| 
 | |
|   findConfigFile: function findConfigFile(from) {
 | |
|     return eachParent(
 | |
|       from,
 | |
|       function (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 config
 | |
|         } else if (isFile(rc)) {
 | |
|           return rc
 | |
|         } else if (pkgBrowserslist) {
 | |
|           return pkg
 | |
|         }
 | |
|       },
 | |
|       configPathCache
 | |
|     )
 | |
|   },
 | |
| 
 | |
|   findConfig: function findConfig(from) {
 | |
|     var configFile = this.findConfigFile(from)
 | |
| 
 | |
|     return configFile ? parsePackageOrReadConfig(configFile) : undefined
 | |
|   },
 | |
| 
 | |
|   clearCaches: function clearCaches() {
 | |
|     dataTimeChecked = false
 | |
|     statCache = {}
 | |
|     configPathCache = {}
 | |
|     parseConfigCache = {}
 | |
| 
 | |
|     this.cache = {}
 | |
|   },
 | |
| 
 | |
|   oldDataWarning: function oldDataWarning(agentsObj) {
 | |
|     if (dataTimeChecked) return
 | |
|     dataTimeChecked = true
 | |
|     if (process.env.BROWSERSLIST_IGNORE_OLD_DATA) return
 | |
| 
 | |
|     var latest = latestReleaseTime(agentsObj)
 | |
|     var monthsPassed = getMonthsPassed(latest)
 | |
| 
 | |
|     if (latest !== 0 && monthsPassed >= 6) {
 | |
|       var months = monthsPassed + ' ' + (monthsPassed > 1 ? 'months' : 'month')
 | |
|       console.warn(
 | |
|         'Browserslist: browsers data (caniuse-lite) is ' +
 | |
|           months +
 | |
|           ' old. 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
 | |
|   },
 | |
| 
 | |
|   env: process.env
 | |
| }
 |