diff --git a/Cargo.lock b/Cargo.lock index 22c18625..0c327ed4 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -260,6 +260,9 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ac2e663e3e3bed2d32d065a8404024dad306e699a04263ec59919529f803aee9" dependencies = [ "clap", + "clap_lex", + "is_executable", + "shlex", ] [[package]] @@ -1095,6 +1098,15 @@ dependencies = [ "windows-sys 0.52.0", ] +[[package]] +name = "is_executable" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fa9acdc6d67b75e626ad644734e8bc6df893d9cd2a834129065d3dd6158ea9c8" +dependencies = [ + "winapi", +] + [[package]] name = "is_terminal_polyfill" version = "1.70.1" diff --git a/Cargo.toml b/Cargo.toml index eba70491..4335e5d0 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -7,6 +7,7 @@ license = "MIT" keywords = ["julia"] categories = ["command-line-utilities"] edition = "2021" +rust-version = "1.80" default-run = "juliaup" authors = ["David Anthoff "] exclude = [ @@ -27,7 +28,7 @@ codegen-units = 1 [dependencies] clap = { version = "4.5", features = ["derive"] } -clap_complete = "4.5" +clap_complete = { version = "4.5", features = ["unstable-dynamic"] } dirs = "5.0.1" dunce = "1.0" serde = { version = "1.0", features = ["derive"] } diff --git a/README.md b/README.md index 32d473fa..5c06e86b 100644 --- a/README.md +++ b/README.md @@ -123,7 +123,6 @@ Here are some of the things you can do with `juliaup`: - `juliaup override set --path foo/bar lts` sets a directory override for the path `foo/bar` to the `lts` channel. - `juliaup override unset --path foo/bar` removes a directory override for the path `foo/bar`. - `juliaup override unset --nonexistent` removes all directory overrides for paths that no longer exist. -- `juliaup completions bash > ~/.local/share/bash-completion/completions/juliaup` generates Bash completions for `juliaup` and saves them to a file. To use them, simply source this file in your `~/.bashrc`. Other supported shells are `zsh`, `fish`, `elvish` and `powershell`. - `juliaup` shows you what other commands are available. The available system provided channels are: @@ -168,6 +167,19 @@ the `JULIAUP_DEPOT_PATH` environment variable. Caution: Previous versions of Jul Juliaup by default downloads julia binary tarballs from the official server "https://julialang-s3.julialang.org". If requested, the environment variable `JULIAUP_SERVER` can be used to tell Juliaup to use a third-party mirror server. +## Shell completions + +To generate shell completions, load `COMPLETE=$SHELL juliaup` at shell launch. For example, with bash, it could look like `source <(COMPLETE=bash juliaup)`. + +For more specific information on adding completions to your shell, see https://docs.rs/clap_complete/latest/clap_complete/env/index.html + +Note: MacOS ships with an ancient version of bash that does not support process substitution. To work around this, you can create a temporary file and `source` that instead like: + +```bash +COMPLETE=bash juliaup > /tmp/juliaup_completion.sh +source /tmp/juliaup_completion.sh +``` + ## Development guides For juliaup developers, information on how to build juliaup locally, update julia versions, and release updates diff --git a/src/bin/juliaup.rs b/src/bin/juliaup.rs index 5e964152..6c67b503 100644 --- a/src/bin/juliaup.rs +++ b/src/bin/juliaup.rs @@ -1,5 +1,6 @@ use anyhow::{Context, Result}; -use clap::Parser; +use clap::{CommandFactory, Parser}; +use clap_complete::CompleteEnv; use juliaup::cli::{ConfigSubCmd, Juliaup, OverrideSubCmd, SelfSubCmd}; use juliaup::command_api::run_command_api; #[cfg(not(windows))] @@ -36,6 +37,8 @@ use juliaup::command_selfuninstall::run_command_selfuninstall_unavailable; use log::info; fn main() -> Result<()> { + CompleteEnv::with_factory(|| Juliaup::command()).complete(); + human_panic::setup_panic!( human_panic::Metadata::new("Juliaup", env!("CARGO_PKG_VERSION")) .support("https://github.com/JuliaLang/juliaup") diff --git a/src/cli.rs b/src/cli.rs index 789f486b..95451dff 100644 --- a/src/cli.rs +++ b/src/cli.rs @@ -1,7 +1,7 @@ use clap::{Parser, ValueEnum}; #[derive(Parser)] -#[clap(name = "Juliaup", version)] +#[clap(name = "juliaup", version)] #[command( after_help = "To launch a specific Julia version, use `julia +{channel}` e.g. `julia +1.6`. Entering just `julia` uses the default channel set via `juliaup default`."