From 6d88687403bf39402c0b87501e9bb96211bc2c8d Mon Sep 17 00:00:00 2001 From: Gar <gar+gh@danger.computer> Date: Tue, 28 Feb 2023 08:47:12 -0800 Subject: [PATCH 1/4] fix(cmd-list): alias only to real commands --- lib/utils/cmd-list.js | 4 ++-- tap-snapshots/test/lib/docs.js.test.cjs | 8 ++++---- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/lib/utils/cmd-list.js b/lib/utils/cmd-list.js index 03fe8ed07c930..68074fe9a4286 100644 --- a/lib/utils/cmd-list.js +++ b/lib/utils/cmd-list.js @@ -36,7 +36,7 @@ const aliases = { v: 'view', run: 'run-script', 'clean-install': 'ci', - 'clean-install-test': 'cit', + 'clean-install-test': 'install-ci-test', x: 'exec', why: 'explain', la: 'll', @@ -62,7 +62,7 @@ const aliases = { upgrade: 'update', udpate: 'update', rum: 'run-script', - sit: 'cit', + sit: 'install-ci-test', urn: 'run-script', ogr: 'org', 'add-user': 'adduser', diff --git a/tap-snapshots/test/lib/docs.js.test.cjs b/tap-snapshots/test/lib/docs.js.test.cjs index a07aab8dd9757..fe9830b1fba8c 100644 --- a/tap-snapshots/test/lib/docs.js.test.cjs +++ b/tap-snapshots/test/lib/docs.js.test.cjs @@ -384,7 +384,7 @@ Object { "c": "config", "cit": "install-ci-test", "clean-install": "ci", - "clean-install-test": "cit", + "clean-install-test": "install-ci-test", "create": "init", "ddp": "dedupe", "dist-tags": "dist-tag", @@ -421,7 +421,7 @@ Object { "s": "search", "se": "search", "show": "view", - "sit": "cit", + "sit": "install-ci-test", "t": "test", "tst": "test", "udpate": "update", @@ -3239,14 +3239,14 @@ Options: [-w|--workspace <workspace-name> [-w|--workspace <workspace-name> ...]] [-ws|--workspaces] [--include-workspace-root] [--install-links] -alias: cit +aliases: cit, clean-install-test, sit Run "npm help install-ci-test" for more info \`\`\`bash npm install-ci-test -alias: cit +aliases: cit, clean-install-test, sit \`\`\` #### \`save\` From fc1673eac1ac4380726bcab8604c96c89e881327 Mon Sep 17 00:00:00 2001 From: Gar <gar+gh@danger.computer> Date: Tue, 28 Feb 2023 08:48:33 -0800 Subject: [PATCH 2/4] fix(access): only complete once --- lib/commands/access.js | 30 ++++++++++++++++-------------- test/lib/commands/access.js | 1 + 2 files changed, 17 insertions(+), 14 deletions(-) diff --git a/lib/commands/access.js b/lib/commands/access.js index 23e51f071b112..c8ac83529e58d 100644 --- a/lib/commands/access.js +++ b/lib/commands/access.js @@ -53,20 +53,22 @@ class Access extends BaseCommand { return commands } - switch (argv[2]) { - case 'grant': - return ['read-only', 'read-write'] - case 'revoke': - return [] - case 'list': - case 'ls': - return ['packages', 'collaborators'] - case 'get': - return ['status'] - case 'set': - return setCommands - default: - throw new Error(argv[2] + ' not recognized') + if (argv.length === 3) { + switch (argv[2]) { + case 'grant': + return ['read-only', 'read-write'] + case 'revoke': + return [] + case 'list': + case 'ls': + return ['packages', 'collaborators'] + case 'get': + return ['status'] + case 'set': + return setCommands + default: + throw new Error(argv[2] + ' not recognized') + } } } diff --git a/test/lib/commands/access.js b/test/lib/commands/access.js index b0057545ba026..f0117098a5b55 100644 --- a/test/lib/commands/access.js +++ b/test/lib/commands/access.js @@ -30,6 +30,7 @@ t.test('completion', async t => { ]) testComp(['npm', 'access', 'grant'], ['read-only', 'read-write']) testComp(['npm', 'access', 'revoke'], []) + testComp(['npm', 'access', 'grant', ''], []) await t.rejects( access.completion({ conf: { argv: { remain: ['npm', 'access', 'foobar'] } } }), From 95d28272afe0278fb4e83207df2f18afad353eac Mon Sep 17 00:00:00 2001 From: Gar <gar+gh@danger.computer> Date: Tue, 28 Feb 2023 08:48:51 -0800 Subject: [PATCH 3/4] fix(audit): add signatures to completion --- lib/commands/audit.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/lib/commands/audit.js b/lib/commands/audit.js index 05830fff69c2e..dab871669eba5 100644 --- a/lib/commands/audit.js +++ b/lib/commands/audit.js @@ -389,11 +389,12 @@ class Audit extends ArboristWorkspaceCmd { const argv = opts.conf.argv.remain if (argv.length === 2) { - return ['fix'] + return ['fix', 'signatures'] } switch (argv[2]) { case 'fix': + case 'signatures': return [] default: throw Object.assign(new Error(argv[2] + ' not recognized'), { From 4501ad1f51d9d6c070b9cda7367ee4054d940079 Mon Sep 17 00:00:00 2001 From: Gar <gar+gh@danger.computer> Date: Tue, 28 Feb 2023 08:49:20 -0800 Subject: [PATCH 4/4] feat: add preliminary fish shell completion --- lib/commands/completion.js | 7 ++--- lib/commands/run-script.js | 3 ++ lib/utils/completion.fish | 40 ++++++++++++++++++++++++ scripts/fish-completion.js | 54 +++++++++++++++++++++++++++++++++ test/lib/commands/run-script.js | 11 +++++-- 5 files changed, 109 insertions(+), 6 deletions(-) create mode 100644 lib/utils/completion.fish create mode 100644 scripts/fish-completion.js diff --git a/lib/commands/completion.js b/lib/commands/completion.js index f5604e099f9a2..49a66627cca2c 100644 --- a/lib/commands/completion.js +++ b/lib/commands/completion.js @@ -79,12 +79,10 @@ class Completion extends BaseCommand { }) } - const { COMP_CWORD, COMP_LINE, COMP_POINT } = process.env + const { COMP_CWORD, COMP_LINE, COMP_POINT, COMP_FISH } = process.env // if the COMP_* isn't in the env, then just dump the script. - if (COMP_CWORD === undefined || - COMP_LINE === undefined || - COMP_POINT === undefined) { + if (COMP_CWORD === undefined || COMP_LINE === undefined || COMP_POINT === undefined) { return dumpScript(resolve(this.npm.npmRoot, 'lib', 'utils', 'completion.sh')) } @@ -111,6 +109,7 @@ class Completion extends BaseCommand { partialWords.push(partialWord) const opts = { + isFish: COMP_FISH === 'true', words, w, word, diff --git a/lib/commands/run-script.js b/lib/commands/run-script.js index 51746c5e5285d..40e18e1ea0644 100644 --- a/lib/commands/run-script.js +++ b/lib/commands/run-script.js @@ -51,6 +51,9 @@ class RunScript extends BaseCommand { // find the script name const json = resolve(this.npm.localPrefix, 'package.json') const { scripts = {} } = await rpj(json).catch(er => ({})) + if (opts.isFish) { + return Object.keys(scripts).map(s => `${s}\t${scripts[s].slice(0, 30)}`) + } return Object.keys(scripts) } } diff --git a/lib/utils/completion.fish b/lib/utils/completion.fish new file mode 100644 index 0000000000000..5e274ad77e5fd --- /dev/null +++ b/lib/utils/completion.fish @@ -0,0 +1,40 @@ +# npm completions for Fish shell +# This script is a work in progress and does not fall under the normal semver contract as the rest of npm. + +# __fish_npm_needs_command taken from: +# https://stackoverflow.com/questions/16657803/creating-autocomplete-script-with-sub-commands +function __fish_npm_needs_command + set -l cmd (commandline -opc) + + if test (count $cmd) -eq 1 + return 0 + end + + return 1 +end + +# Taken from https://github.com/fish-shell/fish-shell/blob/HEAD/share/completions/npm.fish +function __fish_complete_npm -d "Complete the commandline using npm's 'completion' tool" + # tell npm we are fish shell + set -lx COMP_FISH true + if command -sq npm + # npm completion is bash-centric, so we need to translate fish's "commandline" stuff to bash's $COMP_* stuff + # COMP_LINE is an array with the words in the commandline + set -lx COMP_LINE (commandline -opc) + # COMP_CWORD is the index of the current word in COMP_LINE + # bash starts arrays with 0, so subtract 1 + set -lx COMP_CWORD (math (count $COMP_LINE) - 1) + # COMP_POINT is the index of point/cursor when the commandline is viewed as a string + set -lx COMP_POINT (commandline -C) + # If the cursor is after the last word, the empty token will disappear in the expansion + # Readd it + if test (commandline -ct) = "" + set COMP_CWORD (math $COMP_CWORD + 1) + set COMP_LINE $COMP_LINE "" + end + command npm completion -- $COMP_LINE 2>/dev/null + end +end + +# flush out what ships with fish +complete -e npm diff --git a/scripts/fish-completion.js b/scripts/fish-completion.js new file mode 100644 index 0000000000000..6357c1032fe56 --- /dev/null +++ b/scripts/fish-completion.js @@ -0,0 +1,54 @@ +/* eslint-disable no-console */ +const fs = require('fs/promises') +const { resolve } = require('path') + +const { commands, aliases } = require('../lib/utils/cmd-list.js') +const { definitions } = require('../lib/utils/config/index.js') + +async function main () { + const file = resolve(__dirname, '..', 'lib', 'utils', 'completion.fish') + console.log(await fs.readFile(file, 'utf-8')) + const cmds = {} + for (const cmd of commands) { + cmds[cmd] = { aliases: [cmd] } + const cmdClass = require(`../lib/commands/${cmd}.js`) + cmds[cmd].description = cmdClass.description + cmds[cmd].params = cmdClass.params + } + for (const alias in aliases) { + cmds[aliases[alias]].aliases.push(alias) + } + for (const cmd in cmds) { + console.log(`# ${cmd}`) + const { aliases: cmdAliases, description, params = [] } = cmds[cmd] + // If npm completion could return all commands in a fish friendly manner + // like we do w/ run-script these wouldn't be needed. + /* eslint-disable-next-line max-len */ + console.log(`complete -x -c npm -n __fish_npm_needs_command -a '${cmdAliases.join(' ')}' -d '${description}'`) + const shorts = params.map(p => { + // Our multi-character short params (e.g. -ws) are not very standard and + // don't work with things that assume short params are only ever single + // characters. + if (definitions[p].short?.length === 1) { + return `-s ${definitions[p].short}` + } + }).filter(p => p).join(' ') + // The config descriptions are not appropriate for -d here. We may want to + // consider having a more terse description for these. + // We can also have a mechanism to auto-generate the long form of options + // that have predefined values. + // params completion + /* eslint-disable-next-line max-len */ + console.log(`complete -x -c npm -n '__fish_seen_subcommand_from ${cmdAliases.join(' ')}' ${params.map(p => `-l ${p}`).join(' ')} ${shorts}`) + // builtin npm completion + /* eslint-disable-next-line max-len */ + console.log(`complete -x -c npm -n '__fish_seen_subcommand_from ${cmdAliases.join(' ')}' -a '(__fish_complete_npm)'`) + } +} + +main().then(() => { + return process.exit() +}).catch(err => { + console.error(err) + process.exit(1) +}) diff --git a/test/lib/commands/run-script.js b/test/lib/commands/run-script.js index a265db3cc040d..6e2bf22adddcf 100644 --- a/test/lib/commands/run-script.js +++ b/test/lib/commands/run-script.js @@ -34,12 +34,12 @@ const mockRs = async (t, { windows = false, runScript, ...opts } = {}) => { } t.test('completion', async t => { - const completion = async (t, remain, pkg) => { + const completion = async (t, remain, pkg, isFish = false) => { const { npm } = await mockRs(t, pkg ? { prefixDir: { 'package.json': JSON.stringify(pkg) } } : {} ) const cmd = await npm.cmd('run-script') - return cmd.completion({ conf: { argv: { remain } } }) + return cmd.completion({ conf: { argv: { remain } }, isFish }) } t.test('already have a script name', async t => { @@ -60,6 +60,13 @@ t.test('completion', async t => { }) t.strictSame(res, ['hello', 'world']) }) + + t.test('fish shell', async t => { + const res = await completion(t, ['npm', 'run'], { + scripts: { hello: 'echo hello', world: 'echo world' }, + }, true) + t.strictSame(res, ['hello\techo hello', 'world\techo world']) + }) }) t.test('fail if no package.json', async t => {