diff --git a/core/Cargo.lock b/core/Cargo.lock index f6bbdb6..f00fb54 100644 --- a/core/Cargo.lock +++ b/core/Cargo.lock @@ -51,12 +51,27 @@ dependencies = [ "windows-sys 0.52.0", ] +[[package]] +name = "base64" +version = "0.21.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d297deb1925b89f2ccc13d7635fa0714f12c87adce1c75356b39ca9b7178567" + [[package]] name = "bitflags" version = "2.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b048fb63fd8b5923fc5aa7b340d8e156aec7ec02f0c78fa8a6ddc2613f6f71de" +[[package]] +name = "block-buffer" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71" +dependencies = [ + "generic-array", +] + [[package]] name = "cc" version = "1.1.15" @@ -118,6 +133,45 @@ version = "1.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d3fd119d74b830634cea2a0f58bbd0d54540518a14397557951e79340abc28c0" +[[package]] +name = "colored" +version = "2.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cbf2150cce219b664a8a70df7a1f933836724b503f8a413af9365b4dcc4d90b8" +dependencies = [ + "lazy_static", + "windows-sys 0.48.0", +] + +[[package]] +name = "cpufeatures" +version = "0.2.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "51e852e6dc9a5bed1fae92dd2375037bf2b768725bf3be87811edee3249d09ad" +dependencies = [ + "libc", +] + +[[package]] +name = "crypto-common" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3" +dependencies = [ + "generic-array", + "typenum", +] + +[[package]] +name = "digest" +version = "0.10.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292" +dependencies = [ + "block-buffer", + "crypto-common", +] + [[package]] name = "dirs" version = "5.0.1" @@ -139,6 +193,22 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "errno" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "534c5cf6194dfab3db3242765c03bbe257cf92f22b38f6bc0c58d59108a820ba" +dependencies = [ + "libc", + "windows-sys 0.52.0", +] + +[[package]] +name = "fastrand" +version = "2.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e8c02a5121d4ea3eb16a80748c74f5549a5665e4c21333c6098f283870fbdea6" + [[package]] name = "foreign-types" version = "0.3.2" @@ -154,6 +224,16 @@ version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "00b0228411908ca8685dba7fc2cdd70ec9990a6e753e89b6ac91a84c40fbaf4b" +[[package]] +name = "generic-array" +version = "0.14.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a" +dependencies = [ + "typenum", + "version_check", +] + [[package]] name = "getrandom" version = "0.2.15" @@ -199,6 +279,12 @@ dependencies = [ "libc", ] +[[package]] +name = "linux-raw-sys" +version = "0.4.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "78b3ae25bc7c8c38cec158d1f2757ee79e9b3740fbc7ccf0e59e4b08d793fa89" + [[package]] name = "once_cell" version = "1.19.0" @@ -284,6 +370,30 @@ dependencies = [ "thiserror", ] +[[package]] +name = "rustix" +version = "0.38.35" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a85d50532239da68e9addb745ba38ff4612a242c1c7ceea689c4bc7c2f43c36f" +dependencies = [ + "bitflags", + "errno", + "libc", + "linux-raw-sys", + "windows-sys 0.52.0", +] + +[[package]] +name = "sha2" +version = "0.10.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest", +] + [[package]] name = "shlex" version = "1.3.0" @@ -307,6 +417,19 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "tempfile" +version = "3.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "04cbcdd0c794ebb0d4cf35e88edd2f7d2c4c3e9a5a6dab322839b321c6a87a64" +dependencies = [ + "cfg-if", + "fastrand", + "once_cell", + "rustix", + "windows-sys 0.59.0", +] + [[package]] name = "thiserror" version = "1.0.63" @@ -327,6 +450,12 @@ dependencies = [ "syn", ] +[[package]] +name = "typenum" +version = "1.17.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825" + [[package]] name = "unicode-ident" version = "1.0.12" @@ -343,10 +472,14 @@ checksum = "06abde3611657adf66d383f00b093d7faecc7fa57071cce2578660c9f1010821" name = "vanish" version = "0.1.2" dependencies = [ + "base64", "clap", + "colored", "dirs", "lazy_static", "openssl", + "sha2", + "tempfile", ] [[package]] @@ -355,6 +488,12 @@ version = "0.2.15" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426" +[[package]] +name = "version_check" +version = "0.9.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b928f33d975fc6ad9f86c8f283853ad26bdd5b10b7f1542aa2fa15e2289105a" + [[package]] name = "wasi" version = "0.11.0+wasi-snapshot-preview1" @@ -379,6 +518,15 @@ dependencies = [ "windows-targets 0.52.6", ] +[[package]] +name = "windows-sys" +version = "0.59.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e38bc4d79ed67fd075bcc251a1c39b32a1776bbe92e5bef1f0bf1f8c531853b" +dependencies = [ + "windows-targets 0.52.6", +] + [[package]] name = "windows-targets" version = "0.48.5" diff --git a/core/Cargo.toml b/core/Cargo.toml index a8a9623..f924eba 100644 --- a/core/Cargo.toml +++ b/core/Cargo.toml @@ -10,10 +10,14 @@ categories = ["development-tools", "command-line-utilities", "parsing", "os", "h keywords = ["https", "certificates", "localhost", "local-development", "root-ca"] readme = "README.md" -# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html - [dependencies] openssl = "0.10" clap = { version = "4.4.8", features = ["derive"] } dirs = "5.0" -lazy_static = "1.4" \ No newline at end of file +sha2 = "0.10" +lazy_static = "1.4" +colored = "2.0" +base64 = "0.21" + +[dev-dependencies] +tempfile = "3.3" \ No newline at end of file diff --git a/core/src/commands/generate.rs b/core/src/commands/generate.rs index 2d95356..6e7cf22 100644 --- a/core/src/commands/generate.rs +++ b/core/src/commands/generate.rs @@ -9,11 +9,15 @@ use crate::{ Certificate, }, }; +use colored::*; use openssl::{ pkey::{PKey, Private}, x509::{X509Req, X509}, }; -use std::error; +use std::{ + error::{self, Error}, + path::PathBuf, +}; pub fn generate( domains: Vec, @@ -28,58 +32,136 @@ pub fn generate( request: bool, install: bool, ) -> Result<(), Box> { + println!(); if request { + println!("Generated Certificate Requests for :"); for domain in &domains { let distinguished_name: DistinguishedName = create_distinguished_name(&commonname, &country, &state); let (ca_req_certificate, private_key) = CAReq::new(distinguished_name)?.generate_certificate()?; - save_csr_certificate(domain.to_string(), &output, ca_req_certificate, private_key)?; + let is_saved: Result> = + save_csr_certificate(domain.to_string(), &output, ca_req_certificate, private_key); + match is_saved { + Ok(path) => { + println!(" - \"{}\" ✅", domain); + println!(); + println!( + "{}: Your request certs and their corresponding keys are saved at: {:?}", + "Note".green(), + path + ); + } + Err(_err) => { + println!(" - \"{}\" ❌", domain); + } + } } + println!(); return Ok(()); } if let Some(certfile) = certfile { if let Some(keyfile) = keyfile { - let (cert, pkey) = CACert::load_ca_cert(&certfile, &keyfile)?; + let (cert, pkey) = match CACert::load_ca_cert(&certfile, &keyfile) { + Ok(cert_pkey) => cert_pkey, + Err(err) => { + eprintln!("{}", err); + std::process::exit(1); + } + }; if let Some(csr) = &csr { let distinguished_name: DistinguishedName = create_distinguished_name(&commonname, &country, &state); - let csr_object: X509Req = CAReq::read_csr_from_file(csr)?; + let csr_object: X509Req = match CAReq::read_csr_from_file(csr) { + Ok(csr) => csr, + Err(err) => { + eprintln!("{}", err); + std::process::exit(1); + } + }; let leaf_cert_object: LeafCert = LeafCert::new(distinguished_name)?; - let (leaf_certificate, _private_key) = LeafCert::generate_certificate( + println!(); + let (leaf_certificate, _private_key) = match LeafCert::generate_certificate( leaf_cert_object, &cert, &pkey, Some(&csr_object), - )?; - save_pem_certificate("csr_cert.pem".to_string(), output, leaf_certificate)?; + ) { + Ok((a, b)) => { + println!("Generating Certificate for Signing Request Successful! 👍"); + (a, b) + } + Err(err) => { + println!("Generating Certificate for Signing Request Failed! 👎"); + eprintln!("{}", err); + std::process::exit(1); + } + }; + match save_pem_certificate("csr_cert.pem".to_string(), output, leaf_certificate) { + Ok(()) => {} + Err(err) => { + println!("{}", err); + std::process::exit(1); + } + }; } else { + println!(); + println!("Generated Certificate for : "); for domain in &domains { let distinguished_name: DistinguishedName = create_distinguished_name(&commonname, &country, &state); let leaf_cert_object: LeafCert = LeafCert::new(distinguished_name)?; - let (leaf_certificate, private_key) = - LeafCert::generate_certificate(leaf_cert_object, &cert, &pkey, None)?; + let (leaf_certificate, private_key) = match LeafCert::generate_certificate( + leaf_cert_object, + &cert, + &pkey, + None, + ) { + Ok((a, b)) => { + println!(" - \"{}\" ✅", domain); + (a, b) + } + Err(err) => { + println!(" - \"{}\" ❌", domain); + eprintln!("{}", err); + std::process::exit(1); + } + }; if let Some(private_key) = private_key { - save_pem_key_pair( + match save_pem_key_pair( &output, leaf_certificate, domain.to_string(), private_key, - )?; + ) { + Ok(()) => {} + Err(err) => { + println!("{}", err); + } + }; } else { eprintln!( - "Oops! We lost your private key for domain {}. Please try again!", - domain + "{}{}{}", + "Oops! We lost your private key for domain ".yellow(), + domain.yellow(), + ". Please try again!".yellow() ) } } + println!(); + println!( + "{}: All Successful Certificates and their corresponding keys are saved at : {}", + "Note".green(), + output.unwrap() + ); } if install {} + println!(); return Ok(()); } else { - eprintln!("Corresponding KeyFile Not Found"); + eprintln!("{}: Corresponding KeyFile Not Found", "Error".red()); + println!(); std::process::exit(1); } } @@ -91,20 +173,47 @@ pub fn generate( create_distinguished_name(&commonname, &country, &state); let csr_object: X509Req = CAReq::read_csr_from_file(csr)?; let leaf_cert_object: LeafCert = LeafCert::new(distinguished_name)?; - let (leaf_certificate, _private_key) = LeafCert::generate_certificate( + println!(); + let (leaf_certificate, _private_key) = match LeafCert::generate_certificate( leaf_cert_object, &d_cert, &d_pkey, Some(&csr_object), - )?; + ) { + Ok((a, b)) => { + println!("Generating Certificate for Signing Request Successful! 👍"); + (a, b) + } + Err(err) => { + println!("Generating Certificate for Signing Request Failed! 👎"); + eprintln!("{}", err); + std::process::exit(1); + } + }; save_pem_certificate("csr_cert.pem".to_string(), output, leaf_certificate)?; } else { + println!(); + println!("Generated Certificate for : "); for domain in &domains { let distinguished_name: DistinguishedName = create_distinguished_name(&commonname, &country, &state); let leaf_cert_object: LeafCert = LeafCert::new(distinguished_name)?; - let (leaf_certificate, private_key) = - LeafCert::generate_certificate(leaf_cert_object, &d_cert, &d_pkey, None)?; + let (leaf_certificate, private_key) = match LeafCert::generate_certificate( + leaf_cert_object, + &d_cert, + &d_pkey, + None, + ) { + Ok((a, b)) => { + println!(" - \"{}\" ✅", domain); + (a, b) + } + Err(err) => { + println!(" - \"{}\" ❌", domain); + eprintln!("{}", err); + std::process::exit(1); + } + }; if let Some(private_key) = private_key { save_pem_key_pair(&output, leaf_certificate, domain.to_string(), private_key)?; } else { @@ -114,6 +223,12 @@ pub fn generate( ) } } + println!(); + println!( + "{}: All Successful Certificates and their corresponding keys are saved at : {}", + "Note".green(), + output.unwrap() + ); } if install { @@ -121,7 +236,10 @@ pub fn generate( } } else { if noca { - eprintln!("Error: No CA Certificates found and generation of a new one is disabled by `--no-ca`"); + eprintln!( + "{}: No CA Certificates found and generation of a new one is disabled by `--no-ca`", + "Error".red() + ); std::process::exit(1) } let distinguished_name: DistinguishedName = @@ -134,37 +252,80 @@ pub fn generate( create_distinguished_name(&commonname, &country, &state); let csr_object: X509Req = CAReq::read_csr_from_file(csr)?; let leaf_cert_object: LeafCert = LeafCert::new(distinguished_name)?; - let (leaf_certificate, _private_key) = LeafCert::generate_certificate( + println!(); + let (leaf_certificate, _private_key) = match LeafCert::generate_certificate( leaf_cert_object, &created_cert, &created_key, Some(&csr_object), - )?; + ) { + Ok((a, b)) => { + println!("Generating Certificate for Signing Request Successful! 👍"); + (a, b) + } + Err(err) => { + println!("Generating Certificate for Signing Request Failed! 👎"); + eprintln!("{}", err); + std::process::exit(1); + } + }; save_pem_certificate("csr_cert.pem".to_string(), output, leaf_certificate)?; } else { + println!(); + println!("Generated Certificate for : "); for domain in &domains { let distinguished_name: DistinguishedName = create_distinguished_name(&commonname, &country, &state); let leaf_cert_object: LeafCert = LeafCert::new(distinguished_name)?; - let (leaf_certificate, private_key) = LeafCert::generate_certificate( + let (leaf_certificate, private_key) = match LeafCert::generate_certificate( leaf_cert_object, &created_cert, &created_key, None, - )?; + ) { + Ok((a, b)) => { + println!(" - \"{}\" ✅", domain); + (a, b) + } + Err(err) => { + println!(" - \"{}\" ❌", domain); + eprintln!("{}", err); + std::process::exit(1); + } + }; if let Some(private_key) = private_key { - save_pem_key_pair(&output, leaf_certificate, domain.to_string(), private_key)?; + match save_pem_key_pair( + &output, + leaf_certificate, + domain.to_string(), + private_key, + ) { + Ok(()) => {} + Err(err) => { + println!("{}", err); + } + }; } else { eprintln!( - "Oops! We lost your private key for domain {}. Please try again!", - domain + "{}{}{}", + "Oops! We lost your private key for domain ".yellow(), + domain.yellow(), + ". Please try again!".yellow() ) } } + + println!(); + println!( + "{}: All Successful Certificates and their corresponding keys are saved at : {}", + "Note".green(), + output.unwrap() + ); } if install { generate_install(created_cert)?; } } + println!(); Ok(()) } diff --git a/core/src/commands/utils.rs b/core/src/commands/utils.rs index 3af1d49..d5a535a 100644 --- a/core/src/commands/utils.rs +++ b/core/src/commands/utils.rs @@ -2,9 +2,9 @@ use crate::{ trust_stores::{ firefox::FirefoxTrustStore, nss::NSSValue, nss_profile::NSSProfile, utils::check_if_firefox_exists, CAValue, - }, - x509::{ca_req::CAReq, distinguished_name::DistinguishedName, leaf_cert::LeafCert}, + }, utils::get_unique_hash, x509::{ca_req::CAReq, distinguished_name::DistinguishedName, leaf_cert::LeafCert} }; +use colored::*; use openssl::{ pkey::{PKey, Private}, x509::{X509Req, X509}, @@ -19,8 +19,8 @@ pub fn generate_install(cert: X509) -> Result<(), Box> { let ca_value_object: CAValue = CAValue { certificate: cert }; ca_value_object.install_certificate()?; let nss_profile_object: NSSProfile = NSSProfile::new(); - let ca_unique_name: String = "vanish-root-test-123456-shubham-brr".to_string(); let caroot: String = "/home/jerry/.local/share/vanish/ca_cert.pem".to_string(); + let ca_unique_name: String = get_unique_hash(&caroot)?; let mkcert: NSSValue = NSSValue::new(nss_profile_object, ca_unique_name.clone(), caroot.clone()); let success: bool = mkcert.install_nss(); @@ -36,7 +36,7 @@ pub fn generate_install(cert: X509) -> Result<(), Box> { ); } if success { - println!("Certificate installed successfully."); + println!("Certificate installed successfully 👍"); } else { eprintln!("Failed to install the certificate."); } @@ -61,18 +61,56 @@ pub fn save_pem_certificate( let file_name: PathBuf = output_path.join(name); let file_name_str: Option<&str> = file_name.to_str(); if let Some(file_name_str) = file_name_str { - LeafCert::save_cert(&leaf_certificate, file_name_str)?; + match LeafCert::save_cert(&leaf_certificate, file_name_str) { + Ok(()) => { + println!(); + println!( + "{}: Your local certificate from request is saved at: {:?}", + "Note".green(), + file_name + ); + println!( + "{}: You may use the Private Key of the Certificate Provided as the Private Key of your Local Certificate.", + "Note".green() + ); + } + Err(err) => { + eprintln!("{}", err); + } + }; } else { - eprintln!("Error: Error creating file for generated Certificate :"); + eprintln!( + "{}: Error creating file for Generated Certificate :", + "Error".red() + ); } } else { let output_path: PathBuf = std::env::current_dir()?; let file_name: PathBuf = output_path.join("csr_cert.pem"); let file_name_str: Option<&str> = file_name.to_str(); if let Some(file_name_str) = file_name_str { - LeafCert::save_cert(&leaf_certificate, file_name_str)?; + match LeafCert::save_cert(&leaf_certificate, file_name_str) { + Ok(()) => { + println!(); + println!( + "{}: Your local certificate from request is saved at: {:?}", + "Note".green(), + file_name + ); + println!( + "{}: You may use the Private Key of the Certificate Provided as the Private Key of your Local Certificate.", + "Note".green() + ); + } + Err(err) => { + eprintln!("{}", err); + } + }; } else { - eprintln!("Error: Error creating file for generated Certificate"); + eprintln!( + "{}: Error creating file for Generated Certificate", + "Error".red() + ); } } Ok(()) @@ -133,7 +171,7 @@ pub fn save_csr_certificate( output: &Option, ca_req_certificate: X509Req, private_key: PKey, -) -> Result<(), Box> { +) -> Result> { if let Some(output) = &output { let output_path: &Path = Path::new(output); if !output_path.exists() { @@ -158,6 +196,7 @@ pub fn save_csr_certificate( } else { eprintln!("Error: Error creating file for key : {}", name); } + return Ok(output_path); } else { let output_path: PathBuf = std::env::current_dir()?; let file_name: PathBuf = output_path.join(format!("csr-{}.pem", name)); @@ -174,8 +213,8 @@ pub fn save_csr_certificate( } else { eprintln!("Error: Error creating file for key : {}", name); } + Ok(output_path) } - Ok(()) } pub fn create_distinguished_name( diff --git a/core/src/errors.rs b/core/src/errors.rs index ed44b19..c6784b5 100644 --- a/core/src/errors.rs +++ b/core/src/errors.rs @@ -1,7 +1,7 @@ +use colored::*; use core::fmt; use openssl::error::ErrorStack; use std::error::Error; - pub type CertKeyResult = Result; pub type SerialNumberResult = Result; @@ -14,9 +14,16 @@ pub enum CertKeyPairError { impl fmt::Display for CertKeyPairError { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { match self { - Self::PKeyCreationError(err) => write!(f, "Error Creating Key File : {}", err), + Self::PKeyCreationError(err) => { + write!(f, "{}: Creating Key File : {}", "Error".red(), err) + } Self::RSAGenerationError(err) => { - write!(f, "Error Generating Certificate File : {}", err) + write!( + f, + "{}: Generating Certificate File : {}", + "Error".red(), + err + ) } } } @@ -35,13 +42,23 @@ impl fmt::Display for SerialNumberError { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { match self { Self::BigNumberInitializationError(err) => { - write!(f, "Error Initializing Big Number : {}", err) + write!(f, "{}: Initializing Big Number : {}", "Error".red(), err) } Self::RandomBigNumberGenerationError(err) => { - write!(f, "Error creating a Random Big Number : {}", err) + write!( + f, + "{}: creating a Random Big Number : {}", + "Error".red(), + err + ) } Self::ConvertBigNumberToASN1Error(err) => { - write!(f, "Error Converting Big Number to ASN1Integer: {}", err) + write!( + f, + "{}: Converting Big Number to ASN1Integer: {}", + "Error".red(), + err + ) } } } diff --git a/core/src/main.rs b/core/src/main.rs index 051386f..065dec6 100644 --- a/core/src/main.rs +++ b/core/src/main.rs @@ -6,6 +6,8 @@ mod x509; use clap::{Parser, Subcommand}; use commands::generate::generate; use std::env; +#[cfg(test)] +mod utils_tests; #[derive(Parser)] #[clap( @@ -94,20 +96,20 @@ fn main() { ); std::process::exit(1); } - + if !domains.is_empty() && csr.is_some() { eprintln!("Error: `-d` (domains) and `--csr` cannot be used together."); std::process::exit(1); } if request && csr.is_some() { - eprint!("Error: `--req-only` and `csr` are incompatible. You can't generate requests from a request certificate."); + eprintln!("Error: `--req-only` and `csr` are incompatible. You can't generate requests from a request certificate."); std::process::exit(1); } if request && install { //CORRECT THIS - eprint!("Error: `--req-only` and `install` are incompatible. You can't generate requests from a request certificate."); + eprintln!("Error: `--req-only` and `install` are incompatible. You can't generate requests from a request certificate."); std::process::exit(1); } diff --git a/core/src/trust_stores/errors.rs b/core/src/trust_stores/errors.rs index 85d97fd..de9ea26 100644 --- a/core/src/trust_stores/errors.rs +++ b/core/src/trust_stores/errors.rs @@ -1,3 +1,4 @@ +use colored::*; use openssl::error::ErrorStack; use std::{env::VarError, error::Error, fmt, io}; @@ -12,13 +13,28 @@ impl fmt::Display for TrustStoreError { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { match self { Self::PEMEncodingError(err) => { - write!(f, "Trust Store PEM Encoding Failed: {}", err) + write!( + f, + "{}: Trust Store PEM Encoding Failed: {}", + "Error".red(), + err + ) } Self::PEMFileCreationError(err) => { - write!(f, "Trust Store PEM File creation Failed: {}", err) + write!( + f, + "{}: Trust Store PEM File creation Failed: {}", + "Error".red(), + err + ) } Self::WriteToFileError(err) => { - write!(f, "Trust Store Writing to PEM file: {}", err) + write!( + f, + "{}: Trust Store Writing to PEM file: {}", + "Error".red(), + err + ) } } } @@ -38,12 +54,19 @@ impl fmt::Display for FirefoxTrustStoreError { Self::ENVVariableNotFound(err, variable) => { write!( f, - "Error getting Environment varibale {} : {}", - variable, err + "{}: getting Environment varibale {} : {}", + "Error".red(), + variable, + err ) } Self::IOError(err) => { - write!(f, "Error reading the default firefox directoryL {}", err) + write!( + f, + "{}: reading the default firefox directoryL {}", + "Error".red(), + err + ) } } } diff --git a/core/src/trust_stores/firefox.rs b/core/src/trust_stores/firefox.rs index 7552ef6..2c87895 100644 --- a/core/src/trust_stores/firefox.rs +++ b/core/src/trust_stores/firefox.rs @@ -1,5 +1,5 @@ use super::errors::FirefoxTrustStoreError; -use std::process::exit; +use std::process::{exit, Stdio}; use std::{ env, fs, io, path::{Path, PathBuf}, @@ -82,12 +82,9 @@ impl FirefoxTrustStore { pub fn find_cert_directories(&self) -> Result, FirefoxTrustStoreError> { let mut cert_dirs: Vec = Vec::new(); - println!("Firefox Profiles: {:?}", &self.firefox_profile); for profile_dir in &self.firefox_profile { let path: &Path = Path::new(profile_dir); - println!("{:?} exists: {:?}", &path, &path.exists()); if path.exists() && path.is_dir() { - println!("Path is a dir: {:?}", &path); for entry in fs::read_dir(path) .map_err(|err: io::Error| FirefoxTrustStoreError::IOError(err))? { @@ -127,6 +124,7 @@ impl FirefoxTrustStore { .arg(&self.ca_unique_name) .arg("-i") .arg(&self.vanish_ca_path) + .stdout(Stdio::null()) .status(); match cmd_result { diff --git a/core/src/trust_stores/linux.rs b/core/src/trust_stores/linux.rs index 0204e41..6f0577e 100644 --- a/core/src/trust_stores/linux.rs +++ b/core/src/trust_stores/linux.rs @@ -1,3 +1,4 @@ +use colored::*; use openssl::error::ErrorStack; use openssl::x509::X509; use std::fs; @@ -35,7 +36,7 @@ impl CAValue { match store { Some(store) => { let path: String = store.get_path(); - println!("Adding file to trust store: {}", path); + println!("{}: Adding file to trust store: {}", "Note".green(), path); let pem_path: PathBuf = Path::new(&path).join("vanish-root.crt"); if let Err(err) = fs::create_dir_all(&path) { @@ -45,7 +46,7 @@ impl CAValue { match CAValue::save_cert(&self.certificate, pem_path.to_str().unwrap()) { Ok(_) => println!("Certificate saved at: {}", pem_path.display()), - Err(err) => eprintln!("Failed to save certificate: {}", err), + Err(err) => eprintln!("{}", err), } } None => { @@ -85,4 +86,4 @@ impl PossibleStores { PossibleStores::Other => "/usr/share/pki/trust/anchors/".to_string(), } } -} \ No newline at end of file +} diff --git a/core/src/trust_stores/nss.rs b/core/src/trust_stores/nss.rs index 4ee0008..95c7bbc 100644 --- a/core/src/trust_stores/nss.rs +++ b/core/src/trust_stores/nss.rs @@ -1,8 +1,9 @@ use crate::trust_stores::nss_profile::NSSProfile; +use colored::*; use std::{ fs, io, path::Path, - process::{Command, ExitStatus}, + process::{Command, ExitStatus, Stdio}, }; pub struct NSSValue { @@ -28,7 +29,6 @@ impl NSSValue { let mut success: bool = true; if self.for_each_nss_profile(|profile: &str| { - println!("{:?}", self.profile.certutil_path); let cmd: Result = Command::new(self.profile.certutil_path.as_ref().unwrap()) .arg("-V") @@ -38,6 +38,7 @@ impl NSSValue { .arg("L") .arg("-n") .arg(&self.ca_unique_name) + .stdout(Stdio::null()) .status(); if cmd.is_err() || !cmd.unwrap().success() { @@ -64,6 +65,7 @@ impl NSSValue { .arg(&self.ca_unique_name) .arg("-i") .arg(&self.caroot) + .stdout(Stdio::null()) .status(); if let Err(err) = cmd { @@ -71,7 +73,7 @@ impl NSSValue { } }) == 0 { - eprintln!("ERROR: no NSS security databases found"); + eprintln!("{}: No NSS security databases found", "Error".red()); return false; } @@ -94,6 +96,7 @@ impl NSSValue { .arg(profile) .arg("-n") .arg(&self.ca_unique_name) + .stdout(Stdio::null()) .status(); if let Err(err) = cmd { @@ -107,9 +110,7 @@ impl NSSValue { F: FnMut(&str), { let mut found: usize = 0; - let profiles = &self.profile.nss_dbs; - - println!("{:?}", profiles); + let profiles: &Vec = &self.profile.nss_dbs; for profile in profiles { let stat: Result = Path::new(profile).metadata(); diff --git a/core/src/trust_stores/nss_profile.rs b/core/src/trust_stores/nss_profile.rs index 0933c9a..620af81 100644 --- a/core/src/trust_stores/nss_profile.rs +++ b/core/src/trust_stores/nss_profile.rs @@ -2,7 +2,7 @@ use std::path::{Path, PathBuf}; use std::process::Command; pub struct NSSProfile { - pub has_nss: bool, + pub _has_nss: bool, pub has_certutil: bool, pub certutil_path: Option, pub nss_dbs: Vec, @@ -60,7 +60,7 @@ impl NSSProfile { } Self { - has_nss, + _has_nss: has_nss, has_certutil, certutil_path, nss_dbs, diff --git a/core/src/utils.rs b/core/src/utils.rs index f519e68..dbf5309 100644 --- a/core/src/utils.rs +++ b/core/src/utils.rs @@ -2,15 +2,25 @@ use crate::{ errors::{CertKeyPairError, CertKeyResult, SerialNumberError, SerialNumberResult}, x509::{self, ca_cert::CACert}, }; +use base64::engine::general_purpose::URL_SAFE; +use base64::Engine; +use colored::*; use openssl::{ asn1::Asn1Integer, bn::BigNum, error::ErrorStack, pkey::{PKey, Private}, rsa::Rsa, + sha::Sha256, x509::X509, }; -use std::{error, fs, io, path::Path, process::Output}; +use std::{ + error, + fs::{self, File}, + io::{self, Read}, + path::Path, + process::Output, +}; use std::{path::PathBuf, process::Command}; pub fn generate_cert_key_pair() -> CertKeyResult<(Rsa, PKey)> { @@ -36,7 +46,11 @@ pub fn get_certificates_from_data_dir() -> Option<(X509, PKey)> { if let Some(ref data_dir) = *x509::DATA_DIR { if !data_dir.exists() { if let Err(err) = fs::create_dir_all(data_dir) { - eprintln!("Failed to create data directory: {}", err); + eprintln!( + "{}: Failed to create data directory: {}", + "Error".red(), + err + ); return None; } } @@ -46,7 +60,10 @@ pub fn get_certificates_from_data_dir() -> Option<(X509, PKey)> { let ca_cert_file_str: &str = match ca_certfile.to_str() { Some(s) => s, None => { - eprintln!("Failed to convert ca_certfile path to string"); + eprintln!( + "{}: Failed to convert ca_certfile path to string", + "Error".red() + ); return None; } }; @@ -54,7 +71,10 @@ pub fn get_certificates_from_data_dir() -> Option<(X509, PKey)> { let ca_key_file_str: &str = match ca_keyfile.to_str() { Some(s) => s, None => { - eprintln!("Failed to convert ca_keyfile path to string"); + eprintln!( + "{}: Failed to convert ca_keyfile path to string", + "Error".red() + ); return None; } }; @@ -62,12 +82,12 @@ pub fn get_certificates_from_data_dir() -> Option<(X509, PKey)> { match CACert::load_ca_cert(ca_cert_file_str, ca_key_file_str) { Ok((cert, pkey)) => Some((cert, pkey)), Err(_err) => { - eprintln!("Warning: Generating new certificates"); + eprintln!("{}: Generating new certificates ", "Warning".yellow()); None } } } else { - eprintln!("Unable to get Data Directory"); + eprintln!("{}: Unable to get Data Directory", "Error".red()); None } } @@ -76,10 +96,15 @@ pub fn save_generated_cert_key_files( cert: &X509, key: &PKey, ) -> Result<(), Box> { + println!(); if let Some(ref data_dir) = *x509::DATA_DIR { if !data_dir.exists() { fs::create_dir_all(data_dir).map_err(|err| { - eprintln!("Failed to create data directory: {}", err); + eprintln!( + "{}: Failed to create data directory: {}", + "Error".red(), + err + ); err })?; } @@ -89,33 +114,73 @@ pub fn save_generated_cert_key_files( let ca_cert_file_str: &str = ca_certfile.to_str().ok_or_else(|| { let err: String = "Failed to convert ca_certfile path to string".to_string(); - eprintln!("{}", err); + eprintln!("{}: {}", "Error".red(), err); io::Error::new(io::ErrorKind::InvalidInput, err) })?; let ca_key_file_str: &str = ca_keyfile.to_str().ok_or_else(|| { let err: String = "Failed to convert ca_keyfile path to string".to_string(); - eprintln!("{}", err); + eprintln!("{}: {}", "Error".red(), err); io::Error::new(io::ErrorKind::InvalidInput, err) })?; - CACert::save_cert(cert, ca_cert_file_str)?; - CACert::save_key(key, ca_key_file_str)?; + match CACert::save_cert(cert, ca_cert_file_str) { + Ok(()) => { + println!( + "{}: CA Root Certificate saved at: {} 👍", + "Note".green(), + ca_cert_file_str + ); + } + Err(err) => { + eprintln!("{}", err); + std::process::exit(1); + } + }; + match CACert::save_key(key, ca_key_file_str) { + Ok(()) => { + println!( + "{}: CA Root Private Key saved at: {} 👍", + "Note".green(), + ca_cert_file_str + ); + } + Err(err) => { + eprintln!("{}", err); + std::process::exit(1); + } + }; Ok(()) } else { let err: String = "Unable to get Data Directory".to_string(); - eprintln!("{}", err); + eprintln!("{}: {}", "Error".red(), err); Err(Box::new(io::Error::new(io::ErrorKind::NotFound, err))) } } -pub fn _path_exists(path: &str) -> bool { +#[allow(dead_code)] +pub fn path_exists(path: &str) -> bool { Path::new(path).exists() } -pub fn _binary_exists(binary: &str) -> bool { +#[allow(dead_code)] +pub fn binary_exists(binary: &str) -> bool { Command::new(binary) .output() .map(|output: Output| output.status.success()) .unwrap_or(false) } + +#[allow(dead_code)] +pub fn get_unique_hash(csr_path: &str) -> Result { + let mut file: File = File::open(csr_path)?; + let mut csr_contents: Vec = Vec::new(); + file.read_to_end(&mut csr_contents)?; + let mut hasher: Sha256 = Sha256::new(); + hasher.update(&csr_contents); + let result: [u8; 32] = hasher.finish(); + let mut unique_name: String = URL_SAFE.encode(result); + unique_name = unique_name.trim_end_matches('=').to_string(); + + Ok(unique_name) +} diff --git a/core/src/utils_tests.rs b/core/src/utils_tests.rs new file mode 100644 index 0000000..402f25e --- /dev/null +++ b/core/src/utils_tests.rs @@ -0,0 +1,81 @@ +mod get_unique_hash { + use crate::utils::get_unique_hash; + use std::io::Write; + use tempfile::NamedTempFile; + + const CSR_DATA: &[u8; 954] = b"-----BEGIN CERTIFICATE REQUEST-----\n +MIICfTCCAWUCAQAwETEPMA0GA1UECgwGVmFuaXNoMIIBIjANBgkqhkiG9w0BAQEF\n +AAOCAQ8AMIIBCgKCAQEAu6yXUMzVfqCBGOJh50MziGFKShMWQt7iIYe2pAeFBoeo\n +uuwuVq4xSdqpgTFvkhekH8tpTX8n+Y8pY7rur6Nosi/G2A6AxvvwG/ahDuHYg9Il\n +XmTYrqKnK32n/8ALPT6W3J6wKq6Nm2y5yXhGX/fSVSrmNw78hl0osKLUa7dLmCs0\n +zBghTfwjSz+qkUXvwEKuGPqgXLh6hirjiNhbtbR1XgV3p/w/7u3qH7fJcDlrhQDW\n +QhJnxcdYGCMGGcIzEdGaNzTHNAlpU7lLR2FTEjk7OCYwDgzLFYy6QFdeK2swdkfn\n +OlipnXQfzFUEbTKDQUckLafodJBzLhqyKCulDPxqoQIDAQABoCcwJQYJKoZIhvcN\n +AQkOMRgwFjAUBgNVHREEDTALgglsb2NhbGhvc3QwDQYJKoZIhvcNAQELBQADggEB\n +ACwI/gFmKb5oAahbq2LlnkGtAwmbV67xgxTmXXUIvJs37cz4Rde/CdSDct5g7mYh\n +8EghzWLEfHmg9K9kMiV58HHYIkcQ421v+pPub3wDVU3yvq3/DpIqHDBs/Wz3nFrI\n +GZEfI1cwUTqTkK9Lk2z3waUkt8adk4pvof01tiEq9A06GjK5tJfUIGef4lkOfzKE\n +ld7i9gqfMk+9pmSjBYVG0YCfc0ye/oPhmSNtFc9dTlBb9bqcN3C0dgqdhM3E/4Gz\n +YBni+0GjtUKZT9NsekFh/O14P6j9MrtbHsMLWU6KXgHc+t75kD+XKy2U/IPd8nzt\n +ncq6d7m7pfbZs5a+gBnsPqA=\n +-----END CERTIFICATE REQUEST-----"; + + #[test] + fn returns_ok() { + let mut temp_file: NamedTempFile = NamedTempFile::new().unwrap(); + temp_file.write_all(CSR_DATA).unwrap(); + let csr_path: &str = temp_file.path().to_str().unwrap(); + + let hash_result: Result = get_unique_hash(csr_path); + assert!(hash_result.is_ok(), "Expected Ok result, got an error"); + } + + #[test] + fn hash_length() { + let mut temp_file: NamedTempFile = NamedTempFile::new().unwrap(); + temp_file.write_all(CSR_DATA).unwrap(); + let csr_path: &str = temp_file.path().to_str().unwrap(); + + let unique_hash: String = get_unique_hash(csr_path).unwrap(); + assert_eq!(unique_hash.len(), 43, "Expected hash length of 43"); + } + + #[test] + fn consistent() { + let mut temp_file: NamedTempFile = NamedTempFile::new().unwrap(); + temp_file.write_all(CSR_DATA).unwrap(); + let csr_path: &str = temp_file.path().to_str().unwrap(); + + let hash1: String = get_unique_hash(csr_path).unwrap(); + let hash2: String = get_unique_hash(csr_path).unwrap(); + assert_eq!(hash1, hash2, "Hashes for the same file should be equal"); + } +} + +mod serial_number { + use crate::utils::generate_certificate_serial_number; + + #[test] + fn returns_ok() { + let result: Result = + generate_certificate_serial_number(); + assert!(result.is_ok(), "Expected Ok result"); + } + + #[test] + fn is_valid() { + let result: Result = + generate_certificate_serial_number(); + let asn1_integer: openssl::asn1::Asn1Integer = result.unwrap(); + let serial_bytes: Vec = asn1_integer.to_bn().unwrap().to_vec(); + assert!(!serial_bytes.is_empty(), "ASN1 Integer should not be empty"); + + let is_negative: bool = asn1_integer.to_bn().unwrap().is_negative(); + assert!(!is_negative, "ASN1 Integer should not be negative"); + + assert!( + serial_bytes.len() <= 16, + "ASN1 Integer should fit within 128 bits (16 bytes)" + ); + } +} diff --git a/core/src/x509/ca_cert.rs b/core/src/x509/ca_cert.rs index 2c4de7e..bb6a34c 100644 --- a/core/src/x509/ca_cert.rs +++ b/core/src/x509/ca_cert.rs @@ -112,19 +112,40 @@ impl Certificate for CACert { impl CACert { pub fn load_ca_cert(cert_path: &str, key_path: &str) -> X509Result<(X509, PKey)> { - let cert: X509 = X509::from_pem(&std::fs::read(cert_path).map_err(|err: io::Error| { - X509Error::ErrorReadingCertFile(err, cert_path.to_string()) - })?) - .map_err(|err: ErrorStack| { - X509Error::ErrorConvertingFileToData(err, cert_path.to_string()) - })?; - let key: PKey = - PKey::private_key_from_pem(&std::fs::read(key_path).map_err(|err: io::Error| { + let cert: X509 = + match X509::from_pem(&std::fs::read(cert_path).map_err(|err: io::Error| { X509Error::ErrorReadingCertFile(err, cert_path.to_string()) - })?) - .map_err(|err: ErrorStack| { - X509Error::ErrorConvertingFileToData(err, key_path.to_string()) - })?; + })?) { + Ok(certificate) => { + println!("Reading Certificate at {} ✅", cert_path); + certificate + } + Err(err) => { + println!("Reading Certificate at {} ❌", cert_path); + return Err(X509Error::ErrorConvertingFileToData( + err, + cert_path.to_string(), + )); + } + }; + + let key: PKey = + match PKey::private_key_from_pem(&std::fs::read(key_path).map_err( + |err: io::Error| X509Error::ErrorReadingCertFile(err, key_path.to_string()), + )?) { + Ok(key) => { + println!("Reading Key at {} ✅", key_path); + key + }, + Err(err) => { + println!("Reading Key at {} ❌", key_path); + return Err(X509Error::ErrorConvertingFileToData( + err, + key_path.to_string(), + )); + } + }; + Ok((cert, key)) } diff --git a/core/src/x509/ca_req.rs b/core/src/x509/ca_req.rs index 53f7100..ba1dbd5 100644 --- a/core/src/x509/ca_req.rs +++ b/core/src/x509/ca_req.rs @@ -88,17 +88,37 @@ impl CAReq { } pub fn read_csr_from_file(file_name: &str) -> X509Result { - let mut file: File = File::open(file_name).map_err(|err: io::Error| { - X509Error::ErrorReadingCertFile(err, file_name.to_string()) - })?; + println!(); + let mut file: File = match File::open(file_name) { + Ok(f) => f, + Err(err) => { + eprintln!("Reading Signing Request at {} ❌", file_name); + return Err(X509Error::ErrorReadingCertFile(err, file_name.to_string())); + } + }; + let mut buffer: Vec = Vec::new(); - file.read_to_end(&mut buffer).map_err(|err: io::Error| { - X509Error::ErrorReadingCertFile(err, file_name.to_string()) - })?; - let csr: X509Req = X509Req::from_pem(&buffer).map_err(|err: ErrorStack| { - X509Error::ErrorConvertingFileToData(err, file_name.to_string()) - })?; - Ok(csr) + match file.read_to_end(&mut buffer) { + Ok(_) => (), + Err(err) => { + eprintln!("Reading Signing Request at {} ❌", file_name); + return Err(X509Error::ErrorReadingCertFile(err, file_name.to_string())); + } + }; + + match X509Req::from_pem(&buffer) { + Ok(csr) => { + println!("Reading Signing Request at {} ✅", file_name); + Ok(csr) + } + Err(err) => { + eprintln!("Reading Signing Request at {} ❌", file_name); + Err(X509Error::ErrorConvertingFileToData( + err, + file_name.to_string(), + )) + } + } } pub fn save_key(key: &PKey, path: &str) -> X509Result<()> { diff --git a/core/src/x509/errors.rs b/core/src/x509/errors.rs index 49b9456..48a0c8a 100644 --- a/core/src/x509/errors.rs +++ b/core/src/x509/errors.rs @@ -1,4 +1,5 @@ use crate::errors::{CertKeyPairError, SerialNumberError}; +use colored::*; use core::fmt; use openssl::error::ErrorStack; use std::{error::Error, io}; @@ -35,94 +36,170 @@ impl fmt::Display for X509Error { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { match self { Self::CertificateStackPushError(err) => { - write!(f, "Failed to push item to stack: {}", err) + write!( + f, + "{}: Failed to push item to stack: {}", + "Error".red(), + err + ) } Self::CertificateStackInitializationError(err) => { - write!(f, "Failed to initialize extension stack: {}", err) + write!( + f, + "{}: Failed to initialize extension stack: {}", + "Error".red(), + err + ) } Self::InitCARequestCertKeyPairError(err) => { write!( f, - "Failed to initialize Certificate Signing Request : {}", + "{}: Failed to initialize Certificate Signing Request : {}", + "Error".red(), err ) } Self::X509NameBuilderInitializeError(err) => { - write!(f, "Failed to initialize Name Builder: {}", err) + write!( + f, + "{}: Failed to initialize Name Builder: {}", + "Error".red(), + err + ) } Self::X509NameBuilderEntryError(err, entry, value) => { write!( f, - "Error adding entry {} in name builder with value {} : {}", - entry, value, err + "{}: Adding entry {} in name builder with value {} : {}", + "Error".red(), + entry, + value, + err ) } Self::X509CertificateBuilderInitializeError(err) => { - write!(f, "Failed to initialize Certificate Builder: {}", err) + write!( + f, + "{}: Failed to initialize Certificate Builder: {}", + "Error".red(), + err + ) } Self::X509CertificateBuilerEntryError(err, entry) => { write!( f, - "Error adding entry {} in Certificate builder : {}", - entry, err + "{}: adding entry {} in Certificate builder : {}", + "Error".red(), + entry, + err ) } Self::X509CSRToPEMError(err) => { - write!(f, "Error Converting Certificate to PEM : {}", err) + write!( + f, + "{}: Converting Certificate to PEM : {}", + "Error".red(), + err + ) } Self::X509PEMFileCreationError(err) => { - write!(f, "Error creating PEM file at specified location: {}", err) + write!( + f, + "{}: creating PEM file at specified location: {}", + "Error".red(), + err + ) } Self::X509WriteToFileError(err) => { - write!(f, "Error writing to specified PEM file: {}", err) + write!( + f, + "{}: writing to specified PEM file: {}", + "Error".red(), + err + ) } Self::InitSerialNumberGenerationError(err) => { - write!(f, "Error Generating Random Serial Number -> {}", err) + write!( + f, + "{}: Generating Random Serial Number -> {}", + "Error".red(), + err + ) } Self::GenerateNotBeforeError(err) => { - write!(f, "Error Generating Not Before Time: {}", err) + write!(f, "{}: Generating Not Before Time: {}", "Error".red(), err) } Self::GenerateNotAfterError(err) => { - write!(f, "Error Generating Not After Time: {}", err) + write!(f, "{}: Generating Not After Time: {}", "Error".red(), err) } Self::BasicConstraintsInitializeError(err) => { write!( f, - "Error Initializing Basic Constrainsts for Certificate: {}", + "{}: Initializing Basic Constrainsts for Certificate: {}", + "Error".red(), err ) } Self::ErrorGettingPublicKeyFromCSR(err) => { - write!(f, "Error getting Public Key From CSR: {}", err) + write!(f, "{}: getting Public Key From CSR: {}", "Error".red(), err) } Self::KeyUsageBuildError(err) => { - write!(f, "Error building Key Usage for Certificate: {}", err) + write!( + f, + "{}: building Key Usage for Certificate: {}", + "Error".red(), + err + ) } Self::ExtendedKeyUsageBuildError(err) => { write!( f, - "Error building Extended Key Usage for Certificate: {}", + "{}: building Extended Key Usage for Certificate: {}", + "Error".red(), err ) } Self::ErrorReadingCertFile(err, path) => { - writeln!(f, "Error Reading Cert File at Path {} : {}", path, err) + writeln!( + f, + "{}: Reading Cert File at Path {} : {}", + "Error".red(), + path, + err + ) } Self::ErrorConvertingFileToData(err, path) => { - writeln!(f, "Error Converting file {} to desired data: {}", path, err) + writeln!( + f, + "{}: Converting file {} to desired data: {}", + "Error".red(), + path, + err + ) } Self::PKCS8EncodingError(err) => { - writeln!(f, "Failed to encode generated key to PKCS8 Format: {}", err) + writeln!( + f, + "{}: Failed to encode generated key to PKCS8 Format: {}", + "Error".red(), + err + ) } Self::PEMEncodingError(err) => { writeln!( f, - "Failed to encode generated certificate to PEM Format: {}", + "{}: Failed to encode generated certificate to PEM Format: {}", + "Error".red(), err ) } Self::SANCouldNotBuildError(err) => { - write!(f, "Error building Subject Alternative Name : {}", err) + write!( + f, + "{}: building Subject Alternative Name : {}", + "Error".red(), + err + ) } } }