radicle_cli/commands/node/
args.rs1use std::ffi::OsString;
2use std::fmt::Debug;
3use std::path::PathBuf;
4use std::str::FromStr;
5
6use thiserror::Error;
7
8use clap::{Parser, Subcommand};
9
10use radicle::crypto::{PublicKey, PublicKeyError};
11use radicle::node::{Address, NodeId, PeerAddr, PeerAddrParseError};
12use radicle::prelude::RepoId;
13
14const ABOUT: &str = "Control and query the Radicle Node";
15
16#[derive(Parser, Debug)]
17#[command(about = ABOUT, long_about, disable_version_flag = true)]
18pub struct Args {
19 #[command(subcommand)]
20 pub(super) command: Option<Command>,
21}
22
23#[derive(Clone, Debug)]
25pub(super) enum Addr {
26 Peer(PeerAddr<NodeId, Address>),
28 Node(NodeId),
30}
31
32#[derive(Error, Debug)]
33pub(super) enum AddrParseError {
34 #[error("{0}, expected <NID> or <NID>@<ADDR>")]
35 PeerAddr(#[from] PeerAddrParseError<PublicKey>),
36 #[error(transparent)]
37 NodeId(#[from] PublicKeyError),
38}
39
40impl FromStr for Addr {
41 type Err = AddrParseError;
42
43 fn from_str(s: &str) -> Result<Self, Self::Err> {
44 if s.contains("@") {
45 PeerAddr::from_str(s)
46 .map(Self::Peer)
47 .map_err(AddrParseError::PeerAddr)
48 } else {
49 NodeId::from_str(s)
50 .map(Self::Node)
51 .map_err(AddrParseError::NodeId)
52 }
53 }
54}
55
56#[derive(Clone, Debug)]
57pub enum Only {
58 Nid,
59}
60
61#[derive(Error, Debug)]
62#[error("could not parse value `{0}`")]
63pub struct OnlyParseError(String);
64
65impl FromStr for Only {
66 type Err = OnlyParseError;
67
68 fn from_str(value: &str) -> Result<Self, Self::Err> {
69 match value {
70 "nid" => Ok(Only::Nid),
71 _ => Err(OnlyParseError(value.to_string())),
72 }
73 }
74}
75
76#[derive(Clone, Debug)]
77struct OnlyParser;
78
79impl clap::builder::TypedValueParser for OnlyParser {
80 type Value = Only;
81
82 fn parse_ref(
83 &self,
84 cmd: &clap::Command,
85 arg: Option<&clap::Arg>,
86 value: &std::ffi::OsStr,
87 ) -> Result<Self::Value, clap::Error> {
88 <Only as std::str::FromStr>::from_str.parse_ref(cmd, arg, value)
89 }
90
91 fn possible_values(
92 &self,
93 ) -> Option<Box<dyn Iterator<Item = clap::builder::PossibleValue> + '_>> {
94 use clap::builder::PossibleValue;
95 Some(Box::new([PossibleValue::new("nid")].into_iter()))
96 }
97}
98
99#[derive(Subcommand, Debug)]
100pub(super) enum Command {
101 Connect {
103 #[arg(value_name = "NID[@ADDR]")]
105 addr: Addr,
106
107 #[arg(long, value_name = "SECS")]
109 timeout: Option<u64>,
110 },
111
112 Config {
114 #[arg(long)]
116 addresses: bool,
117 },
118
119 #[command(subcommand, hide = true)]
121 Db(DbOperation),
122
123 Events {
130 #[arg(long, value_name = "SECS")]
132 timeout: Option<u64>,
133
134 #[arg(long, short = 'n')]
136 count: Option<usize>,
137 },
138
139 Routing {
141 #[arg(long)]
143 json: bool,
144
145 #[arg(long)]
147 rid: Option<RepoId>,
148
149 #[arg(long)]
151 nid: Option<NodeId>,
152 },
153
154 Start {
156 #[arg(long)]
158 foreground: bool,
159
160 #[arg(long, short)]
162 verbose: bool,
163
164 #[arg(long, default_value = "radicle-node")]
166 path: PathBuf,
167
168 #[arg(value_name = "NODE_OPTIONS", last = true, num_args = 1..)]
172 options: Vec<OsString>,
173 },
174
175 Logs {
177 #[arg(long, value_name = "COUNT", default_value_t = 60)]
179 lines: usize,
180 },
181
182 Status {
184 #[arg(long, value_parser = OnlyParser)]
186 only: Option<Only>,
187 },
188
189 Inventory {
191 #[arg(long)]
193 nid: Option<NodeId>,
194 },
195
196 Debug,
200
201 #[command(hide = true)]
205 Sessions,
206
207 Stop,
209}
210
211impl Default for Command {
212 fn default() -> Self {
213 Command::Status { only: None }
214 }
215}
216
217#[derive(Debug, Subcommand)]
219pub(super) enum DbOperation {
220 Exec {
228 #[arg(value_name = "SQL")]
229 query: String,
230 },
231}