fix(REPL): Don't buffer lines until a trailing semicolon is found and add history hinting (#3630)

* fix(REPL): Don't buffer lines until a trailing semicolon is found

The repl would silently buffer all lines until a trailing semicolon were found which
resulted in some very confusing error messages as I would input invalid commands followed
by a command I thought were valid, except I'd still get an error due to the previous command being buffered.

This uses rustyline's helper feature to detect incomplete input (no trailing semicolon) and makes
it accept multiline input until the input is completed.

I also included some of rustyline's default hint and highlighting while I was at it.

* chore: cargo clippy

Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
pull/24376/head
Markus Westerlind 2022-02-03 18:11:01 +01:00 committed by GitHub
parent b3b2d9b623
commit 0bd7941a18
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 133 additions and 60 deletions

1
Cargo.lock generated
View File

@ -1752,6 +1752,7 @@ dependencies = [
name = "influxdb_iox"
version = "0.1.0"
dependencies = [
"ansi_term",
"arrow",
"arrow-flight",
"arrow_util",

View File

@ -46,6 +46,7 @@ trogging = { path = "../trogging", default-features = false, features = ["clap"]
write_buffer = { path = "../write_buffer" }
# Crates.io dependencies, in alphabetical order
ansi_term = "0.12"
arrow = { version = "8.0", features = ["prettyprint"] }
arrow-flight = "8.0"
async-trait = "0.1"

View File

@ -1,4 +1,4 @@
use std::{convert::TryInto, path::PathBuf, sync::Arc, time::Instant};
use std::{borrow::Cow, convert::TryInto, path::PathBuf, sync::Arc, time::Instant};
use arrow::{
array::{ArrayRef, StringArray},
@ -74,11 +74,95 @@ enum QueryEngine {
Observer(super::observer::Observer),
}
struct RustylineHelper {
hinter: rustyline::hint::HistoryHinter,
highlighter: rustyline::highlight::MatchingBracketHighlighter,
}
impl Default for RustylineHelper {
fn default() -> Self {
Self {
hinter: rustyline::hint::HistoryHinter {},
highlighter: rustyline::highlight::MatchingBracketHighlighter::default(),
}
}
}
impl rustyline::Helper for RustylineHelper {}
impl rustyline::validate::Validator for RustylineHelper {
fn validate(
&self,
ctx: &mut rustyline::validate::ValidationContext<'_>,
) -> rustyline::Result<rustyline::validate::ValidationResult> {
let input = ctx.input();
if input.trim_end().ends_with(';') {
match ReplCommand::try_from(input) {
Ok(_) => Ok(rustyline::validate::ValidationResult::Valid(None)),
Err(err) => Ok(rustyline::validate::ValidationResult::Invalid(Some(err))),
}
} else {
Ok(rustyline::validate::ValidationResult::Incomplete)
}
}
}
impl rustyline::hint::Hinter for RustylineHelper {
type Hint = String;
fn hint(&self, line: &str, pos: usize, ctx: &rustyline::Context<'_>) -> Option<String> {
self.hinter.hint(line, pos, ctx)
}
}
impl rustyline::highlight::Highlighter for RustylineHelper {
fn highlight<'l>(&self, line: &'l str, pos: usize) -> Cow<'l, str> {
self.highlighter.highlight(line, pos)
}
fn highlight_prompt<'b, 's: 'b, 'p: 'b>(
&'s self,
prompt: &'p str,
default: bool,
) -> Cow<'b, str> {
self.highlighter.highlight_prompt(prompt, default)
}
fn highlight_hint<'h>(&self, hint: &'h str) -> Cow<'h, str> {
// TODO Detect when windows supports ANSI escapes
#[cfg(windows)]
{
Cow::Borrowed(hint)
}
#[cfg(not(windows))]
{
use ansi_term::Style;
Cow::Owned(Style::new().dimmed().paint(hint).to_string())
}
}
fn highlight_candidate<'c>(
&self,
candidate: &'c str,
completion: rustyline::CompletionType,
) -> Cow<'c, str> {
self.highlighter.highlight_candidate(candidate, completion)
}
fn highlight_char(&self, line: &str, pos: usize) -> bool {
self.highlighter.highlight_char(line, pos)
}
}
impl rustyline::completion::Completer for RustylineHelper {
type Candidate = String;
}
/// Captures the state of the repl, gathers commands and executes them
/// one by one
pub struct Repl {
/// Rustyline editor for interacting with user on command line
rl: Editor<()>,
rl: Editor<RustylineHelper>,
/// Current prompt
prompt: String,
@ -109,7 +193,8 @@ impl Repl {
let management_client = influxdb_iox_client::management::Client::new(connection.clone());
let flight_client = influxdb_iox_client::flight::Client::new(connection.clone());
let mut rl = Editor::<()>::new();
let mut rl = Editor::new();
rl.set_helper(Some(RustylineHelper::default()));
let history_file = history_file();
if let Err(e) = rl.load_history(&history_file) {
@ -172,37 +257,26 @@ impl Repl {
/// Parss the next command;
fn next_command(&mut self) -> Result<ReplCommand> {
let mut request = "".to_owned();
loop {
match self.rl.readline(&self.prompt) {
Ok(ref line) if is_exit_command(line) && request.is_empty() => {
return Ok(ReplCommand::Exit);
}
Ok(ref line) if line.trim_end().ends_with(';') => {
request.push_str(line.trim_end());
self.rl.add_history_entry(request.clone());
return request
.try_into()
.map_err(|message| Error::ParsingCommand { message });
}
Ok(ref line) if is_exit_command(line) => Ok(ReplCommand::Exit),
Ok(ref line) => {
request.push_str(line);
request.push(' ');
let request = line.trim_end();
self.rl.add_history_entry(request.to_owned());
request
.try_into()
.map_err(|message| Error::ParsingCommand { message })
}
Err(ReadlineError::Eof) => {
debug!("Received Ctrl-D");
return Ok(ReplCommand::Exit);
Ok(ReplCommand::Exit)
}
Err(ReadlineError::Interrupted) => {
debug!("Received Ctrl-C");
return Ok(ReplCommand::Exit);
Ok(ReplCommand::Exit)
}
// Some sort of real underlying error
Err(e) => {
return Err(Error::Readline { source: e });
}
}
Err(e) => Err(Error::Readline { source: e }),
}
}

View File

@ -1,5 +1,3 @@
use std::convert::TryInto;
use observability_deps::tracing::{debug, warn};
/// Represents the parsed command from the user (which may be over many lines)
@ -14,23 +12,31 @@ pub enum ReplCommand {
Exit,
}
impl TryInto<ReplCommand> for String {
type Error = Self;
impl TryFrom<String> for ReplCommand {
type Error = String;
fn try_from(input: String) -> Result<Self, Self::Error> {
Self::try_from(&input[..])
}
}
impl TryFrom<&str> for ReplCommand {
type Error = String;
#[allow(clippy::if_same_then_else)]
fn try_into(self) -> Result<ReplCommand, Self::Error> {
debug!(%self, "tokenizing to ReplCommand");
fn try_from(input: &str) -> Result<Self, Self::Error> {
debug!(%input, "tokenizing to ReplCommand");
if self.trim().is_empty() {
if input.trim().is_empty() {
return Err("No command specified".to_string());
}
// tokenized commands, normalized whitespace but original case
let raw_commands = self
let raw_commands = input
.trim()
// chop off trailing semicolon
.strip_suffix(';')
.unwrap_or(&self)
.unwrap_or(input)
// tokenize on whitespace
.split(' ')
.map(|c| c.trim())
@ -49,37 +55,37 @@ impl TryInto<ReplCommand> for String {
let commands = commands.iter().map(|s| s.as_str()).collect::<Vec<_>>();
match commands.as_slice() {
["help"] => Ok(ReplCommand::Help),
["help"] => Ok(Self::Help),
["help", ..] => {
let extra_content = commands[1..].join(" ");
warn!(%extra_content, "ignoring tokens after 'help'");
Ok(ReplCommand::Help)
Ok(Self::Help)
}
["observer"] => Ok(ReplCommand::Observer),
["exit"] => Ok(ReplCommand::Exit),
["quit"] => Ok(ReplCommand::Exit),
["observer"] => Ok(Self::Observer),
["exit"] => Ok(Self::Exit),
["quit"] => Ok(Self::Exit),
["use", "database"] => {
Err("name not specified. Usage: USE DATABASE <name>".to_string())
} // USE DATABASE
["use", "database", _name] => {
// USE DATABASE <name>
Ok(ReplCommand::UseDatabase {
Ok(Self::UseDatabase {
db_name: raw_commands[2].to_string(),
})
}
["use", _command] => {
// USE <name>
Ok(ReplCommand::UseDatabase {
Ok(Self::UseDatabase {
db_name: raw_commands[1].to_string(),
})
}
["show", "databases"] => Ok(ReplCommand::ShowDatabases),
["set", "format", _format] => Ok(ReplCommand::SetFormat {
["show", "databases"] => Ok(Self::ShowDatabases),
["set", "format", _format] => Ok(Self::SetFormat {
format: raw_commands[2].to_string(),
}),
_ => {
// By default, treat the entire string as SQL
Ok(ReplCommand::SqlCommand { sql: self })
Ok(Self::SqlCommand { sql: input.into() })
}
}
}
@ -130,15 +136,6 @@ LIMIT 20
}
}
#[cfg(test)]
impl TryInto<ReplCommand> for &str {
type Error = String;
fn try_into(self) -> Result<ReplCommand, Self::Error> {
self.to_string().try_into()
}
}
#[cfg(test)]
mod tests {
use super::*;