diff options
Diffstat (limited to 'rust/src')
-rw-r--r-- | rust/src/main.rs | 351 |
1 files changed, 242 insertions, 109 deletions
diff --git a/rust/src/main.rs b/rust/src/main.rs index 3873850..083a99f 100644 --- a/rust/src/main.rs +++ b/rust/src/main.rs @@ -1,38 +1,60 @@ -use clap::{value_parser, Arg, ArgAction, Command}; +// +// Author : Dennis Eriksen <d@ennis.no> +// File : main.rs +// Bin : makepass.rs +// Created : 2023-09-05 +// Licence : BSD-3-Clause +// +// Copyright (c) 2018-2023 Dennis Eriksen <d@ennis.no> + +// +// Imports +// +use clap::{Arg, ArgAction, Command, crate_authors, value_parser}; use rand::{seq::SliceRandom, Rng}; use std::env; -use std::fs::File; +use std::fs; +use std::io; use std::io::prelude::*; -use std::io::BufReader; -use std::path::Path; -use std::process; +use std::process::exit; use terminal_size::{terminal_size, Width}; +// +// Constants +// const MAX: u8 = 255; const RANGE_MAX: u8 = 42; const RANGE_MIN: u8 = 8; const PASS_WORDS: u8 = 8; +// Character sets const LOWER: &str = "abcdefghijklmnopqrstuvwxyz"; const UPPER: &str = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"; const DIGIT: &str = "0123456789"; const OTHER: &str = "!$%&#/()=?+-_,.;:<>[]{}|@*"; const EXTRA: &str = "-_"; -const DEF_LENGTH: u8 = 0; -const DEF_NUMBER: u8 = 10; -const DEF_WORDLIST: &str = "/usr/share/dict/words"; +// Defaults +const DEFAULT_LENGTH: u8 = 0; +const DEFAULT_NUMBER: u8 = 10; +const DEFAULT_WORDLIST: &str = "/usr/share/dict/words"; +// Structure used to pass common data to the print-function struct CommonPrintData<'a> { length: u8, - alpha: &'a Vec<char>, - alnum: &'a Vec<char>, + alpha: &'a [char], + alnum: &'a [char], + printbool: bool, printlen: u8, col_width: u8, col_num: u16, } +// +// Main function. This is where the magic happens +// fn main() { + // Construct arrays of the character sets let lower = LOWER.chars().collect::<Vec<_>>(); let upper = UPPER.chars().collect::<Vec<_>>(); let digit = DIGIT.chars().collect::<Vec<_>>(); @@ -42,133 +64,111 @@ fn main() { let alnum = [&alpha[..], &digit[..]].concat(); let every = [&alnum[..], &other[..]].concat(); + // These are the sets we actually use let mut normal = [&alnum[..], &extra[..]].concat(); let mut special = every; - let mut length = DEF_LENGTH; - let mut number = DEF_NUMBER; - - let mut wordlist = DEF_WORDLIST.to_string(); + // Set the defaults + let mut length = DEFAULT_LENGTH; + let mut number = DEFAULT_NUMBER; + let mut wordlist = DEFAULT_WORDLIST.to_string(); // // env // Deal with env before cli-args, because cli-args take precedence. // - let env_length = env::var("MAKEPASS_LENGTH").unwrap_or_default(); - let env_number = env::var("MAKEPASS_NUMBER").unwrap_or_default(); - let env_wordlist = env::var("MAKEPASS_WORDLIST").unwrap_or_default(); - let env_normal = env::var("MAKEPASS_NORMAL").unwrap_or_default(); - let env_special = env::var("MAKEPASS_SPECIAL").unwrap_or_default(); - - if !env_length.is_empty() { - length = env_length.parse().unwrap_or_else(|_| { + let env = ( + env::var("MAKEPASS_LENGTH").unwrap_or_default(), + env::var("MAKEPASS_NUMBER").unwrap_or_default(), + env::var("MAKEPASS_WORDLIST").unwrap_or_default(), + env::var("MAKEPASS_NORMAL").unwrap_or_default(), + env::var("MAKEPASS_SPECIAL").unwrap_or_default(), + ); + + // length + if !env.0.is_empty() { + length = env.0.parse().unwrap_or_else(|_| { + // length is a <u8>, which is 0-255, so no need for further error-checking eprintln!( "Error: MAKEPASS_LENGTH is not a valid number. Valid numbers are between 0 and 255" ); - process::exit(1); + exit(1); }); } - if !env_number.is_empty() { - match env_number.parse::<u8>() { + // number + if !env.1.is_empty() { + // number is a <u8>, but takes a minimum value of 1 + match env.1.parse::<u8>() { Ok(n) if n >= 1 => number = n, _ => { eprintln!("Error: MAKEPASS_NUMBER is not a valid number. Valid numbers are between 1 and 255"); - process::exit(1); + exit(1); } } } - if !env_wordlist.is_empty() { - wordlist = env_wordlist; + if !env.2.is_empty() { // wordlist + wordlist = env.2; } - if !env_normal.is_empty() { - normal = env_number.chars().collect::<Vec<_>>(); + if !env.3.is_empty() { // Overwrite normal character array if MAKEPASS_NORMAL is present + normal = env.3.chars().collect::<Vec<_>>(); } - if !env_special.is_empty() { - special = env_special.chars().collect::<Vec<_>>(); + if !env.4.is_empty() { // Overwrite special character array if MAKEPASS_SPECIAL is present + special = env.4.chars().collect::<Vec<_>>(); } // // Args // + let opts = cli().get_matches(); - let opts = Command::new("makepass") - .arg( - Arg::new("length_flag") - .short('l') - .value_name("length") - .help(format!("Length of passwords (0..={MAX}). 0 = random.")) - .value_parser(value_parser!(u8).range(0..)), - ) - .arg( - Arg::new("number") - .short('n') - .help(format!("Number of passwords (1..={MAX}). [default: 10]")) - .value_parser(value_parser!(u8).range(1..)), - ) - .arg( - Arg::new("printlen") - .short('p') - .help("Print length") - .action(ArgAction::SetTrue), - ) - .arg( - Arg::new("length") - .help(format!( - "Length of passwords (0..={MAX}). 0 = random length." - )) - .value_parser(value_parser!(u8).range(0..)), - ) - .get_matches(); - - match opts.get_one::<u8>("length_flag") { - Some(n) => length = *n, - _ => (), - } - match opts.get_one::<u8>("number") { - Some(n) => number = *n, - _ => (), - } - match opts.get_one::<u8>("length") { - Some(n) => length = *n, - _ => (), - } - let printbool = if opts.get_flag("printlen") { - true - } else { - false - }; + // Set to new value if exists, or just set to old value + length = *opts.get_one::<u8>("length_flag").unwrap_or(&length); + number = *opts.get_one::<u8>("number").unwrap_or(&number); + length = *opts.get_one::<u8>("length_arg").unwrap_or(&length); + let printbool = opts.get_flag("printlen"); // // Other logic // let printlen: u8 = if printbool && length < 100 { - 3 + 2 } else if printbool { - 4 + 3 } else { 0 }; + // Get the width of the terminal let size = terminal_size(); let term_width: u16 = match size { Some((Width(w), _)) => w, None => 1, }; + // Calculate how wide each column has to be let col_width: u8 = match length { 0 => RANGE_MAX + 2, _ => length + 2, }; - let col_num: u16 = match term_width / (col_width + printlen) as u16 { - 0 => 1, - n => n, - }; + // Number of columns to print + // If printbool is true, col_num is term_width / (col_width + printlen + 1) + // Else, col_num is term_width / col_width + // col_num is never 0. Use std::cmp::max to use the largest value of 1 and the above. + let col_num: u16 = std::cmp::max( + 1, + match printbool { + true => term_width / (col_width + printlen + 1) as u16, + false => term_width / col_width as u16, + }, + ); + // Fill in common print data let data = CommonPrintData { length: length, alpha: &alpha, alnum: &alnum, + printbool: printbool, printlen: printlen, col_width: col_width, col_num: col_num, @@ -178,10 +178,12 @@ fn main() { // Do the dirty! // + // Normal passwords print_columns("Normal passwords", number, &normal, &data); println!(""); + // Special passwords print_columns( "Passwords with special characters", number / 3 * 2 + 1, @@ -189,10 +191,20 @@ fn main() { &data, ); - if Path::new(&wordlist).exists() { + // Passphrases + if fs::metadata(&wordlist).is_ok() { + // Open the file + let file = fs::File::open(&wordlist).unwrap(); + + // Create a buffered reader to handle file contents + let reader = io::BufReader::new(file); + + // Read words into a Vector + let words = reader.lines().map(|l| l.unwrap()).collect(); + + // Print println!(""); println!("Passphrases:"); - let words = read_file(&wordlist); for _ in 0..number / 2 { println!("{}", passphrase(&words)); @@ -200,6 +212,9 @@ fn main() { } } +// +// Print passwords in neat columns +// fn print_columns(title: &str, num: u8, chars: &Vec<char>, data: &CommonPrintData) { let mut strings: Vec<String> = Vec::new(); for _ in 0..num { @@ -210,51 +225,169 @@ fn print_columns(title: &str, num: u8, chars: &Vec<char>, data: &CommonPrintData let mut i: u16 = 0; for s in &strings { - i = i + 1; + i += 1; - if data.printlen > 0 { - print!("{1:00$} ", (data.printlen - 1) as usize, s.len()); + if data.printbool { + print!("{1:00$} ", data.printlen as usize, s.len()); } print!("{1:<0$}", data.col_width as usize, s); - if (i % data.col_num == 0) || (i == num as u16 && i % data.col_num > 0) { - println!(""); + if i % data.col_num == 0 || (i == num as u16 && i % data.col_num > 0) { + println!(); } } } -fn randstring(len: u8, chars: &Vec<char>, alpha: &Vec<char>, alnum: &Vec<char>) -> String { +// +// Generate random strings to use as passwords +// +fn randstring(len: u8, chars: &[char], alpha: &[char], alnum: &[char]) -> String { let mut rng = rand::thread_rng(); let length = if len == 0 { - rng.gen_range(RANGE_MIN..RANGE_MAX) + rng.gen_range(RANGE_MIN..=RANGE_MAX) } else { len }; let mut s = String::with_capacity(length as usize); for i in 0..length { - if i == 0 { - s.push(*alpha.choose(&mut rng).unwrap()); + let c = if i == 0 { + alpha.choose(&mut rng) } else if i == length - 1 { - s.push(*alnum.choose(&mut rng).unwrap()); + alnum.choose(&mut rng) } else { - s.push(*chars.choose(&mut rng).unwrap()); - } + chars.choose(&mut rng) + }; + s.push(*c.unwrap()); } s } -fn read_file(filename: &str) -> Vec<String> { - let file = File::open(filename).unwrap(); - let reader = BufReader::new(file); +// +// Generate passphrases +// +fn passphrase(wordlist: &Vec<String>) -> String { + let mut rng = rand::thread_rng(); + let mut passphrase = Vec::new(); + + for _ in 0..PASS_WORDS { + let word = wordlist.choose(&mut rng).unwrap().clone(); + passphrase.push(word); + } - reader.lines().map(|l| l.unwrap()).collect() + passphrase.join("-") } -fn passphrase(wordlist: &Vec<String>) -> String { - let mut rng = rand::thread_rng(); - (0..PASS_WORDS) - .map(|_| wordlist.choose(&mut rng).unwrap()) - .cloned() - .collect::<Vec<_>>() - .join("-") +// +// Handle command line arguments +// +fn cli() -> Command { + Command::new("makepass") + .about("makepass - create random passwords") + .author(crate_authors!()) + .arg( + Arg::new("length_flag") + .short('l') + .value_name("LENGTH") + .help(format!("Length of passwords (0..={0}). 0 = random.", MAX)) + .value_parser(value_parser!(u8).range(0..)), + ) + .arg( + Arg::new("number") + .short('n') + .value_name("NUM") + .help(format!("Number of passwords (1..={0}). [default: 10]", MAX)) + .value_parser(value_parser!(u8).range(1..)) + .long_help("this is the long help. What is it?"), + ) + .arg( + Arg::new("printlen") + .short('p') + .help("Print length") + .action(ArgAction::SetTrue), + ) + .arg( + Arg::new("length_arg") + .value_name("LENGTH") + .help(format!( + "Length of passwords (0..={MAX}). 0 = random length." + )) + .value_parser(value_parser!(u8).range(0..)), + ) + .override_help("NAME + makepass - create several random passwords + +SYNOPSIS + makepass [OPTIONS] [NUM] + + If a NUM is provided, passwords will be NUM characters long. + + By default `makepass` will output passwords from the three following classes: + + - Normal passwords - random strings with letters (both lower and upper + case), numbers, and dashes and underscores. + + - Passwords with special characters - random strings generated from lower + and upper case letters, numbers, and the following characters: + !#$%&/()=?+-_,.;:<>[]{}|@* + + - Passphrases - if we find a dictionary, a series of eight random words + from the dictionary, separated by dashes. The number of words can not be + changed, but you do not have to use all of them. Use as mane as you want. + + The first character will always be alphabetic, and the last will always be + alphanumeric. + +DESCRIPTION + makepass has the following options: + + -h + output this help-text + -l + length of passwords. See MAKEPASS_LENGTH below + -n + number of passwords. See MAKEPASS_NUMBER below + -p + print length of number + +ENVIRONMENT + makepass examines the following environmental variables. + + MAKEPASS_LENGTH + Specifies the length of passwords. Valid values are 0-255. If 0, a + random value between 8 and 42 will be used for each password. -l + overrides this environmental variable, and the argument NUM overrides + that again. So `MAKEPASS_LENGTH=10 makepass -l 12 14` will give + passwords that are 14 characters long, even though both -l and + MAKEPASS_LENGTH also specifies a length. + + MAKEPASS_NUMBER + The number of passwords to generate. This formula is used to determine + how many passwords from each group should be generated: + - (n) normal passwords + - (n / 3 * 2 + 1) special passwords + - (n / 2) passphrases + Where n is 10 by default. Valid values for n are 1-255. Floating-poing + math is not used, so results may vary. + + MAKEPASS_PRINTLEN + If 1, print length of all passwords. If 0, don't. + + MAKEPASS_NORMAL + String of characters from which to generate \"normal\" passwords. + Defaults to: + abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-_ + + MAKEPASS_SPECIAL + String of characters from which to generate passwords with special + characters. Defaults to the same characters as in MAKEPASS_NORMAL, plus + these: + !#$%&/()=?+-_,.;:<>[]{}|@* + + MAKEPASS_WORDLIST + Specifies the dictionary we find words for passphrases in. If this is + unset or empty, we try \"/usr/share/dict/words\". If that file does not + exist, no passphrases will be provided. + +AUTHOR + Dennis Eriksen <https://dnns.no>") } |