Skip to content
Permalink

Comparing changes

Choose two branches to see what’s changed or to start a new pull request. If you need to, you can also or learn more about diff comparisons.

Open a pull request

Create a new pull request by comparing changes across two branches. If you need to, you can also . Learn more about diff comparisons here.
base repository: raineorshine/npm-check-updates
Failed to load repositories. Confirm that selected base ref is valid, then try again.
Loading
base: v15.2.2
Choose a base ref
...
head repository: raineorshine/npm-check-updates
Failed to load repositories. Confirm that selected head ref is valid, then try again.
Loading
compare: v15.2.3
Choose a head ref
  • 2 commits
  • 9 files changed
  • 1 contributor

Commits on Jul 9, 2022

  1. Verified

    This commit was created on GitHub.com and signed with GitHub’s verified signature. The key has expired.
    Copy the full SHA
    109236c View commit details
  2. 15.2.3

    raineorshine committed Jul 9, 2022
    Copy the full SHA
    a620a7e View commit details
Showing with 296 additions and 338 deletions.
  1. +2 −2 package-lock.json
  2. +1 −1 package.json
  3. +89 −2 src/index.ts
  4. +1 −7 src/lib/doctor.ts
  5. +155 −79 src/lib/runLocal.ts
  6. +4 −160 src/lib/upgradePackageData.ts
  7. +1 −1 src/logging.ts
  8. +43 −9 test/interactive.test.ts
  9. +0 −77 test/upgradePackageData.test.ts
4 changes: 2 additions & 2 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "npm-check-updates",
"version": "15.2.2",
"version": "15.2.3",
"author": "Tomas Junnonen <tomas1@gmail.com>",
"license": "Apache-2.0",
"contributors": [
91 changes: 89 additions & 2 deletions src/index.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
import fs from 'fs'
import os from 'os'
import path from 'path'
import globby from 'globby'
import spawn from 'spawn-please'
import _ from 'lodash'
import Chalk from 'chalk'
import prompts from 'prompts-ncu'
@@ -56,6 +58,83 @@ function checkIfVolta(options: Options): void {
}
}

/** Returns true if a file exists. */
const exists = (path: string) =>
fs.promises.stat(path).then(
() => true,
() => false,
)

/** Returns the package manager that should be used to install packages after running "ncu -u". Detects pnpm via pnpm-lock.yarn. This is the one place that pnpm needs to be detected, since otherwise it is backwards compatible with npm. */
const getPackageManagerForInstall = async (options: Options, pkgFile: string) => {
if (options.packageManager === 'yarn') return 'yarn'
const pnpmLockFile = path.resolve(pkgFile, '../pnpm-lock.yaml')
const pnpm = await exists(pnpmLockFile)
return pnpm ? 'pnpm' : 'npm'
}

/** Either suggest an install command based on the package manager, or in interactive mode, prompt to autoinstall. */
const npmInstallHint = async (
pkgs: string[],
analysis: Index<PackageFile> | PackageFile,
options: Options,
): Promise<unknown> => {
const chalk = options.color ? new Chalk.Instance({ level: 1 }) : Chalk

// if no packages were upgraded (i.e. all dependendencies deselected in interactive mode), then bail without suggesting an install.
// normalize the analysis for one or many packages
const analysisNormalized =
pkgs.length === 1 ? { [pkgs[0]]: analysis as PackageFile } : (analysis as Index<PackageFile>)
const someUpgraded = Object.values(analysisNormalized).some(upgrades => Object.keys(upgrades).length > 0)
if (!someUpgraded) return

let showInstallHint = true

// for the purpose of the install hint, just use the package manager used in the first subproject
// if autoinstalling, the actual package manager in each subproject will be used
const packageManager = await getPackageManagerForInstall(options, pkgs[0])

// by default, show an install hint after upgrading
// this will be disabled in interactive mode if the user chooses to have npm-check-updates execute the install command
const installHint = `${chalk.cyan(packageManager + ' install')}${
pkgs.length > 1 ? ' in each project directory' : ''
} to install new versions`

// prompt the user if they want ncu to run "npm install"
if (options.interactive) {
console.info('')
const response = await prompts({
type: 'confirm',
name: 'value',
message: `${installHint}?`,
initial: true,
// allow Ctrl+C to kill the process
onState: (state: any) => {
if (state.aborted) {
process.nextTick(() => process.exit(1))
}
},
})

// autoinstall
if (response.value) {
showInstallHint = false
pkgs.forEach(async pkgFile => {
const packageManager = await getPackageManagerForInstall(options, pkgFile)
const cmd = packageManager + (process.platform === 'win32' ? '.cmd' : '')
const cwd = options.cwd || path.resolve(pkgFile, '..')
const stdout = await spawn(cmd, ['install'], { cwd })
print(options, stdout, 'verbose')
})
}
}

// show the install hint unless autoinstall occurred
if (showInstallHint) {
print(options, `\nRun ${installHint}.`)
}
}

/** Main entry point.
*
* @returns Promise<
@@ -113,9 +192,11 @@ export async function run(
ignore: ['**/node_modules/**'],
},
)

// enable --deep if multiple package files are found
options.deep = options.deep || pkgs.length > 1

let analysis: Index<string> | PackageFile | void
let analysis: Index<PackageFile> | PackageFile | void
if (options.global) {
const analysis = await runGlobal(options)
clearTimeout(timeout)
@@ -146,7 +227,7 @@ export async function run(
.replace(/\\/g, '/')
: pkgFile!]: await runLocal(pkgOptions, pkgData, pkgFile),
}
}, Promise.resolve({} as Index<string> | PackageFile))
}, Promise.resolve({} as Index<PackageFile> | PackageFile))
if (options.json) {
printJson(options, analysis)
}
@@ -159,6 +240,12 @@ export async function run(
analysis = await runLocal(options, pkgData, pkgFile)
}
clearTimeout(timeout)

// suggest install command or autoinstall
if (options.upgrade) {
npmInstallHint(pkgs, analysis, options)
}

return analysis
}

8 changes: 1 addition & 7 deletions src/lib/doctor.ts
Original file line number Diff line number Diff line change
@@ -197,13 +197,7 @@ const doctor = async (run: Run, options: Options) => {
console.log(` ${chalk.green('✓')} ${name} ${allDependencies[name]}${version}`)

// save upgraded package data so that passing versions can still be saved even when there is a failure
lastPkgFile = await upgradePackageData(
lastPkgFile,
{ [name]: allDependencies[name] },
{ [name]: version },
{},
{ ...options, interactive: false },
)
lastPkgFile = await upgradePackageData(lastPkgFile, { [name]: allDependencies[name] }, { [name]: version })

// save working lock file
lockFile = fs.readFileSync(lockFileName, 'utf-8')
Loading