From c34735aff0d90c813e8b09f9078360ff35675a19 Mon Sep 17 00:00:00 2001 From: Chris Thielen Date: Tue, 22 Dec 2020 12:59:17 -0800 Subject: [PATCH] feat(findSolutions): Add a toggle to find solutions and print installation commands. BREAKING CHANGE: no longer prints installation commands by default, instead prints a message about using --install --- src/checkPeerDependencies.ts | 108 +++++++++++++++++++++-------------- src/cli.ts | 6 ++ src/solution.ts | 1 + 3 files changed, 71 insertions(+), 44 deletions(-) diff --git a/src/checkPeerDependencies.ts b/src/checkPeerDependencies.ts index 8f18d02..6e3925f 100755 --- a/src/checkPeerDependencies.ts +++ b/src/checkPeerDependencies.ts @@ -51,24 +51,8 @@ const reportPeerDependencyStatusByDependee = (dep: Dependency, options: CliOptio } }; -export function checkPeerDependencies(packageManager: string, options: CliOptions) { - const allNestedPeerDependencies = getAllNestedPeerDependencies(options); - - if (options.orderBy === 'depender') { - allNestedPeerDependencies.sort((a, b) => `${a.depender}${a.name}`.localeCompare(`${b.depender}${b.name}`)); - allNestedPeerDependencies.forEach(dep => reportPeerDependencyStatusByDepender(dep, options)); - } else if (options.orderBy === 'dependee') { - allNestedPeerDependencies.sort((a, b) => `${a.name}${a.depender}`.localeCompare(`${b.name}${b.depender}`)); - allNestedPeerDependencies.forEach(dep => reportPeerDependencyStatusByDependee(dep, options)); - } - - const problems = allNestedPeerDependencies.filter(dep => !dep.semverSatisfies && !dep.isYalc); - - if (!problems.length) { - console.log(' ✅ All peer dependencies are met'); - return; - } +function findSolutions(problems: Dependency[], allNestedPeerDependencies: Dependency[]) { console.log(); console.log('Searching for solutions...'); console.log(); @@ -89,40 +73,76 @@ export function checkPeerDependencies(packageManager: string, options: CliOption console.error(); } - const commandLines = getCommandLines(packageManager, resolutionsWithSolutions); - if (options.install && commandLines.length > 0) { - console.log('Installing peerDependencies...'); + return { resolutionsWithSolutions, nosolution }; +} + +function installPeerDependencies(commandLines: any[], options: CliOptions, nosolution: Resolution[], packageManager: string) { + console.log('Installing peerDependencies...'); + console.log(); + commandLines.forEach(command => { + console.log(`$ ${command}`); + exec(command); console.log(); - commandLines.forEach(command => { - console.log(`$ ${command}`); - exec(command); - console.log(); - }); + }); - const newUnsatisfiedDeps = getAllNestedPeerDependencies(options) - .filter(dep => !dep.semverSatisfies) - .filter(dep => !nosolution.some(x => isSameDep(x.problem, dep))); + const newUnsatisfiedDeps = getAllNestedPeerDependencies(options) + .filter(dep => !dep.semverSatisfies) + .filter(dep => !nosolution.some(x => isSameDep(x.problem, dep))); - if (nosolution.length === 0 && newUnsatisfiedDeps.length === 0) { - console.log('All peer dependencies are met'); - } + if (nosolution.length === 0 && newUnsatisfiedDeps.length === 0) { + console.log('All peer dependencies are met'); + } - if (newUnsatisfiedDeps.length > 0) { - console.log(`Found ${newUnsatisfiedDeps.length} new unmet peerDependencies...`); - if (++recursiveCount < 5) { - return checkPeerDependencies(packageManager, options); - } else { - console.error('Recursion limit reached (5)'); - process.exit(5) - } + if (newUnsatisfiedDeps.length > 0) { + console.log(`Found ${newUnsatisfiedDeps.length} new unmet peerDependencies...`); + if (++recursiveCount < 5) { + return checkPeerDependencies(packageManager, options); + } else { + console.error('Recursion limit reached (5)'); + process.exit(5) } + } + return; +} + +export function checkPeerDependencies(packageManager: string, options: CliOptions) { + const allNestedPeerDependencies = getAllNestedPeerDependencies(options); + + if (options.orderBy === 'depender') { + allNestedPeerDependencies.sort((a, b) => `${a.depender}${a.name}`.localeCompare(`${b.depender}${b.name}`)); + allNestedPeerDependencies.forEach(dep => reportPeerDependencyStatusByDepender(dep, options)); + } else if (options.orderBy === 'dependee') { + allNestedPeerDependencies.sort((a, b) => `${a.name}${a.depender}`.localeCompare(`${b.name}${b.depender}`)); + allNestedPeerDependencies.forEach(dep => reportPeerDependencyStatusByDependee(dep, options)); + } + + const problems = allNestedPeerDependencies.filter(dep => !dep.semverSatisfies && !dep.isYalc); + + if (!problems.length) { + console.log(' ✅ All peer dependencies are met'); return; + } - } else if (commandLines.length > 0) { - console.log(`Install peerDependencies using ${commandLines.length > 1 ? 'these commands:' : 'this command'}:`); - console.log(); - commandLines.forEach(command => console.log(command)); - console.log(); + if (options.install) { + const { nosolution, resolutionsWithSolutions } = findSolutions(problems, allNestedPeerDependencies); + const commandLines = getCommandLines(packageManager, resolutionsWithSolutions); + + if (commandLines.length) { + return installPeerDependencies(commandLines, options, nosolution, packageManager); + } + } else if (options.findSolutions) { + const { resolutionsWithSolutions } = findSolutions(problems, allNestedPeerDependencies); + const commandLines = getCommandLines(packageManager, resolutionsWithSolutions); + + if (commandLines.length) { + console.log(`Install peerDependencies using ${commandLines.length > 1 ? 'these commands:' : 'this command'}:`); + console.log(); + commandLines.forEach(command => console.log(command)); + console.log(); + } + } else { + console.log(`Install peerDependencies using "npx check-peer-dependencies --install"`); } + process.exit(1); } diff --git a/src/cli.ts b/src/cli.ts index 51c1b71..1418e88 100644 --- a/src/cli.ts +++ b/src/cli.ts @@ -38,6 +38,11 @@ const options = yarrrrgs default: false, description: 'Run tool only on package root dependencies', }) + .option('findSolutions', { + boolean: true, + default: false, + description: 'Search for solutions and print package installation commands', + }) .option('install', { boolean: true, default: false, @@ -58,6 +63,7 @@ export interface CliOptions { npm: boolean; runOnlyOnRootDependencies: boolean; orderBy: 'depender' | 'dependee'; + findSolutions: boolean; install: boolean; } diff --git a/src/solution.ts b/src/solution.ts index 38a1615..ec51557 100644 --- a/src/solution.ts +++ b/src/solution.ts @@ -33,6 +33,7 @@ export function findPossibleResolutions(problems: Dependency[], allPeerDependenc function findPossibleResolution(packageName, allPeerDeps) { const requiredPeerVersions = allPeerDeps.filter(dep => dep.name === packageName); + // todo: skip this step if only one required peer version and it's an exact version const command = `npm view ${packageName} versions`; let rawVersionsInfo; try {