Skip to content

Commit 619993b

Browse files
feat: clap_completion
- autocompletion for the cli
1 parent d332358 commit 619993b

File tree

7 files changed

+261
-36
lines changed

7 files changed

+261
-36
lines changed

Cargo.lock

+10
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Cargo.toml

+2-1
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,7 @@ bzip2 = { version = "*", features = ["static"] }
4242
cookie = "0.18.1"
4343
chrono = "0.4"
4444
chrono-humanize = "0.2"
45-
clap = { version = "4.1", default-features = false, features = [
45+
clap = { version = "4.5", default-features = false, features = [
4646
"std",
4747
"color",
4848
"help",
@@ -51,6 +51,7 @@ clap = { version = "4.1", default-features = false, features = [
5151
"cargo",
5252
"error-context",
5353
] }
54+
clap_complete = "4.5"
5455
clokwerk = "0.4"
5556
crossterm = "0.28.1"
5657
derive_more = "0.99.18"

src/completions.rs

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
/*
2+
* Parseable Server (C) 2022 - 2024 Parseable, Inc.
3+
*
4+
* This program is free software: you can redistribute it and/or modify
5+
* it under the terms of the GNU Affero General Public License as
6+
* published by the Free Software Foundation, either version 3 of the
7+
* License, or (at your option) any later version.
8+
*
9+
* This program is distributed in the hope that it will be useful,
10+
* but WITHOUT ANY WARRANTY; without even the implied warranty of
11+
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12+
* GNU Affero General Public License for more details.
13+
*
14+
* You should have received a copy of the GNU Affero General Public License
15+
* along with this program. If not, see <http://www.gnu.org/licenses/>.
16+
*
17+
*/
18+
19+
use clap::Command;
20+
use clap_complete::{generate, Shell};
21+
use std::io;
22+
use std::path::PathBuf;
23+
24+
/// Generates a completion script for the specified shell.
25+
///
26+
/// If `output` is `Some(PathBuf)`, the script is written to the specified file.
27+
/// Otherwise, it's written to `stdout`.
28+
pub fn generate_completion_script(
29+
cmd: &mut Command,
30+
bin_name: &str,
31+
shell: Shell,
32+
output: Option<PathBuf>,
33+
) -> Result<(), io::Error> {
34+
if let Some(file_path) = output {
35+
let mut file = std::fs::File::create(file_path)?;
36+
generate(shell, cmd, bin_name, &mut file);
37+
} else {
38+
generate(shell, cmd, bin_name, &mut io::stdout());
39+
}
40+
Ok(())
41+
}

src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ pub mod analytics;
2222
pub mod banner;
2323
mod catalog;
2424
mod cli;
25+
mod completions;
2526
mod event;
2627
pub mod handlers;
2728
pub mod hottier;

src/main.rs

+42-21
Original file line numberDiff line numberDiff line change
@@ -30,28 +30,49 @@ async fn main() -> anyhow::Result<()> {
3030
.compact()
3131
.init();
3232

33-
// these are empty ptrs so mem footprint should be minimal
34-
let server: Box<dyn ParseableServer> = match CONFIG.parseable.mode {
35-
Mode::Query => Box::new(QueryServer),
36-
Mode::Ingest => Box::new(IngestServer),
37-
Mode::All => Box::new(Server),
38-
};
39-
40-
// load metadata from persistence
41-
let parseable_json = server.load_metadata().await?;
42-
let metadata = storage::resolve_parseable_metadata(&parseable_json).await?;
43-
banner::print(&CONFIG, &metadata).await;
44-
// initialize the rbac map
45-
rbac::map::init(&metadata);
46-
// keep metadata info in mem
47-
metadata.set_global();
48-
49-
// load kafka server
50-
if CONFIG.parseable.mode != Mode::Query {
51-
tokio::task::spawn(kafka::setup_integration());
52-
}
33+
match CONFIG.as_ref() {
34+
ParseableCommand::Completion { shell, output } => {
35+
let mut cmd = parseable::create_parseable_cli_command();
36+
let bin_name = cmd.get_name().to_owned();
37+
38+
parseable::completions::generate_completion_script(
39+
&mut cmd,
40+
&bin_name,
41+
*shell,
42+
output.clone(),
43+
)?;
44+
return Ok(());
45+
}
46+
ParseableCommand::LocalStore(config)
47+
| ParseableCommand::S3Store(config)
48+
| ParseableCommand::BlobStore(config) => {
49+
// these are empty ptrs so mem footprint should be minimal
50+
let server: Box<dyn ParseableServer> = match config.parseable.mode {
51+
Mode::Query => Box::new(QueryServer),
52+
Mode::Ingest => Box::new(IngestServer),
53+
Mode::All => Box::new(Server),
54+
};
55+
56+
// Load metadata from persistence
57+
let parseable_json = server.load_metadata().await?;
58+
let metadata = storage::resolve_parseable_metadata(&parseable_json).await?;
59+
banner::print(&CONFIG, &metadata).await;
60+
61+
// Initialize the RBAC map
62+
rbac::map::init(&metadata);
5363

54-
server.init().await?;
64+
// Keep metadata info in memory
65+
metadata.set_global();
66+
67+
// Load Kafka server if not in Query mode
68+
if config.parseable.mode != Mode::Query {
69+
tokio::task::spawn(kafka::setup_integration());
70+
}
71+
72+
// Initialize the server
73+
server.init().await?;
74+
}
75+
}
5576

5677
Ok(())
5778
}

0 commit comments

Comments
 (0)