UNPKG

10.2 kBJavaScriptView Raw
1var region = require('caniuse-lite/dist/unpacker/region').default
2var path = require('path')
3var fs = require('fs')
4
5var BrowserslistError = require('./error')
6
7var IS_SECTION = /^\s*\[(.+)]\s*$/
8var CONFIG_PATTERN = /^browserslist-config-/
9var SCOPED_CONFIG__PATTERN = /@[^/]+\/browserslist-config(-|$|\/)/
10var TIME_TO_UPDATE_CANIUSE = 6 * 30 * 24 * 60 * 60 * 1000
11var FORMAT = 'Browserslist config should be a string or an array ' +
12 'of strings with browser queries'
13
14var dataTimeChecked = false
15var filenessCache = { }
16var configCache = { }
17function checkExtend (name) {
18 var use = ' Use `dangerousExtend` option to disable.'
19 if (!CONFIG_PATTERN.test(name) && !SCOPED_CONFIG__PATTERN.test(name)) {
20 throw new BrowserslistError(
21 'Browserslist config needs `browserslist-config-` prefix. ' + use)
22 }
23 if (name.replace(/^@[^/]+\//, '').indexOf('.') !== -1) {
24 throw new BrowserslistError(
25 '`.` not allowed in Browserslist config name. ' + use)
26 }
27 if (name.indexOf('node_modules') !== -1) {
28 throw new BrowserslistError(
29 '`node_modules` not allowed in Browserslist config.' + use)
30 }
31}
32
33function isFile (file) {
34 if (file in filenessCache) {
35 return filenessCache[file]
36 }
37 var result = fs.existsSync(file) && fs.statSync(file).isFile()
38 if (!process.env.BROWSERSLIST_DISABLE_CACHE) {
39 filenessCache[file] = result
40 }
41 return result
42}
43
44function eachParent (file, callback) {
45 var dir = isFile(file) ? path.dirname(file) : file
46 var loc = path.resolve(dir)
47 do {
48 var result = callback(loc)
49 if (typeof result !== 'undefined') return result
50 } while (loc !== (loc = path.dirname(loc)))
51 return undefined
52}
53
54function check (section) {
55 if (Array.isArray(section)) {
56 for (var i = 0; i < section.length; i++) {
57 if (typeof section[i] !== 'string') {
58 throw new BrowserslistError(FORMAT)
59 }
60 }
61 } else if (typeof section !== 'string') {
62 throw new BrowserslistError(FORMAT)
63 }
64}
65
66function pickEnv (config, opts) {
67 if (typeof config !== 'object') return config
68
69 var name
70 if (typeof opts.env === 'string') {
71 name = opts.env
72 } else if (process.env.BROWSERSLIST_ENV) {
73 name = process.env.BROWSERSLIST_ENV
74 } else if (process.env.NODE_ENV) {
75 name = process.env.NODE_ENV
76 } else {
77 name = 'production'
78 }
79
80 return config[name] || config.defaults
81}
82
83function parsePackage (file) {
84 var config = JSON.parse(fs.readFileSync(file))
85 if (config.browserlist && !config.browserslist) {
86 throw new BrowserslistError(
87 '`browserlist` key instead of `browserslist` in ' + file
88 )
89 }
90 var list = config.browserslist
91 if (Array.isArray(list) || typeof list === 'string') {
92 list = { defaults: list }
93 }
94 for (var i in list) {
95 check(list[i])
96 }
97
98 return list
99}
100
101function latestReleaseTime (agents) {
102 var latest = 0
103 for (var name in agents) {
104 var dates = agents[name].releaseDate || { }
105 for (var key in dates) {
106 if (latest < dates[key]) {
107 latest = dates[key]
108 }
109 }
110 }
111 return latest * 1000
112}
113
114function normalizeStats (data, stats) {
115 if (stats && 'dataByBrowser' in stats) {
116 stats = stats.dataByBrowser
117 }
118
119 if (typeof stats !== 'object') return undefined
120
121 var normalized = { }
122 for (var i in stats) {
123 var versions = Object.keys(stats[i])
124 if (
125 versions.length === 1 &&
126 data[i] &&
127 data[i].versions.length === 1
128 ) {
129 var normal = Object.keys(data[i].versions)[0]
130 normalized[i] = { }
131 normalized[i][normal] = stats[i][versions[0]]
132 } else {
133 normalized[i] = stats[i]
134 }
135 }
136
137 return normalized
138}
139
140function normalizeUsageData (usageData, data) {
141 for (var browser in usageData) {
142 var browserUsage = usageData[browser]
143 // eslint-disable-next-line max-len
144 // https://github.com/browserslist/browserslist/issues/431#issuecomment-565230615
145 // caniuse-db returns { 0: "percentage" } for `and_*` regional stats
146 if ('0' in browserUsage) {
147 var versions = data[browser].versions
148 browserUsage[versions[versions.length - 1]] = browserUsage[0]
149 delete browserUsage[0]
150 }
151 }
152}
153
154module.exports = {
155 loadQueries: function loadQueries (context, name) {
156 if (!context.dangerousExtend) checkExtend(name)
157 // eslint-disable-next-line security/detect-non-literal-require
158 var queries = require(require.resolve(name, { paths: ['.'] }))
159 if (!Array.isArray(queries)) {
160 throw new BrowserslistError(
161 '`' + name + '` config exports not an array of queries'
162 )
163 }
164 return queries
165 },
166
167 loadStat: function loadStat (context, name, data) {
168 if (!context.dangerousExtend) checkExtend(name)
169 // eslint-disable-next-line security/detect-non-literal-require
170 var stats = require(
171 require.resolve(
172 path.join(name, 'browserslist-stats.json'),
173 { paths: ['.'] }
174 )
175 )
176 return normalizeStats(data, stats)
177 },
178
179 getStat: function getStat (opts, data) {
180 var stats
181 if (opts.stats) {
182 stats = opts.stats
183 } else if (process.env.BROWSERSLIST_STATS) {
184 stats = process.env.BROWSERSLIST_STATS
185 } else if (opts.path && path.resolve && fs.existsSync) {
186 stats = eachParent(opts.path, function (dir) {
187 var file = path.join(dir, 'browserslist-stats.json')
188 return isFile(file) ? file : undefined
189 })
190 }
191 if (typeof stats === 'string') {
192 try {
193 stats = JSON.parse(fs.readFileSync(stats))
194 } catch (e) {
195 throw new BrowserslistError('Can\'t read ' + stats)
196 }
197 }
198 return normalizeStats(data, stats)
199 },
200
201 loadConfig: function loadConfig (opts) {
202 if (process.env.BROWSERSLIST) {
203 return process.env.BROWSERSLIST
204 } else if (opts.config || process.env.BROWSERSLIST_CONFIG) {
205 var file = opts.config || process.env.BROWSERSLIST_CONFIG
206 if (path.basename(file) === 'package.json') {
207 return pickEnv(parsePackage(file), opts)
208 } else {
209 return pickEnv(module.exports.readConfig(file), opts)
210 }
211 } else if (opts.path) {
212 return pickEnv(module.exports.findConfig(opts.path), opts)
213 } else {
214 return undefined
215 }
216 },
217
218 loadCountry: function loadCountry (usage, country, data) {
219 var code = country.replace(/[^\w-]/g, '')
220 if (!usage[code]) {
221 // eslint-disable-next-line security/detect-non-literal-require
222 var compressed = require('caniuse-lite/data/regions/' + code + '.js')
223 var usageData = region(compressed)
224 normalizeUsageData(usageData, data)
225 usage[country] = { }
226 for (var i in usageData) {
227 for (var j in usageData[i]) {
228 usage[country][i + ' ' + j] = usageData[i][j]
229 }
230 }
231 }
232 },
233
234 parseConfig: function parseConfig (string) {
235 var result = { defaults: [] }
236 var sections = ['defaults']
237
238 string.toString()
239 .replace(/#[^\n]*/g, '')
240 .split(/\n|,/)
241 .map(function (line) {
242 return line.trim()
243 })
244 .filter(function (line) {
245 return line !== ''
246 })
247 .forEach(function (line) {
248 if (IS_SECTION.test(line)) {
249 sections = line.match(IS_SECTION)[1].trim().split(' ')
250 sections.forEach(function (section) {
251 if (result[section]) {
252 throw new BrowserslistError(
253 'Duplicate section ' + section + ' in Browserslist config'
254 )
255 }
256 result[section] = []
257 })
258 } else {
259 sections.forEach(function (section) {
260 result[section].push(line)
261 })
262 }
263 })
264
265 return result
266 },
267
268 readConfig: function readConfig (file) {
269 if (!isFile(file)) {
270 throw new BrowserslistError('Can\'t read ' + file + ' config')
271 }
272 return module.exports.parseConfig(fs.readFileSync(file))
273 },
274
275 findConfig: function findConfig (from) {
276 from = path.resolve(from)
277
278 var passed = []
279 var resolved = eachParent(from, function (dir) {
280 if (dir in configCache) {
281 return configCache[dir]
282 }
283
284 passed.push(dir)
285
286 var config = path.join(dir, 'browserslist')
287 var pkg = path.join(dir, 'package.json')
288 var rc = path.join(dir, '.browserslistrc')
289
290 var pkgBrowserslist
291 if (isFile(pkg)) {
292 try {
293 pkgBrowserslist = parsePackage(pkg)
294 } catch (e) {
295 if (e.name === 'BrowserslistError') throw e
296 console.warn(
297 '[Browserslist] Could not parse ' + pkg + '. Ignoring it.'
298 )
299 }
300 }
301
302 if (isFile(config) && pkgBrowserslist) {
303 throw new BrowserslistError(
304 dir + ' contains both browserslist and package.json with browsers'
305 )
306 } else if (isFile(rc) && pkgBrowserslist) {
307 throw new BrowserslistError(
308 dir + ' contains both .browserslistrc and package.json with browsers'
309 )
310 } else if (isFile(config) && isFile(rc)) {
311 throw new BrowserslistError(
312 dir + ' contains both .browserslistrc and browserslist'
313 )
314 } else if (isFile(config)) {
315 return module.exports.readConfig(config)
316 } else if (isFile(rc)) {
317 return module.exports.readConfig(rc)
318 } else {
319 return pkgBrowserslist
320 }
321 })
322 if (!process.env.BROWSERSLIST_DISABLE_CACHE) {
323 passed.forEach(function (dir) {
324 configCache[dir] = resolved
325 })
326 }
327 return resolved
328 },
329
330 clearCaches: function clearCaches () {
331 dataTimeChecked = false
332 filenessCache = { }
333 configCache = { }
334 },
335
336 oldDataWarning: function oldDataWarning (agentsObj) {
337 if (dataTimeChecked) return
338 dataTimeChecked = true
339 if (process.env.BROWSERSLIST_IGNORE_OLD_DATA) return
340
341 var latest = latestReleaseTime(agentsObj)
342 var halfYearAgo = Date.now() - TIME_TO_UPDATE_CANIUSE
343
344 if (latest !== 0 && latest < halfYearAgo) {
345 var command = 'npx browserslist --update-db'
346 console.warn(
347 'Browserslist: caniuse-lite is outdated. ' +
348 'Please run the following command: `' + command + '`'
349 )
350 }
351 },
352
353 currentNode: function currentNode () {
354 return 'node ' + process.versions.node
355 }
356}