#include "command_manager.hh" #include "alias_registry.hh" #include "assert.hh" #include "buffer_utils.hh" #include "context.hh" #include "flags.hh" #include "file.hh" #include "optional.hh" #include "option_types.hh" #include "ranges.hh" #include "regex.hh" #include "register_manager.hh" #include "shell_manager.hh" #include "utils.hh" #include "unit_tests.hh" #include namespace Kakoune { bool CommandManager::command_defined(StringView command_name) const { return m_commands.find(command_name) != m_commands.end(); } void CommandManager::register_command(String command_name, CommandFunc func, String docstring, ParameterDesc param_desc, CommandFlags flags, CommandHelper helper, CommandCompleter completer) { m_commands[command_name] = { std::move(func), std::move(docstring), std::move(param_desc), flags, std::move(helper), std::move(completer) }; } bool CommandManager::module_defined(StringView module_name) const { return m_modules.find(module_name) != m_modules.end(); } void CommandManager::register_module(String module_name, String commands) { auto module = m_modules.find(module_name); if (module != m_modules.end() and module->value.loaded) throw runtime_error{format("module already loaded: '{}'", module_name)}; m_modules[module_name] = { false, std::move(commands) }; } void CommandManager::load_module(StringView module_name, Context& context) { auto module = m_modules.find(module_name); if (module == m_modules.end()) throw runtime_error{format("no such module: '{}'", module_name)}; if (module->value.loaded) return; module->value.loaded = true; Context empty_context{Context::EmptyContextFlag{}}; execute(module->value.commands, empty_context); module->value.commands.clear(); context.hooks().run_hook(Hook::ModuleLoaded, module_name, context); } struct parse_error : runtime_error { parse_error(StringView error) : runtime_error{format("parse error: {}", error)} {} }; Codepoint Reader::operator*() const { kak_assert(pos < str.end()); return utf8::codepoint(pos, str.end()); } Codepoint Reader::peek_next() const { return utf8::codepoint(utf8::next(pos, str.end()), str.end()); } Reader& Reader::operator++() { kak_assert(pos < str.end()); if (*pos == '\n') ++line; utf8::to_next(pos, str.end()); return *this; } namespace { bool is_command_separator(Codepoint c) { return c == ';' or c == '\n'; } struct QuotedResult { String content; bool terminated; }; QuotedResult parse_quoted(Reader& reader, Codepoint delimiter) { auto beg = reader.pos; String str; while (reader) { const Codepoint c = *reader; if (c == delimiter) { if (reader.peek_next() != delimiter) { str += reader.substr_from(beg); ++reader; return {str, true}; } str += (++reader).substr_from(beg); beg = reader.pos+1; } ++reader; } if (beg < reader.str.end()) str += reader.substr_from(beg); return {str, false}; } QuotedResult parse_quoted_balanced(Reader& reader, char opening_delimiter, char closing_delimiter) { kak_assert(utf8::codepoint(utf8::previous(reader.pos, reader.str.begin()), reader.str.end()) == opening_delimiter); int level = 0; auto start = reader.pos; while (reader) { const char c = *reader.pos; if (c == opening_delimiter) ++level; else if (c == closing_delimiter and level-- == 0) { auto content = reader.substr_from(start); ++reader.pos; return {String{String::NoCopy{}, content}, true}; } ++reader.pos; } return {String{String::NoCopy{}, reader.substr_from(start)}, false}; } String parse_unquoted(Reader& reader) { auto beg = reader.pos; String str; while (reader) { const char c = *reader.pos; if (is_command_separator(c) or is_horizontal_blank(c)) { str += reader.substr_from(beg); if (reader.pos != reader.str.begin() and *(reader.pos - 1) == '\\') { str.back() = c; beg = reader.pos+1; } else return str; } ++reader.pos; } if (beg < reader.str.end()) str += reader.substr_from(beg); return str; } Token::Type token_type(StringView type_name, bool throw_on_invalid) { if (type_name == "") return Token::Type::RawQuoted; else if (type_name == "sh") return Token::Type::ShellExpand; else if (type_name == "reg") return Token::Type::RegisterExpand; else if (type_name == "opt") return Token::Type::OptionExpand; else if (type_name == "val") return Token::Type::ValExpand; else if (type_name == "arg") return Token::Type::ArgExpand; else if (type_name == "file") return Token::Type::FileExpand; else if (throw_on_invalid) throw parse_error{format("unknown expand '{}'", type_name)}; else return Token::Type::RawQuoted; } void skip_blanks_and_comments(Reader& reader) { while (reader) { const Codepoint c = *reader; if (is_horizontal_blank(c)) ++reader; else if (c == '\\' and reader.pos + 1 != reader.str.end() and *(reader.pos + 1) == '\n') ++(++reader); else if (c == '#') { while (reader and *reader != '\n') ++reader; } else break; } } Token parse_percent_token(Reader& reader, bool throw_on_unterminated) { kak_assert(*reader == '%'); ++reader; const auto type_start = reader.pos; while (reader and iswalpha(*reader)) ++reader; StringView type_name = reader.substr_from(type_start); if (not reader or is_blank(*reader)) { if (throw_on_unterminated) throw parse_error{format("expected a string delimiter after '%{}'", type_name)}; return {}; } Token::Type type = token_type(type_name, throw_on_unterminated); constexpr struct CharPair { Codepoint opening; Codepoint closing; } matching_pairs[] = { { '(', ')' }, { '[', ']' }, { '{', '}' }, { '<', '>' } }; const Codepoint opening_delimiter = *reader; auto coord = reader.coord(); ++reader; auto start = reader.pos; auto it = find_if(matching_pairs, [opening_delimiter](const CharPair& cp) { return opening_delimiter == cp.opening; }); const auto str_beg = reader.str.begin(); if (it != std::end(matching_pairs)) { const Codepoint closing_delimiter = it->closing; auto quoted = parse_quoted_balanced(reader, opening_delimiter, closing_delimiter); if (throw_on_unterminated and not quoted.terminated) throw parse_error{format("{}:{}: unterminated string '%{}{}...{}'", coord.line, coord.column, type_name, opening_delimiter, closing_delimiter)}; return {type, start - str_beg, coord, std::move(quoted.content)}; } else { auto quoted = parse_quoted(reader, opening_delimiter); if (throw_on_unterminated and not quoted.terminated) throw parse_error{format("{}:{}: unterminated string '%{}{}...{}'", coord.line, coord.column, type_name, opening_delimiter, opening_delimiter)}; return {type, start - str_beg, coord, std::move(quoted.content)}; } } auto expand_option(const Option& opt, std::true_type) { return opt.get_as_string(Quoting::Raw); } auto expand_option(const Option& opt, std::false_type) { return opt.get_as_strings(); } auto expand_register(StringView reg, const Context& context, std::true_type) { return context.main_sel_register_value(reg).str(); } auto expand_register(StringView reg, const Context& context, std::false_type) { return RegisterManager::instance()[reg].get(context) | gather>(); } String expand_arobase(ConstArrayView params, std::true_type) { return join(params, ' ', false); } Vector expand_arobase(ConstArrayView params, std::false_type) { return {params.begin(), params.end()}; } template std::conditional_t> expand_token(const Token& token, const Context& context, const ShellContext& shell_context) { using IsSingle = std::integral_constant; auto& content = token.content; switch (token.type) { case Token::Type::ShellExpand: { auto str = ShellManager::instance().eval( content, context, {}, ShellManager::Flags::WaitForStdout, shell_context).first; int trailing_eol_count = 0; for (auto c : str | reverse()) { if (c != '\n') break; ++trailing_eol_count; } str.resize(str.length() - trailing_eol_count, 0); return {str}; } case Token::Type::RegisterExpand: return expand_register(content, context, IsSingle{}); case Token::Type::OptionExpand: return expand_option(context.options()[content], IsSingle{}); case Token::Type::ValExpand: { auto it = shell_context.env_vars.find(content); if (it != shell_context.env_vars.end()) return {it->value}; return {ShellManager::instance().get_val(content, context, Quoting::Kakoune)}; } case Token::Type::ArgExpand: { auto& params = shell_context.params; if (content == '@') return expand_arobase(params, IsSingle{}); const int arg = str_to_int(content)-1; if (arg < 0) throw runtime_error("invalid argument index"); return {arg < params.size() ? params[arg] : String{}}; } case Token::Type::FileExpand: return {read_file(content)}; case Token::Type::RawEval: return {expand(content, context, shell_context)}; case Token::Type::Raw: case Token::Type::RawQuoted: return {content}; default: kak_assert(false); } return {}; } } CommandParser::CommandParser(StringView command_line) : m_reader{command_line} {} Optional CommandParser::read_token(bool throw_on_unterminated) { skip_blanks_and_comments(m_reader); if (not m_reader) return {}; const StringView line = m_reader.str; const char* start = m_reader.pos; auto coord = m_reader.coord(); const Codepoint c = *m_reader; if (c == '"' or c == '\'') { start = (++m_reader).pos; QuotedResult quoted = parse_quoted(m_reader, c); if (throw_on_unterminated and not quoted.terminated) throw parse_error{format("unterminated string {0}...{0}", c)}; return Token{c == '"' ? Token::Type::RawEval : Token::Type::RawQuoted, start - line.begin(), coord, std::move(quoted.content)}; } else if (c == '%') { auto token = parse_percent_token(m_reader, throw_on_unterminated); return token; } else if (is_command_separator(*m_reader)) { ++m_reader; return Token{Token::Type::CommandSeparator, m_reader.pos - line.begin(), coord, {}}; } else { if (c == '\\') { auto next = m_reader.peek_next(); if (next == '%' or next == '\'' or next == '"') ++m_reader; } return Token{Token::Type::Raw, start - line.begin(), coord, parse_unquoted(m_reader)}; } return {}; } template String expand_impl(StringView str, const Context& context, const ShellContext& shell_context, Postprocess postprocess) { Reader reader{str}; String res; auto beg = str.begin(); while (reader) { Codepoint c = *reader; if (c == '%') { if (reader.peek_next() == '%') { res += (++reader).substr_from(beg); beg = (++reader).pos; } else { res += reader.substr_from(beg); res += postprocess(expand_token(parse_percent_token(reader, true), context, shell_context)); beg = reader.pos; } } else ++reader; } res += reader.substr_from(beg); return res; } String expand(StringView str, const Context& context, const ShellContext& shell_context) { return expand_impl(str, context, shell_context, [](String s){ return s; }); } String expand(StringView str, const Context& context, const ShellContext& shell_context, const FunctionRef& postprocess) { return expand_impl(str, context, shell_context, postprocess); } struct command_not_found : runtime_error { command_not_found(StringView name) : runtime_error(format("no such command: '{}'", name)) {} }; CommandManager::CommandMap::const_iterator CommandManager::find_command(const Context& context, StringView name) const { auto alias = context.aliases()[name]; StringView cmd_name = alias.empty() ? name : alias; return m_commands.find(cmd_name); } void CommandManager::execute_single_command(CommandParameters params, Context& context, const ShellContext& shell_context, BufferCoord pos) { if (params.empty()) return; constexpr int max_command_depth = 100; if (m_command_depth > max_command_depth) throw runtime_error("maximum nested command depth hit"); ++m_command_depth; auto pop_cmd = on_scope_end([this] { --m_command_depth; }); ParameterList param_view(params.begin()+1, params.end()); auto command_it = find_command(context, params[0]); if (command_it == m_commands.end()) throw command_not_found(params[0]); const DebugFlags debug_flags = context.options()["debug"].get(); if (debug_flags & DebugFlags::Commands) write_to_debug_buffer(format("command {} {}", params[0], join(param_view, ' '))); try { ParametersParser parameter_parser(param_view, command_it->value.param_desc); command_it->value.func(parameter_parser, context, shell_context); } catch (failure& error) { throw; } catch (runtime_error& error) { error.set_what(format("{}:{}: '{}' {}", pos.line+1, pos.column+1, params[0], error.what())); throw; } } void CommandManager::execute(StringView command_line, Context& context, const ShellContext& shell_context) { CommandParser parser(command_line); BufferCoord command_coord; Vector params; while (Optional token_opt = parser.read_token(true)) { auto& token = *token_opt; if (params.empty()) command_coord = token.coord; if (token.type == Token::Type::CommandSeparator) { execute_single_command(params, context, shell_context, command_coord); params.clear(); } else if (token.type == Token::Type::ArgExpand and token.content == '@') params.insert(params.end(), shell_context.params.begin(), shell_context.params.end()); else { auto tokens = expand_token(token, context, shell_context); params.insert(params.end(), std::make_move_iterator(tokens.begin()), std::make_move_iterator(tokens.end())); } } execute_single_command(params, context, shell_context, command_coord); } Optional CommandManager::command_info(const Context& context, StringView command_line) const { CommandParser parser{command_line}; Vector tokens; while (auto token = parser.read_token(false)) { if (token->type == Token::Type::CommandSeparator) tokens.clear(); else tokens.push_back(std::move(*token)); } if (tokens.empty() or (tokens.front().type != Token::Type::Raw and tokens.front().type != Token::Type::RawQuoted)) return {}; auto cmd = find_command(context, tokens.front().content); if (cmd == m_commands.end()) return {}; CommandInfo res; res.name = cmd->key; if (not cmd->value.docstring.empty()) res.info += cmd->value.docstring + "\n"; if (cmd->value.helper) { Vector params; for (auto it = tokens.begin() + 1; it != tokens.end(); ++it) { if (it->type == Token::Type::Raw or it->type == Token::Type::RawQuoted or it->type == Token::Type::RawEval) params.push_back(it->content); } String helpstr = cmd->value.helper(context, params); if (not helpstr.empty()) res.info += format("{}\n", helpstr); } String aliases; for (auto& alias : context.aliases().aliases_for(cmd->key)) aliases += " " + alias; if (not aliases.empty()) res.info += format("Aliases:{}\n", aliases); auto& switches = cmd->value.param_desc.switches; if (not switches.empty()) res.info += format("Switches:\n{}", indent(generate_switches_doc(switches))); return res; } Completions CommandManager::complete_command_name(const Context& context, StringView query) const { auto commands = m_commands | filter([](const CommandMap::Item& cmd) { return not (cmd.value.flags & CommandFlags::Hidden); }) | transform(&CommandMap::Item::key); auto aliases = context.aliases().flatten_aliases() | transform(&HashItem::key) | filter([](auto& alias) { return alias.length() > 3; }); return {0, query.length(), Kakoune::complete(query, query.length(), concatenated(commands, aliases))}; } Completions CommandManager::complete_module_name(StringView query) const { return {0, query.length(), Kakoune::complete(query, query.length(), m_modules | filter([](auto&& item) { return not item.value.loaded; }) | transform(&ModuleMap::Item::key))}; } Completions CommandManager::complete(const Context& context, CompletionFlags flags, StringView command_line, ByteCount cursor_pos) { CommandParser parser{command_line}; const char* cursor = command_line.begin() + (int)cursor_pos; Vector tokens; bool is_last_token = true; while (auto token = parser.read_token(false)) { if (token->type == Token::Type::CommandSeparator) { tokens.clear(); continue; } tokens.push_back(std::move(*token)); if (parser.pos() >= cursor) { is_last_token = false; break; } } if (is_last_token) tokens.push_back({Token::Type::Raw, command_line.length(), parser.coord(), {}}); kak_assert(not tokens.empty()); const auto& token = tokens.back(); // command name completion if (tokens.size() == 1 and (token.type == Token::Type::Raw or token.type == Token::Type::RawQuoted)) { auto cmd_start = token.pos; StringView query = command_line.substr(cmd_start, cursor_pos - cmd_start); return offset_pos(complete_command_name(context, query), cmd_start); } const ByteCount start = token.pos; const ByteCount cursor_pos_in_token = cursor_pos - start; switch (token.type) { case Token::Type::RegisterExpand: return {start , cursor_pos, RegisterManager::instance().complete_register_name( token.content, cursor_pos_in_token) }; case Token::Type::OptionExpand: return {start , cursor_pos, GlobalScope::instance().option_registry().complete_option_name( token.content, cursor_pos_in_token) }; case Token::Type::ShellExpand: return offset_pos(shell_complete(context, flags, token.content, cursor_pos_in_token), start); case Token::Type::ValExpand: return {start , cursor_pos, ShellManager::instance().complete_env_var( token.content, cursor_pos_in_token) }; case Token::Type::FileExpand: { const auto& ignored_files = context.options()["ignored_files"].get(); return {start , cursor_pos, complete_filename( token.content, ignored_files, cursor_pos_in_token, FilenameFlags::Expand) }; } case Token::Type::Raw: case Token::Type::RawQuoted: case Token::Type::RawEval: { if (token.type != Token::Type::Raw and token.type != Token::Type::RawQuoted) return Completions{}; StringView command_name = tokens.front().content; if (command_name != m_last_complete_command) { m_last_complete_command = command_name.str(); flags |= CompletionFlags::Start; } auto command_it = find_command(context, command_name); if (command_it == m_commands.end()) return Completions{}; if (token.content.substr(0_byte, 1_byte) == "-") { auto switches = Kakoune::complete(token.content.substr(1_byte), cursor_pos_in_token, command_it->value.param_desc.switches | transform(&SwitchMap::Item::key)); if (not switches.empty()) return Completions{start+1, cursor_pos, std::move(switches)}; } if (not command_it->value.completer) return Completions{}; Vector params; for (auto it = tokens.begin() + 1; it != tokens.end(); ++it) params.push_back(it->content); Completions completions = offset_pos(command_it->value.completer( context, flags, params, tokens.size() - 2, cursor_pos_in_token), start); if (not (completions.flags & Completions::Flags::Quoted) and token.type == Token::Type::Raw) { for (auto& c : completions.candidates) c = (not c.empty() and contains("%'\"", c[0]) ? "\\" : "") + escape(c, "; \t", '\\'); } return completions; } default: break; } return Completions{}; } Completions CommandManager::complete(const Context& context, CompletionFlags flags, CommandParameters params, size_t token_to_complete, ByteCount pos_in_token) { StringView prefix = params[token_to_complete].substr(0, pos_in_token); if (token_to_complete == 0) return complete_command_name(context, prefix); else { StringView command_name = params[0]; if (command_name != m_last_complete_command) { m_last_complete_command = command_name.str(); flags |= CompletionFlags::Start; } auto command_it = find_command(context, command_name); if (command_it != m_commands.end() and command_it->value.completer) return command_it->value.completer( context, flags, params.subrange(1), token_to_complete-1, pos_in_token); } return Completions{}; } UnitTest test_command_parsing{[] { auto check_quoted = [](StringView str, bool terminated, StringView content) { Reader reader{str}; const Codepoint delimiter = *reader; auto quoted = parse_quoted(++reader, delimiter); kak_assert(quoted.terminated == terminated); kak_assert(quoted.content == content); }; check_quoted("'abc'", true, "abc"); check_quoted("'abc''def", false, "abc'def"); check_quoted("'abc''def'''", true, "abc'def'"); auto check_balanced = [](StringView str, Codepoint opening, Codepoint closing, bool terminated, StringView content) { Reader reader{str}; auto quoted = parse_quoted_balanced(++reader, opening, closing); kak_assert(quoted.terminated == terminated); kak_assert(quoted.content == content); }; check_balanced("{abc}", '{', '}', true, "abc"); check_balanced("{abc{def}}", '{', '}', true, "abc{def}"); check_balanced("{{abc}{def}", '{', '}', false, "{abc}{def}"); auto check_unquoted = [](StringView str, StringView content) { Reader reader{str}; auto res = parse_unquoted(reader); kak_assert(res == content); }; check_unquoted("abc def", "abc"); check_unquoted("abc; def", "abc"); check_unquoted("abc\\; def", "abc;"); check_unquoted("abc\\;\\ def", "abc; def"); { CommandParser parser(R"(foo 'bar' "baz" qux)"); kak_assert(parser.read_token(false)->content == "foo"); kak_assert(parser.read_token(false)->content == "bar"); kak_assert(parser.read_token(false)->content == "baz"); kak_assert(parser.read_token(false)->content == "qux"); kak_assert(not parser.read_token(false)); } }}; }