The Wayback Machine - https://web.archive.org./web/20220306020444/https://github.com/npm/cli/commit/6734ba36dd6e07a859ab4d6eb4f264d2c0022276
Skip to content
Permalink
Browse files
feat: streaming debug logfile
This decouples the log file writing from the terminal logging. We now
open an append only file at the start of the process and stream logs to
it. We still only display the log file message in timing mode or if
there is an error, but the file is still written regardless.

All logging now goes through `proc-log` and this is the first step to
removing `npmlog`. For now `npmlog` is still used for the terminal
logging but with a shim in front of it to make it easier to test and
use in conjunction with `proc-log`. Ref: npm/statusboard#366

This also refactors many of the tests to always use an explicit
`t.testdir` for their cache since the file is opened on each `new Npm()`.
Tests are also refactored to use more of `MockNpm` with behavior to
add config items and load `npm` if necessary. A new fixture `mockGlobals`
was also added to make much of this more ergonomic. Ref: npm/statusboard#410

Closes npm/statusboard#411
Closes npm/statusboard#367

PR-URL: #4062
Credit: @lukekarrys
Close: #4062
Reviewed-by: @wraithgar
  • Loading branch information
lukekarrys committed Dec 2, 2021
1 parent 037f2cc commit 6734ba36dd6e07a859ab4d6eb4f264d2c0022276
Showing with 4,631 additions and 3,286 deletions.
  1. +13 −1 .eslintrc.json
  2. +3 −6 lib/auth/legacy.js
  3. +1 −2 lib/auth/sso.js
  4. +17 −16 lib/cli.js
  5. +2 −1 lib/commands/adduser.js
  6. +1 −0 lib/commands/bin.js
  7. +1 −1 lib/commands/bugs.js
  8. +2 −2 lib/commands/cache.js
  9. +2 −3 lib/commands/ci.js
  10. +6 −5 lib/commands/config.js
  11. +2 −1 lib/commands/dedupe.js
  12. +7 −9 lib/commands/diff.js
  13. +1 −2 lib/commands/dist-tag.js
  14. +1 −2 lib/commands/docs.js
  15. +9 −8 lib/commands/doctor.js
  16. +1 −1 lib/commands/exec.js
  17. +4 −3 lib/commands/explore.js
  18. +2 −1 lib/commands/fund.js
  19. +8 −8 lib/commands/init.js
  20. +3 −3 lib/commands/install.js
  21. +6 −5 lib/commands/link.js
  22. +1 −1 lib/commands/logout.js
  23. +1 −2 lib/commands/owner.js
  24. +3 −6 lib/commands/pack.js
  25. +1 −1 lib/commands/ping.js
  26. +1 −1 lib/commands/profile.js
  27. +2 −1 lib/commands/prune.js
  28. +3 −3 lib/commands/publish.js
  29. +1 −2 lib/commands/repo.js
  30. +1 −1 lib/commands/run-script.js
  31. +1 −1 lib/commands/search.js
  32. +1 −1 lib/commands/set-script.js
  33. +1 −2 lib/commands/shrinkwrap.js
  34. +1 −2 lib/commands/star.js
  35. +1 −2 lib/commands/stars.js
  36. +1 −1 lib/commands/token.js
  37. +2 −1 lib/commands/uninstall.js
  38. +4 −4 lib/commands/unpublish.js
  39. +2 −2 lib/commands/update.js
  40. +2 −2 lib/commands/view.js
  41. +101 −52 lib/npm.js
  42. +3 −1 lib/utils/audit-error.js
  43. +0 −35 lib/utils/cleanup-log-files.js
  44. +8 −1 lib/utils/config/definitions.js
  45. +1 −1 lib/utils/deref-command.js
  46. +119 −0 lib/utils/display.js
  47. +6 −3 lib/utils/error-message.js
  48. +77 −130 lib/utils/exit-handler.js
  49. +245 −0 lib/utils/log-file.js
  50. +59 −0 lib/utils/log-shim.js
  51. +0 −22 lib/utils/proc-log-listener.js
  52. +1 −1 lib/utils/pulse-till-done.js
  53. +7 −7 lib/utils/read-user-info.js
  54. +1 −1 lib/utils/reify-output.js
  55. +0 −66 lib/utils/setup-log.js
  56. +2 −2 lib/utils/tar.js
  57. +111 −0 lib/utils/timers.js
  58. +11 −5 lib/utils/unsupported.js
  59. +2 −1 lib/utils/update-notifier.js
  60. +1 −1 lib/utils/usage.js
  61. +13 −0 lib/utils/with-chown-sync.js
  62. +2 −0 package-lock.json
  63. +2 −0 package.json
  64. +4 −2 tap-snapshots/test/lib/commands/config.js.test.cjs
  65. +35 −35 tap-snapshots/test/lib/commands/shrinkwrap.js.test.cjs
  66. +4 −4 tap-snapshots/test/lib/commands/view.js.test.cjs
  67. +60 −23 tap-snapshots/test/lib/utils/error-message.js.test.cjs
  68. +51 −11 tap-snapshots/test/lib/utils/exit-handler.js.test.cjs
  69. +68 −0 tap-snapshots/test/lib/utils/log-file.js.test.cjs
  70. +19 −0 test/fixtures/clean-snapshot.js
  71. +210 −0 test/fixtures/mock-globals.js
  72. +71 −0 test/fixtures/mock-logs.js
  73. +115 −75 test/fixtures/mock-npm.js
  74. +25 −21 test/fixtures/sandbox.js
  75. +3 −1 test/index.js
  76. +1 −1 test/lib/auth/legacy.js
  77. +1 −1 test/lib/auth/sso.js
  78. +79 −103 test/lib/cli.js
  79. +151 −156 test/lib/commands/access.js
  80. +9 −0 test/lib/commands/adduser.js
  81. +70 −73 test/lib/commands/audit.js
  82. +7 −6 test/lib/commands/birthday.js
  83. +8 −11 test/lib/commands/cache.js
  84. +1 −1 test/lib/commands/ci.js
  85. +87 −162 test/lib/commands/completion.js
  86. +32 −29 test/lib/commands/dedupe.js
  87. +1 −1 test/lib/commands/diff.js
  88. +1 −1 test/lib/commands/dist-tag.js
  89. +40 −43 test/lib/commands/doctor.js
  90. +19 −11 test/lib/commands/exec.js
  91. +7 −4 test/lib/commands/explore.js
  92. +18 −17 test/lib/commands/find-dupes.js
  93. +2 −4 test/lib/commands/get.js
  94. +17 −15 test/lib/commands/init.js
  95. +21 −41 test/lib/commands/install.js
  96. +70 −61 test/lib/commands/logout.js
  97. +11 −11 test/lib/commands/owner.js
  98. +73 −73 test/lib/commands/pack.js
  99. +3 −3 test/lib/commands/ping.js
  100. +2 −3 test/lib/commands/prefix.js
  101. +17 −15 test/lib/commands/profile.js
  102. +14 −12 test/lib/commands/prune.js
  103. +22 −9 test/lib/commands/publish.js
  104. +45 −46 test/lib/commands/repo.js
  105. +17 −17 test/lib/commands/restart.js
  106. +2 −3 test/lib/commands/root.js
  107. +9 −3 test/lib/commands/run-script.js
  108. +1 −1 test/lib/commands/set-script.js
  109. +1 −0 test/lib/commands/set.js
  110. +13 −21 test/lib/commands/shrinkwrap.js
  111. +5 −5 test/lib/commands/star.js
  112. +6 −6 test/lib/commands/stars.js
  113. +17 −16 test/lib/commands/start.js
  114. +15 −15 test/lib/commands/stop.js
  115. +16 −16 test/lib/commands/test.js
  116. +17 −7 test/lib/commands/token.js
  117. +5 −4 test/lib/commands/unpublish.js
  118. +15 −13 test/lib/commands/update.js
  119. +256 −248 test/lib/commands/version.js
  120. +10 −8 test/lib/commands/view.js
  121. +9 −11 test/lib/commands/whoami.js
  122. +321 −0 test/lib/fixtures/mock-globals.js
  123. +4 −9 test/lib/load-all-commands.js
  124. +18 −21 test/lib/load-all.js
  125. +261 −267 test/lib/npm.js
  126. +5 −4 test/lib/utils/audit-error.js
  127. +0 −79 test/lib/utils/cleanup-log-files.js
  128. +67 −40 test/lib/utils/config/definitions.js
  129. +2 −4 test/lib/utils/did-you-mean.js
  130. +85 −0 test/lib/utils/display.js
  131. +104 −160 test/lib/utils/error-message.js
  132. +332 −286 test/lib/utils/exit-handler.js
  133. +18 −16 test/lib/utils/is-windows-bash.js
  134. +333 −0 test/lib/utils/log-file.js
  135. +100 −0 test/lib/utils/log-shim.js
  136. +2 −4 test/lib/utils/npm-usage.js
  137. +0 −41 test/lib/utils/proc-log-listener.js
  138. +9 −10 test/lib/utils/pulse-till-done.js
  139. +14 −16 test/lib/utils/read-user-info.js
  140. +4 −3 test/lib/utils/reify-output.js
  141. +0 −296 test/lib/utils/setup-log.js
  142. +16 −36 test/lib/utils/tar.js
  143. +82 −0 test/lib/utils/timers.js
  144. +21 −31 test/lib/utils/unsupported.js
  145. +31 −39 test/lib/utils/update-notifier.js
@@ -1,3 +1,15 @@
{
"extends": ["@npmcli"]
"extends": ["@npmcli"],
"overrides": [{
"files": "test/**",
"rules": {
"no-extend-native": "off",
"no-global-assign": "off"
}
}, {
"files": ["lib/**"],
"rules": {
"no-console": "warn"
}
}]
}
@@ -1,15 +1,12 @@
const log = require('npmlog')
const profile = require('npm-profile')

const log = require('../utils/log-shim')
const openUrl = require('../utils/open-url.js')
const read = require('../utils/read-user-info.js')

const loginPrompter = async (creds) => {
const opts = { log: log }

creds.username = await read.username('Username:', creds.username, opts)
creds.username = await read.username('Username:', creds.username)
creds.password = await read.password('Password:', creds.password)
creds.email = await read.email('Email: (this IS public) ', creds.email, opts)
creds.email = await read.email('Email: (this IS public) ', creds.email)

return creds
}
@@ -7,10 +7,9 @@
// CLI, we can remove this, and fold the lib/auth/legacy.js back into
// lib/adduser.js

const log = require('npmlog')
const profile = require('npm-profile')
const npmFetch = require('npm-registry-fetch')

const log = require('../utils/log-shim')
const openUrl = require('../utils/open-url.js')
const otplease = require('../utils/otplease.js')

@@ -4,20 +4,23 @@ module.exports = async process => {
// leak any private CLI configs to other programs
process.title = 'npm'

const { checkForBrokenNode, checkForUnsupportedNode } = require('../lib/utils/unsupported.js')

// We used to differentiate between known broken and unsupported
// versions of node and attempt to only log unsupported but still run.
// After we dropped node 10 support, we can use new features
// (like static, private, etc) which will only give vague syntax errors,
// so now both broken and unsupported use console, but only broken
// will process.exit. It is important to now perform *both* of these
// checks as early as possible so the user gets the error message.
const { checkForBrokenNode, checkForUnsupportedNode } = require('./utils/unsupported.js')
checkForBrokenNode()

const log = require('npmlog')
// pause it here so it can unpause when we've loaded the configs
// and know what loglevel we should be printing.
log.pause()

checkForUnsupportedNode()

const Npm = require('../lib/npm.js')
const exitHandler = require('./utils/exit-handler.js')
process.on('uncaughtException', exitHandler)
process.on('unhandledRejection', exitHandler)

const Npm = require('./npm.js')
const npm = new Npm()
const exitHandler = require('../lib/utils/exit-handler.js')
exitHandler.setNpm(npm)

// if npm is called as "npmg" or "npm_g", then
@@ -26,16 +29,14 @@ module.exports = async process => {
process.argv.splice(1, 1, 'npm', '-g')
}

const replaceInfo = require('../lib/utils/replace-info.js')
const log = require('./utils/log-shim.js')
const replaceInfo = require('./utils/replace-info.js')
log.verbose('cli', replaceInfo(process.argv))

log.info('using', 'npm@%s', npm.version)
log.info('using', 'node@%s', process.version)

process.on('uncaughtException', exitHandler)
process.on('unhandledRejection', exitHandler)

const updateNotifier = require('../lib/utils/update-notifier.js')
const updateNotifier = require('./utils/update-notifier.js')

let cmd
// now actually fire up npm and run the command.
@@ -63,7 +64,7 @@ module.exports = async process => {
}

await npm.exec(cmd, npm.argv)
exitHandler()
return exitHandler()
} catch (err) {
if (err.code === 'EUNKNOWNCOMMAND') {
const didYouMean = require('./utils/did-you-mean.js')
@@ -1,4 +1,4 @@
const log = require('npmlog')
const log = require('../utils/log-shim.js')
const replaceInfo = require('../utils/replace-info.js')
const BaseCommand = require('../base-command.js')
const authTypes = {
@@ -31,6 +31,7 @@ class AddUser extends BaseCommand {
creds,
registry,
scope,
log,
})

await this.updateConfig({
@@ -10,6 +10,7 @@ class Bin extends BaseCommand {
const b = this.npm.bin
this.npm.output(b)
if (this.npm.config.get('global') && !envPath.includes(b)) {
// XXX: does this need to be console?
console.error('(not in PATH env variable)')

Check warning on line 14 in lib/commands/bin.js

GitHub Actions / lint

lib/commands/bin.js#L14

Unexpected console statement
}
}
@@ -1,5 +1,5 @@
const log = require('npmlog')
const pacote = require('pacote')
const log = require('../utils/log-shim')
const openUrl = require('../utils/open-url.js')
const hostedFromMani = require('../utils/hosted-git-info-from-manifest.js')
const BaseCommand = require('../base-command.js')
@@ -1,6 +1,5 @@
const cacache = require('cacache')
const { promisify } = require('util')
const log = require('npmlog')
const pacote = require('pacote')
const path = require('path')
const rimraf = promisify(require('rimraf'))
@@ -9,6 +8,7 @@ const BaseCommand = require('../base-command.js')
const npa = require('npm-package-arg')
const jsonParse = require('json-parse-even-better-errors')
const localeCompare = require('@isaacs/string-locale-compare')('en')
const log = require('../utils/log-shim')

const searchCachePackage = async (path, spec, cacheKeys) => {
const parsed = npa(spec)
@@ -141,7 +141,7 @@ class Cache extends BaseCommand {
try {
entry = await cacache.get(cachePath, key)
} catch (err) {
this.npm.log.warn(`Not Found: ${key}`)
log.warn(`Not Found: ${key}`)
break
}
this.npm.output(`Deleted: ${key}`)
@@ -5,8 +5,7 @@ const reifyFinish = require('../utils/reify-finish.js')
const runScript = require('@npmcli/run-script')
const fs = require('fs')
const readdir = util.promisify(fs.readdir)

const log = require('npmlog')
const log = require('../utils/log-shim.js')

const removeNodeModules = async where => {
const rimrafOpts = { glob: false }
@@ -39,7 +38,7 @@ class CI extends ArboristWorkspaceCmd {
const opts = {
...this.npm.flatOptions,
path: where,
log: this.npm.log,
log,
save: false, // npm ci should never modify the lockfile or package.json
workspaces: this.workspaceNames,
}
@@ -11,6 +11,7 @@ const { spawn } = require('child_process')
const { EOL } = require('os')
const ini = require('ini')
const localeCompare = require('@isaacs/string-locale-compare')('en')
const log = require('../utils/log-shim.js')

// take an array of `[key, value, k2=v2, k3, v3, ...]` and turn into
// { key: value, k2: v2, k3: v3 }
@@ -87,12 +88,12 @@ class Config extends BaseCommand {
}

async execWorkspaces (args, filters) {
this.npm.log.warn('config', 'This command does not support workspaces.')
log.warn('config', 'This command does not support workspaces.')
return this.exec(args)
}

async exec ([action, ...args]) {
this.npm.log.disableProgress()
log.disableProgress()
try {
switch (action) {
case 'set':
@@ -117,7 +118,7 @@ class Config extends BaseCommand {
throw this.usageError()
}
} finally {
this.npm.log.enableProgress()
log.enableProgress()
}
}

@@ -128,10 +129,10 @@ class Config extends BaseCommand {

const where = this.npm.flatOptions.location
for (const [key, val] of Object.entries(keyValues(args))) {
this.npm.log.info('config', 'set %j %j', key, val)
log.info('config', 'set %j %j', key, val)
this.npm.config.set(key, val || '', where)
if (!this.npm.config.validate(where)) {
this.npm.log.warn('config', 'omitting invalid config values')
log.warn('config', 'omitting invalid config values')
}
}

@@ -1,6 +1,7 @@
// dedupe duplicated packages, or find them in the tree
const Arborist = require('@npmcli/arborist')
const reifyFinish = require('../utils/reify-finish.js')
const log = require('../utils/log-shim.js')

const ArboristWorkspaceCmd = require('../arborist-cmd.js')

@@ -32,7 +33,7 @@ class Dedupe extends ArboristWorkspaceCmd {
const where = this.npm.prefix
const opts = {
...this.npm.flatOptions,
log: this.npm.log,
log,
path: where,
dryRun,
workspaces: this.workspaceNames,
@@ -1,13 +1,11 @@
const { resolve } = require('path')

const semver = require('semver')
const libnpmdiff = require('libnpmdiff')
const npa = require('npm-package-arg')
const Arborist = require('@npmcli/arborist')
const npmlog = require('npmlog')
const pacote = require('pacote')
const pickManifest = require('npm-pick-manifest')

const log = require('../utils/log-shim')
const readPackageName = require('../utils/read-package-name.js')
const BaseCommand = require('../base-command.js')

@@ -57,7 +55,7 @@ class Diff extends BaseCommand {
}

const [a, b] = await this.retrieveSpecs(specs)
npmlog.info('diff', { src: a, dst: b })
log.info('diff', { src: a, dst: b })

const res = await libnpmdiff([a, b], {
...this.npm.flatOptions,
@@ -83,7 +81,7 @@ class Diff extends BaseCommand {
try {
name = await readPackageName(this.prefix)
} catch (e) {
npmlog.verbose('diff', 'could not read project dir package.json')
log.verbose('diff', 'could not read project dir package.json')
}

if (!name) {
@@ -116,7 +114,7 @@ class Diff extends BaseCommand {
try {
pkgName = await readPackageName(this.prefix)
} catch (e) {
npmlog.verbose('diff', 'could not read project dir package.json')
log.verbose('diff', 'could not read project dir package.json')
noPackageJson = true
}

@@ -154,7 +152,7 @@ class Diff extends BaseCommand {
actualTree.inventory.query('name', spec.name)
.values().next().value
} catch (e) {
npmlog.verbose('diff', 'failed to load actual install tree')
log.verbose('diff', 'failed to load actual install tree')
}

if (!node || !node.name || !node.package || !node.package.version) {
@@ -227,7 +225,7 @@ class Diff extends BaseCommand {
try {
pkgName = await readPackageName(this.prefix)
} catch (e) {
npmlog.verbose('diff', 'could not read project dir package.json')
log.verbose('diff', 'could not read project dir package.json')
}

if (!pkgName) {
@@ -261,7 +259,7 @@ class Diff extends BaseCommand {
const arb = new Arborist(opts)
actualTree = await arb.loadActual(opts)
} catch (e) {
npmlog.verbose('diff', 'failed to load actual install tree')
log.verbose('diff', 'failed to load actual install tree')
}

return specs.map(i => {
@@ -1,8 +1,7 @@
const log = require('npmlog')
const npa = require('npm-package-arg')
const regFetch = require('npm-registry-fetch')
const semver = require('semver')

const log = require('../utils/log-shim')
const otplease = require('../utils/otplease.js')
const readPackageName = require('../utils/read-package-name.js')
const BaseCommand = require('../base-command.js')
@@ -1,8 +1,7 @@
const log = require('npmlog')
const pacote = require('pacote')
const openUrl = require('../utils/open-url.js')
const hostedFromMani = require('../utils/hosted-git-info-from-manifest.js')

const log = require('../utils/log-shim')
const BaseCommand = require('../base-command.js')
class Docs extends BaseCommand {
static description = 'Open documentation for a package in a web browser'

0 comments on commit 6734ba3

Please sign in to comment.