//===-- CommandObjectRegexCommand.cpp -------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "CommandObjectRegexCommand.h" #include "lldb/Interpreter/CommandInterpreter.h" #include "lldb/Interpreter/CommandReturnObject.h" #include "llvm/Support/Errc.h" #include "llvm/Support/Error.h" using namespace lldb; using namespace lldb_private; // CommandObjectRegexCommand constructor CommandObjectRegexCommand::CommandObjectRegexCommand( CommandInterpreter &interpreter, llvm::StringRef name, llvm::StringRef help, llvm::StringRef syntax, uint32_t max_matches, uint32_t completion_type_mask, bool is_removable) : CommandObjectRaw(interpreter, name, help, syntax), m_max_matches(max_matches), m_completion_type_mask(completion_type_mask), m_is_removable(is_removable) {} // Destructor CommandObjectRegexCommand::~CommandObjectRegexCommand() = default; llvm::Expected CommandObjectRegexCommand::SubstituteVariables( llvm::StringRef input, const llvm::SmallVectorImpl &replacements) { std::string buffer; llvm::raw_string_ostream output(buffer); llvm::SmallVector parts; input.split(parts, '%'); output << parts[0]; for (llvm::StringRef part : drop_begin(parts)) { size_t idx = 0; if (part.consumeInteger(10, idx)) output << '%'; else if (idx < replacements.size()) output << replacements[idx]; else return llvm::make_error( llvm::formatv("%{0} is out of range: not enough arguments specified", idx), llvm::errc::invalid_argument); output << part; } return output.str(); } bool CommandObjectRegexCommand::DoExecute(llvm::StringRef command, CommandReturnObject &result) { EntryCollection::const_iterator pos, end = m_entries.end(); for (pos = m_entries.begin(); pos != end; ++pos) { llvm::SmallVector matches; if (pos->regex.Execute(command, &matches)) { llvm::Expected new_command = SubstituteVariables(pos->command, matches); if (!new_command) { result.SetError(new_command.takeError()); return false; } // Interpret the new command and return this as the result! if (m_interpreter.GetExpandRegexAliases()) result.GetOutputStream().Printf("%s\n", new_command->c_str()); // We don't have to pass an override_context here, as the command that // called us should have set up the context appropriately. return m_interpreter.HandleCommand(new_command->c_str(), eLazyBoolNo, result); } } result.SetStatus(eReturnStatusFailed); if (!GetSyntax().empty()) result.AppendError(GetSyntax()); else result.GetErrorStream() << "Command contents '" << command << "' failed to match any " "regular expression in the '" << m_cmd_name << "' regex "; return false; } bool CommandObjectRegexCommand::AddRegexCommand(llvm::StringRef re_cstr, llvm::StringRef command_cstr) { m_entries.resize(m_entries.size() + 1); // Only add the regular expression if it compiles m_entries.back().regex = RegularExpression(re_cstr); if (m_entries.back().regex.IsValid()) { m_entries.back().command = command_cstr.str(); return true; } // The regex didn't compile... m_entries.pop_back(); return false; } void CommandObjectRegexCommand::HandleCompletion(CompletionRequest &request) { if (m_completion_type_mask) { CommandCompletions::InvokeCommonCompletionCallbacks( GetCommandInterpreter(), m_completion_type_mask, request, nullptr); } }