forked from talaia-labs/rust-teos
-
Notifications
You must be signed in to change notification settings - Fork 0
/
config.rs
445 lines (388 loc) · 14.1 KB
/
config.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
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
//! Logic related to the tower configuration and command line parameter parsing.
use serde::{Deserialize, Serialize};
use std::path::PathBuf;
use structopt::StructOpt;
pub fn data_dir_absolute_path(data_dir: String) -> PathBuf {
if let Some(a) = data_dir.strip_prefix('~') {
if let Some(b) = data_dir.strip_prefix("~/") {
home::home_dir().unwrap().join(b)
} else {
home::home_dir().unwrap().join(a)
}
} else {
PathBuf::from(&data_dir)
}
}
pub fn from_file<T: Default + serde::de::DeserializeOwned>(path: &PathBuf) -> T {
match std::fs::read(path) {
Ok(file_content) => toml::from_slice::<T>(&file_content).map_or_else(
|e| {
eprintln!("Couldn't parse config file: {e}");
T::default()
},
|config| config,
),
Err(_) => T::default(),
}
}
/// Error raised if something is wrong with the configuration.
#[derive(PartialEq, Eq, Debug)]
pub struct ConfigError(String);
impl std::fmt::Display for ConfigError {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
write!(f, "Configuration error: {}", self.0)
}
}
impl std::error::Error for ConfigError {}
#[derive(PartialEq)]
pub enum AuthMethod {
UserPass,
CookieFile,
Multiple,
Invalid,
}
/// Holds all the command line options.
#[derive(StructOpt, Debug, Clone)]
#[structopt(rename_all = "lowercase")]
#[structopt(version = env!("CARGO_PKG_VERSION"), about = "The Eye of Satoshi - Lightning watchtower")]
pub struct Opt {
/// Address teos HTTP(s) API will bind to [default: localhost]
#[structopt(long)]
pub api_bind: Option<String>,
/// Port teos HTTP(s) API will bind to [default: 9814]
#[structopt(long)]
pub api_port: Option<u16>,
/// Address teos RPC server will bind to [default: localhost]
#[structopt(long)]
pub rpc_bind: Option<String>,
/// Port teos RPC server will bind to [default: 8814]
#[structopt(long)]
pub rpc_port: Option<u16>,
/// Network bitcoind is connected to. Either mainnet, testnet, signet or regtest [default: mainnet]
#[structopt(long)]
pub btc_network: Option<String>,
/// bitcoind rpcuser
#[structopt(long)]
pub btc_rpc_user: Option<String>,
/// bitcoind rpcpassword
#[structopt(long)]
pub btc_rpc_password: Option<String>,
/// bitcoind rpccookie
#[structopt(long)]
pub btc_rpc_cookie: Option<String>,
/// bitcoind rpcconnect [default: localhost]
#[structopt(long)]
pub btc_rpc_connect: Option<String>,
/// bitcoind rpcport [default: 8332]
#[structopt(long)]
pub btc_rpc_port: Option<u16>,
/// Specify data directory
#[structopt(long, default_value = "~/.teos")]
pub data_dir: String,
/// Runs teos in debug mode
#[structopt(long)]
pub debug: bool,
/// Runs third party libs in debug mode
#[structopt(long)]
pub deps_debug: bool,
/// Overwrites the tower secret key. THIS IS IRREVERSIBLE AND WILL CHANGE YOUR TOWER ID
#[structopt(long)]
pub overwrite_key: bool,
/// If set, creates a Tor endpoint to serve API data. This endpoint is additional to the clearnet HTTP API
#[structopt(long)]
pub tor_support: bool,
/// Forces the tower to run even if the underlying chain has gone too far out of sync. This can only happen
/// if the node is being run in pruned mode.
#[structopt(long)]
pub force_update: bool,
/// Tor control port [default: 9051]
#[structopt(long)]
pub tor_control_port: Option<u16>,
/// Port for the onion hidden service to listen on [default: 9814]
#[structopt(long)]
pub onion_hidden_service_port: Option<u16>,
}
/// Holds all configuration options.
///
/// The overwrite policy goes, from less to more:
/// - Defaults
/// - Configuration file
/// - Command line options
#[derive(Debug, Deserialize, Serialize, Clone, PartialEq, Eq)]
#[serde(default)]
pub struct Config {
// API
pub api_bind: String,
pub api_port: u16,
// RPC
pub rpc_bind: String,
pub rpc_port: u16,
// Bitcoind
pub btc_network: String,
pub btc_rpc_user: String,
pub btc_rpc_cookie: String,
pub btc_rpc_password: String,
pub btc_rpc_connect: String,
pub btc_rpc_port: u16,
// Flags
pub debug: bool,
pub deps_debug: bool,
pub overwrite_key: bool,
pub force_update: bool,
// General
pub subscription_slots: u32,
pub subscription_duration: u32,
pub expiry_delta: u32,
pub min_to_self_delay: u16,
pub polling_delta: u16,
// Internal API
pub internal_api_bind: String,
pub internal_api_port: u32,
// Tor
pub tor_support: bool,
pub tor_control_port: u16,
pub onion_hidden_service_port: u16,
}
impl Config {
/// The only combinations of valid authentication methods are:
/// - User **AND** password
/// - **OR** Cookie file
//
/// Any other combination will be rejected
pub fn get_auth_method(&self) -> AuthMethod {
match (
self.btc_rpc_user.is_empty(),
self.btc_rpc_password.is_empty(),
self.btc_rpc_cookie.is_empty(),
) {
(false, false, true) => AuthMethod::UserPass,
(true, true, false) => AuthMethod::CookieFile,
(true, true, true) => AuthMethod::Invalid,
_ => AuthMethod::Multiple,
}
}
/// Patches the configuration options with the command line options.
pub fn patch_with_options(&mut self, options: Opt) {
if options.api_bind.is_some() {
self.api_bind = options.api_bind.unwrap();
}
if options.api_port.is_some() {
self.api_port = options.api_port.unwrap();
}
if options.rpc_bind.is_some() {
self.rpc_bind = options.rpc_bind.unwrap();
}
if options.rpc_port.is_some() {
self.rpc_port = options.rpc_port.unwrap();
}
if options.btc_network.is_some() {
self.btc_network = options.btc_network.unwrap();
}
if options.btc_rpc_user.is_some() {
self.btc_rpc_user = options.btc_rpc_user.unwrap();
}
if options.btc_rpc_password.is_some() {
self.btc_rpc_password = options.btc_rpc_password.unwrap();
}
if options.btc_rpc_cookie.is_some() {
self.btc_rpc_cookie = options.btc_rpc_cookie.unwrap();
}
if options.btc_rpc_connect.is_some() {
self.btc_rpc_connect = options.btc_rpc_connect.unwrap();
}
if options.btc_rpc_port.is_some() {
self.btc_rpc_port = options.btc_rpc_port.unwrap();
}
if options.tor_control_port.is_some() {
self.tor_control_port = options.tor_control_port.unwrap();
}
if options.onion_hidden_service_port.is_some() {
self.onion_hidden_service_port = options.onion_hidden_service_port.unwrap();
}
self.tor_support |= options.tor_support;
self.debug |= options.debug;
self.deps_debug |= options.deps_debug;
self.overwrite_key = options.overwrite_key;
self.force_update = options.force_update;
}
/// Verifies that [Config] is properly built.
///
/// This includes:
/// - `bitcoind` credentials have been set
/// - The Bitcoin network has been properly set (to either bitcoin, testnet, signet or regtest)
///
/// This will also assign the default `btc_rpc_port` depending on the network if it has not
/// been overwritten at this point.
pub fn verify(&mut self) -> Result<(), ConfigError> {
let auth_method = self.get_auth_method();
if auth_method == AuthMethod::Invalid {
return Err(ConfigError("No valid bitcoind auth provided. Set either both btc_rpc_user/btc_rpc_password or btc_rpc_cookie".to_owned()));
} else if auth_method == AuthMethod::Multiple {
return Err(ConfigError(
"Multiple bitcoind auth provided. Pick a single one (either btc_rpc_user/btc_rpc_password or btc_rpc_cookie)"
.to_owned(),
));
}
// Normalize the network option to the ones used by bitcoind.
if ["mainnet", "testnet"].contains(&self.btc_network.as_str()) {
self.btc_network = self.btc_network.trim_end_matches("net").into();
}
let default_rpc_port = match self.btc_network.as_str() {
"main" => 8332,
"test" => 18332,
"regtest" => 18443,
"signet" => 38332,
_ => return Err(ConfigError(format!("btc_network not recognized. Expected {{mainnet, testnet, signet, regtest}}, received {}", self.btc_network)))
};
// Set the port to it's default (depending on the network) if it has not been
// overwritten at this point.
if self.btc_rpc_port == 0 {
self.btc_rpc_port = default_rpc_port;
}
Ok(())
}
/// Checks whether the config has been set with only with default values.
pub fn is_default(&self) -> bool {
self == &Config::default()
}
/// Logs non-default options.
pub fn log_non_default_options(&self) {
let json_default_config = serde_json::json!(&Config::default());
let json_config = serde_json::json!(&self);
let sensitive_args = ["btc_rpc_user", "btc_rpc_password"];
for (key, value) in json_config.as_object().unwrap().iter() {
if *value != json_default_config[key] {
log::info!(
"Custom config arg: {}: {}",
key,
if sensitive_args.contains(&key.as_str()) {
"****".to_owned()
} else {
value.to_string()
}
);
}
}
}
}
impl Default for Config {
/// Sets the tower [Config] defaults.
///
/// Notice the defaults are not enough, and the tower will refuse to run on them.
/// For instance, the defaults do set the `bitcoind` `rpu_user` and `rpc_password`
/// to empty strings so the user is forced the set them (and most importantly so the
/// user does not use any values provided here).
fn default() -> Self {
Self {
api_bind: "127.0.0.1".into(),
api_port: 9814,
tor_support: false,
tor_control_port: 9051,
onion_hidden_service_port: 9814,
rpc_bind: "127.0.0.1".into(),
rpc_port: 8814,
btc_network: "mainnet".into(),
btc_rpc_user: String::new(),
btc_rpc_password: String::new(),
btc_rpc_cookie: String::new(),
btc_rpc_connect: "localhost".into(),
btc_rpc_port: 0,
debug: false,
deps_debug: false,
overwrite_key: false,
force_update: false,
subscription_slots: 10000,
subscription_duration: 4320,
expiry_delta: 6,
min_to_self_delay: 20,
polling_delta: 60,
internal_api_bind: "127.0.0.1".into(),
internal_api_port: 50051,
}
}
}
#[cfg(test)]
mod tests {
use super::*;
impl Default for Opt {
fn default() -> Self {
Self {
api_bind: None,
api_port: None,
tor_support: false,
tor_control_port: None,
onion_hidden_service_port: None,
rpc_bind: None,
rpc_port: None,
btc_network: None,
btc_rpc_user: None,
btc_rpc_password: None,
btc_rpc_cookie: None,
btc_rpc_connect: None,
btc_rpc_port: None,
data_dir: String::from("~/.teos"),
debug: false,
deps_debug: false,
overwrite_key: false,
force_update: false,
}
}
}
#[test]
fn test_config_patch_with_options() {
// Tests that a given Config is overwritten with Opts if the options are present
let mut config = Config::default();
let config_clone = config.clone();
let mut opt = Opt::default();
let expected_value = String::from("test");
opt.api_bind = Some(expected_value.clone());
config.patch_with_options(opt);
// Check the field has been updated
assert_eq!(config.api_bind, expected_value);
// Check the rest of fields are equal. The easiest is to just the field back and compare with a clone
config.api_bind = config_clone.api_bind.clone();
assert_eq!(config, config_clone);
}
#[test]
fn test_config_default_not_verify() {
// Tests that the default configuration does not pass verification checks. This is on purpose so some fields are
// required to be updated by the user.
let mut config = Config::default();
assert!(
matches!(config.verify(), Err(ConfigError(e)) if e.contains("No valid bitcoind auth provided"))
);
}
#[test]
fn test_config_default_verify_overwrite_required() {
// Tests that setting a some btc_rpc_user and btc_rpc_password results in a Config object that verifies
let mut config = Config {
btc_rpc_user: "user".to_owned(),
btc_rpc_password: "password".to_owned(),
..Default::default()
};
config.verify().unwrap();
}
#[test]
fn test_config_verify_wrong_network() {
// Tests that setting a wrong network will make verify fail
let mut config = Config {
btc_rpc_user: "user".to_owned(),
btc_rpc_password: "password".to_owned(),
btc_network: "wrong_network".to_owned(),
..Default::default()
};
assert!(
matches!(config.verify(), Err(ConfigError(e)) if e.contains("btc_network not recognized"))
);
}
#[test]
fn test_config_verify_tor_set() {
let mut config = Config {
btc_rpc_user: "user".to_owned(),
btc_rpc_password: "password".to_owned(),
tor_support: true,
..Default::default()
};
config.verify().unwrap()
}
}