UNPKG

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