Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: clap_completion #1056

Closed
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
91 changes: 83 additions & 8 deletions src/cli.rs
Original file line number Diff line number Diff line change
Expand Up @@ -37,14 +37,14 @@
all(target_os = "linux", target_arch = "x86_64"),
all(target_os = "macos", target_arch = "aarch64")
)))]
use std::string::String as KafkaSslProtocol;

Check warning on line 40 in src/cli.rs

View workflow job for this annotation

GitHub Actions / Build for aarch64-unknown-linux-gnu

unused import: `std::string::String as KafkaSslProtocol`

/// Default username and password for Parseable server, used by default for local mode.
/// NOTE: obviously not recommended for production
pub const DEFAULT_USERNAME: &str = "admin";
pub const DEFAULT_PASSWORD: &str = "admin";

#[derive(Parser)]
#[derive(Parser, Debug)]
#[command(
name = "parseable",
bin_name = "parseable",
Expand Down Expand Up @@ -73,13 +73,88 @@
"#,
subcommand_required = true,
)]
pub struct Cli {
pub enum Cli {
#[clap(name = "storage")]
/// Storage options
Storage(StorageOptions),

/// Generate shell completions
#[clap(name = "completions")]
Completion(CommandCompletionOptions),
}

// taken generously from https://github.com/jj-vcs/jj/blob/be32d4e3efbb9a51deadcc63635a5fb1526d0d6c/cli/src/commands/util/completion.rs#L23C1-L47C36
// Using an explicit `doc` attribute prevents rustfmt from mangling the list
// formatting without disabling rustfmt for the entire struct.
#[doc = r#"Print a command-line-completion script

Apply it by running one of these:

- Bash: `source <(pb util completion bash)`
- Fish: `pb util completion fish | source`
- Nushell:
```nu
pb util completion nushell | save "completions-pb.nu"
Copy link
Contributor

@de-sh de-sh Jan 21, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

pb is a different cli tool, a client to the parseable(server) actually.

Can we have a argument/flag similar to --help that when set will print the completions for the specific shell. i.e.

parseable --shell-completion-config nushell | save "parseable-completions.nu"

this way we can continue to use the CLI as is, setting the default behavior of the CLI to be "don't output" when the flag is not set.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is also a good way to do this, rather than adding a command. I'll implement this.

use "completions-pb.nu" * # Or `source "completions-pb.nu"`
```
- Zsh:
```shell
autoload -U compinit
compinit
source <(pb util completion zsh)
```
"#]

#[derive(clap::Args, Debug)]
#[command(verbatim_doc_comment)]
pub struct CommandCompletionOptions {
pub shell: Option<ShellCompletion>,
}

/// Available shell completions
#[derive(clap::ValueEnum, Clone, Copy, Debug, Eq, Hash, PartialEq)]
pub enum ShellCompletion {
Bash,
Elvish,
Fish,
// Nushell,
PowerShell,
Zsh,
}

impl ShellCompletion {
pub fn generate(&self, cmd: &mut clap::Command) -> Vec<u8> {
use clap_complete::generate;

Check failure on line 127 in src/cli.rs

View workflow job for this annotation

GitHub Actions / Build for aarch64-unknown-linux-gnu

unresolved import `clap_complete`

Check failure on line 127 in src/cli.rs

View workflow job for this annotation

GitHub Actions / Build for x86_64-unknown-linux-gnu

unresolved import `clap_complete`
Copy link
Contributor

@de-sh de-sh Jan 21, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

clap_complete is missing?

use clap_complete::Shell;

Check failure on line 128 in src/cli.rs

View workflow job for this annotation

GitHub Actions / Build for aarch64-unknown-linux-gnu

unresolved import `clap_complete`

Check failure on line 128 in src/cli.rs

View workflow job for this annotation

GitHub Actions / Build for x86_64-unknown-linux-gnu

unresolved import `clap_complete`
// use clap_complete_nushell::Nushell;

let mut buf = Vec::new();

let bin_name = "pb";

match self {
Self::Bash => generate(Shell::Bash, cmd, bin_name, &mut buf),
Self::Elvish => generate(Shell::Elvish, cmd, bin_name, &mut buf),
Self::Fish => generate(Shell::Fish, cmd, bin_name, &mut buf),
// Self::Nushell => generate(Nushell, cmd, bin_name, &mut buf),
Self::PowerShell => generate(Shell::PowerShell, cmd, bin_name, &mut buf),
Self::Zsh => generate(Shell::Zsh, cmd, bin_name, &mut buf),
}

buf
}
}

// todo remove caution
#[derive(clap::Args, Debug)]
pub struct StorageOptions {
#[command(subcommand)]
pub storage: StorageOptions,
pub storage: StorageOptionsEnum,
}

#[derive(Parser)]
pub enum StorageOptions {
// todo remove caution
#[derive(clap::Subcommand, Debug)]
pub enum StorageOptionsEnum {
#[command(name = "local-store")]
Local(LocalStoreArgs),

Expand All @@ -90,23 +165,23 @@
Blob(BlobStoreArgs),
}

#[derive(Parser)]
#[derive(Parser, Debug)]
pub struct LocalStoreArgs {
#[command(flatten)]
pub options: Options,
#[command(flatten)]
pub storage: FSConfig,
}

#[derive(Parser)]
#[derive(Parser, Debug)]
pub struct S3StoreArgs {
#[command(flatten)]
pub options: Options,
#[command(flatten)]
pub storage: S3Config,
}

#[derive(Parser)]
#[derive(Parser, Debug)]
pub struct BlobStoreArgs {
#[command(flatten)]
pub options: Options,
Expand Down
80 changes: 47 additions & 33 deletions src/option.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,12 +16,13 @@
*
*/

use crate::cli::{Cli, Options, StorageOptions, DEFAULT_PASSWORD, DEFAULT_USERNAME};
use crate::cli::StorageOptionsEnum;
use crate::cli::{Cli, Options, DEFAULT_PASSWORD, DEFAULT_USERNAME};
use crate::storage::object_storage::parseable_json_path;
use crate::storage::{ObjectStorageError, ObjectStorageProvider};
use bytes::Bytes;
use clap::error::ErrorKind;
use clap::Parser;
use clap::CommandFactory;
use clap::{error::ErrorKind, Parser};
use once_cell::sync::Lazy;
use parquet::basic::{BrotliLevel, GzipLevel, ZstdLevel};
use serde::{Deserialize, Serialize};
Expand All @@ -41,40 +42,53 @@

impl Config {
fn new() -> Self {
match Cli::parse().storage {
StorageOptions::Local(args) => {
if args.options.local_staging_path == args.storage.root {
clap::Error::raw(
ErrorKind::ValueValidation,
"Cannot use same path for storage and staging",
)
.exit();
let result = Cli::parse();
match result {
Cli::Storage(storage_opts) => match storage_opts.storage {
StorageOptionsEnum::Local(args) => {
if args.options.local_staging_path == args.storage.root {
clap::Error::raw(
ErrorKind::ValueValidation,
"Cannot use same path for storage and staging",
)
.exit();
}

if args.options.hot_tier_storage_path.is_some() {
clap::Error::raw(
ErrorKind::ValueValidation,
"Cannot use hot tier with local-store subcommand.",
)
.exit();
}

Config {
options: args.options,
storage: Arc::new(args.storage),
storage_name: "drive",
}
}

if args.options.hot_tier_storage_path.is_some() {
clap::Error::raw(
ErrorKind::ValueValidation,
"Cannot use hot tier with local-store subcommand.",
)
.exit();
}

Config {
StorageOptionsEnum::S3(args) => Config {
options: args.options,
storage: Arc::new(args.storage),
storage_name: "drive",
}
}
StorageOptions::S3(args) => Config {
options: args.options,
storage: Arc::new(args.storage),
storage_name: "s3",
},
StorageOptions::Blob(args) => Config {
options: args.options,
storage: Arc::new(args.storage),
storage_name: "blob_store",
storage_name: "s3",
},
StorageOptionsEnum::Blob(args) => Config {
options: args.options,
storage: Arc::new(args.storage),
storage_name: "blob_store",
},
},
Cli::Completion(completion_opts) => {
let shell = completion_opts
.shell
.unwrap_or(crate::cli::ShellCompletion::Bash);
let mut cmd = Cli::command();
let output = shell.generate(&mut cmd);
println!("{}", String::from_utf8_lossy(&output));
std::process::exit(0);
unreachable!()

Check warning on line 90 in src/option.rs

View workflow job for this annotation

GitHub Actions / Build for aarch64-unknown-linux-gnu

unreachable expression

Check warning on line 90 in src/option.rs

View workflow job for this annotation

GitHub Actions / Build for x86_64-unknown-linux-gnu

unreachable expression
}
}
}

Expand Down
Loading