index.js 34 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235
  1. var jsReleases = require('node-releases/data/processed/envs.json')
  2. var agents = require('caniuse-lite/dist/unpacker/agents').agents
  3. var e2c = require('electron-to-chromium/versions')
  4. var jsEOL = require('node-releases/data/release-schedule/release-schedule.json')
  5. var path = require('path')
  6. var BrowserslistError = require('./error')
  7. var env = require('./node')
  8. var parseWithoutCache = require('./parse') // Will load browser.js in webpack
  9. var YEAR = 365.259641 * 24 * 60 * 60 * 1000
  10. var ANDROID_EVERGREEN_FIRST = '37'
  11. var OP_MOB_BLINK_FIRST = 14
  12. // Helpers
  13. function isVersionsMatch(versionA, versionB) {
  14. return (versionA + '.').indexOf(versionB + '.') === 0
  15. }
  16. function isEolReleased(name) {
  17. var version = name.slice(1)
  18. return browserslist.nodeVersions.some(function (i) {
  19. return isVersionsMatch(i, version)
  20. })
  21. }
  22. function normalize(versions) {
  23. return versions.filter(function (version) {
  24. return typeof version === 'string'
  25. })
  26. }
  27. function normalizeElectron(version) {
  28. var versionToUse = version
  29. if (version.split('.').length === 3) {
  30. versionToUse = version.split('.').slice(0, -1).join('.')
  31. }
  32. return versionToUse
  33. }
  34. function nameMapper(name) {
  35. return function mapName(version) {
  36. return name + ' ' + version
  37. }
  38. }
  39. function getMajor(version) {
  40. return parseInt(version.split('.')[0])
  41. }
  42. function getMajorVersions(released, number) {
  43. if (released.length === 0) return []
  44. var majorVersions = uniq(released.map(getMajor))
  45. var minimum = majorVersions[majorVersions.length - number]
  46. if (!minimum) {
  47. return released
  48. }
  49. var selected = []
  50. for (var i = released.length - 1; i >= 0; i--) {
  51. if (minimum > getMajor(released[i])) break
  52. selected.unshift(released[i])
  53. }
  54. return selected
  55. }
  56. function uniq(array) {
  57. var filtered = []
  58. for (var i = 0; i < array.length; i++) {
  59. if (filtered.indexOf(array[i]) === -1) filtered.push(array[i])
  60. }
  61. return filtered
  62. }
  63. function fillUsage(result, name, data) {
  64. for (var i in data) {
  65. result[name + ' ' + i] = data[i]
  66. }
  67. }
  68. function generateFilter(sign, version) {
  69. version = parseFloat(version)
  70. if (sign === '>') {
  71. return function (v) {
  72. return parseLatestFloat(v) > version
  73. }
  74. } else if (sign === '>=') {
  75. return function (v) {
  76. return parseLatestFloat(v) >= version
  77. }
  78. } else if (sign === '<') {
  79. return function (v) {
  80. return parseFloat(v) < version
  81. }
  82. } else {
  83. return function (v) {
  84. return parseFloat(v) <= version
  85. }
  86. }
  87. function parseLatestFloat(v) {
  88. return parseFloat(v.split('-')[1] || v)
  89. }
  90. }
  91. function generateSemverFilter(sign, version) {
  92. version = version.split('.').map(parseSimpleInt)
  93. version[1] = version[1] || 0
  94. version[2] = version[2] || 0
  95. if (sign === '>') {
  96. return function (v) {
  97. v = v.split('.').map(parseSimpleInt)
  98. return compareSemver(v, version) > 0
  99. }
  100. } else if (sign === '>=') {
  101. return function (v) {
  102. v = v.split('.').map(parseSimpleInt)
  103. return compareSemver(v, version) >= 0
  104. }
  105. } else if (sign === '<') {
  106. return function (v) {
  107. v = v.split('.').map(parseSimpleInt)
  108. return compareSemver(version, v) > 0
  109. }
  110. } else {
  111. return function (v) {
  112. v = v.split('.').map(parseSimpleInt)
  113. return compareSemver(version, v) >= 0
  114. }
  115. }
  116. }
  117. function parseSimpleInt(x) {
  118. return parseInt(x)
  119. }
  120. function compare(a, b) {
  121. if (a < b) return -1
  122. if (a > b) return +1
  123. return 0
  124. }
  125. function compareSemver(a, b) {
  126. return (
  127. compare(parseInt(a[0]), parseInt(b[0])) ||
  128. compare(parseInt(a[1] || '0'), parseInt(b[1] || '0')) ||
  129. compare(parseInt(a[2] || '0'), parseInt(b[2] || '0'))
  130. )
  131. }
  132. // this follows the npm-like semver behavior
  133. function semverFilterLoose(operator, range) {
  134. range = range.split('.').map(parseSimpleInt)
  135. if (typeof range[1] === 'undefined') {
  136. range[1] = 'x'
  137. }
  138. // ignore any patch version because we only return minor versions
  139. // range[2] = 'x'
  140. switch (operator) {
  141. case '<=':
  142. return function (version) {
  143. version = version.split('.').map(parseSimpleInt)
  144. return compareSemverLoose(version, range) <= 0
  145. }
  146. case '>=':
  147. default:
  148. return function (version) {
  149. version = version.split('.').map(parseSimpleInt)
  150. return compareSemverLoose(version, range) >= 0
  151. }
  152. }
  153. }
  154. // this follows the npm-like semver behavior
  155. function compareSemverLoose(version, range) {
  156. if (version[0] !== range[0]) {
  157. return version[0] < range[0] ? -1 : +1
  158. }
  159. if (range[1] === 'x') {
  160. return 0
  161. }
  162. if (version[1] !== range[1]) {
  163. return version[1] < range[1] ? -1 : +1
  164. }
  165. return 0
  166. }
  167. function resolveVersion(data, version) {
  168. if (data.versions.indexOf(version) !== -1) {
  169. return version
  170. } else if (browserslist.versionAliases[data.name][version]) {
  171. return browserslist.versionAliases[data.name][version]
  172. } else {
  173. return false
  174. }
  175. }
  176. function normalizeVersion(data, version) {
  177. var resolved = resolveVersion(data, version)
  178. if (resolved) {
  179. return resolved
  180. } else if (data.versions.length === 1) {
  181. return data.versions[0]
  182. } else {
  183. return false
  184. }
  185. }
  186. function filterByYear(since, context) {
  187. since = since / 1000
  188. return Object.keys(agents).reduce(function (selected, name) {
  189. var data = byName(name, context)
  190. if (!data) return selected
  191. var versions = Object.keys(data.releaseDate).filter(function (v) {
  192. var date = data.releaseDate[v]
  193. return date !== null && date >= since
  194. })
  195. return selected.concat(versions.map(nameMapper(data.name)))
  196. }, [])
  197. }
  198. function cloneData(data) {
  199. return {
  200. name: data.name,
  201. versions: data.versions,
  202. released: data.released,
  203. releaseDate: data.releaseDate
  204. }
  205. }
  206. function byName(name, context) {
  207. name = name.toLowerCase()
  208. name = browserslist.aliases[name] || name
  209. if (context.mobileToDesktop && browserslist.desktopNames[name]) {
  210. var desktop = browserslist.data[browserslist.desktopNames[name]]
  211. if (name === 'android') {
  212. return normalizeAndroidData(cloneData(browserslist.data[name]), desktop)
  213. } else {
  214. var cloned = cloneData(desktop)
  215. cloned.name = name
  216. return cloned
  217. }
  218. }
  219. return browserslist.data[name]
  220. }
  221. function normalizeAndroidVersions(androidVersions, chromeVersions) {
  222. var iFirstEvergreen = chromeVersions.indexOf(ANDROID_EVERGREEN_FIRST)
  223. return androidVersions
  224. .filter(function (version) {
  225. return /^(?:[2-4]\.|[34]$)/.test(version)
  226. })
  227. .concat(chromeVersions.slice(iFirstEvergreen))
  228. }
  229. function copyObject(obj) {
  230. var copy = {}
  231. for (var key in obj) {
  232. copy[key] = obj[key]
  233. }
  234. return copy
  235. }
  236. function normalizeAndroidData(android, chrome) {
  237. android.released = normalizeAndroidVersions(android.released, chrome.released)
  238. android.versions = normalizeAndroidVersions(android.versions, chrome.versions)
  239. android.releaseDate = copyObject(android.releaseDate)
  240. android.released.forEach(function (v) {
  241. if (android.releaseDate[v] === undefined) {
  242. android.releaseDate[v] = chrome.releaseDate[v]
  243. }
  244. })
  245. return android
  246. }
  247. function checkName(name, context) {
  248. var data = byName(name, context)
  249. if (!data) throw new BrowserslistError('Unknown browser ' + name)
  250. return data
  251. }
  252. function unknownQuery(query) {
  253. return new BrowserslistError(
  254. 'Unknown browser query `' +
  255. query +
  256. '`. ' +
  257. 'Maybe you are using old Browserslist or made typo in query.'
  258. )
  259. }
  260. // Adjusts last X versions queries for some mobile browsers,
  261. // where caniuse data jumps from a legacy version to the latest
  262. function filterJumps(list, name, nVersions, context) {
  263. var jump = 1
  264. switch (name) {
  265. case 'android':
  266. if (context.mobileToDesktop) return list
  267. var released = browserslist.data.chrome.released
  268. jump = released.length - released.indexOf(ANDROID_EVERGREEN_FIRST)
  269. break
  270. case 'op_mob':
  271. var latest = browserslist.data.op_mob.released.slice(-1)[0]
  272. jump = getMajor(latest) - OP_MOB_BLINK_FIRST + 1
  273. break
  274. default:
  275. return list
  276. }
  277. if (nVersions <= jump) {
  278. return list.slice(-1)
  279. }
  280. return list.slice(jump - 1 - nVersions)
  281. }
  282. function isSupported(flags, withPartial) {
  283. return (
  284. typeof flags === 'string' &&
  285. (flags.indexOf('y') >= 0 || (withPartial && flags.indexOf('a') >= 0))
  286. )
  287. }
  288. function resolve(queries, context) {
  289. return parseQueries(queries).reduce(function (result, node, index) {
  290. if (node.not && index === 0) {
  291. throw new BrowserslistError(
  292. 'Write any browsers query (for instance, `defaults`) ' +
  293. 'before `' +
  294. node.query +
  295. '`'
  296. )
  297. }
  298. var type = QUERIES[node.type]
  299. var array = type.select.call(browserslist, context, node).map(function (j) {
  300. var parts = j.split(' ')
  301. if (parts[1] === '0') {
  302. return parts[0] + ' ' + byName(parts[0], context).versions[0]
  303. } else {
  304. return j
  305. }
  306. })
  307. if (node.compose === 'and') {
  308. if (node.not) {
  309. return result.filter(function (j) {
  310. return array.indexOf(j) === -1
  311. })
  312. } else {
  313. return result.filter(function (j) {
  314. return array.indexOf(j) !== -1
  315. })
  316. }
  317. } else {
  318. if (node.not) {
  319. var filter = {}
  320. array.forEach(function (j) {
  321. filter[j] = true
  322. })
  323. return result.filter(function (j) {
  324. return !filter[j]
  325. })
  326. }
  327. return result.concat(array)
  328. }
  329. }, [])
  330. }
  331. function prepareOpts(opts) {
  332. if (typeof opts === 'undefined') opts = {}
  333. if (typeof opts.path === 'undefined') {
  334. opts.path = path.resolve ? path.resolve('.') : '.'
  335. }
  336. return opts
  337. }
  338. function prepareQueries(queries, opts) {
  339. if (typeof queries === 'undefined' || queries === null) {
  340. var config = browserslist.loadConfig(opts)
  341. if (config) {
  342. queries = config
  343. } else {
  344. queries = browserslist.defaults
  345. }
  346. }
  347. return queries
  348. }
  349. function checkQueries(queries) {
  350. if (!(typeof queries === 'string' || Array.isArray(queries))) {
  351. throw new BrowserslistError(
  352. 'Browser queries must be an array or string. Got ' + typeof queries + '.'
  353. )
  354. }
  355. }
  356. var cache = {}
  357. var parseCache = {}
  358. function browserslist(queries, opts) {
  359. opts = prepareOpts(opts)
  360. queries = prepareQueries(queries, opts)
  361. checkQueries(queries)
  362. var needsPath = parseQueries(queries).some(function (node) {
  363. return QUERIES[node.type].needsPath
  364. })
  365. var context = {
  366. ignoreUnknownVersions: opts.ignoreUnknownVersions,
  367. dangerousExtend: opts.dangerousExtend,
  368. mobileToDesktop: opts.mobileToDesktop,
  369. env: opts.env
  370. }
  371. // Removing to avoid using context.path without marking query as needsPath
  372. if (needsPath) {
  373. context.path = opts.path
  374. }
  375. env.oldDataWarning(browserslist.data)
  376. var stats = env.getStat(opts, browserslist.data)
  377. if (stats) {
  378. context.customUsage = {}
  379. for (var browser in stats) {
  380. fillUsage(context.customUsage, browser, stats[browser])
  381. }
  382. }
  383. var cacheKey = JSON.stringify([queries, context])
  384. if (cache[cacheKey]) return cache[cacheKey]
  385. var result = uniq(resolve(queries, context)).sort(function (name1, name2) {
  386. name1 = name1.split(' ')
  387. name2 = name2.split(' ')
  388. if (name1[0] === name2[0]) {
  389. // assumptions on caniuse data
  390. // 1) version ranges never overlaps
  391. // 2) if version is not a range, it never contains `-`
  392. var version1 = name1[1].split('-')[0]
  393. var version2 = name2[1].split('-')[0]
  394. return compareSemver(version2.split('.'), version1.split('.'))
  395. } else {
  396. return compare(name1[0], name2[0])
  397. }
  398. })
  399. if (!env.env.BROWSERSLIST_DISABLE_CACHE) {
  400. cache[cacheKey] = result
  401. }
  402. return result
  403. }
  404. function parseQueries(queries) {
  405. var cacheKey = JSON.stringify(queries)
  406. if (cacheKey in parseCache) return parseCache[cacheKey]
  407. var result = parseWithoutCache(QUERIES, queries)
  408. if (!env.env.BROWSERSLIST_DISABLE_CACHE) {
  409. parseCache[cacheKey] = result
  410. }
  411. return result
  412. }
  413. browserslist.parse = function (queries, opts) {
  414. opts = prepareOpts(opts)
  415. queries = prepareQueries(queries, opts)
  416. checkQueries(queries)
  417. return parseQueries(queries)
  418. }
  419. // Will be filled by Can I Use data below
  420. browserslist.cache = {}
  421. browserslist.data = {}
  422. browserslist.usage = {
  423. global: {},
  424. custom: null
  425. }
  426. // Default browsers query
  427. browserslist.defaults = ['> 0.5%', 'last 2 versions', 'Firefox ESR', 'not dead']
  428. // Browser names aliases
  429. browserslist.aliases = {
  430. fx: 'firefox',
  431. ff: 'firefox',
  432. ios: 'ios_saf',
  433. explorer: 'ie',
  434. blackberry: 'bb',
  435. explorermobile: 'ie_mob',
  436. operamini: 'op_mini',
  437. operamobile: 'op_mob',
  438. chromeandroid: 'and_chr',
  439. firefoxandroid: 'and_ff',
  440. ucandroid: 'and_uc',
  441. qqandroid: 'and_qq'
  442. }
  443. // Can I Use only provides a few versions for some browsers (e.g. and_chr).
  444. // Fallback to a similar browser for unknown versions
  445. // Note op_mob is not included as its chromium versions are not in sync with Opera desktop
  446. browserslist.desktopNames = {
  447. and_chr: 'chrome',
  448. and_ff: 'firefox',
  449. ie_mob: 'ie',
  450. android: 'chrome' // has extra processing logic
  451. }
  452. // Aliases to work with joined versions like `ios_saf 7.0-7.1`
  453. browserslist.versionAliases = {}
  454. browserslist.clearCaches = env.clearCaches
  455. browserslist.parseConfig = env.parseConfig
  456. browserslist.readConfig = env.readConfig
  457. browserslist.findConfigFile = env.findConfigFile
  458. browserslist.findConfig = env.findConfig
  459. browserslist.loadConfig = env.loadConfig
  460. browserslist.coverage = function (browsers, stats) {
  461. var data
  462. if (typeof stats === 'undefined') {
  463. data = browserslist.usage.global
  464. } else if (stats === 'my stats') {
  465. var opts = {}
  466. opts.path = path.resolve ? path.resolve('.') : '.'
  467. var customStats = env.getStat(opts)
  468. if (!customStats) {
  469. throw new BrowserslistError('Custom usage statistics was not provided')
  470. }
  471. data = {}
  472. for (var browser in customStats) {
  473. fillUsage(data, browser, customStats[browser])
  474. }
  475. } else if (typeof stats === 'string') {
  476. if (stats.length > 2) {
  477. stats = stats.toLowerCase()
  478. } else {
  479. stats = stats.toUpperCase()
  480. }
  481. env.loadCountry(browserslist.usage, stats, browserslist.data)
  482. data = browserslist.usage[stats]
  483. } else {
  484. if ('dataByBrowser' in stats) {
  485. stats = stats.dataByBrowser
  486. }
  487. data = {}
  488. for (var name in stats) {
  489. for (var version in stats[name]) {
  490. data[name + ' ' + version] = stats[name][version]
  491. }
  492. }
  493. }
  494. return browsers.reduce(function (all, i) {
  495. var usage = data[i]
  496. if (usage === undefined) {
  497. usage = data[i.replace(/ \S+$/, ' 0')]
  498. }
  499. return all + (usage || 0)
  500. }, 0)
  501. }
  502. function nodeQuery(context, node) {
  503. var matched = browserslist.nodeVersions.filter(function (i) {
  504. return isVersionsMatch(i, node.version)
  505. })
  506. if (matched.length === 0) {
  507. if (context.ignoreUnknownVersions) {
  508. return []
  509. } else {
  510. throw new BrowserslistError(
  511. 'Unknown version ' + node.version + ' of Node.js'
  512. )
  513. }
  514. }
  515. return ['node ' + matched[matched.length - 1]]
  516. }
  517. function sinceQuery(context, node) {
  518. var year = parseInt(node.year)
  519. var month = parseInt(node.month || '01') - 1
  520. var day = parseInt(node.day || '01')
  521. return filterByYear(Date.UTC(year, month, day, 0, 0, 0), context)
  522. }
  523. function coverQuery(context, node) {
  524. var coverage = parseFloat(node.coverage)
  525. var usage = browserslist.usage.global
  526. if (node.place) {
  527. if (node.place.match(/^my\s+stats$/i)) {
  528. if (!context.customUsage) {
  529. throw new BrowserslistError('Custom usage statistics was not provided')
  530. }
  531. usage = context.customUsage
  532. } else {
  533. var place
  534. if (node.place.length === 2) {
  535. place = node.place.toUpperCase()
  536. } else {
  537. place = node.place.toLowerCase()
  538. }
  539. env.loadCountry(browserslist.usage, place, browserslist.data)
  540. usage = browserslist.usage[place]
  541. }
  542. }
  543. var versions = Object.keys(usage).sort(function (a, b) {
  544. return usage[b] - usage[a]
  545. })
  546. var coveraged = 0
  547. var result = []
  548. var version
  549. for (var i = 0; i < versions.length; i++) {
  550. version = versions[i]
  551. if (usage[version] === 0) break
  552. coveraged += usage[version]
  553. result.push(version)
  554. if (coveraged >= coverage) break
  555. }
  556. return result
  557. }
  558. var QUERIES = {
  559. last_major_versions: {
  560. matches: ['versions'],
  561. regexp: /^last\s+(\d+)\s+major\s+versions?$/i,
  562. select: function (context, node) {
  563. return Object.keys(agents).reduce(function (selected, name) {
  564. var data = byName(name, context)
  565. if (!data) return selected
  566. var list = getMajorVersions(data.released, node.versions)
  567. list = list.map(nameMapper(data.name))
  568. list = filterJumps(list, data.name, node.versions, context)
  569. return selected.concat(list)
  570. }, [])
  571. }
  572. },
  573. last_versions: {
  574. matches: ['versions'],
  575. regexp: /^last\s+(\d+)\s+versions?$/i,
  576. select: function (context, node) {
  577. return Object.keys(agents).reduce(function (selected, name) {
  578. var data = byName(name, context)
  579. if (!data) return selected
  580. var list = data.released.slice(-node.versions)
  581. list = list.map(nameMapper(data.name))
  582. list = filterJumps(list, data.name, node.versions, context)
  583. return selected.concat(list)
  584. }, [])
  585. }
  586. },
  587. last_electron_major_versions: {
  588. matches: ['versions'],
  589. regexp: /^last\s+(\d+)\s+electron\s+major\s+versions?$/i,
  590. select: function (context, node) {
  591. var validVersions = getMajorVersions(Object.keys(e2c), node.versions)
  592. return validVersions.map(function (i) {
  593. return 'chrome ' + e2c[i]
  594. })
  595. }
  596. },
  597. last_node_major_versions: {
  598. matches: ['versions'],
  599. regexp: /^last\s+(\d+)\s+node\s+major\s+versions?$/i,
  600. select: function (context, node) {
  601. return getMajorVersions(browserslist.nodeVersions, node.versions).map(
  602. function (version) {
  603. return 'node ' + version
  604. }
  605. )
  606. }
  607. },
  608. last_browser_major_versions: {
  609. matches: ['versions', 'browser'],
  610. regexp: /^last\s+(\d+)\s+(\w+)\s+major\s+versions?$/i,
  611. select: function (context, node) {
  612. var data = checkName(node.browser, context)
  613. var validVersions = getMajorVersions(data.released, node.versions)
  614. var list = validVersions.map(nameMapper(data.name))
  615. list = filterJumps(list, data.name, node.versions, context)
  616. return list
  617. }
  618. },
  619. last_electron_versions: {
  620. matches: ['versions'],
  621. regexp: /^last\s+(\d+)\s+electron\s+versions?$/i,
  622. select: function (context, node) {
  623. return Object.keys(e2c)
  624. .slice(-node.versions)
  625. .map(function (i) {
  626. return 'chrome ' + e2c[i]
  627. })
  628. }
  629. },
  630. last_node_versions: {
  631. matches: ['versions'],
  632. regexp: /^last\s+(\d+)\s+node\s+versions?$/i,
  633. select: function (context, node) {
  634. return browserslist.nodeVersions
  635. .slice(-node.versions)
  636. .map(function (version) {
  637. return 'node ' + version
  638. })
  639. }
  640. },
  641. last_browser_versions: {
  642. matches: ['versions', 'browser'],
  643. regexp: /^last\s+(\d+)\s+(\w+)\s+versions?$/i,
  644. select: function (context, node) {
  645. var data = checkName(node.browser, context)
  646. var list = data.released.slice(-node.versions).map(nameMapper(data.name))
  647. list = filterJumps(list, data.name, node.versions, context)
  648. return list
  649. }
  650. },
  651. unreleased_versions: {
  652. matches: [],
  653. regexp: /^unreleased\s+versions$/i,
  654. select: function (context) {
  655. return Object.keys(agents).reduce(function (selected, name) {
  656. var data = byName(name, context)
  657. if (!data) return selected
  658. var list = data.versions.filter(function (v) {
  659. return data.released.indexOf(v) === -1
  660. })
  661. list = list.map(nameMapper(data.name))
  662. return selected.concat(list)
  663. }, [])
  664. }
  665. },
  666. unreleased_electron_versions: {
  667. matches: [],
  668. regexp: /^unreleased\s+electron\s+versions?$/i,
  669. select: function () {
  670. return []
  671. }
  672. },
  673. unreleased_browser_versions: {
  674. matches: ['browser'],
  675. regexp: /^unreleased\s+(\w+)\s+versions?$/i,
  676. select: function (context, node) {
  677. var data = checkName(node.browser, context)
  678. return data.versions
  679. .filter(function (v) {
  680. return data.released.indexOf(v) === -1
  681. })
  682. .map(nameMapper(data.name))
  683. }
  684. },
  685. last_years: {
  686. matches: ['years'],
  687. regexp: /^last\s+(\d*.?\d+)\s+years?$/i,
  688. select: function (context, node) {
  689. return filterByYear(Date.now() - YEAR * node.years, context)
  690. }
  691. },
  692. since_y: {
  693. matches: ['year'],
  694. regexp: /^since (\d+)$/i,
  695. select: sinceQuery
  696. },
  697. since_y_m: {
  698. matches: ['year', 'month'],
  699. regexp: /^since (\d+)-(\d+)$/i,
  700. select: sinceQuery
  701. },
  702. since_y_m_d: {
  703. matches: ['year', 'month', 'day'],
  704. regexp: /^since (\d+)-(\d+)-(\d+)$/i,
  705. select: sinceQuery
  706. },
  707. popularity: {
  708. matches: ['sign', 'popularity'],
  709. regexp: /^(>=?|<=?)\s*(\d+|\d+\.\d+|\.\d+)%$/,
  710. select: function (context, node) {
  711. var popularity = parseFloat(node.popularity)
  712. var usage = browserslist.usage.global
  713. return Object.keys(usage).reduce(function (result, version) {
  714. if (node.sign === '>') {
  715. if (usage[version] > popularity) {
  716. result.push(version)
  717. }
  718. } else if (node.sign === '<') {
  719. if (usage[version] < popularity) {
  720. result.push(version)
  721. }
  722. } else if (node.sign === '<=') {
  723. if (usage[version] <= popularity) {
  724. result.push(version)
  725. }
  726. } else if (usage[version] >= popularity) {
  727. result.push(version)
  728. }
  729. return result
  730. }, [])
  731. }
  732. },
  733. popularity_in_my_stats: {
  734. matches: ['sign', 'popularity'],
  735. regexp: /^(>=?|<=?)\s*(\d+|\d+\.\d+|\.\d+)%\s+in\s+my\s+stats$/,
  736. select: function (context, node) {
  737. var popularity = parseFloat(node.popularity)
  738. if (!context.customUsage) {
  739. throw new BrowserslistError('Custom usage statistics was not provided')
  740. }
  741. var usage = context.customUsage
  742. return Object.keys(usage).reduce(function (result, version) {
  743. var percentage = usage[version]
  744. if (percentage == null) {
  745. return result
  746. }
  747. if (node.sign === '>') {
  748. if (percentage > popularity) {
  749. result.push(version)
  750. }
  751. } else if (node.sign === '<') {
  752. if (percentage < popularity) {
  753. result.push(version)
  754. }
  755. } else if (node.sign === '<=') {
  756. if (percentage <= popularity) {
  757. result.push(version)
  758. }
  759. } else if (percentage >= popularity) {
  760. result.push(version)
  761. }
  762. return result
  763. }, [])
  764. }
  765. },
  766. popularity_in_config_stats: {
  767. matches: ['sign', 'popularity', 'config'],
  768. regexp: /^(>=?|<=?)\s*(\d+|\d+\.\d+|\.\d+)%\s+in\s+(\S+)\s+stats$/,
  769. select: function (context, node) {
  770. var popularity = parseFloat(node.popularity)
  771. var stats = env.loadStat(context, node.config, browserslist.data)
  772. if (stats) {
  773. context.customUsage = {}
  774. for (var browser in stats) {
  775. fillUsage(context.customUsage, browser, stats[browser])
  776. }
  777. }
  778. if (!context.customUsage) {
  779. throw new BrowserslistError('Custom usage statistics was not provided')
  780. }
  781. var usage = context.customUsage
  782. return Object.keys(usage).reduce(function (result, version) {
  783. var percentage = usage[version]
  784. if (percentage == null) {
  785. return result
  786. }
  787. if (node.sign === '>') {
  788. if (percentage > popularity) {
  789. result.push(version)
  790. }
  791. } else if (node.sign === '<') {
  792. if (percentage < popularity) {
  793. result.push(version)
  794. }
  795. } else if (node.sign === '<=') {
  796. if (percentage <= popularity) {
  797. result.push(version)
  798. }
  799. } else if (percentage >= popularity) {
  800. result.push(version)
  801. }
  802. return result
  803. }, [])
  804. }
  805. },
  806. popularity_in_place: {
  807. matches: ['sign', 'popularity', 'place'],
  808. regexp: /^(>=?|<=?)\s*(\d+|\d+\.\d+|\.\d+)%\s+in\s+((alt-)?\w\w)$/,
  809. select: function (context, node) {
  810. var popularity = parseFloat(node.popularity)
  811. var place = node.place
  812. if (place.length === 2) {
  813. place = place.toUpperCase()
  814. } else {
  815. place = place.toLowerCase()
  816. }
  817. env.loadCountry(browserslist.usage, place, browserslist.data)
  818. var usage = browserslist.usage[place]
  819. return Object.keys(usage).reduce(function (result, version) {
  820. var percentage = usage[version]
  821. if (percentage == null) {
  822. return result
  823. }
  824. if (node.sign === '>') {
  825. if (percentage > popularity) {
  826. result.push(version)
  827. }
  828. } else if (node.sign === '<') {
  829. if (percentage < popularity) {
  830. result.push(version)
  831. }
  832. } else if (node.sign === '<=') {
  833. if (percentage <= popularity) {
  834. result.push(version)
  835. }
  836. } else if (percentage >= popularity) {
  837. result.push(version)
  838. }
  839. return result
  840. }, [])
  841. }
  842. },
  843. cover: {
  844. matches: ['coverage'],
  845. regexp: /^cover\s+(\d+|\d+\.\d+|\.\d+)%$/i,
  846. select: coverQuery
  847. },
  848. cover_in: {
  849. matches: ['coverage', 'place'],
  850. regexp: /^cover\s+(\d+|\d+\.\d+|\.\d+)%\s+in\s+(my\s+stats|(alt-)?\w\w)$/i,
  851. select: coverQuery
  852. },
  853. supports: {
  854. matches: ['supportType', 'feature'],
  855. regexp: /^(?:(fully|partially)\s+)?supports\s+([\w-]+)$/,
  856. select: function (context, node) {
  857. env.loadFeature(browserslist.cache, node.feature)
  858. var withPartial = node.supportType !== 'fully'
  859. var features = browserslist.cache[node.feature]
  860. var result = []
  861. for (var name in features) {
  862. var data = byName(name, context)
  863. // Only check desktop when latest released mobile has support
  864. var iMax = data.released.length - 1
  865. while (iMax >= 0) {
  866. if (data.released[iMax] in features[name]) break
  867. iMax--
  868. }
  869. var checkDesktop =
  870. context.mobileToDesktop &&
  871. name in browserslist.desktopNames &&
  872. isSupported(features[name][data.released[iMax]], withPartial)
  873. data.versions.forEach(function (version) {
  874. var flags = features[name][version]
  875. if (flags === undefined && checkDesktop) {
  876. flags = features[browserslist.desktopNames[name]][version]
  877. }
  878. if (isSupported(flags, withPartial)) {
  879. result.push(name + ' ' + version)
  880. }
  881. })
  882. }
  883. return result
  884. }
  885. },
  886. electron_range: {
  887. matches: ['from', 'to'],
  888. regexp: /^electron\s+([\d.]+)\s*-\s*([\d.]+)$/i,
  889. select: function (context, node) {
  890. var fromToUse = normalizeElectron(node.from)
  891. var toToUse = normalizeElectron(node.to)
  892. var from = parseFloat(node.from)
  893. var to = parseFloat(node.to)
  894. if (!e2c[fromToUse]) {
  895. throw new BrowserslistError('Unknown version ' + from + ' of electron')
  896. }
  897. if (!e2c[toToUse]) {
  898. throw new BrowserslistError('Unknown version ' + to + ' of electron')
  899. }
  900. return Object.keys(e2c)
  901. .filter(function (i) {
  902. var parsed = parseFloat(i)
  903. return parsed >= from && parsed <= to
  904. })
  905. .map(function (i) {
  906. return 'chrome ' + e2c[i]
  907. })
  908. }
  909. },
  910. node_range: {
  911. matches: ['from', 'to'],
  912. regexp: /^node\s+([\d.]+)\s*-\s*([\d.]+)$/i,
  913. select: function (context, node) {
  914. return browserslist.nodeVersions
  915. .filter(semverFilterLoose('>=', node.from))
  916. .filter(semverFilterLoose('<=', node.to))
  917. .map(function (v) {
  918. return 'node ' + v
  919. })
  920. }
  921. },
  922. browser_range: {
  923. matches: ['browser', 'from', 'to'],
  924. regexp: /^(\w+)\s+([\d.]+)\s*-\s*([\d.]+)$/i,
  925. select: function (context, node) {
  926. var data = checkName(node.browser, context)
  927. var from = parseFloat(normalizeVersion(data, node.from) || node.from)
  928. var to = parseFloat(normalizeVersion(data, node.to) || node.to)
  929. function filter(v) {
  930. var parsed = parseFloat(v)
  931. return parsed >= from && parsed <= to
  932. }
  933. return data.released.filter(filter).map(nameMapper(data.name))
  934. }
  935. },
  936. electron_ray: {
  937. matches: ['sign', 'version'],
  938. regexp: /^electron\s*(>=?|<=?)\s*([\d.]+)$/i,
  939. select: function (context, node) {
  940. var versionToUse = normalizeElectron(node.version)
  941. return Object.keys(e2c)
  942. .filter(generateFilter(node.sign, versionToUse))
  943. .map(function (i) {
  944. return 'chrome ' + e2c[i]
  945. })
  946. }
  947. },
  948. node_ray: {
  949. matches: ['sign', 'version'],
  950. regexp: /^node\s*(>=?|<=?)\s*([\d.]+)$/i,
  951. select: function (context, node) {
  952. return browserslist.nodeVersions
  953. .filter(generateSemverFilter(node.sign, node.version))
  954. .map(function (v) {
  955. return 'node ' + v
  956. })
  957. }
  958. },
  959. browser_ray: {
  960. matches: ['browser', 'sign', 'version'],
  961. regexp: /^(\w+)\s*(>=?|<=?)\s*([\d.]+)$/,
  962. select: function (context, node) {
  963. var version = node.version
  964. var data = checkName(node.browser, context)
  965. var alias = browserslist.versionAliases[data.name][version]
  966. if (alias) version = alias
  967. return data.released
  968. .filter(generateFilter(node.sign, version))
  969. .map(function (v) {
  970. return data.name + ' ' + v
  971. })
  972. }
  973. },
  974. firefox_esr: {
  975. matches: [],
  976. regexp: /^(firefox|ff|fx)\s+esr$/i,
  977. select: function () {
  978. return ['firefox 128']
  979. }
  980. },
  981. opera_mini_all: {
  982. matches: [],
  983. regexp: /(operamini|op_mini)\s+all/i,
  984. select: function () {
  985. return ['op_mini all']
  986. }
  987. },
  988. electron_version: {
  989. matches: ['version'],
  990. regexp: /^electron\s+([\d.]+)$/i,
  991. select: function (context, node) {
  992. var versionToUse = normalizeElectron(node.version)
  993. var chrome = e2c[versionToUse]
  994. if (!chrome) {
  995. throw new BrowserslistError(
  996. 'Unknown version ' + node.version + ' of electron'
  997. )
  998. }
  999. return ['chrome ' + chrome]
  1000. }
  1001. },
  1002. node_major_version: {
  1003. matches: ['version'],
  1004. regexp: /^node\s+(\d+)$/i,
  1005. select: nodeQuery
  1006. },
  1007. node_minor_version: {
  1008. matches: ['version'],
  1009. regexp: /^node\s+(\d+\.\d+)$/i,
  1010. select: nodeQuery
  1011. },
  1012. node_patch_version: {
  1013. matches: ['version'],
  1014. regexp: /^node\s+(\d+\.\d+\.\d+)$/i,
  1015. select: nodeQuery
  1016. },
  1017. current_node: {
  1018. matches: [],
  1019. regexp: /^current\s+node$/i,
  1020. select: function (context) {
  1021. return [env.currentNode(resolve, context)]
  1022. }
  1023. },
  1024. maintained_node: {
  1025. matches: [],
  1026. regexp: /^maintained\s+node\s+versions$/i,
  1027. select: function (context) {
  1028. var now = Date.now()
  1029. var queries = Object.keys(jsEOL)
  1030. .filter(function (key) {
  1031. return (
  1032. now < Date.parse(jsEOL[key].end) &&
  1033. now > Date.parse(jsEOL[key].start) &&
  1034. isEolReleased(key)
  1035. )
  1036. })
  1037. .map(function (key) {
  1038. return 'node ' + key.slice(1)
  1039. })
  1040. return resolve(queries, context)
  1041. }
  1042. },
  1043. phantomjs_1_9: {
  1044. matches: [],
  1045. regexp: /^phantomjs\s+1.9$/i,
  1046. select: function () {
  1047. return ['safari 5']
  1048. }
  1049. },
  1050. phantomjs_2_1: {
  1051. matches: [],
  1052. regexp: /^phantomjs\s+2.1$/i,
  1053. select: function () {
  1054. return ['safari 6']
  1055. }
  1056. },
  1057. browser_version: {
  1058. matches: ['browser', 'version'],
  1059. regexp: /^(\w+)\s+(tp|[\d.]+)$/i,
  1060. select: function (context, node) {
  1061. var version = node.version
  1062. if (/^tp$/i.test(version)) version = 'TP'
  1063. var data = checkName(node.browser, context)
  1064. var alias = normalizeVersion(data, version)
  1065. if (alias) {
  1066. version = alias
  1067. } else {
  1068. if (version.indexOf('.') === -1) {
  1069. alias = version + '.0'
  1070. } else {
  1071. alias = version.replace(/\.0$/, '')
  1072. }
  1073. alias = normalizeVersion(data, alias)
  1074. if (alias) {
  1075. version = alias
  1076. } else if (context.ignoreUnknownVersions) {
  1077. return []
  1078. } else {
  1079. throw new BrowserslistError(
  1080. 'Unknown version ' + version + ' of ' + node.browser
  1081. )
  1082. }
  1083. }
  1084. return [data.name + ' ' + version]
  1085. }
  1086. },
  1087. browserslist_config: {
  1088. matches: [],
  1089. regexp: /^browserslist config$/i,
  1090. needsPath: true,
  1091. select: function (context) {
  1092. return browserslist(undefined, context)
  1093. }
  1094. },
  1095. extends: {
  1096. matches: ['config'],
  1097. regexp: /^extends (.+)$/i,
  1098. needsPath: true,
  1099. select: function (context, node) {
  1100. return resolve(env.loadQueries(context, node.config), context)
  1101. }
  1102. },
  1103. defaults: {
  1104. matches: [],
  1105. regexp: /^defaults$/i,
  1106. select: function (context) {
  1107. return resolve(browserslist.defaults, context)
  1108. }
  1109. },
  1110. dead: {
  1111. matches: [],
  1112. regexp: /^dead$/i,
  1113. select: function (context) {
  1114. var dead = [
  1115. 'Baidu >= 0',
  1116. 'ie <= 11',
  1117. 'ie_mob <= 11',
  1118. 'bb <= 10',
  1119. 'op_mob <= 12.1',
  1120. 'samsung 4'
  1121. ]
  1122. return resolve(dead, context)
  1123. }
  1124. },
  1125. unknown: {
  1126. matches: [],
  1127. regexp: /^(\w+)$/i,
  1128. select: function (context, node) {
  1129. if (byName(node.query, context)) {
  1130. throw new BrowserslistError(
  1131. 'Specify versions in Browserslist query for browser ' + node.query
  1132. )
  1133. } else {
  1134. throw unknownQuery(node.query)
  1135. }
  1136. }
  1137. }
  1138. }
  1139. // Get and convert Can I Use data
  1140. ;(function () {
  1141. for (var name in agents) {
  1142. var browser = agents[name]
  1143. browserslist.data[name] = {
  1144. name: name,
  1145. versions: normalize(agents[name].versions),
  1146. released: normalize(agents[name].versions.slice(0, -3)),
  1147. releaseDate: agents[name].release_date
  1148. }
  1149. fillUsage(browserslist.usage.global, name, browser.usage_global)
  1150. browserslist.versionAliases[name] = {}
  1151. for (var i = 0; i < browser.versions.length; i++) {
  1152. var full = browser.versions[i]
  1153. if (!full) continue
  1154. if (full.indexOf('-') !== -1) {
  1155. var interval = full.split('-')
  1156. for (var j = 0; j < interval.length; j++) {
  1157. browserslist.versionAliases[name][interval[j]] = full
  1158. }
  1159. }
  1160. }
  1161. }
  1162. browserslist.nodeVersions = jsReleases.map(function (release) {
  1163. return release.version
  1164. })
  1165. })()
  1166. module.exports = browserslist