update-db.js 9.4 KB

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