UNPKG

npm

Version:

a package manager for JavaScript

330 lines (281 loc) 11.1 kB
// Each command has a completion function that takes an options object and a cb The callback gets called with an error and an array of possible completions. // The options object is built up based on the environment variables set by zsh or bash when calling a function for completion, based on the cursor position and the command line thus far. // These are: // COMP_CWORD: the index of the "word" in the command line being completed // COMP_LINE: the full command line thus far as a string // COMP_POINT: the cursor index at the point of triggering completion // // We parse the command line with nopt, like npm does, and then create an options object containing: // words: array of words in the command line // w: the index of the word being completed (ie, COMP_CWORD) // word: the word being completed // line: the COMP_LINE // lineLength // point: the COMP_POINT, usually equal to line length, but not always, eg if the user has pressed the left-arrow to complete an earlier word // partialLine: the line up to the point // partialWord: the word being completed (which might be ''), up to the point // conf: a nopt parse of the command line // // When the implementation completion method returns its list of strings, and arrays of strings, we filter that by any that start with the partialWord, since only those can possibly be valid matches. // // Matches are wrapped with ' to escape them, if necessary, and then printed one per line for the shell completion method to consume in IFS=$'\n' mode as an array. const fs = require('node:fs/promises') const nopt = require('nopt') const { resolve } = require('node:path') const { output } = require('proc-log') const Npm = require('../npm.js') const { definitions, shorthands } = require('@npmcli/config/lib/definitions') const { commands, aliases, deref } = require('../utils/cmd-list.js') const { isWindowsShell } = require('../utils/is-windows.js') const BaseCommand = require('../base-cmd.js') const fileExists = (file) => fs.stat(file).then(s => s.isFile()).catch(() => false) class Completion extends BaseCommand { static description = 'Tab Completion for npm' static name = 'completion' // Completion command uses args differently - they represent the command line being completed, not actual arguments to this command, so we use an empty definitions object to prevent flag validation static definitions = [] // completion for the completion command static async completion (opts) { if (opts.w > 2) { return } const [bashExists, zshExists] = await Promise.all([ fileExists(resolve(process.env.HOME, '.bashrc')), fileExists(resolve(process.env.HOME, '.zshrc')), ]) const out = [] if (zshExists) { out.push(['>>', '~/.zshrc']) } if (bashExists) { out.push(['>>', '~/.bashrc']) } return out } async exec (args) { if (isWindowsShell) { const msg = 'npm completion supported only in MINGW / Git bash on Windows' throw Object.assign(new Error(msg), { code: 'ENOTSUP', }) } 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) { return dumpScript(resolve(this.npm.npmRoot, 'lib', 'utils', 'completion.sh')) } // ok we're actually looking at the envs and outputting the suggestions get the partial line and partial word, if the point isn't at the end. // ie, tabbing at: npm foo b|ar const w = +COMP_CWORD const line = COMP_LINE // Use COMP_LINE to get words if args doesn't include flags (e.g., in tests) const hasFlags = line.includes(' -') && !args.some(arg => arg.startsWith('-')) const words = (hasFlags ? line.split(/\s+/) : args).map(unescape) const word = words[w] || '' const point = +COMP_POINT const partialLine = line.slice(0, point) const partialWords = words.slice(0, w) // figure out where in that last word the point is. const partialWordRaw = args[w] || '' let i = partialWordRaw.length while (partialWordRaw.slice(0, i) !== partialLine.slice(-1 * i) && i > 0) { i-- } const partialWord = unescape(partialWordRaw.slice(0, i)) partialWords.push(partialWord) const opts = { isFish: COMP_FISH === 'true', words, w, word, line, lineLength: line.length, point, partialLine, partialWords, partialWord, raw: args, } // try to find the npm command and subcommand early for flag completion this helps with custom command definitions from subcommands const types = Object.entries(definitions).reduce((acc, [key, def]) => { acc[key] = def.type return acc }, {}) const parsed = opts.conf = nopt(types, shorthands, partialWords.slice(0, -1), 0) const cmd = parsed.argv.remain[1] const subCmd = parsed.argv.remain[2] if (partialWords.slice(0, -1).indexOf('--') === -1) { if (word && word.charAt(0) === '-') { return this.wrap(opts, configCompl(opts, cmd, subCmd, this.npm)) } if (words[w - 1] && words[w - 1].charAt(0) === '-' && !isFlag(words[w - 1], cmd, subCmd, this.npm)) { // awaiting a value for a non-bool config. // don't even try to do this for now return this.wrap(opts, configValueCompl(opts)) } } // check if there's a command already. if (!cmd) { return this.wrap(opts, cmdCompl(opts, this.npm)) } Object.keys(parsed).forEach(k => this.npm.config.set(k, parsed[k])) // at this point, if words[1] is some kind of npm command, then complete on it. // otherwise, do nothing try { const { completion } = Npm.cmd(cmd) if (completion) { const comps = await completion(opts, this.npm) return this.wrap(opts, comps) } } catch { // it wasn't a valid command, so do nothing } } // The command should respond with an array. // Loop over that, wrapping quotes around any that have spaces, and writing them to stdout. // If any of the items are arrays, then join them with a space. // e.g. returning ['a', 'b c', ['d', 'e']] would allow it to expand to: 'a', 'b c', or 'd' 'e' wrap (opts, compls) { if (opts.partialWord) { compls = compls.filter(c => c.startsWith(opts.partialWord)) } if (compls.length > 0) { output.standard(compls.join('\n')) } } } const dumpScript = async (p) => { const d = (await fs.readFile(p, 'utf8')).replace(/^#!.*?\n/, '') await new Promise((res, rej) => { let done = false process.stdout.on('error', er => { if (done) { return } done = true // Darwin is a pain sometimes. // // This is necessary because the "source" or "." program in bash on OS X closes its file argument before reading from it, meaning that you get exactly 1 write, which will work most of the time, and will always raise an EPIPE. // // Really, one should not be tossing away EPIPE errors, or any errors, so casually. // But, without this, `. <(npm completion)` can never ever work on OS X. // TODO Ignoring coverage, see 'non EPIPE errors cause failures' test. /* istanbul ignore next */ if (er.errno === 'EPIPE') { res() } else { rej(er) } }) process.stdout.write(d, () => { if (done) { return } done = true res() }) }) } const unescape = w => w.charAt(0) === '\'' ? w.replace(/^'|'$/g, '') : w.replace(/\\ /g, ' ') // Helper to get custom definitions from a command/subcommand const getCustomDefinitions = (cmd, subCmd) => { if (!cmd) { return [] } try { const command = Npm.cmd(cmd) // Check if the command has subcommands if (subCmd && command.subcommands && command.subcommands[subCmd]) { const subcommand = command.subcommands[subCmd] // All subcommands have definitions return subcommand.definitions } // Check if the command itself has definitions if (command.definitions) { return command.definitions } } catch { // Command not found or no definitions } return [] } // Helper to get all config names including aliases from custom definitions const getCustomConfigNames = (customDefs) => { const names = new Set() for (const def of customDefs) { names.add(def.key) if (def.alias && Array.isArray(def.alias)) { def.alias.forEach(a => names.add(a)) } } return [...names] } // the current word has a dash. // Return the config names with the same number of dashes as the current word has. const configCompl = (opts, cmd, subCmd, npm) => { const word = opts.word const split = word.match(/^(-+)((?:no-)*)(.*)$/) const dashes = split[1] const no = split[2] // Get custom definitions from the command/subcommand const customDefs = getCustomDefinitions(cmd, subCmd, npm) const customNames = getCustomConfigNames(customDefs) // If there are custom definitions, return only those (new feature) // Otherwise, return empty array (historical behavior - no global flag completion) if (customNames.length > 0) { const flags = customNames.filter(name => isFlag(name, cmd, subCmd, npm)) return customNames.map(c => dashes + c) .concat(flags.map(f => dashes + (no || 'no-') + f)) } return [] } // expand with the valid values of various config values. // not yet implemented. const configValueCompl = () => [] // check if the thing is a flag or not. const isFlag = (word, cmd, subCmd, npm) => { // shorthands never take args. const split = word.match(/^(-*)((?:no-)+)?(.*)$/) const no = split[2] const conf = split[3] // Check custom definitions first const customDefs = getCustomDefinitions(cmd, subCmd, npm) // Check if conf is in custom definitions or is an alias let customDef = customDefs.find(d => d.key === conf) if (!customDef) { // Check if conf is an alias for any of the custom definitions for (const def of customDefs) { if (def.alias && Array.isArray(def.alias) && def.alias.includes(conf)) { customDef = def break } } } if (customDef) { const { type } = customDef return no || type === Boolean || (Array.isArray(type) && type.includes(Boolean)) } // No custom definitions found, should not reach here in normal flow since configCompl returns empty array when no custom defs exist return false } // complete against the npm commands // if they all resolve to the same thing, just return the thing it already is const cmdCompl = (opts) => { const allCommands = commands.concat(Object.keys(aliases)) const matches = allCommands.filter(c => c.startsWith(opts.partialWord)) if (!matches.length) { return matches } const derefs = new Set([...matches.map(c => deref(c))]) if (derefs.size === 1) { return [...derefs] } return allCommands } module.exports = Completion