fix(xonsh): Add xonsh to auto import, respect $HISTFILE in xonsh import, and fix issue with up-arrow keybinding in xonsh (#1711)

* add xonsh to `atuin import auto`

* respect $HISTFILE in xonsh importers

* disable up-arrow binding in xonsh when completion menu is active

* include xonsh logic in the same conditional as other shells

* format and fix clippy lints
This commit is contained in:
jfmontanaro 2024-02-15 11:33:30 -08:00 committed by GitHub
parent 0d31499a62
commit 4512cd5c7f
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 47 additions and 21 deletions

View File

@ -10,7 +10,7 @@ use time::OffsetDateTime;
use uuid::timestamp::{context::NoContext, Timestamp}; use uuid::timestamp::{context::NoContext, Timestamp};
use uuid::Uuid; use uuid::Uuid;
use super::{Importer, Loader}; use super::{get_histpath, Importer, Loader};
use crate::history::History; use crate::history::History;
// Note: both HistoryFile and HistoryData have other keys present in the JSON, we don't // Note: both HistoryFile and HistoryData have other keys present in the JSON, we don't
@ -41,7 +41,7 @@ pub struct Xonsh {
hostname: String, hostname: String,
} }
fn get_hist_dir(xonsh_data_dir: Option<String>) -> Result<PathBuf> { fn xonsh_hist_dir(xonsh_data_dir: Option<String>) -> Result<PathBuf> {
// if running within xonsh, this will be available // if running within xonsh, this will be available
if let Some(d) = xonsh_data_dir { if let Some(d) = xonsh_data_dir {
let mut path = PathBuf::from(d); let mut path = PathBuf::from(d);
@ -107,7 +107,9 @@ impl Importer for Xonsh {
const NAME: &'static str = "xonsh"; const NAME: &'static str = "xonsh";
async fn new() -> Result<Self> { async fn new() -> Result<Self> {
let hist_dir = get_hist_dir(env::var("XONSH_DATA_DIR").ok())?; // wrap xonsh-specific path resolver in general one so that it respects $HISTPATH
let xonsh_data_dir = env::var("XONSH_DATA_DIR").ok();
let hist_dir = get_histpath(|| xonsh_hist_dir(xonsh_data_dir))?;
let sessions = load_sessions(&hist_dir)?; let sessions = load_sessions(&hist_dir)?;
let hostname = get_hostname(); let hostname = get_hostname();
Ok(Xonsh { sessions, hostname }) Ok(Xonsh { sessions, hostname })
@ -167,7 +169,7 @@ mod tests {
#[test] #[test]
fn test_hist_dir_xonsh() { fn test_hist_dir_xonsh() {
let hist_dir = get_hist_dir(Some("/home/user/xonsh_data".to_string())).unwrap(); let hist_dir = xonsh_hist_dir(Some("/home/user/xonsh_data".to_string())).unwrap();
assert_eq!( assert_eq!(
hist_dir, hist_dir,
PathBuf::from("/home/user/xonsh_data/history_json") PathBuf::from("/home/user/xonsh_data/history_json")

View File

@ -10,7 +10,7 @@ use time::OffsetDateTime;
use uuid::timestamp::{context::NoContext, Timestamp}; use uuid::timestamp::{context::NoContext, Timestamp};
use uuid::Uuid; use uuid::Uuid;
use super::{Importer, Loader}; use super::{get_histpath, Importer, Loader};
use crate::history::History; use crate::history::History;
#[derive(Debug, FromRow)] #[derive(Debug, FromRow)]
@ -57,7 +57,7 @@ impl HistDbEntry {
} }
} }
fn get_db_path(xonsh_data_dir: Option<String>) -> Result<PathBuf> { fn xonsh_db_path(xonsh_data_dir: Option<String>) -> Result<PathBuf> {
// if running within xonsh, this will be available // if running within xonsh, this will be available
if let Some(d) = xonsh_data_dir { if let Some(d) = xonsh_data_dir {
let mut path = PathBuf::from(d); let mut path = PathBuf::from(d);
@ -98,7 +98,9 @@ impl Importer for XonshSqlite {
const NAME: &'static str = "xonsh_sqlite"; const NAME: &'static str = "xonsh_sqlite";
async fn new() -> Result<Self> { async fn new() -> Result<Self> {
let db_path = get_db_path(env::var("XONSH_DATA_DIR").ok())?; // wrap xonsh-specific path resolver in general one so that it respects $HISTPATH
let xonsh_data_dir = env::var("XONSH_DATA_DIR").ok();
let db_path = get_histpath(|| xonsh_db_path(xonsh_data_dir))?;
let connection_str = db_path.to_str().ok_or_else(|| { let connection_str = db_path.to_str().ok_or_else(|| {
eyre!( eyre!(
"Invalid path for SQLite database: {}", "Invalid path for SQLite database: {}",
@ -151,7 +153,7 @@ mod tests {
#[test] #[test]
fn test_db_path_xonsh() { fn test_db_path_xonsh() {
let db_path = get_db_path(Some("/home/user/xonsh_data".to_string())).unwrap(); let db_path = xonsh_db_path(Some("/home/user/xonsh_data".to_string())).unwrap();
assert_eq!( assert_eq!(
db_path, db_path,
PathBuf::from("/home/user/xonsh_data/xonsh-history.sqlite") PathBuf::from("/home/user/xonsh_data/xonsh-history.sqlite")

View File

@ -59,8 +59,18 @@ impl Cmd {
return Ok(()); return Ok(());
} }
// $XONSH_HISTORY_BACKEND isn't always set, but $XONSH_HISTORY_FILE is
let xonsh_histfile =
env::var("XONSH_HISTORY_FILE").unwrap_or_else(|_| String::new());
let shell = env::var("SHELL").unwrap_or_else(|_| String::from("NO_SHELL")); let shell = env::var("SHELL").unwrap_or_else(|_| String::from("NO_SHELL"));
if shell.ends_with("/zsh") {
if xonsh_histfile.to_lowercase().ends_with(".json") {
println!("Detected Xonsh",);
import::<Xonsh, DB>(db).await
} else if xonsh_histfile.to_lowercase().ends_with(".sqlite") {
println!("Detected Xonsh (SQLite backend)");
import::<XonshSqlite, DB>(db).await
} else if shell.ends_with("/zsh") {
if ZshHistDb::histpath().is_ok() { if ZshHistDb::histpath().is_ok() {
println!( println!(
"Detected Zsh-HistDb, using :{}", "Detected Zsh-HistDb, using :{}",

View File

@ -1,8 +1,11 @@
import subprocess import subprocess
from prompt_toolkit.application.current import get_app
from prompt_toolkit.filters import Condition
from prompt_toolkit.keys import Keys from prompt_toolkit.keys import Keys
$ATUIN_SESSION=$(atuin uuid).rstrip('\n')
$ATUIN_SESSION=$(atuin uuid).rstrip('\n')
@events.on_precommand @events.on_precommand
def _atuin_precommand(cmd: str): def _atuin_precommand(cmd: str):
@ -52,16 +55,25 @@ def _search(event, extra_args: list[str]):
@events.on_ptk_create @events.on_ptk_create
def _custom_keybindings(bindings, **kw): def _custom_keybindings(bindings, **kw):
@bindings.add(Keys.ControlR, filter=_ATUIN_BIND_CTRL_R) if _ATUIN_BIND_CTRL_R:
@bindings.add(Keys.ControlR)
def r_search(event): def r_search(event):
_search(event, extra_args=[]) _search(event, extra_args=[])
@bindings.add(Keys.Up, filter=_ATUIN_BIND_UP_ARROW) if _ATUIN_BIND_UP_ARROW:
def up_search(event): @Condition
# Only trigger if the buffer is a single line def should_search():
if not '\n' in buffer.text: buffer = get_app().current_buffer
_search(event, extra_args=["--shell-up-key-binding"]) # disable keybind when there is an active completion, so
return # that up arrow can be used to navigate completion menu
if buffer.complete_state is not None:
return False
# similarly, disable when buffer text contains multiple lines
if '\n' in buffer.text:
return False
# Run the default behavior for up arrow return True
event.current_buffer.auto_up(count=event.arg)
@bindings.add(Keys.Up, filter=should_search)
def up_search(event):
_search(event, extra_args=["--shell-up-key-binding"])