summaryrefslogtreecommitdiffstats
path: root/src/readline_curses.cc
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-04 17:44:55 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-05-04 17:44:55 +0000
commit5068d34c08f951a7ea6257d305a1627b09a95817 (patch)
tree08213e2be853396a3b07ce15dbe222644dcd9a89 /src/readline_curses.cc
parentInitial commit. (diff)
downloadlnav-5068d34c08f951a7ea6257d305a1627b09a95817.tar.xz
lnav-5068d34c08f951a7ea6257d305a1627b09a95817.zip
Adding upstream version 0.11.1.upstream/0.11.1upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to '')
-rw-r--r--src/readline_curses.cc1508
1 files changed, 1508 insertions, 0 deletions
diff --git a/src/readline_curses.cc b/src/readline_curses.cc
new file mode 100644
index 0000000..ed91b3c
--- /dev/null
+++ b/src/readline_curses.cc
@@ -0,0 +1,1508 @@
+/**
+ * Copyright (c) 2007-2012, Timothy Stack
+ *
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *
+ * * Redistributions of source code must retain the above copyright notice, this
+ * list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above copyright notice,
+ * this list of conditions and the following disclaimer in the documentation
+ * and/or other materials provided with the distribution.
+ * * Neither the name of Timothy Stack nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ''AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+ * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ *
+ * @file readline_curses.cc
+ */
+
+#include <errno.h>
+#include <signal.h>
+#include <stdlib.h>
+#include <string.h>
+#include <sys/socket.h>
+#include <sys/wait.h>
+#include <unistd.h>
+
+#include "config.h"
+
+#ifdef HAVE_PTY_H
+# include <pty.h>
+#endif
+
+#ifdef HAVE_UTIL_H
+# include <util.h>
+#endif
+
+#ifdef HAVE_LIBUTIL_H
+# include <libutil.h>
+#endif
+
+#include <string>
+#include <utility>
+
+#include "base/ansi_scrubber.hh"
+#include "base/auto_mem.hh"
+#include "base/lnav_log.hh"
+#include "base/paths.hh"
+#include "base/string_util.hh"
+#include "fmt/format.h"
+#include "fts_fuzzy_match.hh"
+#include "lnav_util.hh"
+#include "readline_curses.hh"
+#include "shlex.hh"
+#include "spookyhash/SpookyV2.h"
+
+static int got_line = 0;
+static int got_abort = 0;
+static bool alt_done = 0;
+static sig_atomic_t got_timeout = 0;
+static sig_atomic_t got_winch = 0;
+static readline_curses* child_this;
+static sig_atomic_t looping = 1;
+static const int HISTORY_SIZE = 256;
+static int completion_start;
+static const int FUZZY_PEER_THRESHOLD = 30;
+
+static const char* RL_INIT[] = {
+ /*
+ * XXX Need to keep the input on a single line since the display screws
+ * up if it wraps around.
+ */
+ "set horizontal-scroll-mode on",
+ "set bell-style none",
+ "set show-all-if-ambiguous on",
+ "set show-all-if-unmodified on",
+ "set menu-complete-display-prefix on",
+ "TAB: menu-complete",
+ "\"\\e[Z\": menu-complete-backward",
+};
+
+readline_context* readline_context::loaded_context;
+std::set<std::string>* readline_context::arg_possibilities;
+static std::string last_match_str;
+static bool last_match_str_valid;
+static bool arg_needs_shlex;
+static nonstd::optional<std::string> rewrite_line_start;
+
+static void
+sigalrm(int sig)
+{
+ got_timeout = 1;
+}
+
+static void
+sigwinch(int sig)
+{
+ got_winch = 1;
+}
+
+static void
+sigterm(int sig)
+{
+ looping = 0;
+}
+
+static void
+line_ready_tramp(char* line)
+{
+ child_this->line_ready(line);
+ got_line = 1;
+ rl_callback_handler_remove();
+}
+
+static int
+sendall(int sock, const char* buf, size_t len)
+{
+ off_t offset = 0;
+
+ while (len > 0) {
+ int rc = send(sock, &buf[offset], len, 0);
+
+ if (rc == -1) {
+ switch (errno) {
+ case EAGAIN:
+ case EINTR:
+ break;
+ default:
+ return -1;
+ }
+ } else {
+ len -= rc;
+ offset += rc;
+ }
+ }
+
+ return 0;
+}
+
+static int
+sendstring(int sock, const char* buf, size_t len)
+{
+ if (sendall(sock, (char*) &len, sizeof(len)) == -1) {
+ return -1;
+ } else if (sendall(sock, buf, len) == -1) {
+ return -1;
+ }
+
+ return 0;
+}
+
+static int
+sendcmd(int sock, char cmd, const char* buf, size_t len)
+{
+ size_t total_len = len + 2;
+ char prefix[2] = {cmd, ':'};
+
+ if (sendall(sock, (char*) &total_len, sizeof(total_len)) == -1) {
+ return -1;
+ } else if (sendall(sock, prefix, sizeof(prefix)) == -1
+ || sendall(sock, buf, len) == -1)
+ {
+ return -1;
+ }
+
+ return 0;
+}
+
+static int
+recvall(int sock, char* buf, size_t len)
+{
+ off_t offset = 0;
+
+ while (len > 0) {
+ ssize_t rc = recv(sock, &buf[offset], len, 0);
+
+ if (rc == -1) {
+ switch (errno) {
+ case EAGAIN:
+ case EINTR:
+ break;
+ default:
+ return -1;
+ }
+ } else if (rc == 0) {
+ errno = EIO;
+ return -1;
+ } else {
+ len -= rc;
+ offset += rc;
+ }
+ }
+
+ return 0;
+}
+
+static ssize_t
+recvstring(int sock, char* buf, size_t len)
+{
+ ssize_t retval;
+
+ if (recvall(sock, (char*) &retval, sizeof(retval)) == -1) {
+ return -1;
+ } else if (retval > (ssize_t) len) {
+ return -1;
+ } else if (recvall(sock, buf, retval) == -1) {
+ return -1;
+ }
+
+ return retval;
+}
+
+char*
+readline_context::completion_generator(const char* text_in, int state)
+{
+ static std::vector<std::string> matches;
+
+ std::vector<std::string> long_matches;
+ char* retval = nullptr;
+
+ if (state == 0) {
+ auto text_str = std::string(text_in);
+
+ if (arg_needs_shlex) {
+ shlex arg_lexer(text_str);
+ std::map<std::string, std::string> scope;
+ std::string result;
+
+ if (arg_lexer.eval(result, scope)) {
+ text_str = result;
+ }
+ }
+
+ matches.clear();
+ if (arg_possibilities != nullptr) {
+ for (const auto& poss : (*arg_possibilities)) {
+ auto cmpfunc
+ = (loaded_context->is_case_sensitive() ? strncmp
+ : strncasecmp);
+ auto poss_str = poss.c_str();
+
+ // Check for an exact match and for the quoted version.
+ if (cmpfunc(text_str.c_str(), poss_str, text_str.length()) == 0
+ || ((strchr(loaded_context->rc_quote_chars, poss_str[0])
+ != nullptr)
+ && cmpfunc(text_str.c_str(),
+ &poss_str[1],
+ text_str.length())
+ == 0))
+ {
+ auto poss_slash_count
+ = std::count(poss.begin(), poss.end(), '/');
+
+ if (endswith(poss, "/")) {
+ poss_slash_count -= 1;
+ }
+ if (std::count(text_str.begin(), text_str.end(), '/')
+ == poss_slash_count)
+ {
+ matches.emplace_back(poss);
+ } else {
+ long_matches.emplace_back(poss);
+ }
+ }
+ }
+
+ if (matches.empty()) {
+ matches = std::move(long_matches);
+ }
+
+ if (matches.empty()) {
+ std::vector<std::pair<int, std::string>> fuzzy_matches;
+ std::vector<std::pair<int, std::string>> fuzzy_long_matches;
+
+ for (const auto& poss : (*arg_possibilities)) {
+ std::string poss_str = tolower(poss);
+ int score = 0;
+
+ if (fts::fuzzy_match(
+ text_str.c_str(), poss_str.c_str(), score)
+ && score > 0)
+ {
+ if (score <= 0) {
+ continue;
+ }
+
+ auto poss_slash_count
+ = std::count(poss_str.begin(), poss_str.end(), '/');
+ if (endswith(poss, "/")) {
+ poss_slash_count -= 1;
+ }
+ if (std::count(text_str.begin(), text_str.end(), '/')
+ == poss_slash_count)
+ {
+ fuzzy_matches.emplace_back(score, poss);
+ } else {
+ fuzzy_long_matches.emplace_back(score, poss);
+ }
+ }
+ }
+
+ if (fuzzy_matches.empty()) {
+ fuzzy_matches = std::move(fuzzy_long_matches);
+ }
+
+ if (!fuzzy_matches.empty()) {
+ stable_sort(
+ begin(fuzzy_matches),
+ end(fuzzy_matches),
+ [](auto l, auto r) { return r.first < l.first; });
+
+ int highest = fuzzy_matches[0].first;
+
+ for (const auto& pair : fuzzy_matches) {
+ if (highest - pair.first < FUZZY_PEER_THRESHOLD) {
+ matches.push_back(pair.second);
+ } else {
+ break;
+ }
+ }
+ }
+ }
+ }
+
+ if (matches.size() == 1) {
+ if (text_str == matches[0]) {
+ matches.pop_back();
+ }
+
+ last_match_str_valid = false;
+ if (sendstring(
+ child_this->rc_command_pipe[readline_curses::RCF_SLAVE],
+ "m:0:0:0",
+ 7)
+ == -1)
+ {
+ _exit(1);
+ }
+ }
+ }
+
+ if (!matches.empty()) {
+ retval = strdup(matches.back().c_str());
+ matches.pop_back();
+ }
+
+ return retval;
+}
+
+char**
+readline_context::attempted_completion(const char* text, int start, int end)
+{
+ char** retval = nullptr;
+
+ completion_start = start;
+ if (start == 0
+ && loaded_context->rc_possibilities.find("__command")
+ != loaded_context->rc_possibilities.end())
+ {
+ arg_possibilities = &loaded_context->rc_possibilities["__command"];
+ arg_needs_shlex = false;
+ rl_completion_append_character = loaded_context->rc_append_character;
+ } else {
+ char* space;
+ std::string cmd;
+ std::vector<std::string> prefix;
+ int point = rl_point;
+ while (point > 0 && rl_line_buffer[point] != ' ') {
+ point -= 1;
+ }
+ shlex lexer(rl_line_buffer, point);
+ std::map<std::string, std::string> scope;
+
+ arg_possibilities = nullptr;
+ rl_completion_append_character = 0;
+ if (lexer.split(prefix, scope)) {
+ auto prefix2
+ = fmt::format(FMT_STRING("{}"), fmt::join(prefix, "\x1f"));
+ auto prefix_iter = loaded_context->rc_prefixes.find(prefix2);
+
+ if (prefix_iter != loaded_context->rc_prefixes.end()) {
+ arg_possibilities
+ = &(loaded_context->rc_possibilities[prefix_iter->second]);
+ arg_needs_shlex = false;
+ }
+ }
+
+ if (arg_possibilities == nullptr) {
+ space = strchr(rl_line_buffer, ' ');
+ if (space == nullptr) {
+ space = rl_line_buffer + strlen(rl_line_buffer);
+ }
+ cmd = std::string(rl_line_buffer, space - rl_line_buffer);
+
+ auto iter = loaded_context->rc_prototypes.find(cmd);
+
+ if (iter == loaded_context->rc_prototypes.end()) {
+ if (loaded_context->rc_possibilities.find("*")
+ != loaded_context->rc_possibilities.end())
+ {
+ arg_possibilities = &loaded_context->rc_possibilities["*"];
+ arg_needs_shlex = false;
+ rl_completion_append_character
+ = loaded_context->rc_append_character;
+ }
+ } else {
+ std::vector<std::string>& proto
+ = loaded_context->rc_prototypes[cmd];
+
+ if (proto.empty()) {
+ arg_possibilities = nullptr;
+ } else if (proto[0] == "filename") {
+ shlex fn_lexer(rl_line_buffer, rl_point);
+ std::vector<std::string> fn_list;
+ int found = 0;
+
+ fn_lexer.split(fn_list, scope);
+
+ const auto& last_fn = fn_list.size() <= 1 ? ""
+ : fn_list.back();
+
+ if (last_fn.find(':') != std::string::npos) {
+ auto rp_iter = loaded_context->rc_possibilities.find(
+ "remote-path");
+ if (rp_iter != loaded_context->rc_possibilities.end()) {
+ for (const auto& poss : rp_iter->second) {
+ if (startswith(poss, last_fn.c_str())) {
+ found += 1;
+ }
+ }
+ if (found) {
+ arg_possibilities = &rp_iter->second;
+ arg_needs_shlex = false;
+ }
+ }
+ if (!found || (endswith(last_fn, "/") && found == 1)) {
+ char msg[2048];
+
+ snprintf(
+ msg, sizeof(msg), "\t:%s", last_fn.c_str());
+ sendstring(child_this->rc_command_pipe[1],
+ msg,
+ strlen(msg));
+ }
+ }
+ if (!found) {
+ static std::set<std::string> file_name_set;
+
+ file_name_set.clear();
+ auto_mem<char> completed_fn;
+ int fn_state = 0;
+ auto recent_netlocs_iter
+ = loaded_context->rc_possibilities.find(
+ "recent-netlocs");
+
+ if (recent_netlocs_iter
+ != loaded_context->rc_possibilities.end())
+ {
+ file_name_set.insert(
+ recent_netlocs_iter->second.begin(),
+ recent_netlocs_iter->second.end());
+ }
+ while ((completed_fn = rl_filename_completion_function(
+ last_fn.c_str(), fn_state))
+ != nullptr)
+ {
+ file_name_set.insert(completed_fn.in());
+ fn_state += 1;
+ }
+ arg_possibilities = &file_name_set;
+ arg_needs_shlex = true;
+ }
+ } else {
+ arg_possibilities
+ = &(loaded_context->rc_possibilities[proto[0]]);
+ arg_needs_shlex = false;
+ }
+ }
+ }
+ }
+
+ retval = rl_completion_matches(text, completion_generator);
+ if (retval == nullptr) {
+ rl_attempted_completion_over = 1;
+ }
+
+ return retval;
+}
+
+static int
+rubout_char_or_abort(int count, int key)
+{
+ if (rl_line_buffer[0] == '\0') {
+ rl_done = true;
+ got_abort = 1;
+ got_line = 0;
+ return 0;
+ } else {
+ return rl_rubout(count, '\b');
+ }
+}
+
+static int
+alt_done_func(int count, int key)
+{
+ alt_done = true;
+ rl_newline(count, key);
+ return 0;
+}
+
+int
+readline_context::command_complete(int count, int key)
+{
+ if (loaded_context->rc_possibilities.find("__command")
+ != loaded_context->rc_possibilities.end())
+ {
+ char* space = strchr(rl_line_buffer, ' ');
+
+ if (space == nullptr) {
+ return rl_menu_complete(count, key);
+ }
+ }
+ return rl_insert(count, key);
+}
+
+readline_context::readline_context(std::string name,
+ readline_context::command_map_t* commands,
+ bool case_sensitive)
+ : rc_name(std::move(name)), rc_case_sensitive(case_sensitive),
+ rc_quote_chars("\"'"), rc_highlighter(nullptr)
+{
+ if (commands != nullptr) {
+ command_map_t::iterator iter;
+
+ for (iter = commands->begin(); iter != commands->end(); ++iter) {
+ std::string cmd = iter->first;
+
+ this->rc_possibilities["__command"].insert(cmd);
+ iter->second->c_func(
+ INIT_EXEC_CONTEXT, cmd, this->rc_prototypes[cmd]);
+ }
+ }
+
+ memset(&this->rc_history, 0, sizeof(this->rc_history));
+ history_set_history_state(&this->rc_history);
+
+ auto config_dir = lnav::paths::dotlnav();
+ auto hpath = (config_dir / this->rc_name).string() + ".history";
+ read_history(hpath.c_str());
+ this->save();
+
+ this->rc_append_character = ' ';
+}
+
+void
+readline_context::load()
+{
+ char buffer[128];
+
+ rl_completer_word_break_characters = (char*) " \t\n|()"; /* XXX */
+ /*
+ * XXX Need to keep the input on a single line since the display screws
+ * up if it wraps around.
+ */
+ snprintf(buffer,
+ sizeof(buffer),
+ "set completion-ignore-case %s",
+ this->rc_case_sensitive ? "off" : "on");
+ rl_parse_and_bind(buffer); /* NOTE: buffer is modified */
+
+ loaded_context = this;
+ rl_attempted_completion_function = attempted_completion;
+ history_set_history_state(&this->rc_history);
+ for (auto& rc_var : this->rc_vars) {
+ *(rc_var.rv_dst.ch) = (char*) rc_var.rv_val.ch;
+ }
+}
+
+void
+readline_context::save()
+{
+ HISTORY_STATE* hs = history_get_history_state();
+
+ this->rc_history = *hs;
+ free(hs);
+ hs = nullptr;
+}
+
+readline_curses::readline_curses(
+ std::shared_ptr<pollable_supervisor> supervisor)
+ : pollable(supervisor, pollable::category::interactive),
+ rc_focus(noop_func{}), rc_change(noop_func{}), rc_perform(noop_func{}),
+ rc_alt_perform(noop_func{}), rc_timeout(noop_func{}),
+ rc_abort(noop_func{}), rc_display_match(noop_func{}),
+ rc_display_next(noop_func{}), rc_blur(noop_func{}),
+ rc_completion_request(noop_func{})
+{
+}
+
+readline_curses::~readline_curses()
+{
+ this->rc_pty[RCF_MASTER].reset();
+ this->rc_command_pipe[RCF_MASTER].reset();
+ if (this->rc_child == 0) {
+ _exit(0);
+ } else if (this->rc_child > 0) {
+ int status;
+
+ log_debug("terminating readline child %d", this->rc_child);
+ log_perror(kill(this->rc_child, SIGTERM));
+ this->rc_child = -1;
+
+ while (wait(&status) < 0 && (errno == EINTR)) {
+ ;
+ }
+ log_debug(" child %d has exited", this->rc_child);
+ }
+}
+
+void
+readline_curses::store_matches(char** matches, int num_matches, int max_len)
+{
+ static int match_index = 0;
+ char msg[64];
+ int rc;
+
+ max_len = 0;
+ for (int lpc = 0; lpc <= num_matches; lpc++) {
+ max_len = std::max(max_len, (int) strlen(matches[lpc]));
+ }
+
+ if (last_match_str_valid && strcmp(last_match_str.c_str(), matches[0]) == 0)
+ {
+ match_index += 1;
+ rc = snprintf(msg, sizeof(msg), "n:%d", match_index);
+
+ if (sendstring(child_this->rc_command_pipe[RCF_SLAVE], msg, rc) == -1) {
+ _exit(1);
+ }
+ } else {
+ match_index = 0;
+ rc = snprintf(msg,
+ sizeof(msg),
+ "m:%d:%d:%d",
+ completion_start,
+ num_matches,
+ max_len);
+ if (sendstring(child_this->rc_command_pipe[RCF_SLAVE], msg, rc) == -1) {
+ _exit(1);
+ }
+ for (int lpc = 1; lpc <= num_matches; lpc++) {
+ if (sendstring(child_this->rc_command_pipe[RCF_SLAVE],
+ matches[lpc],
+ strlen(matches[lpc]))
+ == -1)
+ {
+ _exit(1);
+ }
+ }
+
+ last_match_str = matches[0];
+ last_match_str_valid = true;
+ }
+}
+
+void
+readline_curses::start()
+{
+ if (this->rc_child > 0) {
+ return;
+ }
+
+ struct winsize ws;
+ int sp[2];
+
+ if (socketpair(PF_UNIX, SOCK_STREAM, 0, sp) < 0) {
+ throw error(errno);
+ }
+
+ this->rc_command_pipe[RCF_MASTER] = sp[RCF_MASTER];
+ this->rc_command_pipe[RCF_SLAVE] = sp[RCF_SLAVE];
+
+ if (ioctl(STDOUT_FILENO, TIOCGWINSZ, &ws) == -1) {
+ throw error(errno);
+ }
+
+ if (openpty(this->rc_pty[RCF_MASTER].out(),
+ this->rc_pty[RCF_SLAVE].out(),
+ nullptr,
+ nullptr,
+ &ws)
+ < 0)
+ {
+ perror("error: failed to open terminal(openpty)");
+ throw error(errno);
+ }
+
+ if ((this->rc_child = fork()) == -1) {
+ throw error(errno);
+ }
+
+ if (this->rc_child != 0) {
+ this->rc_command_pipe[RCF_SLAVE].reset();
+ this->rc_pty[RCF_SLAVE].reset();
+ return;
+ }
+
+ {
+ char buffer[1024];
+
+ this->rc_command_pipe[RCF_MASTER].reset();
+ this->rc_pty[RCF_MASTER].reset();
+
+ signal(SIGALRM, sigalrm);
+ signal(SIGWINCH, sigwinch);
+ signal(SIGINT, sigterm);
+ signal(SIGTERM, sigterm);
+
+ dup2(this->rc_pty[RCF_SLAVE], STDIN_FILENO);
+ dup2(this->rc_pty[RCF_SLAVE], STDOUT_FILENO);
+
+ setenv("TERM", "vt52", 1);
+
+ rl_initialize();
+ using_history();
+ stifle_history(HISTORY_SIZE);
+
+ rl_add_defun("rubout-char-or-abort", rubout_char_or_abort, '\b');
+ rl_add_defun("alt-done", alt_done_func, '\x0a');
+ // rl_add_defun("command-complete", readline_context::command_complete,
+ // ' ');
+
+ for (const auto* init_cmd : RL_INIT) {
+ snprintf(buffer, sizeof(buffer), "%s", init_cmd);
+ rl_parse_and_bind(buffer); /* NOTE: buffer is modified */
+ }
+
+ child_this = this;
+ }
+
+ std::map<int, readline_context*>::iterator current_context;
+ int maxfd;
+
+ require(!this->rc_contexts.empty());
+
+ rl_completion_display_matches_hook = store_matches;
+
+ current_context = this->rc_contexts.end();
+
+ maxfd = std::max(STDIN_FILENO, this->rc_command_pipe[RCF_SLAVE].get());
+
+ while (looping) {
+ fd_set ready_rfds;
+ int rc;
+
+ FD_ZERO(&ready_rfds);
+ if (current_context != this->rc_contexts.end()) {
+ FD_SET(STDIN_FILENO, &ready_rfds);
+ }
+ FD_SET(this->rc_command_pipe[RCF_SLAVE], &ready_rfds);
+
+ rc = select(maxfd + 1, &ready_rfds, nullptr, nullptr, nullptr);
+ if (rc < 0) {
+ switch (errno) {
+ case EINTR:
+ break;
+ }
+ } else {
+ if (FD_ISSET(STDIN_FILENO, &ready_rfds)) {
+ static uint64_t last_h1, last_h2;
+
+ struct itimerval itv;
+
+ itv.it_value.tv_sec = 0;
+ itv.it_value.tv_usec = KEY_TIMEOUT;
+ itv.it_interval.tv_sec = 0;
+ itv.it_interval.tv_usec = 0;
+ setitimer(ITIMER_REAL, &itv, nullptr);
+
+ rl_callback_read_char();
+ if (RL_ISSTATE(RL_STATE_DONE) && !got_line) {
+ got_line = 1;
+ this->line_ready("");
+ rl_callback_handler_remove();
+ } else {
+ uint64_t h1 = 1, h2 = 2;
+
+ if (rl_last_func == readline_context::command_complete) {
+ rl_last_func = rl_menu_complete;
+ }
+
+ bool complete_done
+ = (rl_last_func != rl_menu_complete
+ && rl_last_func != rl_backward_menu_complete);
+
+ if (complete_done) {
+ last_match_str_valid = false;
+ } else if (rewrite_line_start
+ && !startswith(rl_line_buffer,
+ rewrite_line_start->c_str()))
+ {
+ // If the line was rewritten, the extra text stays on
+ // the screen, so we need to delete it, make sure the
+ // append character is there, and redisplay. For
+ // example, ':co<TAB>' will complete ':comment' and
+ // append the current comment. Pressing '<TAB>' again
+ // would switch to ':config' and the comment text would
+ // be left on the display.
+ rl_delete_text(rl_point, rl_end);
+ if (rl_completion_append_character
+ && rl_line_buffer[rl_point]
+ != rl_completion_append_character)
+ {
+ char buf[2]
+ = {(char) rl_completion_append_character, '\0'};
+
+ rl_insert_text(buf);
+ }
+ rl_redisplay();
+ }
+ rewrite_line_start = nonstd::nullopt;
+
+ SpookyHash::Hash128(rl_line_buffer, rl_end, &h1, &h2);
+
+ if (h1 == last_h1 && h2 == last_h2) {
+ // do nothing
+ } else if (sendcmd(this->rc_command_pipe[RCF_SLAVE],
+ complete_done ? 'l' : 'c',
+ rl_line_buffer,
+ rl_end)
+ != 0)
+ {
+ perror("line: write failed");
+ _exit(1);
+ }
+ last_h1 = h1;
+ last_h2 = h2;
+ if (sendcmd(this->rc_command_pipe[RCF_SLAVE], 'w', "", 0)
+ != 0)
+ {
+ perror("line: write failed");
+ _exit(1);
+ }
+ }
+ }
+ if (FD_ISSET(this->rc_command_pipe[RCF_SLAVE], &ready_rfds)) {
+ char msg[1024 + 1];
+
+ if ((rc = recvstring(this->rc_command_pipe[RCF_SLAVE],
+ msg,
+ sizeof(msg) - 1))
+ < 0)
+ {
+ looping = false;
+ } else {
+ int context, prompt_start = 0;
+ char type[1024];
+
+ msg[rc] = '\0';
+ if (sscanf(msg, "i:%d:%n", &rl_point, &prompt_start) == 1) {
+ const char* initial = &msg[prompt_start];
+
+ rl_extend_line_buffer(strlen(initial) + 1);
+ strcpy(rl_line_buffer, initial);
+ rl_end = strlen(initial);
+ rewrite_line_start
+ = std::string(rl_line_buffer, rl_point);
+ rl_redisplay();
+ if (sendcmd(this->rc_command_pipe[RCF_SLAVE],
+ 'c',
+ rl_line_buffer,
+ rl_end)
+ != 0)
+ {
+ perror("line: write failed");
+ _exit(1);
+ }
+ } else if (sscanf(msg, "f:%d:%n", &context, &prompt_start)
+ == 1
+ && prompt_start != 0
+ && (current_context
+ = this->rc_contexts.find(context))
+ != this->rc_contexts.end())
+ {
+ got_abort = 0;
+ current_context->second->load();
+ rl_callback_handler_install(&msg[prompt_start],
+ line_ready_tramp);
+ last_match_str_valid = false;
+ if (sendcmd(this->rc_command_pipe[RCF_SLAVE],
+ 'l',
+ rl_line_buffer,
+ rl_end)
+ != 0)
+ {
+ perror("line: write failed");
+ _exit(1);
+ }
+ if (sendcmd(
+ this->rc_command_pipe[RCF_SLAVE], 'w', "", 0)
+ != 0)
+ {
+ perror("line: write failed");
+ _exit(1);
+ }
+ } else if (strcmp(msg, "a") == 0) {
+ char reply[4];
+
+ rl_done = 1;
+ got_timeout = 0;
+ got_line = 1;
+ rl_callback_handler_remove();
+
+ snprintf(reply, sizeof(reply), "a");
+
+ if (sendstring(this->rc_command_pipe[RCF_SLAVE],
+ reply,
+ strlen(reply))
+ == -1)
+ {
+ perror("abort: write failed");
+ _exit(1);
+ }
+ } else if (sscanf(msg,
+ "apre:%d:%1023[^\x1d]\x1d%n",
+ &context,
+ type,
+ &prompt_start)
+ == 2)
+ {
+ require(this->rc_contexts[context] != nullptr);
+
+ this->rc_contexts[context]
+ ->rc_prefixes[std::string(type)]
+ = std::string(&msg[prompt_start]);
+ } else if (sscanf(msg,
+ "ap:%d:%31[^:]:%n",
+ &context,
+ type,
+ &prompt_start)
+ == 2)
+ {
+ require(this->rc_contexts[context] != nullptr);
+
+ this->rc_contexts[context]->add_possibility(
+ std::string(type), std::string(&msg[prompt_start]));
+ if (rl_last_func == rl_complete
+ || rl_last_func == rl_menu_complete)
+ {
+ rl_last_func = NULL;
+ }
+ } else if (sscanf(msg,
+ "rp:%d:%31[^:]:%n",
+ &context,
+ type,
+ &prompt_start)
+ == 2)
+ {
+ require(this->rc_contexts[context] != nullptr);
+
+ this->rc_contexts[context]->rem_possibility(
+ std::string(type), std::string(&msg[prompt_start]));
+ } else if (sscanf(msg, "cpre:%d", &context) == 1) {
+ this->rc_contexts[context]->rc_prefixes.clear();
+ } else if (sscanf(msg, "cp:%d:%s", &context, type)) {
+ this->rc_contexts[context]->clear_possibilities(type);
+ } else {
+ log_error("unhandled message: %s", msg);
+ }
+ }
+ }
+ }
+
+ if (got_timeout) {
+ got_timeout = 0;
+ if (sendcmd(this->rc_command_pipe[RCF_SLAVE],
+ 't',
+ rl_line_buffer,
+ rl_end)
+ == -1)
+ {
+ _exit(1);
+ }
+ }
+ if (got_line) {
+ struct itimerval itv;
+
+ got_line = 0;
+ itv.it_value.tv_sec = 0;
+ itv.it_value.tv_usec = 0;
+ itv.it_interval.tv_sec = 0;
+ itv.it_interval.tv_usec = 0;
+ if (setitimer(ITIMER_REAL, &itv, nullptr) < 0) {
+ log_error("setitimer: %s", strerror(errno));
+ }
+ if (current_context != this->rc_contexts.end()) {
+ current_context->second->save();
+ current_context = this->rc_contexts.end();
+ }
+ }
+ if (got_winch) {
+ struct winsize new_ws;
+
+ if (ioctl(STDOUT_FILENO, TIOCGWINSZ, &new_ws) == -1) {
+ throw error(errno);
+ }
+ got_winch = 0;
+ rl_set_screen_size(new_ws.ws_row, new_ws.ws_col);
+ }
+ }
+
+ if (this->rc_save_history) {
+ auto config_dir = lnav::paths::dotlnav();
+ for (auto& pair : this->rc_contexts) {
+ pair.second->load();
+
+ auto hpath
+ = (config_dir / pair.second->get_name()).string() + ".history";
+ write_history(hpath.c_str());
+ pair.second->save();
+ }
+ }
+
+ _exit(0);
+}
+
+void
+readline_curses::line_ready(const char* line)
+{
+ auto_mem<char> expanded;
+ char msg[1024] = {0};
+ int rc;
+ const char* cmd_ch = alt_done ? "D" : "d";
+
+ alt_done = false;
+ if (got_abort || line == nullptr) {
+ snprintf(msg, sizeof(msg), "a");
+
+ if (sendstring(this->rc_command_pipe[RCF_SLAVE], msg, strlen(msg))
+ == -1)
+ {
+ perror("abort: write failed");
+ _exit(1);
+ }
+ return;
+ }
+
+ if (rl_line_buffer[0] == '^') {
+ rc = -1;
+ } else {
+ rc = history_expand(rl_line_buffer, expanded.out());
+ }
+ switch (rc) {
+#if 0
+ /* TODO: fix clash between history and pcre metacharacters */
+ case -1:
+ /* XXX */
+ snprintf(msg, sizeof(msg),
+ "e:unable to expand history -- %s",
+ expanded.in());
+ break;
+#endif
+
+ case -1:
+ snprintf(msg, sizeof(msg), "%s:%s", cmd_ch, line);
+ break;
+
+ case 0:
+ case 1:
+ case 2: /* XXX */
+ snprintf(msg, sizeof(msg), "%s:%s", cmd_ch, expanded.in());
+ break;
+ }
+
+ if (sendstring(this->rc_command_pipe[RCF_SLAVE], msg, strlen(msg)) == -1) {
+ perror("line_ready: write failed");
+ _exit(1);
+ }
+
+ {
+ HIST_ENTRY* entry;
+
+ if (line[0] != '\0'
+ && (history_length == 0
+ || (entry = history_get(history_base + history_length - 1))
+ == nullptr
+ || strcmp(entry->line, line) != 0))
+ {
+ add_history(line);
+ }
+ }
+}
+
+void
+readline_curses::check_poll_set(const std::vector<struct pollfd>& pollfds)
+{
+ int rc;
+
+ if (pollfd_ready(pollfds, this->rc_pty[RCF_MASTER])) {
+ char buffer[128];
+
+ rc = read(this->rc_pty[RCF_MASTER], buffer, sizeof(buffer));
+ if (rc > 0) {
+ int old_x = this->vc_x;
+
+ this->map_output(buffer, rc);
+ if (this->vc_x != old_x) {
+ this->rc_change(this);
+ }
+ }
+ }
+ if (pollfd_ready(pollfds, this->rc_command_pipe[RCF_MASTER])) {
+ char msg[1024 + 1];
+
+ rc = recvstring(
+ this->rc_command_pipe[RCF_MASTER], msg, sizeof(msg) - 1);
+ if (rc >= 0) {
+ msg[rc] = '\0';
+ if (this->rc_matches_remaining > 0) {
+ this->rc_matches.emplace_back(msg);
+ this->rc_matches_remaining -= 1;
+ if (this->rc_matches_remaining == 0) {
+ this->rc_display_match(this);
+ }
+ } else if (msg[0] == 'm') {
+ if (sscanf(msg,
+ "m:%d:%d:%d",
+ &this->rc_match_start,
+ &this->rc_matches_remaining,
+ &this->rc_max_match_length)
+ != 3)
+ {
+ require(0);
+ }
+ this->rc_matches.clear();
+ if (this->rc_matches_remaining == 0) {
+ this->rc_display_match(this);
+ }
+ this->rc_match_index = 0;
+ } else if (msg[0] == '\t') {
+ char path[2048];
+
+ if (sscanf(msg, "\t:%s", path) != 1) {
+ require(0);
+ }
+ this->rc_remote_complete_path = path;
+ this->rc_completion_request(this);
+ } else if (msg[0] == 'n') {
+ if (sscanf(msg, "n:%d", &this->rc_match_index) != 1) {
+ require(0);
+ }
+ this->rc_display_next(this);
+ } else {
+ switch (msg[0]) {
+ case 't':
+ case 'd':
+ case 'D':
+ this->rc_value = std::string(&msg[2]);
+ break;
+ }
+ switch (msg[0]) {
+ case 'a':
+ curs_set(0);
+ this->vc_line.clear();
+ this->rc_active_context = -1;
+ this->rc_matches.clear();
+ this->rc_abort(this);
+ this->rc_display_match(this);
+ this->rc_blur(this);
+ break;
+
+ case 't':
+ this->rc_timeout(this);
+ break;
+
+ case 'd':
+ case 'D':
+ curs_set(0);
+ this->rc_active_context = -1;
+ this->rc_matches.clear();
+ if (msg[0] == 'D' || this->rc_is_alt_focus) {
+ this->rc_alt_perform(this);
+ } else {
+ this->rc_perform(this);
+ }
+ this->rc_display_match(this);
+ this->rc_blur(this);
+ break;
+
+ case 'l':
+ this->rc_line_buffer = &msg[2];
+ if (this->rc_active_context != -1) {
+ this->rc_change(this);
+ }
+ this->rc_matches.clear();
+ if (this->rc_active_context != -1) {
+ this->rc_display_match(this);
+ }
+ break;
+
+ case 'c':
+ this->rc_line_buffer = &msg[2];
+ this->rc_change(this);
+ this->rc_display_match(this);
+ break;
+ case 'w':
+ this->rc_ready_for_input = true;
+ break;
+ }
+ }
+ }
+ }
+}
+
+void
+readline_curses::handle_key(int ch)
+{
+ const char* bch;
+ int len;
+
+ bch = this->map_input(ch, len);
+ if (write(this->rc_pty[RCF_MASTER], bch, len) == -1) {
+ perror("handle_key: write failed");
+ }
+}
+
+void
+readline_curses::focus(int context,
+ const std::string& prompt,
+ const std::string& initial)
+{
+ char buffer[1024];
+
+ curs_set(1);
+
+ this->rc_active_context = context;
+
+ snprintf(buffer, sizeof(buffer), "f:%d:%s", context, prompt.c_str());
+ if (sendstring(
+ this->rc_command_pipe[RCF_MASTER], buffer, strlen(buffer) + 1)
+ == -1)
+ {
+ perror("focus: write failed");
+ }
+ wmove(this->vc_window, this->get_actual_y(), this->vc_left);
+ wclrtoeol(this->vc_window);
+ if (!initial.empty()) {
+ this->rewrite_line(initial.size(), initial);
+ }
+ this->rc_is_alt_focus = false;
+ this->rc_focus(this);
+}
+
+void
+readline_curses::rewrite_line(int pos, const std::string& value)
+{
+ char buffer[1024];
+
+ snprintf(buffer, sizeof(buffer), "i:%d:%s", pos, value.c_str());
+ if (sendstring(
+ this->rc_command_pipe[RCF_MASTER], buffer, strlen(buffer) + 1)
+ == -1)
+ {
+ perror("rewrite_line: write failed");
+ }
+}
+
+void
+readline_curses::abort()
+{
+ char buffer[1024];
+
+ this->vc_x = 0;
+ snprintf(buffer, sizeof(buffer), "a");
+ if (sendstring(this->rc_command_pipe[RCF_MASTER], buffer, strlen(buffer))
+ == -1)
+ {
+ perror("abort: write failed");
+ }
+}
+
+void
+readline_curses::add_prefix(int context,
+ const std::vector<std::string>& prefix,
+ const std::string& value)
+{
+ char buffer[1024];
+ auto prefix_wire = fmt::format(FMT_STRING("{}"), fmt::join(prefix, "\x1f"));
+
+ snprintf(buffer,
+ sizeof(buffer),
+ "apre:%d:%s\x1d%s",
+ context,
+ prefix_wire.c_str(),
+ value.c_str());
+ if (sendstring(
+ this->rc_command_pipe[RCF_MASTER], buffer, strlen(buffer) + 1)
+ == -1)
+ {
+ perror("add_prefix: write failed");
+ }
+}
+
+void
+readline_curses::clear_prefixes(int context)
+{
+ char buffer[1024];
+
+ snprintf(buffer, sizeof(buffer), "cpre:%d", context);
+ if (sendstring(
+ this->rc_command_pipe[RCF_MASTER], buffer, strlen(buffer) + 1)
+ == -1)
+ {
+ perror("add_possibility: write failed");
+ }
+}
+
+void
+readline_curses::add_possibility(int context,
+ const std::string& type,
+ const std::string& value)
+{
+ char buffer[1024];
+
+ if (value.empty()) {
+ return;
+ }
+
+ snprintf(buffer,
+ sizeof(buffer),
+ "ap:%d:%s:%s",
+ context,
+ type.c_str(),
+ value.c_str());
+ if (sendstring(
+ this->rc_command_pipe[RCF_MASTER], buffer, strlen(buffer) + 1)
+ == -1)
+ {
+ perror("add_possibility: write failed");
+ }
+}
+
+void
+readline_curses::rem_possibility(int context,
+ const std::string& type,
+ const std::string& value)
+{
+ char buffer[1024];
+
+ snprintf(buffer,
+ sizeof(buffer),
+ "rp:%d:%s:%s",
+ context,
+ type.c_str(),
+ value.c_str());
+ if (sendstring(
+ this->rc_command_pipe[RCF_MASTER], buffer, strlen(buffer) + 1)
+ == -1)
+ {
+ perror("rem_possiblity: write failed");
+ }
+}
+
+void
+readline_curses::clear_possibilities(int context, std::string type)
+{
+ char buffer[1024];
+
+ snprintf(buffer, sizeof(buffer), "cp:%d:%s", context, type.c_str());
+ if (sendstring(
+ this->rc_command_pipe[RCF_MASTER], buffer, strlen(buffer) + 1)
+ == -1)
+ {
+ perror("clear_possiblity: write failed");
+ }
+}
+
+void
+readline_curses::do_update()
+{
+ if (!this->vc_visible || this->vc_window == nullptr) {
+ return;
+ }
+
+ if (this->rc_active_context == -1) {
+ int alt_start = -1;
+ struct line_range lr(0, 0);
+ attr_line_t alt_al;
+ view_colors& vc = view_colors::singleton();
+
+ wmove(this->vc_window, this->get_actual_y(), this->vc_left);
+ auto attrs = vc.attrs_for_role(role_t::VCR_TEXT);
+ wattr_set(this->vc_window,
+ attrs.ta_attrs,
+ vc.ensure_color_pair(attrs.ta_fg_color, attrs.ta_bg_color),
+ nullptr);
+ whline(this->vc_window, ' ', this->vc_width);
+
+ if (time(nullptr) > this->rc_value_expiration) {
+ this->rc_value.clear();
+ }
+
+ if (!this->rc_alt_value.empty()) {
+ alt_al.get_string() = this->rc_alt_value;
+ scrub_ansi_string(alt_al.get_string(), &alt_al.get_attrs());
+
+ alt_start = getmaxx(this->vc_window) - alt_al.get_string().size();
+ }
+
+ if (alt_start >= (int) (this->rc_value.length() + 5)) {
+ lr.lr_end = alt_al.get_string().length();
+ view_curses::mvwattrline(
+ this->vc_window, this->get_actual_y(), alt_start, alt_al, lr);
+ }
+
+ lr.lr_end = this->rc_value.length();
+ view_curses::mvwattrline(this->vc_window,
+ this->get_actual_y(),
+ this->vc_left,
+ this->rc_value,
+ lr);
+ this->set_x(0);
+ }
+
+ if (this->rc_active_context != -1) {
+ readline_context* rc = this->rc_contexts[this->rc_active_context];
+ readline_highlighter_t hl = rc->get_highlighter();
+ attr_line_t al = this->vc_line;
+
+ if (hl != nullptr) {
+ hl(al, this->vc_left + this->vc_x);
+ }
+ view_curses::mvwattrline(this->vc_window,
+ this->get_actual_y(),
+ this->vc_left,
+ al,
+ line_range{0, (int) this->vc_width});
+
+ wmove(
+ this->vc_window, this->get_actual_y(), this->vc_left + this->vc_x);
+ }
+}
+
+std::string
+readline_curses::get_match_string() const
+{
+ auto len = std::min((size_t) this->vc_x, this->rc_line_buffer.size())
+ - this->rc_match_start;
+ auto* context = this->get_active_context();
+
+ if (context->get_append_character() != 0) {
+ if (this->rc_line_buffer.length() > (this->rc_match_start + len - 1)
+ && this->rc_line_buffer[this->rc_match_start + len - 1]
+ == context->get_append_character())
+ {
+ len -= 1;
+ } else if (this->rc_line_buffer.length()
+ > (this->rc_match_start + len - 2)
+ && this->rc_line_buffer[this->rc_match_start + len - 2]
+ == context->get_append_character())
+ {
+ len -= 2;
+ }
+ }
+
+ return this->rc_line_buffer.substr(this->rc_match_start, len);
+}
+
+void
+readline_curses::set_value(const std::string& value)
+{
+ this->set_attr_value(attr_line_t::from_ansi_str(value.c_str()));
+}
+
+void
+readline_curses::set_attr_value(const attr_line_t& value)
+{
+ this->rc_value = value;
+ if (this->rc_value.length() > 1024) {
+ this->rc_value = this->rc_value.subline(0, 1024);
+ }
+ this->rc_value_expiration = time(nullptr) + VALUE_EXPIRATION;
+ this->set_needs_update();
+}
+
+void
+readline_curses::update_poll_set(std::vector<struct pollfd>& pollfds)
+{
+ if (this->rc_pty[RCF_MASTER] != -1) {
+ pollfds.push_back((struct pollfd){this->rc_pty[RCF_MASTER], POLLIN, 0});
+ }
+ if (this->rc_command_pipe[RCF_MASTER] != -1) {
+ pollfds.push_back(
+ (struct pollfd){this->rc_command_pipe[RCF_MASTER], POLLIN, 0});
+ }
+}