update-db.js 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363
  1. var childProcess = require('child_process')
  2. var escalade = require('escalade/sync')
  3. var pico = require('picocolors')
  4. var path = require('path')
  5. var fs = require('fs')
  6. var BrowserslistError = require('./error')
  7. function detectLockfile() {
  8. var packageDir = escalade('.', function (dir, names) {
  9. return names.indexOf('package.json') !== -1 ? dir : ''
  10. })
  11. if (!packageDir) {
  12. throw new BrowserslistError(
  13. 'Cannot find package.json. ' +
  14. 'Is this the right directory to run `npx browserslist --update-db` in?'
  15. )
  16. }
  17. var lockfileNpm = path.join(packageDir, 'package-lock.json')
  18. var lockfileShrinkwrap = path.join(packageDir, 'npm-shrinkwrap.json')
  19. var lockfileYarn = path.join(packageDir, 'yarn.lock')
  20. var lockfilePnpm = path.join(packageDir, 'pnpm-lock.yaml')
  21. if (fs.existsSync(lockfilePnpm)) {
  22. return { mode: 'pnpm', file: lockfilePnpm }
  23. } else if (fs.existsSync(lockfileNpm)) {
  24. return { mode: 'npm', file: lockfileNpm }
  25. } else if (fs.existsSync(lockfileYarn)) {
  26. var lock = { mode: 'yarn', file: lockfileYarn }
  27. lock.content = fs.readFileSync(lock.file).toString()
  28. lock.version = /# yarn lockfile v1/.test(lock.content) ? 1 : 2
  29. return lock
  30. } else if (fs.existsSync(lockfileShrinkwrap)) {
  31. return { mode: 'npm', file: lockfileShrinkwrap }
  32. }
  33. throw new BrowserslistError(
  34. 'No lockfile found. Run "npm install", "yarn install" or "pnpm install"'
  35. )
  36. }
  37. function getLatestInfo(lock) {
  38. if (lock.mode === 'yarn') {
  39. if (lock.version === 1) {
  40. return JSON.parse(
  41. childProcess.execSync('yarn info caniuse-lite --json').toString()
  42. ).data
  43. } else {
  44. return JSON.parse(
  45. childProcess.execSync('yarn npm info caniuse-lite --json').toString()
  46. )
  47. }
  48. }
  49. return JSON.parse(
  50. childProcess.execSync('npm show caniuse-lite --json').toString()
  51. )
  52. }
  53. function getBrowsersList() {
  54. return childProcess
  55. .execSync('npx browserslist')
  56. .toString()
  57. .trim()
  58. .split('\n')
  59. .map(function (line) {
  60. return line.trim().split(' ')
  61. })
  62. .reduce(function (result, entry) {
  63. if (!result[entry[0]]) {
  64. result[entry[0]] = []
  65. }
  66. result[entry[0]].push(entry[1])
  67. return result
  68. }, {})
  69. }
  70. function diffBrowsersLists(old, current) {
  71. var browsers = Object.keys(old).concat(
  72. Object.keys(current).filter(function (browser) {
  73. return old[browser] === undefined
  74. })
  75. )
  76. return browsers
  77. .map(function (browser) {
  78. var oldVersions = old[browser] || []
  79. var currentVersions = current[browser] || []
  80. var intersection = oldVersions.filter(function (version) {
  81. return currentVersions.indexOf(version) !== -1
  82. })
  83. var addedVersions = currentVersions.filter(function (version) {
  84. return intersection.indexOf(version) === -1
  85. })
  86. var removedVersions = oldVersions.filter(function (version) {
  87. return intersection.indexOf(version) === -1
  88. })
  89. return removedVersions
  90. .map(function (version) {
  91. return pico.red('- ' + browser + ' ' + version)
  92. })
  93. .concat(
  94. addedVersions.map(function (version) {
  95. return pico.green('+ ' + browser + ' ' + version)
  96. })
  97. )
  98. })
  99. .reduce(function (result, array) {
  100. return result.concat(array)
  101. }, [])
  102. .join('\n')
  103. }
  104. function updateNpmLockfile(lock, latest) {
  105. var metadata = { latest: latest, versions: [] }
  106. var content = deletePackage(JSON.parse(lock.content), metadata)
  107. metadata.content = JSON.stringify(content, null, ' ')
  108. return metadata
  109. }
  110. function deletePackage(node, metadata) {
  111. if (node.dependencies) {
  112. if (node.dependencies['caniuse-lite']) {
  113. var version = node.dependencies['caniuse-lite'].version
  114. metadata.versions[version] = true
  115. delete node.dependencies['caniuse-lite']
  116. }
  117. for (var i in node.dependencies) {
  118. node.dependencies[i] = deletePackage(node.dependencies[i], metadata)
  119. }
  120. }
  121. return node
  122. }
  123. var yarnVersionRe = /version "(.*?)"/
  124. function updateYarnLockfile(lock, latest) {
  125. var blocks = lock.content.split(/(\n{2,})/).map(function (block) {
  126. return block.split('\n')
  127. })
  128. var versions = {}
  129. blocks.forEach(function (lines) {
  130. if (lines[0].indexOf('caniuse-lite@') !== -1) {
  131. var match = yarnVersionRe.exec(lines[1])
  132. versions[match[1]] = true
  133. if (match[1] !== latest.version) {
  134. lines[1] = lines[1].replace(
  135. /version "[^"]+"/,
  136. 'version "' + latest.version + '"'
  137. )
  138. lines[2] = lines[2].replace(
  139. /resolved "[^"]+"/,
  140. 'resolved "' + latest.dist.tarball + '"'
  141. )
  142. lines[3] = latest.dist.integrity
  143. ? lines[3].replace(
  144. /integrity .+/,
  145. 'integrity ' + latest.dist.integrity
  146. )
  147. : ''
  148. }
  149. }
  150. })
  151. var content = blocks
  152. .map(function (lines) {
  153. return lines.join('\n')
  154. })
  155. .join('')
  156. return { content: content, versions: versions }
  157. }
  158. function updatePnpmLockfile(lock, latest) {
  159. var versions = {}
  160. var lines = lock.content.split('\n')
  161. var i
  162. var j
  163. var lineParts
  164. for (i = 0; i < lines.length; i++) {
  165. if (lines[i].indexOf('caniuse-lite:') >= 0) {
  166. lineParts = lines[i].split(/:\s?/, 2)
  167. if (lineParts[1].indexOf('/') >= 0) {
  168. var sublineParts = lineParts[1].split(/([/:])/)
  169. for (j = 0; j < sublineParts.length; j++) {
  170. if (sublineParts[j].indexOf('caniuse-lite') >= 0) {
  171. versions[sublineParts[j + 2]] = true
  172. sublineParts[j + 2] = latest.version
  173. break
  174. }
  175. }
  176. lineParts[1] = sublineParts.join('')
  177. } else {
  178. versions[lineParts[1]] = true
  179. }
  180. lines[i] = lineParts[0] + ': ' + latest.version
  181. } else if (lines[i].indexOf('/caniuse-lite') >= 0) {
  182. lineParts = lines[i].split(/([/:])/)
  183. for (j = 0; j < lineParts.length; j++) {
  184. if (lineParts[j].indexOf('caniuse-lite') >= 0) {
  185. versions[lineParts[j + 2]] = true
  186. lineParts[j + 2] = latest.version
  187. break
  188. }
  189. }
  190. lines[i] = lineParts.join('')
  191. for (i = i + 1; i < lines.length; i++) {
  192. if (lines[i].indexOf('integrity: ') !== -1) {
  193. lines[i] = lines[i].replace(
  194. /integrity: .+/,
  195. 'integrity: ' + latest.dist.integrity
  196. )
  197. } else if (lines[i].indexOf(' /') !== -1) {
  198. break
  199. }
  200. }
  201. }
  202. }
  203. return { content: lines.join('\n'), versions: versions }
  204. }
  205. function updateLockfile(lock, latest) {
  206. if (!lock.content) lock.content = fs.readFileSync(lock.file).toString()
  207. if (lock.mode === 'npm') {
  208. return updateNpmLockfile(lock, latest)
  209. } else if (lock.mode === 'yarn') {
  210. return updateYarnLockfile(lock, latest)
  211. }
  212. return updatePnpmLockfile(lock, latest)
  213. }
  214. function updatePackageManually(print, lock, latest) {
  215. var lockfileData = updateLockfile(lock, latest)
  216. var caniuseVersions = Object.keys(lockfileData.versions).sort()
  217. if (caniuseVersions.length === 1 && caniuseVersions[0] === latest.version) {
  218. print(
  219. 'Installed version: ' +
  220. pico.bold(pico.green(latest.version)) +
  221. '\n' +
  222. pico.bold(pico.green('caniuse-lite is up to date')) +
  223. '\n'
  224. )
  225. return
  226. }
  227. if (caniuseVersions.length === 0) {
  228. caniuseVersions[0] = 'none'
  229. }
  230. print(
  231. 'Installed version' +
  232. (caniuseVersions.length === 1 ? ': ' : 's: ') +
  233. pico.bold(pico.red(caniuseVersions.join(', '))) +
  234. '\n' +
  235. 'Removing old caniuse-lite from lock file\n'
  236. )
  237. fs.writeFileSync(lock.file, lockfileData.content)
  238. var install = lock.mode === 'yarn' ? 'yarn add -W' : lock.mode + ' install'
  239. print(
  240. 'Installing new caniuse-lite version\n' +
  241. pico.yellow('$ ' + install + ' caniuse-lite') +
  242. '\n'
  243. )
  244. try {
  245. childProcess.execSync(install + ' caniuse-lite')
  246. } catch (e) /* istanbul ignore next */ {
  247. print(
  248. pico.red(
  249. '\n' +
  250. e.stack +
  251. '\n\n' +
  252. 'Problem with `' +
  253. install +
  254. ' caniuse-lite` call. ' +
  255. 'Run it manually.\n'
  256. )
  257. )
  258. process.exit(1)
  259. }
  260. var del = lock.mode === 'yarn' ? 'yarn remove -W' : lock.mode + ' uninstall'
  261. print(
  262. 'Cleaning package.json dependencies from caniuse-lite\n' +
  263. pico.yellow('$ ' + del + ' caniuse-lite') +
  264. '\n'
  265. )
  266. childProcess.execSync(del + ' caniuse-lite')
  267. }
  268. module.exports = function updateDB(print) {
  269. var lock = detectLockfile()
  270. var latest = getLatestInfo(lock)
  271. var browsersListRetrievalError
  272. var oldBrowsersList
  273. try {
  274. oldBrowsersList = getBrowsersList()
  275. } catch (e) {
  276. browsersListRetrievalError = e
  277. }
  278. print('Latest version: ' + pico.bold(pico.green(latest.version)) + '\n')
  279. if (lock.mode === 'yarn' && lock.version !== 1) {
  280. var update = 'yarn up -R'
  281. print(
  282. 'Updating caniuse-lite version\n' +
  283. pico.yellow('$ ' + update + ' caniuse-lite') +
  284. '\n'
  285. )
  286. try {
  287. childProcess.execSync(update + ' caniuse-lite')
  288. } catch (e) /* istanbul ignore next */ {
  289. print(
  290. pico.red(
  291. '\n' +
  292. e.stack +
  293. '\n\n' +
  294. 'Problem with `' +
  295. update +
  296. ' caniuse-lite` call. ' +
  297. 'Run it manually.\n'
  298. )
  299. )
  300. process.exit(1)
  301. }
  302. } else {
  303. updatePackageManually(print, lock, latest)
  304. }
  305. print('caniuse-lite has been successfully updated\n')
  306. var currentBrowsersList
  307. if (!browsersListRetrievalError) {
  308. try {
  309. currentBrowsersList = getBrowsersList()
  310. } catch (e) /* istanbul ignore next */ {
  311. browsersListRetrievalError = e
  312. }
  313. }
  314. if (browsersListRetrievalError) {
  315. print(
  316. pico.red(
  317. '\n' +
  318. browsersListRetrievalError.stack +
  319. '\n\n' +
  320. 'Problem with browser list retrieval.\n' +
  321. 'Target browser changes won’t be shown.\n'
  322. )
  323. )
  324. } else {
  325. var targetBrowserChanges = diffBrowsersLists(
  326. oldBrowsersList,
  327. currentBrowsersList
  328. )
  329. if (targetBrowserChanges) {
  330. print('\nTarget browser changes:\n')
  331. print(targetBrowserChanges + '\n')
  332. } else {
  333. print('\n' + pico.green('No target browser changes') + '\n')
  334. }
  335. }
  336. }