From 00c956394d44cd0270eda93fba4498ea2e4f9881 Mon Sep 17 00:00:00 2001 From: Scott Lowe Date: Thu, 10 Sep 2020 14:44:32 +0200 Subject: [PATCH 1/3] Improve command suggestions, add topic help Change-type: minor Resolves: #2021 Signed-off-by: Scott Lowe --- lib/help.ts | 21 ++++++++++++++++ lib/hooks/command-not-found/suggest.ts | 34 +++++++++++++++++++++----- patches/all/@oclif+config+1.17.0.patch | 15 ++++++++++++ 3 files changed, 64 insertions(+), 6 deletions(-) create mode 100644 patches/all/@oclif+config+1.17.0.patch diff --git a/lib/help.ts b/lib/help.ts index cad3bd63..b2f67200 100644 --- a/lib/help.ts +++ b/lib/help.ts @@ -40,6 +40,27 @@ export default class BalenaHelp extends Help { return; } + // If they've typed a topic (e.g. `balena os`) that isn't also a command (e.g. `balena device`) + // then list the associated commands. + const topicCommands = this.config.commands.filter((c) => { + return c.id.startsWith(`${subject}:`); + }); + if (topicCommands.length > 0) { + console.log(`${chalk.yellow(subject)} commands include:`); + console.log(this.formatCommands(topicCommands)); + console.log( + `\nRun ${chalk.cyan.bold( + 'balena help -v', + )} for a list of all available commands,`, + ); + console.log( + ` or ${chalk.cyan.bold( + 'balena help ', + )} for detailed help on a specific command.`, + ); + return; + } + throw new ExpectedError(`command ${chalk.cyan.bold(subject)} not found`); } diff --git a/lib/hooks/command-not-found/suggest.ts b/lib/hooks/command-not-found/suggest.ts index 292f6032..b1c3a735 100644 --- a/lib/hooks/command-not-found/suggest.ts +++ b/lib/hooks/command-not-found/suggest.ts @@ -22,10 +22,12 @@ import type { IConfig } from '@oclif/config'; A modified version of the command-not-found plugin logic, which deals with spaces separators stead of colons, and prints suggested commands instead of prompting interactively. + + Also see help.ts showHelp() for handling of topics. */ const hook: Hook<'command-not-found'> = async function ( - opts: object & { config: IConfig; id?: string }, + opts: object & { config: IConfig; id?: string; argv?: string[] }, ) { const Levenshtein = await import('fast-levenshtein'); const _ = await import('lodash'); @@ -44,16 +46,36 @@ const hook: Hook<'command-not-found'> = async function ( return _.minBy(commandIDs, (c) => Levenshtein.get(cmd, c))!; } + const suggestions: string[] = []; + suggestions.push(closest(commandId).replace(':', ' ') || ''); + + // opts.argv contains everything after the first command word + // if there's something there, also test if it might be a double + // word command spelt wrongly, rather than command args. + if (opts.argv?.[0]) { + suggestions.unshift( + closest(`${commandId}: + ${opts.argv[0]}`).replace(':', ' ') || '', + ); + } + + // Output suggestions console.error( `${color.yellow(command)} is not a recognized balena command.\n`, ); - - const suggestion = closest(commandId).replace(':', ' ') || ''; - console.log(`Did you mean: ${color.cmd(suggestion)} ? `); - console.log( - `Run ${color.cmd('balena help -v')} for a list of available commands.`, + console.error(`Did you mean: ? `); + suggestions.forEach((s) => { + console.error(` ${color.cmd(s)}`); + }); + console.error( + `\nRun ${color.cmd('balena help -v')} for a list of available commands,`, + ); + console.error( + ` or ${color.cmd( + 'balena help ', + )} for detailed help on a specific command.`, ); + // Exit const COMMAND_NOT_FOUND = 127; process.exit(COMMAND_NOT_FOUND); }; diff --git a/patches/all/@oclif+config+1.17.0.patch b/patches/all/@oclif+config+1.17.0.patch new file mode 100644 index 00000000..facffe54 --- /dev/null +++ b/patches/all/@oclif+config+1.17.0.patch @@ -0,0 +1,15 @@ +diff --git a/node_modules/@oclif/config/lib/config.js b/node_modules/@oclif/config/lib/config.js +index aba1da9..830f800 100644 +--- a/node_modules/@oclif/config/lib/config.js ++++ b/node_modules/@oclif/config/lib/config.js +@@ -165,7 +165,9 @@ class Config { + debug('runCommand %s %o', id, argv); + const c = this.findCommand(id); + if (!c) { +- await this.runHook('command_not_found', { id }); ++ // argv added to command_not_found hook ++ // We should try to upstream this change ++ await this.runHook('command_not_found', { id, argv }); + throw new errors_1.CLIError(`command ${id} not found`); + } + const command = c.load(); From 9805854eab2ae7bb6dbc0e545f9f06a9dc2be714 Mon Sep 17 00:00:00 2001 From: Scott Lowe Date: Thu, 10 Sep 2020 16:02:04 +0200 Subject: [PATCH 2/3] Update registry secrets example URL Change-type: patch Signed-off-by: Scott Lowe --- doc/cli.markdown | 6 +++--- lib/utils/messages.ts | 2 +- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/doc/cli.markdown b/doc/cli.markdown index 41631cb5..f8bac81a 100644 --- a/doc/cli.markdown +++ b/doc/cli.markdown @@ -2082,7 +2082,7 @@ Sample registry-secrets YAML file: password: '{escaped contents of the GCR keyfile.json file}' ``` For a sample project using registry secrets with the Google Container Registry, -check: https://github.com/balena-io-playground/sample-gcr-registry-secrets +check: https://github.com/balena-io-examples/sample-gcr-registry-secrets If the --registry-secrets option is not specified, and a secrets.yml or secrets.json file exists in the balena directory (usually $HOME/.balena), @@ -2358,7 +2358,7 @@ Sample registry-secrets YAML file: password: '{escaped contents of the GCR keyfile.json file}' ``` For a sample project using registry secrets with the Google Container Registry, -check: https://github.com/balena-io-playground/sample-gcr-registry-secrets +check: https://github.com/balena-io-examples/sample-gcr-registry-secrets If the --registry-secrets option is not specified, and a secrets.yml or secrets.json file exists in the balena directory (usually $HOME/.balena), @@ -2580,7 +2580,7 @@ Sample registry-secrets YAML file: password: '{escaped contents of the GCR keyfile.json file}' ``` For a sample project using registry secrets with the Google Container Registry, -check: https://github.com/balena-io-playground/sample-gcr-registry-secrets +check: https://github.com/balena-io-examples/sample-gcr-registry-secrets If the --registry-secrets option is not specified, and a secrets.yml or secrets.json file exists in the balena directory (usually $HOME/.balena), diff --git a/lib/utils/messages.ts b/lib/utils/messages.ts index e443c3f3..03096e82 100644 --- a/lib/utils/messages.ts +++ b/lib/utils/messages.ts @@ -61,7 +61,7 @@ Sample registry-secrets YAML file: password: '{escaped contents of the GCR keyfile.json file}' \`\`\` For a sample project using registry secrets with the Google Container Registry, -check: https://github.com/balena-io-playground/sample-gcr-registry-secrets +check: https://github.com/balena-io-examples/sample-gcr-registry-secrets If the --registry-secrets option is not specified, and a secrets.yml or secrets.json file exists in the balena directory (usually $HOME/.balena), From b180eb7b73a7d340d0cde534425d06f5f06b7396 Mon Sep 17 00:00:00 2001 From: Scott Lowe Date: Thu, 10 Sep 2020 16:06:07 +0200 Subject: [PATCH 3/3] Minor fix to device rm Change-type: patch Signed-off-by: Scott Lowe --- lib/actions-oclif/device/rm.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/actions-oclif/device/rm.ts b/lib/actions-oclif/device/rm.ts index 27f59ea1..9b29744a 100644 --- a/lib/actions-oclif/device/rm.ts +++ b/lib/actions-oclif/device/rm.ts @@ -82,7 +82,7 @@ export default class DeviceRmCmd extends Command { ); // Remove - for (const uuid of params.uuid.split(',')) { + for (const uuid of uuids) { try { await balena.models.device.remove(tryAsInteger(uuid)); } catch (err) {