// Functions used for implementing the complete builtin. #include "config.h" // IWYU pragma: keep #include "complete.h" #include #include #include #include #include #include "../builtin.h" #include "../color.h" #include "../common.h" #include "../complete.h" #include "../env.h" #include "../fallback.h" // IWYU pragma: keep #include "../highlight.h" #include "../parse_constants.h" #include "../parse_util.h" #include "../parser.h" #include "../reader.h" #include "../wcstringutil.h" #include "../wgetopt.h" #include "../wutil.h" // IWYU pragma: keep // builtin_complete_* are a set of rather silly looping functions that make sure that all the proper // combinations of complete_add or complete_remove get called. This is needed since complete allows // you to specify multiple switches on a single commandline, like 'complete -s a -s b -s c', but the // complete_add function only accepts one short switch and one long switch. /// Silly function. static void builtin_complete_add2(const wcstring &cmd, bool cmd_is_path, const wchar_t *short_opt, const wcstring_list_t &gnu_opts, const wcstring_list_t &old_opts, completion_mode_t result_mode, const wcstring_list_t &condition, const wchar_t *comp, const wchar_t *desc, complete_flags_t flags) { for (const wchar_t *s = short_opt; *s; s++) { complete_add(cmd, cmd_is_path, wcstring{*s}, option_type_short, result_mode, condition, comp, desc, flags); } for (const wcstring &gnu_opt : gnu_opts) { complete_add(cmd, cmd_is_path, gnu_opt, option_type_double_long, result_mode, condition, comp, desc, flags); } for (const wcstring &old_opt : old_opts) { complete_add(cmd, cmd_is_path, old_opt, option_type_single_long, result_mode, condition, comp, desc, flags); } if (old_opts.empty() && gnu_opts.empty() && short_opt[0] == L'\0') { complete_add(cmd, cmd_is_path, wcstring(), option_type_args_only, result_mode, condition, comp, desc, flags); } } /// Silly function. static void builtin_complete_add(const wcstring_list_t &cmds, const wcstring_list_t &paths, const wchar_t *short_opt, const wcstring_list_t &gnu_opt, const wcstring_list_t &old_opt, completion_mode_t result_mode, const wcstring_list_t &condition, const wchar_t *comp, const wchar_t *desc, complete_flags_t flags) { for (const wcstring &cmd : cmds) { builtin_complete_add2(cmd, false /* not path */, short_opt, gnu_opt, old_opt, result_mode, condition, comp, desc, flags); } for (const wcstring &path : paths) { builtin_complete_add2(path, true /* is path */, short_opt, gnu_opt, old_opt, result_mode, condition, comp, desc, flags); } } static void builtin_complete_remove_cmd(const wcstring &cmd, bool cmd_is_path, const wchar_t *short_opt, const wcstring_list_t &gnu_opt, const wcstring_list_t &old_opt) { bool removed = false; for (const wchar_t *s = short_opt; *s; s++) { complete_remove(cmd, cmd_is_path, wcstring{*s}, option_type_short); removed = true; } for (const wcstring &opt : old_opt) { complete_remove(cmd, cmd_is_path, opt, option_type_single_long); removed = true; } for (const wcstring &opt : gnu_opt) { complete_remove(cmd, cmd_is_path, opt, option_type_double_long); removed = true; } if (!removed) { // This means that all loops were empty. complete_remove_all(cmd, cmd_is_path); } } static void builtin_complete_remove(const wcstring_list_t &cmds, const wcstring_list_t &paths, const wchar_t *short_opt, const wcstring_list_t &gnu_opt, const wcstring_list_t &old_opt) { for (const wcstring &cmd : cmds) { builtin_complete_remove_cmd(cmd, false /* not path */, short_opt, gnu_opt, old_opt); } for (const wcstring &path : paths) { builtin_complete_remove_cmd(path, true /* is path */, short_opt, gnu_opt, old_opt); } } static void builtin_complete_print(const wcstring &cmd, io_streams_t &streams, parser_t &parser) { const wcstring repr = complete_print(cmd); // colorize if interactive if (!streams.out_is_redirected && isatty(STDOUT_FILENO)) { std::vector colors; highlight_shell(repr, colors, parser.context()); streams.out.append(str2wcstring(colorize(repr, colors, parser.vars()))); } else { streams.out.append(repr); } } /// Values used for long-only options. enum { opt_escape = 1, }; /// The complete builtin. Used for specifying programmable tab-completions. Calls the functions in // complete.cpp for any heavy lifting. maybe_t builtin_complete(parser_t &parser, io_streams_t &streams, const wchar_t **argv) { const wchar_t *cmd = argv[0]; int argc = builtin_count_args(argv); completion_mode_t result_mode{}; int remove = 0; wcstring short_opt; wcstring_list_t gnu_opt, old_opt, subcommand; const wchar_t *comp = L"", *desc = L""; wcstring_list_t condition; bool do_complete = false; bool have_do_complete_param = false; wcstring do_complete_param; wcstring_list_t cmd_to_complete; wcstring_list_t path; wcstring_list_t wrap_targets; bool preserve_order = false; bool unescape_output = true; static const wchar_t *const short_options = L":a:c:p:s:l:o:d:fFrxeuAn:C::w:hk"; static const struct woption long_options[] = { {L"exclusive", no_argument, nullptr, 'x'}, {L"no-files", no_argument, nullptr, 'f'}, {L"force-files", no_argument, nullptr, 'F'}, {L"require-parameter", no_argument, nullptr, 'r'}, {L"path", required_argument, nullptr, 'p'}, {L"command", required_argument, nullptr, 'c'}, {L"short-option", required_argument, nullptr, 's'}, {L"long-option", required_argument, nullptr, 'l'}, {L"old-option", required_argument, nullptr, 'o'}, {L"subcommand", required_argument, nullptr, 'S'}, {L"description", required_argument, nullptr, 'd'}, {L"arguments", required_argument, nullptr, 'a'}, {L"erase", no_argument, nullptr, 'e'}, {L"unauthoritative", no_argument, nullptr, 'u'}, {L"authoritative", no_argument, nullptr, 'A'}, {L"condition", required_argument, nullptr, 'n'}, {L"wraps", required_argument, nullptr, 'w'}, {L"do-complete", optional_argument, nullptr, 'C'}, {L"help", no_argument, nullptr, 'h'}, {L"keep-order", no_argument, nullptr, 'k'}, {L"escape", no_argument, nullptr, opt_escape}, {}}; bool have_x = false; int opt; wgetopter_t w; while ((opt = w.wgetopt_long(argc, argv, short_options, long_options, nullptr)) != -1) { switch (opt) { case 'x': { result_mode.no_files = true; result_mode.requires_param = true; // Needed to print an error later. have_x = true; break; } case 'f': { result_mode.no_files = true; break; } case 'F': { result_mode.force_files = true; break; } case 'r': { result_mode.requires_param = true; break; } case 'k': { preserve_order = true; break; } case 'p': case 'c': { wcstring tmp; if (unescape_string(w.woptarg, &tmp, UNESCAPE_SPECIAL)) { if (opt == 'p') path.push_back(tmp); else cmd_to_complete.push_back(tmp); } else { streams.err.append_format(_(L"%ls: Invalid token '%ls'\n"), cmd, w.woptarg); return STATUS_INVALID_ARGS; } break; } case 'd': { desc = w.woptarg; assert(desc); break; } case 'u': { // This option was removed in commit 1911298 and is now a no-op. break; } case 'A': { // This option was removed in commit 1911298 and is now a no-op. break; } case 's': { short_opt.append(w.woptarg); if (w.woptarg[0] == '\0') { streams.err.append_format(_(L"%ls: -s requires a non-empty string\n"), cmd); return STATUS_INVALID_ARGS; } break; } case 'l': { gnu_opt.push_back(w.woptarg); if (w.woptarg[0] == '\0') { streams.err.append_format(_(L"%ls: -l requires a non-empty string\n"), cmd); return STATUS_INVALID_ARGS; } break; } case 'o': { old_opt.push_back(w.woptarg); if (w.woptarg[0] == '\0') { streams.err.append_format(_(L"%ls: -o requires a non-empty string\n"), cmd); return STATUS_INVALID_ARGS; } break; } case 'S': { subcommand.push_back(w.woptarg); if (w.woptarg[0] == '\0') { streams.err.append_format(_(L"%ls: -S requires a non-empty string\n"), cmd); return STATUS_INVALID_ARGS; } break; } case 'a': { comp = w.woptarg; assert(comp); break; } case 'e': { remove = 1; break; } case 'n': { condition.push_back(w.woptarg); assert(w.woptarg); break; } case 'w': { wrap_targets.push_back(w.woptarg); break; } case 'C': { do_complete = true; have_do_complete_param = w.woptarg != nullptr; if (have_do_complete_param) do_complete_param = w.woptarg; break; } case opt_escape: { unescape_output = false; break; } case 'h': { builtin_print_help(parser, streams, cmd); return STATUS_CMD_OK; } case ':': { builtin_missing_argument(parser, streams, cmd, argv[w.woptind - 1]); return STATUS_INVALID_ARGS; } case '?': { builtin_unknown_option(parser, streams, cmd, argv[w.woptind - 1]); return STATUS_INVALID_ARGS; } default: { DIE("unexpected retval from wgetopt_long"); } } } if (result_mode.no_files && result_mode.force_files) { if (!have_x) { streams.err.append_format(BUILTIN_ERR_COMBO2, L"complete", L"'--no-files' and '--force-files'"); } else { // The reason for us not wanting files is `-x`, // which is short for `-rf`. streams.err.append_format(BUILTIN_ERR_COMBO2, L"complete", L"'--exclusive' and '--force-files'"); } return STATUS_INVALID_ARGS; } if (w.woptind != argc) { // Use one left-over arg as the do-complete argument // to enable `complete -C "git check"`. if (do_complete && !have_do_complete_param && argc == w.woptind + 1) { do_complete_param = argv[argc - 1]; have_do_complete_param = true; } else if (!do_complete && cmd_to_complete.empty() && argc == w.woptind + 1) { // Or use one left-over arg as the command to complete cmd_to_complete.push_back(argv[argc - 1]); } else { streams.err.append_format(BUILTIN_ERR_TOO_MANY_ARGUMENTS, cmd); builtin_print_error_trailer(parser, streams.err, cmd); return STATUS_INVALID_ARGS; } } for (const auto &condition_string : condition) { parse_error_list_t errors; if (parse_util_detect_errors(condition_string, &errors)) { streams.err.append_format(L"%ls: Condition '%ls' contained a syntax error", cmd, condition_string.c_str()); for (const auto &error : errors) { streams.err.append_format(L"\n%ls: ", cmd); streams.err.append(error.describe(condition_string, parser.is_interactive())); } return STATUS_CMD_ERROR; } } if (comp && *comp) { wcstring prefix; prefix.append(cmd); prefix.append(L": "); if (maybe_t err_text = parse_util_detect_errors_in_argument_list(comp, prefix)) { streams.err.append_format(L"%ls: Completion '%ls' contained a syntax error\n", cmd, comp); streams.err.append(*err_text); streams.err.push_back(L'\n'); return STATUS_CMD_ERROR; } } if (do_complete) { if (!have_do_complete_param) { // No argument given, try to use the current commandline. auto state = commandline_get_state(); if (!state.initialized) { // This corresponds to using 'complete -C' in non-interactive mode. // See #2361 . builtin_missing_argument(parser, streams, cmd, L"-C"); return STATUS_INVALID_ARGS; } do_complete_param = std::move(state.text); } const wchar_t *token; parse_util_token_extent(do_complete_param.c_str(), do_complete_param.size(), &token, nullptr, nullptr, nullptr); // Create a scoped transient command line, so that builtin_commandline will see our // argument, not the reader buffer. parser.libdata().transient_commandlines.push_back(do_complete_param); cleanup_t remove_transient([&] { parser.libdata().transient_commandlines.pop_back(); }); // Prevent accidental recursion (see #6171). if (!parser.libdata().builtin_complete_current_commandline) { if (!have_do_complete_param) parser.libdata().builtin_complete_current_commandline = true; completion_list_t comp = complete( do_complete_param, completion_request_options_t::normal(), parser.context()); for (const auto &next : comp) { // Make a fake commandline, and then apply the completion to it. const wcstring faux_cmdline = token; size_t tmp_cursor = faux_cmdline.size(); wcstring faux_cmdline_with_completion = completion_apply_to_command_line( next.completion, next.flags, faux_cmdline, &tmp_cursor, false); // completion_apply_to_command_line will append a space unless COMPLETE_NO_SPACE // is set. We don't want to set COMPLETE_NO_SPACE because that won't close // quotes. What we want is to close the quote, but not append the space. So we // just look for the space and clear it. if (!(next.flags & COMPLETE_NO_SPACE) && string_suffixes_string(L" ", faux_cmdline_with_completion)) { faux_cmdline_with_completion.resize(faux_cmdline_with_completion.size() - 1); } if (unescape_output) { // The input data is meant to be something like you would have on the command // line, e.g. includes backslashes. The output should be raw, i.e. unescaped. So // we need to unescape the command line. See #1127. unescape_string_in_place(&faux_cmdline_with_completion, UNESCAPE_DEFAULT); } streams.out.append(faux_cmdline_with_completion); // Append any description. if (!next.description.empty()) { streams.out.push_back(L'\t'); streams.out.append(next.description); } streams.out.push_back(L'\n'); } parser.libdata().builtin_complete_current_commandline = false; } } else if (path.empty() && gnu_opt.empty() && short_opt.empty() && old_opt.empty() && !remove && !*comp && !*desc && condition.empty() && wrap_targets.empty() && !result_mode.no_files && !result_mode.force_files && !result_mode.requires_param) { // No arguments that would add or remove anything specified, so we print the definitions of // all matching completions. if (cmd_to_complete.empty()) { builtin_complete_print(L"", streams, parser); } else { for (auto &cmd : cmd_to_complete) { builtin_complete_print(cmd, streams, parser); } } } else { int flags = COMPLETE_AUTO_SPACE; // HACK: Don't escape tildes because at the beginning of a token they probably mean // $HOME, for example as produced by a recursive call to "complete -C". flags |= COMPLETE_DONT_ESCAPE_TILDES; if (preserve_order) { flags |= COMPLETE_DONT_SORT; } if (remove) { builtin_complete_remove(cmd_to_complete, path, short_opt.c_str(), gnu_opt, old_opt); } else { builtin_complete_add(cmd_to_complete, path, short_opt.c_str(), gnu_opt, old_opt, result_mode, condition, comp, desc, flags); } // Handle wrap targets (probably empty). We only wrap commands, not paths. for (const auto &wrap_target : wrap_targets) { for (const auto &i : cmd_to_complete) { (remove ? complete_remove_wrapper : complete_add_wrapper)(i, wrap_target); } } } return STATUS_CMD_OK; }