plenum-bot/src/config_check.rs

358 lines
14 KiB
Rust
Raw Normal View History

2024-07-29 15:12:26 +02:00
use crate::key_value::KeyValueStore as KV;
use std::error::Error;
use std::io::{self, Write};
2024-07-30 03:57:24 +02:00
/// Text shown for an empty field.
const EMPTY_VALUE : &str = "(empty)";
/// Text shown in place of a password.
const HIDDEN_PASSWORD : &str = "*****";
// highlight some of the info
const ANSI_GROUP: &str = "\x1b[1;31m";
const ANSI_FIELD: &str = "\x1b[1;33m";
const ANSI_NOTICE: &str= "\x1b[33m";
const ANSI_RESET: &str = "\x1b[0m";
const CONFIG_SPEC: CfgSpec<'static> = CfgSpec {
2024-07-29 15:12:26 +02:00
groups: &[
CfgGroup { name: "config",
description: "(internal values)",
fields: &[
CfgField::Silent { key: "version", default: "1" },
],
},
CfgGroup { name: "hedgedoc",
description: "HedgeDoc markdown pad server settings",
fields: &[
CfgField::Default { key: "server-url", default: "https://md.berlin.ccc.de/", description: "Hedgedoc server storing the pads." },
CfgField::Default { key: "template-name", default: "plenum-template", description: "Name of the pad containing the template to use." },
// TODO: make these generators?
CfgField::Optional { key: "last-id", description: "ID of last plenum's pad." },
CfgField::Optional { key: "next-id", description: "ID of next plenum's pad." }
],
},
CfgGroup { name: "email",
description: "Sending emails.",
fields: &[
CfgField::Default { key: "server", default: "mail.berlin.ccc.de", description: "SMTP server used for sending emails." },
CfgField::Default { key: "user", default: "plenum-bot@berlin.ccc.de", description: "User name used for authenticating with the mail server." },
CfgField::Password { key: "password", description: "Password for authenticating with the mail server." },
CfgField::Default { key: "sender", default: "Plenumsbot <plenum-bot@berlin.ccc.de>", description: "Email address to use for \"From:\"." },
CfgField::Default { key: "to", default: "CCCB Intern <intern@berlin.ccc.de>", description: "Recipient of the emails sent." },
CfgField::Optional { key: "last-message-id", description: "Message-Id of last initial announcement to send In-Reply-To (if applicable)." }
],
},
// TODO: Matrix, Wiki
CfgGroup { name: "text",
description: "Various strings used.",
fields: &[
CfgField::Default { key: "email-signature",
default: "\n\n[Diese Nachricht wurde automatisiert vom Plenumsbot erstellt und ist daher ohne Unterschrift gültig.]",
description: "Text added at the bottom of every email."
},
],
},
],
};
/// Ensure all fields with known default values exist.
pub fn populate_defaults(config: &KV) {
for group in CONFIG_SPEC.groups {
for field in group.fields {
match field {
2024-07-30 03:57:24 +02:00
CfgField::Silent { default, .. } => {
config.default(&field.full_key(group.name), default)
2024-07-29 15:12:26 +02:00
},
2024-07-30 03:57:24 +02:00
CfgField::Default { default, .. } => {
config.default(&field.full_key(group.name), default)
2024-07-29 15:12:26 +02:00
},
_ => {}
}
}
}
}
2024-07-30 03:57:24 +02:00
/// Check all groups / fields and interactively edit as needed.
///
/// Will report if the config is fine or has missing values.
2024-07-29 15:12:26 +02:00
pub fn interactive_check(config: KV) -> Result<(), Box<dyn Error>> {
2024-07-30 03:57:24 +02:00
let mut all_valid : bool = true;
2024-07-29 15:12:26 +02:00
for group in CONFIG_SPEC.groups {
group_header( group.name, group.description );
2024-07-30 03:57:24 +02:00
let todo = needs_info(&config, group);
2024-07-30 13:23:51 +02:00
// TODO: add distinction between edit all / edit necessary only
2024-07-30 03:57:24 +02:00
let choices = if !todo.is_empty() {
println!( "{}The following fields need adjustment: {}{}", ANSI_NOTICE, todo.join(", "), ANSI_RESET );
&["[E]dit (default)", "[L]ist all", "[S]kip group"]
} else {
println!( "{}This group looks fine. OK to [S]kip, unless you want to adjust values here.{}", ANSI_NOTICE, ANSI_RESET );
&["[S]kip group (default)", "[E]dit", "[L]ist all", ]
};
loop {
let choice = prompt_action( choices );
match choice {
'L' => { show_group(&config, group); },
'E' => {
for field in group.fields {
check_field( &config, group.name, field, &mut all_valid )?;
}
break;
},
'S' => {
if !todo.is_empty() { all_valid = false; }
break;
},
2024-08-01 06:50:17 +02:00
_ => { unreachable!(); }, // (prompt already checks)
2024-07-30 03:57:24 +02:00
}
2024-07-29 15:12:26 +02:00
}
}
2024-07-30 03:57:24 +02:00
if all_valid {
Ok(())
} else {
Err("WARNING: Config is incomplete. Re-run to fix.".into())
}
2024-07-29 15:12:26 +02:00
}
2024-07-30 03:57:24 +02:00
fn check_field( config: &KV, grpname: &str, field: &CfgField, ok: &mut bool ) -> Result<(), Box<dyn Error>> {
2024-07-29 15:12:26 +02:00
if field.is_silent() { return Ok(()) }
2024-07-30 03:57:24 +02:00
show_field(config, grpname, field);
2024-07-29 15:12:26 +02:00
let key = field.full_key(grpname);
let value = config.get(&key).ok();
2024-07-30 03:57:24 +02:00
let mut actions: Vec<&'static str> = Vec::new();
2024-07-30 13:23:51 +02:00
// TODO: adjust order: empty password should offer input first
// TODO: RandomId should offer generating as [R]egenerate
if value.is_some() | field.is_password() | field.is_optional() { actions.push("[K]eep as-is"); }
if field.default_description().is_some() { actions.push("[R]eset to default"); }
if field.is_password() { actions.push( "[R]emove" ); }
actions.push( "[I]nput new value" );
if !field.is_password() { actions.push( "[L]ong (multiline) input" ) };
match prompt_action(&actions) {
'K' => {
// we allow leaving a password empty, but that means config is incomplete
if value.is_none() & !field.is_optional() { *ok &= false; }
},
'R' => {
match field.default_value()? {
Some(value) => { config.set(&key, &value)?; },
// password again
None => { config.delete(&key)?; *ok &= false; },
}
},
'I' => {
let value = if field.is_password() { prompt_password() } else { prompt_single_line() };
config.set(&key, &value)?;
},
'L' => {
let value = prompt_multiline();
config.set(&key, &value)?;
},
_ => { return Err("Wat.".into()); }
2024-07-29 15:12:26 +02:00
}
Ok(())
}
2024-07-30 03:57:24 +02:00
/* ***** displaying various kinds of info ***** */
2024-07-29 15:12:26 +02:00
fn group_header( name: &str, description: &str ) {
println!("==============================");
println!("{}{}{} - {}", ANSI_GROUP, name, ANSI_RESET, description);
println!("");
}
2024-07-30 03:57:24 +02:00
fn show_group( config: &KV, group: &CfgGroup ) {
for field in group.fields {
show_field( &config, group.name, field );
}
}
fn show_field( config: &KV, grpname: &str, field: &CfgField ) {
if field.is_silent() { return }
let key = field.full_key(grpname);
println!( "{}{}{} - {}", ANSI_FIELD, &key, ANSI_RESET, field.description() );
2024-07-30 13:23:51 +02:00
println!( " default: {}", field.default_description().unwrap_or(EMPTY_VALUE.to_string()));
2024-07-30 03:57:24 +02:00
let value = config.get(&key).ok()
.map(|s| if field.is_password() { HIDDEN_PASSWORD.to_string() } else { s })
.unwrap_or(EMPTY_VALUE.to_string());
println!( " current: {}", value );
}
/* ***** basic validation ***** */
fn needs_info( config: &KV, group: &CfgGroup ) -> Vec<String> {
let mut acc = Vec::new();
for field in group.fields {
if field.is_action_required(config, group.name) {
acc.push( field.key() );
}
}
acc
}
/* ***** prompting (with actual I/O) ***** */
/// Ask for a choice between several options. First option is the default,
/// format options with the first character in brackets like so: `"[E]xample"`.
fn prompt_action( choices: &[&str] ) -> char {
let prompt_message = choices.join(", ");
let default_choice = choices[0].to_uppercase().chars().nth(1).unwrap_or_default();
loop {
println!( "{}", prompt_message );
print!( "Select: " );
io::stdout().flush().ok();
// Read line and take first non-space character
let mut input = String::new();
io::stdin().read_line(&mut input).unwrap();
let input = input.trim().to_uppercase().chars().next().unwrap_or(default_choice);
// Check list of choices for match
if choices.iter().any(|&choice| choice.to_uppercase().chars().nth(1).unwrap_or_default() == input) {
return input;
} else {
println!("Invalid choice. Try again!");
}
}
}
/// Read a single line of text.
2024-07-29 15:12:26 +02:00
fn prompt_single_line( ) -> String {
print!("New value: ");
io::stdout().flush().ok();
let mut input = String::new();
io::stdin().read_line(&mut input).unwrap();
input.trim().to_string()
}
2024-07-30 03:57:24 +02:00
/// Read multiple lines of text, terminated by a line containing just a '.'.
2024-07-29 15:12:26 +02:00
fn prompt_multiline( ) -> String {
println!("Enter new value: (end with '.' on a new line)");
let mut acc = String::new();
loop {
let mut line = String::new();
io::stdin().read_line(&mut line).unwrap();
if line.trim() == "." {
break;
}
acc.push_str(&line);
}
acc
}
2024-07-30 03:57:24 +02:00
/// Read a password without echoing it.
2024-07-29 15:12:26 +02:00
fn prompt_password( ) -> String {
2024-07-30 13:23:51 +02:00
let pass = rpassword::prompt_password("New password (not shown) : ").unwrap();
// disabled echo means the newline also isn't shown
println!("");
pass
2024-07-29 15:12:26 +02:00
}
2024-07-30 03:57:24 +02:00
/* ***** config structure definition ***** */
2024-07-29 15:12:26 +02:00
/// Describes a field in the configuration.
enum CfgField<'a> {
/// Silently writes a default to the DB if absent, never prompts for changes.
Silent { key: &'a str, default: &'a str },
/// Writes a default, asks for changes.
Default { key: &'a str, default: &'a str, description: &'a str },
/// No default, asks for value, but can stay empty.
Optional { key: &'a str, description: &'a str },
/// Empty by default, required, will prompt without echo.
Password { key: &'a str, description: &'a str },
/// Empty by default, can be user-provided, or will be generated randomly.
2024-07-30 13:23:51 +02:00
RandomId { key: &'a str, generator: fn() -> Result<String,Box<dyn Error>>, generator_description: &'a str, description: &'a str },
2024-07-29 15:12:26 +02:00
}
/// A group of related config fields. The final key of an inner value will be
/// `{group.name}-{key}`.
struct CfgGroup<'a> {
name: &'a str,
description: &'a str,
fields: &'a [CfgField<'a>],
}
/// A description of the entire config, as a collection of named groups.
///
/// Used both to populate defaults and to interactively adjust the configuration.
struct CfgSpec<'a> {
groups: &'a [CfgGroup<'a>],
}
impl<'a> CfgField<'a> {
2024-07-30 03:57:24 +02:00
/// Silent fields don't get prompted or shown.
2024-07-29 15:12:26 +02:00
fn is_silent(&self) -> bool {
2024-07-30 03:57:24 +02:00
matches!( self, CfgField::Silent { .. } )
2024-07-29 15:12:26 +02:00
}
2024-07-30 03:57:24 +02:00
/// Password fields will be censored when displayed.
2024-07-29 15:12:26 +02:00
fn is_password(&self) -> bool {
2024-07-30 03:57:24 +02:00
matches!( self, CfgField::Password { .. } )
}
2024-07-30 13:23:51 +02:00
/// Optional fields are allowed to be (and stay) empty.
fn is_optional(&self) -> bool {
matches!( self, CfgField::Optional { .. } )
}
2024-07-30 03:57:24 +02:00
/// Reports if the field needs changing or is ok as-is.
///
/// (Because of the default population at boot, Silent and Default must
/// *necessarily* always have a value. Optional is optional. Only
/// Password and RandomId might need actions if they are missing.)
fn is_action_required(&self, config: &KV, grpname: &str) -> bool {
if matches!( self, CfgField::Silent { .. } | CfgField::Default { .. } | CfgField::Optional { .. } ) {
false
} else {
config.get(self.full_key(grpname).as_str()).ok().is_none()
2024-07-29 15:12:26 +02:00
}
}
2024-07-30 03:57:24 +02:00
/// Gets `key`, which is the local part of the field name. (`.full_key` is used for the DB.)
2024-07-29 15:12:26 +02:00
fn key(&self) -> String {
match self {
CfgField::Silent { key, .. } => key,
CfgField::Default { key, .. } => key,
CfgField::Optional { key, .. } => key,
CfgField::Password { key, .. } => key,
CfgField::RandomId { key, .. } => key,
}.to_string()
}
2024-07-30 03:57:24 +02:00
/// Full field name / key used in the DB (currently `{grpname}-{fieldname}`.)
2024-07-29 15:12:26 +02:00
fn full_key(&self, grpname: &str) -> String {
format!("{}-{}", grpname, self.key())
}
2024-07-30 03:57:24 +02:00
/// Gets `description` or "(none)".
2024-07-29 15:12:26 +02:00
fn description(&self) -> String {
match self {
CfgField::Silent { .. } => "(none)",
CfgField::Default { description, .. } => description,
CfgField::Optional { description, .. } => description,
CfgField::Password { description, .. } => description,
CfgField::RandomId { description, .. } => description,
}.to_string()
}
2024-07-30 03:57:24 +02:00
/// Gets a description of the default value if one exists.
/// (Either the value itself, or a description of the generator making a default value.)
2024-07-30 13:23:51 +02:00
fn default_description(&self) -> Option<String> {
2024-07-29 15:12:26 +02:00
match self {
CfgField::Silent { default, .. } => Some(default.to_string()),
CfgField::Default { default, .. } => Some(default.to_string()),
CfgField::RandomId { generator_description, .. } => Some(format!("{}{}{}",'(',generator_description,')')),
_ => None,
}
}
2024-07-30 13:23:51 +02:00
/// Gets the actual default value, which may involve running a generator.
fn default_value(&self) -> Result<Option<String>, Box<dyn Error>> {
match self {
CfgField::Silent { default, .. } => Ok(Some(default.to_string())),
CfgField::Default { default, .. } => Ok(Some(default.to_string())),
CfgField::RandomId { generator, .. } => generator().map(Some),
_ => Ok(None),
}
}
2024-07-29 15:12:26 +02:00
}