2020-05-26 15:34:44 +02:00
|
|
|
import * as program from "commander";
|
2020-05-26 15:17:41 +02:00
|
|
|
|
2022-06-14 17:10:53 +02:00
|
|
|
import { Response } from "@bitwarden/node/cli/models/response";
|
|
|
|
import { MessageResponse } from "@bitwarden/node/cli/models/response/messageResponse";
|
2020-05-26 15:34:44 +02:00
|
|
|
|
|
|
|
interface IOption {
|
2021-02-03 18:44:33 +01:00
|
|
|
long?: string;
|
|
|
|
short?: string;
|
2020-05-26 15:17:41 +02:00
|
|
|
description: string;
|
2020-05-26 15:34:44 +02:00
|
|
|
}
|
2020-05-26 15:17:41 +02:00
|
|
|
|
2020-05-26 15:34:44 +02:00
|
|
|
interface ICommand {
|
|
|
|
commands?: ICommand[];
|
|
|
|
options?: IOption[];
|
2020-05-26 15:17:41 +02:00
|
|
|
_name: string;
|
|
|
|
_description: string;
|
2020-05-26 15:34:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const validShells = ["zsh"];
|
|
|
|
|
|
|
|
export class CompletionCommand {
|
2021-02-03 18:44:33 +01:00
|
|
|
async run(options: program.OptionValues) {
|
|
|
|
const shell: typeof validShells[number] = options.shell;
|
2020-05-26 15:34:44 +02:00
|
|
|
|
|
|
|
if (!shell) {
|
2022-01-19 16:45:14 +01:00
|
|
|
return Response.badRequest("`shell` option was not provided.");
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
|
2020-05-26 15:34:44 +02:00
|
|
|
if (!validShells.includes(shell)) {
|
|
|
|
return Response.badRequest("Unsupported shell.");
|
|
|
|
}
|
|
|
|
|
|
|
|
let content = "";
|
2021-12-20 18:04:00 +01:00
|
|
|
|
2020-05-26 15:34:44 +02:00
|
|
|
if (shell === "zsh") {
|
|
|
|
content = this.zshCompletion("bw", program as any as ICommand).render();
|
|
|
|
}
|
|
|
|
|
|
|
|
const res = new MessageResponse(content, null);
|
|
|
|
return Response.success(res);
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
|
|
|
|
2020-05-26 15:34:44 +02:00
|
|
|
private zshCompletion(rootName: string, rootCommand: ICommand) {
|
2021-12-20 18:04:00 +01:00
|
|
|
return {
|
2020-05-26 15:34:44 +02:00
|
|
|
render: () => {
|
2021-12-20 18:04:00 +01:00
|
|
|
return [
|
2020-05-26 15:34:44 +02:00
|
|
|
`#compdef _${rootName} ${rootName}`,
|
2021-12-20 18:04:00 +01:00
|
|
|
"",
|
2020-05-26 15:34:44 +02:00
|
|
|
this.renderCommandBlock(rootName, rootCommand),
|
|
|
|
].join("\n");
|
2021-12-20 18:04:00 +01:00
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-05-26 15:34:44 +02:00
|
|
|
private renderCommandBlock(name: string, command: ICommand): string {
|
2020-05-26 15:17:41 +02:00
|
|
|
const { commands = [], options = [] } = command;
|
|
|
|
const hasOptions = options.length > 0;
|
|
|
|
const hasCommands = commands.length > 0;
|
2021-12-20 18:04:00 +01:00
|
|
|
|
2020-05-26 15:34:44 +02:00
|
|
|
const args = options
|
2020-05-26 15:17:41 +02:00
|
|
|
.map(({ long, short, description }) => {
|
2020-05-26 15:34:44 +02:00
|
|
|
const aliases = [short, long].filter(Boolean);
|
|
|
|
const opts = aliases.join(",");
|
|
|
|
const desc = `[${description.replace(`'`, `'"'"'`)}]`;
|
|
|
|
return aliases.length > 1
|
|
|
|
? `'(${aliases.join(" ")})'{${opts}}'${desc}'`
|
|
|
|
: `'${opts}${desc}'`;
|
2021-12-20 18:04:00 +01:00
|
|
|
})
|
|
|
|
.concat(
|
2020-05-26 15:34:44 +02:00
|
|
|
`'(-h --help)'{-h,--help}'[output usage information]'`,
|
|
|
|
hasCommands ? '"1: :->cmnds"' : null,
|
|
|
|
'"*::arg:->args"'
|
2021-12-20 18:04:00 +01:00
|
|
|
)
|
2020-05-26 15:34:44 +02:00
|
|
|
.filter(Boolean);
|
|
|
|
|
2021-02-03 18:44:33 +01:00
|
|
|
const commandBlockFunctionParts = [];
|
2020-05-26 15:34:44 +02:00
|
|
|
|
2020-05-26 15:17:41 +02:00
|
|
|
if (hasCommands) {
|
2020-05-26 15:34:44 +02:00
|
|
|
commandBlockFunctionParts.push("local -a commands");
|
|
|
|
}
|
2020-05-26 15:17:41 +02:00
|
|
|
|
2020-05-26 15:34:44 +02:00
|
|
|
if (hasOptions) {
|
|
|
|
commandBlockFunctionParts.push(`_arguments -C \\\n ${args.join(` \\\n `)}`);
|
|
|
|
}
|
|
|
|
|
2020-05-26 15:17:41 +02:00
|
|
|
if (hasCommands) {
|
2020-05-26 15:34:44 +02:00
|
|
|
commandBlockFunctionParts.push(
|
2020-05-26 15:17:41 +02:00
|
|
|
`case $state in
|
|
|
|
cmnds)
|
|
|
|
commands=(
|
2020-05-26 15:34:44 +02:00
|
|
|
${commands
|
|
|
|
.map(({ _name, _description }) => `"${_name}:${_description}"`)
|
|
|
|
.join("\n ")}
|
2020-05-26 15:17:41 +02:00
|
|
|
)
|
|
|
|
_describe "command" commands
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
|
|
|
|
case "$words[1]" in
|
2020-05-26 15:34:44 +02:00
|
|
|
${commands
|
|
|
|
.map(({ _name }) => [`${_name})`, `_${name}_${_name}`, ";;"].join("\n "))
|
|
|
|
.join("\n ")}
|
|
|
|
esac`
|
2020-05-26 15:17:41 +02:00
|
|
|
);
|
|
|
|
}
|
2021-12-20 18:04:00 +01:00
|
|
|
|
2020-05-26 15:17:41 +02:00
|
|
|
const commandBlocParts = [
|
2020-05-26 15:34:44 +02:00
|
|
|
`function _${name} {\n ${commandBlockFunctionParts.join("\n\n ")}\n}`,
|
2021-12-20 18:04:00 +01:00
|
|
|
];
|
|
|
|
|
2020-05-26 15:17:41 +02:00
|
|
|
if (hasCommands) {
|
|
|
|
commandBlocParts.push(
|
2021-02-04 05:51:59 +01:00
|
|
|
commands.map((c) => this.renderCommandBlock(`${name}_${c._name}`, c)).join("\n\n")
|
2021-12-20 18:04:00 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-05-26 15:34:44 +02:00
|
|
|
return commandBlocParts.join("\n\n");
|
2021-12-20 18:04:00 +01:00
|
|
|
}
|
2020-05-26 15:17:41 +02:00
|
|
|
}
|