Skip to content

Commit 84fbaf2

Browse files
wraithgarfritzy
authored andcommittedMar 2, 2023
feat: add preliminary fish shell completion
1 parent c4c8754 commit 84fbaf2

File tree

5 files changed

+109
-6
lines changed

5 files changed

+109
-6
lines changed
 

‎lib/commands/completion.js

+3-4
Original file line numberDiff line numberDiff line change
@@ -79,12 +79,10 @@ class Completion extends BaseCommand {
7979
})
8080
}
8181

82-
const { COMP_CWORD, COMP_LINE, COMP_POINT } = process.env
82+
const { COMP_CWORD, COMP_LINE, COMP_POINT, COMP_FISH } = process.env
8383

8484
// if the COMP_* isn't in the env, then just dump the script.
85-
if (COMP_CWORD === undefined ||
86-
COMP_LINE === undefined ||
87-
COMP_POINT === undefined) {
85+
if (COMP_CWORD === undefined || COMP_LINE === undefined || COMP_POINT === undefined) {
8886
return dumpScript(resolve(this.npm.npmRoot, 'lib', 'utils', 'completion.sh'))
8987
}
9088

@@ -111,6 +109,7 @@ class Completion extends BaseCommand {
111109
partialWords.push(partialWord)
112110

113111
const opts = {
112+
isFish: COMP_FISH === 'true',
114113
words,
115114
w,
116115
word,

‎lib/commands/run-script.js

+3
Original file line numberDiff line numberDiff line change
@@ -51,6 +51,9 @@ class RunScript extends BaseCommand {
5151
// find the script name
5252
const json = resolve(this.npm.localPrefix, 'package.json')
5353
const { scripts = {} } = await rpj(json).catch(er => ({}))
54+
if (opts.isFish) {
55+
return Object.keys(scripts).map(s => `${s}\t${scripts[s].slice(0, 30)}`)
56+
}
5457
return Object.keys(scripts)
5558
}
5659
}

‎lib/utils/completion.fish

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
# npm completions for Fish shell
2+
# This script is a work in progress and does not fall under the normal semver contract as the rest of npm.
3+
4+
# __fish_npm_needs_command taken from:
5+
# https://stackoverflow.com/questions/16657803/creating-autocomplete-script-with-sub-commands
6+
function __fish_npm_needs_command
7+
set -l cmd (commandline -opc)
8+
9+
if test (count $cmd) -eq 1
10+
return 0
11+
end
12+
13+
return 1
14+
end
15+
16+
# Taken from https://github.com/fish-shell/fish-shell/blob/HEAD/share/completions/npm.fish
17+
function __fish_complete_npm -d "Complete the commandline using npm's 'completion' tool"
18+
# tell npm we are fish shell
19+
set -lx COMP_FISH true
20+
if command -sq npm
21+
# npm completion is bash-centric, so we need to translate fish's "commandline" stuff to bash's $COMP_* stuff
22+
# COMP_LINE is an array with the words in the commandline
23+
set -lx COMP_LINE (commandline -opc)
24+
# COMP_CWORD is the index of the current word in COMP_LINE
25+
# bash starts arrays with 0, so subtract 1
26+
set -lx COMP_CWORD (math (count $COMP_LINE) - 1)
27+
# COMP_POINT is the index of point/cursor when the commandline is viewed as a string
28+
set -lx COMP_POINT (commandline -C)
29+
# If the cursor is after the last word, the empty token will disappear in the expansion
30+
# Readd it
31+
if test (commandline -ct) = ""
32+
set COMP_CWORD (math $COMP_CWORD + 1)
33+
set COMP_LINE $COMP_LINE ""
34+
end
35+
command npm completion -- $COMP_LINE 2>/dev/null
36+
end
37+
end
38+
39+
# flush out what ships with fish
40+
complete -e npm

‎scripts/fish-completion.js

+54
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
/* eslint-disable no-console */
2+
const fs = require('fs/promises')
3+
const { resolve } = require('path')
4+
5+
const { commands, aliases } = require('../lib/utils/cmd-list.js')
6+
const { definitions } = require('../lib/utils/config/index.js')
7+
8+
async function main () {
9+
const file = resolve(__dirname, '..', 'lib', 'utils', 'completion.fish')
10+
console.log(await fs.readFile(file, 'utf-8'))
11+
const cmds = {}
12+
for (const cmd of commands) {
13+
cmds[cmd] = { aliases: [cmd] }
14+
const cmdClass = require(`../lib/commands/${cmd}.js`)
15+
cmds[cmd].description = cmdClass.description
16+
cmds[cmd].params = cmdClass.params
17+
}
18+
for (const alias in aliases) {
19+
cmds[aliases[alias]].aliases.push(alias)
20+
}
21+
for (const cmd in cmds) {
22+
console.log(`# ${cmd}`)
23+
const { aliases: cmdAliases, description, params = [] } = cmds[cmd]
24+
// If npm completion could return all commands in a fish friendly manner
25+
// like we do w/ run-script these wouldn't be needed.
26+
/* eslint-disable-next-line max-len */
27+
console.log(`complete -x -c npm -n __fish_npm_needs_command -a '${cmdAliases.join(' ')}' -d '${description}'`)
28+
const shorts = params.map(p => {
29+
// Our multi-character short params (e.g. -ws) are not very standard and
30+
// don't work with things that assume short params are only ever single
31+
// characters.
32+
if (definitions[p].short?.length === 1) {
33+
return `-s ${definitions[p].short}`
34+
}
35+
}).filter(p => p).join(' ')
36+
// The config descriptions are not appropriate for -d here. We may want to
37+
// consider having a more terse description for these.
38+
// We can also have a mechanism to auto-generate the long form of options
39+
// that have predefined values.
40+
// params completion
41+
/* eslint-disable-next-line max-len */
42+
console.log(`complete -x -c npm -n '__fish_seen_subcommand_from ${cmdAliases.join(' ')}' ${params.map(p => `-l ${p}`).join(' ')} ${shorts}`)
43+
// builtin npm completion
44+
/* eslint-disable-next-line max-len */
45+
console.log(`complete -x -c npm -n '__fish_seen_subcommand_from ${cmdAliases.join(' ')}' -a '(__fish_complete_npm)'`)
46+
}
47+
}
48+
49+
main().then(() => {
50+
return process.exit()
51+
}).catch(err => {
52+
console.error(err)
53+
process.exit(1)
54+
})

‎test/lib/commands/run-script.js

+9-2
Original file line numberDiff line numberDiff line change
@@ -34,12 +34,12 @@ const mockRs = async (t, { windows = false, runScript, ...opts } = {}) => {
3434
}
3535

3636
t.test('completion', async t => {
37-
const completion = async (t, remain, pkg) => {
37+
const completion = async (t, remain, pkg, isFish = false) => {
3838
const { npm } = await mockRs(t,
3939
pkg ? { prefixDir: { 'package.json': JSON.stringify(pkg) } } : {}
4040
)
4141
const cmd = await npm.cmd('run-script')
42-
return cmd.completion({ conf: { argv: { remain } } })
42+
return cmd.completion({ conf: { argv: { remain } }, isFish })
4343
}
4444

4545
t.test('already have a script name', async t => {
@@ -60,6 +60,13 @@ t.test('completion', async t => {
6060
})
6161
t.strictSame(res, ['hello', 'world'])
6262
})
63+
64+
t.test('fish shell', async t => {
65+
const res = await completion(t, ['npm', 'run'], {
66+
scripts: { hello: 'echo hello', world: 'echo world' },
67+
}, true)
68+
t.strictSame(res, ['hello\techo hello', 'world\techo world'])
69+
})
6370
})
6471

6572
t.test('fail if no package.json', async t => {

0 commit comments

Comments
 (0)
Please sign in to comment.