backport the latest fish_key_reader from master

This includes the improvements to `fish_key_reader` as of commit
68e167d576 on the master branch. This makes
the program much friendlier to users.
This commit is contained in:
Kurtis Rader 2016-06-30 21:21:10 -07:00
parent 81dee16d69
commit c429a585e4
3 changed files with 234 additions and 87 deletions

View file

@ -6,16 +6,17 @@
// carriage-return (\cM) and newline (\cJ). // carriage-return (\cM) and newline (\cJ).
// //
// Type "exit" or "quit" to terminate the program. // Type "exit" or "quit" to terminate the program.
#include "config.h" // IWYU pragma: keep
#include <errno.h>
#include <getopt.h> #include <getopt.h>
#include <locale.h>
#include <signal.h> #include <signal.h>
#include <stdbool.h>
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
#include <sys/time.h> #include <unistd.h>
#include <wchar.h>
#include <wctype.h> #include <wctype.h>
#include <string>
#include "common.h" #include "common.h"
#include "env.h" #include "env.h"
@ -24,28 +25,34 @@
#include "input_common.h" #include "input_common.h"
#include "proc.h" #include "proc.h"
#include "reader.h" #include "reader.h"
#include "signal.h"
#include "wutil.h" // IWYU pragma: keep
struct config_paths_t determine_config_directory_paths(const char *argv0); struct config_paths_t determine_config_directory_paths(const char *argv0);
static long long int prev_tstamp = 0; static const char *ctrl_symbolic_names[] = {NULL, NULL, NULL, NULL, NULL, NULL, NULL, "\\a",
static const char *ctrl_equivalents[] = {NULL, NULL, NULL, NULL, NULL, NULL, NULL, "\\a",
"\\b", "\\t", "\\n", "\\v", "\\f", "\\r", NULL, NULL, "\\b", "\\t", "\\n", "\\v", "\\f", "\\r", NULL, NULL,
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL,
NULL, NULL, NULL, "\\e", NULL, NULL, NULL, NULL}; NULL, NULL, NULL, "\\e", NULL, NULL, NULL, NULL};
static bool keep_running = true;
/// Return true if the recent sequence of characters indicates the user wants to exit the program. /// Return true if the recent sequence of characters indicates the user wants to exit the program.
bool should_exit(unsigned char c) { static bool should_exit(wchar_t wc) {
unsigned char c = wc < 0x80 ? wc : 0;
static unsigned char recent_chars[4] = {0}; static unsigned char recent_chars[4] = {0};
recent_chars[0] = recent_chars[1]; recent_chars[0] = recent_chars[1];
recent_chars[1] = recent_chars[2]; recent_chars[1] = recent_chars[2];
recent_chars[2] = recent_chars[3]; recent_chars[2] = recent_chars[3];
recent_chars[3] = c; recent_chars[3] = c;
return memcmp(recent_chars, "exit", 4) == 0 || memcmp(recent_chars, "quit", 4) == 0; return (memcmp(recent_chars, "exit", 4) == 0 || memcmp(recent_chars, "quit", 4) == 0 ||
memcmp(recent_chars + 2, "\x3\x3", 2) == 0 || // ctrl-C, ctrl-C
memcmp(recent_chars + 2, "\x4\x4", 2) == 0); // ctrl-D, ctrl-D
} }
/// Return the key name if the recent sequence of characters matches a known terminfo sequence. /// Return the key name if the recent sequence of characters matches a known terminfo sequence.
char *const key_name(unsigned char c) { static char *const key_name(wchar_t wc) {
unsigned char c = wc < 0x80 ? wc : 0;
static char recent_chars[8] = {0}; static char recent_chars[8] = {0};
recent_chars[0] = recent_chars[1]; recent_chars[0] = recent_chars[1];
@ -69,63 +76,138 @@ char *const key_name(unsigned char c) {
return NULL; return NULL;
} }
/// Process the characters we receive as the user presses keys. /// Return true if the character must be escaped in the sequence of chars to be bound in `bind`
void process_input(bool continuous_mode) { /// command.
bool first_char_seen = false; static bool must_escape(wchar_t wc) {
while (true) { switch (wc) {
wchar_t wc = input_common_readch(first_char_seen && !continuous_mode); case '[':
if (wc == WEOF) { case ']':
return; case '(':
case ')':
case '<':
case '>':
case '{':
case '}':
case '*':
case '\\':
case '?':
case '$':
case '#':
case ';':
case '&':
case '|':
case '\'':
case '"':
return true;
default:
return false;
} }
if (wc > 255) {
printf("\nUnexpected wide character from input_common_readch(): %lld / 0x%llx\n",
(long long)wc, (long long)wc);
return;
} }
long long int curr_tstamp, delta_tstamp; static char *char_to_symbol(wchar_t wc, bool bind_friendly) {
timeval char_tstamp; static char buf[128];
gettimeofday(&char_tstamp, NULL);
curr_tstamp = char_tstamp.tv_sec * 1000000 + char_tstamp.tv_usec;
delta_tstamp = curr_tstamp - prev_tstamp;
if (delta_tstamp >= 1000000) delta_tstamp = 999999;
if (delta_tstamp >= 200000 && continuous_mode) {
printf("\n");
printf("Type 'exit' or 'quit' to terminate this program.\n");
printf("\n");
}
prev_tstamp = curr_tstamp;
unsigned char c = wc; if (wc < ' ') {
if (c < 32) { // ASCII control character.
// Control characters. if (ctrl_symbolic_names[wc]) {
if (ctrl_equivalents[c]) { if (bind_friendly) {
printf("%6lld usec dec: %3u hex: %2x char: %s (aka \\c%c)\n", delta_tstamp, c, c, snprintf(buf, sizeof(buf), "%s", ctrl_symbolic_names[wc]);
ctrl_equivalents[c], c + 64);
} else { } else {
printf("%6lld usec dec: %3u hex: %2x char: \\c%c\n", delta_tstamp, c, c, c + 64); snprintf(buf, sizeof(buf), "\\c%c (or %s)", wc + 64, ctrl_symbolic_names[wc]);
} }
} else if (c == 32) { } else {
// The space character. snprintf(buf, sizeof(buf), "\\c%c", wc + 64);
printf("%6lld usec dec: %3u hex: %2x char: <space>\n", delta_tstamp, c, c); }
} else if (c == 127) { } else if (wc == ' ') {
// The "space" character.
snprintf(buf, sizeof(buf), "\\x%X (aka \"space\")", wc);
} else if (wc == 0x7F) {
// The "del" character. // The "del" character.
printf("%6lld usec dec: %3u hex: %2x char: \\x7f (aka del)\n", delta_tstamp, c, c); snprintf(buf, sizeof(buf), "\\x%X (aka \"del\")", wc);
} else if (c >= 128) { } else if (wc < 0x80) {
// Non-ASCII characters (i.e., those with bit 7 set).
printf("%6lld usec dec: %3u hex: %2x char: non-ASCII\n", delta_tstamp, c, c);
} else {
// ASCII characters that are not control characters. // ASCII characters that are not control characters.
printf("%6lld usec dec: %3u hex: %2x char: %c\n", delta_tstamp, c, c, c); if (bind_friendly && must_escape(wc)) {
snprintf(buf, sizeof(buf), "\\%c", wc);
} else {
snprintf(buf, sizeof(buf), "%c", wc);
}
} else if (wc <= 0xFFFF) {
snprintf(buf, sizeof(buf), "\\u%04X", wc);
} else {
snprintf(buf, sizeof(buf), "\\U%06X", wc);
}
return buf;
} }
char *const name = key_name(c); static void add_char_to_bind_command(wchar_t wc, std::vector<wchar_t> &bind_chars) {
bind_chars.push_back(wc);
}
static void output_bind_command(std::vector<wchar_t> &bind_chars) {
if (bind_chars.size()) {
fputs("bind ", stdout);
for (int i = 0; i < bind_chars.size(); i++) {
fputs(char_to_symbol(bind_chars[i], true), stdout);
}
fputs(" 'do something'\n", stdout);
bind_chars.clear();
}
}
static void output_info_about_char(wchar_t wc) {
printf("hex: %4X char: %s\n", wc, char_to_symbol(wc, false));
}
static bool output_matching_key_name(wchar_t wc) {
char *name = key_name(wc);
if (name) { if (name) {
printf("FYI: Saw sequence for bind key name \"%s\"\n", name); printf("bind -k %s 'do something'\n", name);
free(name); free(name);
return true;
}
return false;
} }
if (should_exit(c)) { static double output_elapsed_time(double prev_tstamp, bool first_char_seen) {
// How much time has passed since the previous char was received in microseconds.
double now = timef();
long long int delta_tstamp_us = 1000000 * (now - prev_tstamp);
if (delta_tstamp_us >= 200000 && first_char_seen) putchar('\n');
if (delta_tstamp_us >= 1000000) {
printf(" ");
} else {
printf("(%3lld.%03lld ms) ", delta_tstamp_us / 1000, delta_tstamp_us % 1000);
}
return now;
}
/// Process the characters we receive as the user presses keys.
static void process_input(bool continuous_mode) {
bool first_char_seen = false;
double prev_tstamp = 0.0;
std::vector<wchar_t> bind_chars;
printf("Press a key\n\n");
while (keep_running) {
wchar_t wc = input_common_readch(true);
if (wc == WEOF) {
output_bind_command(bind_chars);
if (first_char_seen && !continuous_mode) {
return;
} else {
continue;
}
}
prev_tstamp = output_elapsed_time(prev_tstamp, first_char_seen);
add_char_to_bind_command(wc, bind_chars);
output_info_about_char(wc);
if (output_matching_key_name(wc)) {
output_bind_command(bind_chars);
}
if (should_exit(wc)) {
printf("\nExiting at your request.\n"); printf("\nExiting at your request.\n");
break; break;
} }
@ -134,51 +216,83 @@ void process_input(bool continuous_mode) {
} }
} }
/// Make sure we cleanup before exiting if we're signaled. /// Make sure we cleanup before exiting if we receive a signal that should cause us to exit.
void signal_handler(int signo) { /// Otherwise just report receipt of the signal.
printf("\nExiting on receipt of signal #%d\n", signo); static struct sigaction old_sigactions[32];
restore_term_mode(); static void signal_handler(int signo, siginfo_t *siginfo, void *siginfo_arg) {
exit(1); debug(2, L"signal #%d (%ls) received", signo, sig2wcs(signo));
// SIGINT isn't included in the following conditional because it is handled specially by fish.
// Specifically, it causes \cC to be reinserted into the tty input stream.
if (signo == SIGHUP || signo == SIGTERM || signo == SIGABRT || signo == SIGSEGV) {
keep_running = false;
}
if (old_sigactions[signo].sa_handler != SIG_IGN &&
old_sigactions[signo].sa_handler != SIG_DFL) {
int needs_siginfo = old_sigactions[signo].sa_flags & SA_SIGINFO;
if (needs_siginfo) {
old_sigactions[signo].sa_sigaction(signo, siginfo, siginfo_arg);
} else {
old_sigactions[signo].sa_handler(signo);
}
}
}
/// Install a handler for every signal. This allows us to restore the tty modes so the terminal is
/// still usable when we die. If the signal already has a handler arrange to invoke it from within
/// our handler.
static void install_our_signal_handlers() {
struct sigaction new_sa, old_sa;
sigemptyset(&new_sa.sa_mask);
new_sa.sa_flags = SA_SIGINFO;
new_sa.sa_sigaction = signal_handler;
for (int signo = 1; signo < 32; signo++) {
if (sigaction(signo, &new_sa, &old_sa) != -1) {
memcpy(&old_sigactions[signo], &old_sa, sizeof(old_sa));
if (old_sa.sa_handler == SIG_IGN) {
debug(3, "signal #%d (%ls) was being ignored", signo, sig2wcs(signo));
}
if (old_sa.sa_flags && ~SA_SIGINFO != 0) {
debug(3, L"signal #%d (%ls) handler had flags 0x%X", signo, sig2wcs(signo),
old_sa.sa_flags);
}
}
}
} }
/// Setup our environment (e.g., tty modes), process key strokes, then reset the environment. /// Setup our environment (e.g., tty modes), process key strokes, then reset the environment.
void setup_and_process_keys(bool continuous_mode) { static void setup_and_process_keys(bool continuous_mode) {
is_interactive_session = 1; // by definition this is interactive is_interactive_session = 1; // by definition this program is interactive
set_main_thread(); set_main_thread();
setup_fork_guards(); setup_fork_guards();
setlocale(LC_ALL, "POSIX");
env_init(); env_init();
reader_init(); reader_init();
input_init(); input_init();
proc_push_interactive(1);
// Installing our handler for every signal (e.g., SIGSEGV) is dubious because it means that signal_set_handlers();
// signals that might generate a core dump will not do so. On the other hand this allows us install_our_signal_handlers();
// to restore the tty modes so the terminal is still usable when we die.
for (int signo = 1; signo < 32; signo++) {
signal(signo, signal_handler);
}
if (continuous_mode) { if (continuous_mode) {
printf("\n"); printf("\n");
printf("Type 'exit' or 'quit' to terminate this program.\n"); printf("To terminate this program type \"exit\" or \"quit\" in this window,\n");
printf("or press [ctrl-C] or [ctrl-D] twice in a row.\n");
printf("\n"); printf("\n");
printf("Characters such as [ctrl-D] (EOF) and [ctrl-C] (interrupt)\n");
printf("have no special meaning and will not terminate this program.\n");
printf("\n");
} else {
set_wait_on_escape_ms(500);
} }
// TODO: We really should enable keypad mode but see issue #838.
process_input(continuous_mode); process_input(continuous_mode);
restore_term_mode(); restore_term_mode();
restore_term_foreground_process_group();
input_destroy();
reader_destroy();
} }
int main(int argc, char **argv) { int main(int argc, char **argv) {
program_name = L"fish_key_reader"; program_name = L"fish_key_reader";
bool continuous_mode = false; bool continuous_mode = false;
const char *short_opts = "+c"; const char *short_opts = "+cd:";
const struct option long_opts[] = {{"continuous", no_argument, NULL, 'd'}, {NULL, 0, NULL, 0}}; const struct option long_opts[] = {{"continuous", no_argument, NULL, 'c'},
{"debug-level", required_argument, NULL, 'd'},
{NULL, 0, NULL, 0}};
int opt; int opt;
while ((opt = getopt_long(argc, argv, short_opts, long_opts, NULL)) != -1) { while ((opt = getopt_long(argc, argv, short_opts, long_opts, NULL)) != -1) {
switch (opt) { switch (opt) {
@ -190,6 +304,21 @@ int main(int argc, char **argv) {
continuous_mode = true; continuous_mode = true;
break; break;
} }
case 'd': {
char *end;
long tmp;
errno = 0;
tmp = strtol(optarg, &end, 10);
if (tmp >= 0 && tmp <= 10 && !*end && !errno) {
debug_level = (int)tmp;
} else {
fwprintf(stderr, _(L"Invalid value '%s' for debug-level flag"), optarg);
exit(1);
}
break;
}
default: { default: {
// We assume getopt_long() has already emitted a diagnostic msg. // We assume getopt_long() has already emitted a diagnostic msg.
exit(1); exit(1);
@ -199,7 +328,12 @@ int main(int argc, char **argv) {
argc -= optind; argc -= optind;
if (argc != 0) { if (argc != 0) {
fprintf(stderr, "Expected no CLI arguments, got %d\n", argc); fprintf(stderr, "Expected no arguments, got %d\n", argc);
return 1;
}
if (!isatty(STDIN_FILENO) || !isatty(STDOUT_FILENO)) {
fprintf(stderr, "Stdin and stdout must be attached to a tty, redirection not allowed.\n");
return 1; return 1;
} }

View file

@ -4,6 +4,7 @@ log_user 0
log_file -noappend interactive.tmp.log log_file -noappend interactive.tmp.log
set fish ../test/root/bin/fish set fish ../test/root/bin/fish
set fish_key_reader ../test/root/bin/fish_key_reader
set timeout 5 set timeout 5

View file

@ -4,6 +4,9 @@
# should be running it via `make test` to ensure the environment is properly # should be running it via `make test` to ensure the environment is properly
# setup. # setup.
# This is a list of flakey tests that often succeed when rerun.
set TESTS_TO_RETRY bind.expect
# Change to directory containing this script # Change to directory containing this script
cd (dirname (status -f)) cd (dirname (status -f))
@ -69,11 +72,20 @@ function test_file
end end
end end
set -l failed set failed
for i in $files_to_test for i in $files_to_test
if not test_file $i
# Retry flakey tests once.
if contains $i $TESTS_TO_RETRY
say -o cyan "Rerunning test $i since it is known to be flakey"
rm -f $i.tmp.*
if not test_file $i if not test_file $i
set failed $failed $i set failed $failed $i
end end
else
set failed $failed $i
end
end
end end
set failed (count $failed) set failed (count $failed)