-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathcli.rs
executable file
·91 lines (77 loc) · 2.57 KB
/
cli.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
use clap::{AppSettings, Clap};
use std::fs;
use std::io::Write;
use std::path::PathBuf;
use client::{Client, DResult, JobId, JobStartRequest, OutputStream, TlsConfig};
#[derive(Clap)]
#[clap(version, author)]
#[clap(setting = AppSettings::ColoredHelp)]
struct Opts {
server: String,
#[clap(
short = 's',
long = "server-root-ca-crt",
env = "TLS_SERVER_ROOT_CA_CRT"
)]
server_root_ca_crt: PathBuf,
#[clap(short = 'c', long = "client-crt", env = "TLS_CLIENT_CRT")]
client_crt: PathBuf,
#[clap(short = 'k', long = "client-key", env = "TLS_CLIENT_KEY")]
client_key: PathBuf,
#[clap(subcommand)]
subcmd: Operation,
}
#[derive(Clap)]
enum Operation {
Start { path: String, args: Vec<String> },
Stop { jobid: String },
Status { jobid: String },
Wait { jobid: String },
Output { jobid: String },
}
#[tokio::main]
async fn main() {
env_logger::init();
if let Err(err) = inner().await {
eprintln!("Error: {}", err);
std::process::exit(1);
}
}
async fn inner() -> DResult<()> {
let opts: Opts = Opts::parse();
let tls = TlsConfig {
server_root_ca_crt: fs::read(opts.server_root_ca_crt)?,
client_crt: fs::read(opts.client_crt)?,
client_key: fs::read(opts.client_key)?,
};
let mut client = Client::connect(opts.server, tls).await?;
match opts.subcmd {
Operation::Start { path, args } => {
let jobid = client.start(JobStartRequest { path, args }).await?;
println!("{}", jobid);
}
Operation::Stop { jobid } => client.stop(JobId::parse(&jobid)?).await?,
Operation::Status { jobid } => {
println!("{}", client.status(JobId::parse(&jobid)?).await?);
}
Operation::Wait { jobid } => {
println!("{}", client.wait(JobId::parse(&jobid)?).await?);
}
Operation::Output { jobid } => {
let (tx, mut rx) = tokio::sync::mpsc::channel(2);
client.output(JobId::parse(&jobid)?, tx).await?;
let error_msg = "writing out output stream failed";
let mut out = std::io::stdout();
let mut err = std::io::stderr();
while let Some(event) = rx.recv().await {
match event? {
(OutputStream::Stdout, data) => out.write_all(&data).expect(error_msg),
(OutputStream::Stderr, data) => err.write_all(&data).expect(error_msg),
}
}
out.flush().expect(error_msg);
err.flush().expect(error_msg);
}
}
Ok(())
}