From 3bb21ede976a2be51ec2d07d488e8c2d7433cfe6 Mon Sep 17 00:00:00 2001 From: Ajeet D'Souza <98ajeet@gmail.com> Date: Thu, 29 Jul 2021 09:20:40 +0530 Subject: [PATCH] zoxide-{add,remove} should accept multiple arguments (#243) --- CHANGELOG.md | 6 +++ Cargo.lock | 70 +++++++++++++++++++++------------ Cargo.toml | 9 +++-- README.md | 20 +++++----- contrib/completions/_zoxide | 4 +- contrib/completions/zoxide.bash | 4 +- man/zoxide-add.1 | 2 +- man/zoxide-init.1 | 4 +- man/zoxide-remove.1 | 2 +- src/app/_app.rs | 8 ++-- src/app/add.rs | 42 ++++++++++++-------- src/app/import.rs | 8 ++-- src/app/query.rs | 8 +++- src/app/remove.rs | 30 +++++--------- src/db/mod.rs | 20 ++++------ templates/powershell.txt | 4 +- 16 files changed, 134 insertions(+), 107 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 75629b77..2300c4fa 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,5 @@ + + # Changelog All notable changes to this project will be documented in this file. @@ -7,6 +9,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## Unreleased +### Added + +- `zoxide add` and `zoxide remove` now accept multiple arguments. + ### Fixed - Nushell: errors on 0.33.0. diff --git a/Cargo.lock b/Cargo.lock index 9b2ee9f3..cafafbe5 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1,10 +1,12 @@ # This file is automatically @generated by Cargo. # It is not intended for manual editing. +version = 3 + [[package]] name = "anyhow" -version = "1.0.41" +version = "1.0.42" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "15af2628f6890fe2609a3b91bef4c83450512802e59489f9c1cb1fa5df064a61" +checksum = "595d3cfa7a60d4555cb5067b99f07142a08ea778de5cf993f7b75c7d8fabc486" [[package]] name = "arrayvec" @@ -55,9 +57,9 @@ dependencies = [ [[package]] name = "assert_cmd" -version = "1.0.5" +version = "1.0.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a88b6bd5df287567ffdf4ddf4d33060048e1068308e5f62d81c6f9824a045a48" +checksum = "3d20831bd004dda4c7c372c19cdabff369f794a95e955b3f13fe460e3e1ae95f" dependencies = [ "bstr", "doc-comment", @@ -170,10 +172,10 @@ dependencies = [ ] [[package]] -name = "difference" -version = "2.0.0" +name = "difflib" +version = "0.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "524cbf6897b527295dff137cec09ecf3a05f4fddffd7dfcd1585403449e74198" +checksum = "6184e33543162437515c2e2b48714794e37845ec9851711914eec9d308f6ebe8" [[package]] name = "dirs-next" @@ -208,6 +210,12 @@ version = "1.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "453440c271cf5577fd2a40e4942540cb7d0d2f85e27c8d07dd0023c925a67541" +[[package]] +name = "either" +version = "1.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e78d4f1cc4ae33bbfc157ed5d5a5ef3bc29227303d595861deb238fcec4e9457" + [[package]] name = "funty" version = "1.1.0" @@ -233,9 +241,9 @@ checksum = "9b919933a397b79c37e33b77bb2aa3dc8eb6e165ad809e58ff75bc7db2e34574" [[package]] name = "hashbrown" -version = "0.9.1" +version = "0.11.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d7afe4a420e3fe79967a00898cc1f4db7c8a49a9333a29f8a4bd76a253d5cd04" +checksum = "ab5ef0d4909ef3724cc8cce6ccc8572c5c817592e9285f5464f8e86f8bd3726e" [[package]] name = "heck" @@ -248,23 +256,32 @@ dependencies = [ [[package]] name = "hermit-abi" -version = "0.1.18" +version = "0.1.19" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "322f4de77956e22ed0e5032c359a0f1273f1f7f0d79bfa3b8ffbc730d7fbcc5c" +checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33" dependencies = [ "libc", ] [[package]] name = "indexmap" -version = "1.6.2" +version = "1.7.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "824845a0bf897a9042383849b02c1bc219c2383772efcd5c6f9766fa4b81aef3" +checksum = "bc633605454125dec4b66843673f01c7df2b89479b32e0ed634e43a91cff62a5" dependencies = [ "autocfg", "hashbrown", ] +[[package]] +name = "itertools" +version = "0.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "69ddb889f9d0d08a67338271fa9b62996bc788c7796a5c18cf057420aaed5eaf" +dependencies = [ + "either", +] + [[package]] name = "lazy_static" version = "1.4.0" @@ -286,9 +303,9 @@ dependencies = [ [[package]] name = "libc" -version = "0.2.97" +version = "0.2.98" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "12b8adadd720df158f4d70dfe7ccc6adb0472d7c55ca83445f6a5ab3e36f8fb6" +checksum = "320cfe77175da3a483efed4bc0adc1968ca050b098ce4f2f1c13a56626128790" [[package]] name = "memchr" @@ -320,9 +337,9 @@ dependencies = [ [[package]] name = "ordered-float" -version = "2.5.1" +version = "2.7.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f100fcfb41e5385e0991f74981732049f9b896821542a219420491046baafdc2" +checksum = "039f02eb0f69271f26abe3202189275d7aa2258b903cb0281b5de710a2570ff3" dependencies = [ "num-traits", ] @@ -350,11 +367,12 @@ checksum = "ac74c624d6b2d21f425f752262f42188365d7b8ff1aff74c82e45136510a4857" [[package]] name = "predicates" -version = "1.0.8" +version = "2.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f49cfaf7fdaa3bfacc6fa3e7054e65148878354a5cfddcf661df4c851f8021df" +checksum = "bc3d91237f5de3bcd9d927e24d03b495adb6135097b001cea7403e2d573d00a9" dependencies = [ - "difference", + "difflib", + "itertools", "predicates-core", ] @@ -400,9 +418,9 @@ dependencies = [ [[package]] name = "proc-macro2" -version = "1.0.27" +version = "1.0.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f0d8caf72986c1a598726adc988bb5984792ef84f5ee5aa50209145ee8077038" +checksum = "5c7ed8b8c7b886ea3ed7dde405212185f423ab44682667c8c6dd14aa1d9f6612" dependencies = [ "unicode-xid", ] @@ -576,9 +594,9 @@ checksum = "73473c0e59e6d5812c5dfe2a064a6444949f089e20eec9a2e5506596494e4623" [[package]] name = "syn" -version = "1.0.73" +version = "1.0.74" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f71489ff30030d2ae598524f61326b902466f72a0fb1a8564c001cc63425bcc7" +checksum = "1873d832550d4588c3dbc20f01361ab00bfe741048f71e3fecf145a7cc18b29c" dependencies = [ "proc-macro2", "quote", @@ -637,9 +655,9 @@ checksum = "56dee185309b50d1f11bfedef0fe6d036842e3fb77413abef29f8f8d1c5d4c1c" [[package]] name = "unicode-segmentation" -version = "1.7.1" +version = "1.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bb0d2e7be6ae3a5fa87eed5fb451aff96f2573d2694942e40543ae0bbe19c796" +checksum = "8895849a949e7845e06bd6dc1aa51731a103c42707010a5b591c0038fb73385b" [[package]] name = "unicode-width" diff --git a/Cargo.toml b/Cargo.toml index 35c6f48f..b30a3e2c 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -11,18 +11,21 @@ categories = ["command-line-utilities", "filesystem"] [dependencies] anyhow = "1.0.32" -askama = { version="0.10.3", default-features=false } +askama = { version = "0.10.3", default-features = false } bincode = "1.3.1" clap = "3.0.0-beta.2" dirs-next = "2.0.0" dunce = "1.0.1" glob = "0.3.0" ordered-float = "2.0.0" -serde = { version="1.0.116", features=["derive"] } +serde = { version = "1.0.116", features = ["derive"] } tempfile = "3.1.0" [target.'cfg(windows)'.dependencies] -rand = { version="0.8.4", features=["getrandom", "small_rng"], default-features=false } +rand = { version = "0.8.4", features = [ + "getrandom", + "small_rng", +], default-features = false } [dev-dependencies] assert_cmd = "1.0.1" diff --git a/README.md b/README.md index 0732449d..44857ac4 100644 --- a/README.md +++ b/README.md @@ -31,9 +31,9 @@ Read more about the matching algorithm [here][algorithm-matching]. ## Getting started -### Step 1: Install `zoxide` +### *Step 1: Install `zoxide`* -`zoxide` supports most major platforms. If your platform isn't listed below, +`zoxide` runs on most major platforms. If your platform isn't listed below, please [open an issue][issues].
@@ -131,12 +131,12 @@ To install `zoxide`, use a package manager:
-### Step 2: Install `fzf` (optional) +### *Step 2: Install `fzf` (optional)* [`fzf`][fzf] is a command-line fuzzy finder, used by `zoxide` for interactive -selection ([installation instructions][fzf-installation]). +selection. It can be installed from [here][fzf-installation]. -### Step 3: Add `zoxide` to your shell +### *Step 3: Import your data (optional)* If you currently use any of the following utilities, you may want to import your data into `zoxide`: @@ -159,7 +159,9 @@ zoxide import --from z path/to/db -Now, initialize `zoxide` on your shell: +### *Step 4: Add `zoxide` to your shell* + +To start using `zoxide`, add it to your shell.
bash @@ -195,7 +197,7 @@ zoxide init fish | source
-nushell 0.32+ +nushell v0.32+ Initialize the `zoxide` script: @@ -217,7 +219,7 @@ You can replace `__zoxide_prompt` with a custom prompt.
powershell -Add this to your configuration (the location is stored in `$profile`): +Add this to your configuration (find it with `echo $profile`): ```powershell Invoke-Expression (& { @@ -251,7 +253,7 @@ eval "$(zoxide init zsh)"
-Any POSIX shell +any POSIX shell Add this to your configuration: diff --git a/contrib/completions/_zoxide b/contrib/completions/_zoxide index 08650874..1548e9b5 100644 --- a/contrib/completions/_zoxide +++ b/contrib/completions/_zoxide @@ -32,7 +32,7 @@ _zoxide() { _arguments "${_arguments_options[@]}" \ '-h[Prints help information]' \ '--help[Prints help information]' \ -':path:_files -/' \ +'*::paths:_files -/' \ && ret=0 ;; (import) @@ -75,7 +75,7 @@ _arguments "${_arguments_options[@]}" \ '()*--interactive=[]' \ '-h[Prints help information]' \ '--help[Prints help information]' \ -'::path:_files -/' \ +'*::paths:_files -/' \ && ret=0 ;; esac diff --git a/contrib/completions/zoxide.bash b/contrib/completions/zoxide.bash index 2ab89f97..749b1ef0 100644 --- a/contrib/completions/zoxide.bash +++ b/contrib/completions/zoxide.bash @@ -51,7 +51,7 @@ _zoxide() { ;; zoxide__add) - opts=" -h --help " + opts=" -h --help ... " if [[ ${cur} == -* || ${COMP_CWORD} -eq 2 ]] ; then COMPREPLY=( $(compgen -W "${opts}" -- "${cur}") ) return 0 @@ -127,7 +127,7 @@ _zoxide() { return 0 ;; zoxide__remove) - opts=" -i -h --interactive --help " + opts=" -i -h --interactive --help ... " if [[ ${cur} == -* || ${COMP_CWORD} -eq 2 ]] ; then COMPREPLY=( $(compgen -W "${opts}" -- "${cur}") ) return 0 diff --git a/man/zoxide-add.1 b/man/zoxide-add.1 index 707fca8e..d82b12ad 100644 --- a/man/zoxide-add.1 +++ b/man/zoxide-add.1 @@ -2,7 +2,7 @@ .SH NAME zoxide-add - add a new directory or increment its rank .SH SYNOPSIS -.B zoxide add \fIPATH\fR +.B zoxide add \fI[PATHS]\fR .SH DESCRIPTION If the directory is not already in the database, this command creates a new entry for it with a default score of \fI1\fR, otherwise, it increments the diff --git a/man/zoxide-init.1 b/man/zoxide-init.1 index 2286443b..3efffbf3 100644 --- a/man/zoxide-init.1 +++ b/man/zoxide-init.1 @@ -44,7 +44,7 @@ Add this to your configuration (usually \fI~/.config/nu/config.toml\fR): You can replace \fB__zoxide_prompt\fR with a custom prompt. .TP .B powershell -Add this to your configuration (the location is stored in \fI$profile\fR): +Add this to your configuration (find it with \fIecho $profile\fR): .sp .nf \fBInvoke-Expression (& { @@ -67,7 +67,7 @@ Add this to your configuration (usually \fI~/.zshrc\fR): \fBeval "$(zoxide init zsh)"\fR .fi .TP -.B Any POSIX shell +.B any POSIX shell .sp Add this to your configuration: .sp diff --git a/man/zoxide-remove.1 b/man/zoxide-remove.1 index f0f186e5..ee716404 100644 --- a/man/zoxide-remove.1 +++ b/man/zoxide-remove.1 @@ -2,7 +2,7 @@ .SH NAME zoxide-remove - remove a directory from the database .SH SYNOPSIS -.B zoxide remove \fIPATH [OPTIONS]\fR +.B zoxide remove \fI[PATHS] [OPTIONS]\fR .SH DESCRIPTION If you'd like to permanently exclude a directory from the database, see the \fB_ZO_EXCLUDE_DIRS\fR environment variable in \fBzoxide\fR(1). diff --git a/src/app/_app.rs b/src/app/_app.rs index 7362a249..0ce7b2ae 100644 --- a/src/app/_app.rs +++ b/src/app/_app.rs @@ -33,8 +33,8 @@ pub enum App { /// Add a new directory or increment its rank #[derive(Clap, Debug)] pub struct Add { - #[clap(value_hint = ValueHint::DirPath)] - pub path: PathBuf, + #[clap(min_values = 1, required = true, value_hint = ValueHint::DirPath)] + pub paths: Vec, } /// Import entries from another application @@ -126,12 +126,12 @@ pub struct Query { #[derive(Clap, Debug)] pub struct Remove { // Use interactive selection - #[clap(conflicts_with = "path", long, short, value_name = "keywords")] + #[clap(conflicts_with = "paths", long, short, value_name = "keywords")] pub interactive: Option>, #[clap( conflicts_with = "interactive", required_unless_present = "interactive", value_hint = ValueHint::DirPath )] - pub path: Option, + pub paths: Vec, } diff --git a/src/app/add.rs b/src/app/add.rs index 7dff8601..7f734003 100644 --- a/src/app/add.rs +++ b/src/app/add.rs @@ -9,33 +9,41 @@ use std::path::Path; impl Run for Add { fn run(&self) -> Result<()> { - let path = if config::resolve_symlinks() { - util::canonicalize(&self.path) - } else { - util::resolve_path(&self.path) - }?; - let path = util::path_to_str(&path)?; - let now = util::current_time()?; - // These characters can't be printed cleanly to a single line, so they // can cause confusion when writing to fzf / stdout. const EXCLUDE_CHARS: &[char] = &['\n', '\r']; - let mut exclude_dirs = config::exclude_dirs()?.into_iter(); - if exclude_dirs.any(|pattern| pattern.matches(path)) || path.contains(EXCLUDE_CHARS) { - return Ok(()); - } - if !Path::new(path).is_dir() { - bail!("not a directory: {}", path); - } let data_dir = config::data_dir()?; + let exclude_dirs = config::exclude_dirs()?; let max_age = config::maxage()?; + let now = util::current_time()?; let mut db = DatabaseFile::new(data_dir); let mut db = db.open()?; - db.add(path, now); - db.age(max_age); + for path in self.paths.iter() { + let path = if config::resolve_symlinks() { + util::canonicalize(path) + } else { + util::resolve_path(path) + }?; + let path = util::path_to_str(&path)?; + + // Ignore path if it contains unsupported characters, or if it's in + // the exclude list. + if path.contains(EXCLUDE_CHARS) || exclude_dirs.iter().any(|glob| glob.matches(path)) { + continue; + } + if !Path::new(path).is_dir() { + bail!("not a directory: {}", path); + } + db.add(path, now); + } + + if db.modified { + db.age(max_age); + db.save()?; + } Ok(()) } diff --git a/src/app/import.rs b/src/app/import.rs index 128c3ec7..2b237496 100644 --- a/src/app/import.rs +++ b/src/app/import.rs @@ -8,7 +8,7 @@ use std::fs; impl Run for Import { fn run(&self) -> Result<()> { - let buffer = &fs::read_to_string(&self.path).with_context(|| { + let buffer = fs::read_to_string(&self.path).with_context(|| { format!("could not open database for importing: {}", &self.path.display()) })?; @@ -20,12 +20,12 @@ impl Run for Import { } match self.from { - ImportFrom::Autojump => from_autojump(db, buffer), - ImportFrom::Z => from_z(db, buffer), + ImportFrom::Autojump => from_autojump(db, &buffer), + ImportFrom::Z => from_z(db, &buffer), } .context("import error")?; - Ok(()) + db.save() } } diff --git a/src/app/query.rs b/src/app/query.rs index 3dfee242..5abd28d8 100644 --- a/src/app/query.rs +++ b/src/app/query.rs @@ -1,6 +1,6 @@ use crate::app::{Query, Run}; use crate::config; -use crate::db::DatabaseFile; +use crate::db::{Database, DatabaseFile}; use crate::error::BrokenPipeHandler; use crate::fzf::Fzf; use crate::util; @@ -14,6 +14,12 @@ impl Run for Query { let data_dir = config::data_dir()?; let mut db = DatabaseFile::new(data_dir); let mut db = db.open()?; + self.query(&mut db).and(db.save()) + } +} + +impl Query { + fn query(&self, db: &mut Database) -> Result<()> { let now = util::current_time()?; let mut stream = db.stream(now).with_keywords(&self.keywords); diff --git a/src/app/remove.rs b/src/app/remove.rs index 99eb48ce..8d832648 100644 --- a/src/app/remove.rs +++ b/src/app/remove.rs @@ -28,35 +28,25 @@ impl Run for Remove { selection = fzf.wait_select()?; let paths = selection.lines().filter_map(|line| line.get(5..)); - let mut not_found = Vec::new(); for path in paths { - if !db.remove(&path) { - not_found.push(path); + if !db.remove(path) { + bail!("path not found in database: {}", path); } } - - if !not_found.is_empty() { - let mut err = "path not found in database:".to_string(); - for path in not_found { - err.push_str("\n "); - err.push_str(path.as_ref()); - } - bail!(err); - } } None => { - // unwrap is safe here because path is required_unless_present = "interactive" - let path = self.path.as_ref().unwrap(); - if !db.remove(path) { - let path_abs = util::resolve_path(&path)?; - let path_abs = util::path_to_str(&path_abs)?; - if path_abs != path && !db.remove(path) { - bail!("path not found in database:\n {}", &path) + for path in self.paths.iter() { + if !db.remove(path) { + let path_abs = util::resolve_path(path)?; + let path_abs = util::path_to_str(&path_abs)?; + if path_abs != path && !db.remove(path_abs) { + bail!("path not found in database: {} ({})", path, path_abs) + } } } } } - Ok(()) + db.save() } } diff --git a/src/db/mod.rs b/src/db/mod.rs index 6808f067..3d198a36 100644 --- a/src/db/mod.rs +++ b/src/db/mod.rs @@ -15,7 +15,7 @@ use std::path::{Path, PathBuf}; pub struct Database<'file> { pub dirs: DirList<'file>, pub modified: bool, - pub data_dir: &'file PathBuf, + pub data_dir: &'file Path, } impl<'file> Database<'file> { @@ -25,7 +25,7 @@ impl<'file> Database<'file> { } let buffer = self.dirs.to_bytes()?; - let mut file = NamedTempFile::new_in(&self.data_dir).with_context(|| { + let mut file = NamedTempFile::new_in(self.data_dir).with_context(|| { format!("could not create temporary database in: {}", self.data_dir.display()) })?; @@ -125,16 +125,6 @@ impl<'file> Database<'file> { } } -impl Drop for Database<'_> { - fn drop(&mut self) { - // Since the error can't be properly handled here, - // pretty-print it instead. - if let Err(e) = self.save() { - let _ = writeln!(io::stderr(), "zoxide: {:?}", e); - } - } -} - #[cfg(windows)] fn persist>(mut file: NamedTempFile, path: P) -> Result<(), PersistError> { use rand::distributions::{Distribution, Uniform}; @@ -168,7 +158,7 @@ fn persist>(mut file: NamedTempFile, path: P) -> Result<(), Persi #[cfg(unix)] fn persist>(file: NamedTempFile, path: P) -> Result<(), PersistError> { - file.persist(&path)?; + file.persist(path)?; Ok(()) } @@ -231,6 +221,7 @@ mod tests { let mut db = db.open().unwrap(); db.add(path, now); db.add(path, now); + db.save().unwrap(); } { let mut db = DatabaseFile::new(data_dir.path()); @@ -253,17 +244,20 @@ mod tests { let mut db = DatabaseFile::new(data_dir.path()); let mut db = db.open().unwrap(); db.add(path, now); + db.save().unwrap(); } { let mut db = DatabaseFile::new(data_dir.path()); let mut db = db.open().unwrap(); assert!(db.remove(path)); + db.save().unwrap(); } { let mut db = DatabaseFile::new(data_dir.path()); let mut db = db.open().unwrap(); assert!(db.dirs.is_empty()); assert!(!db.remove(path)); + db.save().unwrap(); } } } diff --git a/templates/powershell.txt b/templates/powershell.txt index ede86194..2d30a409 100644 --- a/templates/powershell.txt +++ b/templates/powershell.txt @@ -115,7 +115,7 @@ Set-Alias {{cmd}}i __zoxide_zi {%- endmatch %} {{ section }} -# To initialize zoxide, add this to your configuration (the location is stored -# in $profile): +# To initialize zoxide, add this to your configuration (find it with +# `echo $profile`): # # Invoke-Expression (& { $hook = if ($PSVersionTable.PSVersion.Major -ge 6) { 'pwd' } else { 'prompt' } (zoxide init powershell --hook $hook) -join "`n" })