use std::error::Error;
-use std::path::PathBuf;
+use std::path::{Path, PathBuf};
use std::process;
use structopt::StructOpt;
/// Collect candidate hashes based on options and match them against a calculated hash
mod verify;
+/// Problem running the program
+const EXIT_ERR: i32 = 1;
+/// Verification was performed and was not a match
+const EXIT_MISMATCH: i32 = 2;
+
#[derive(StructOpt)]
#[structopt(name = "hashgood")]
pub struct Opt {
- /// Read the hash from the clipboard
- #[cfg(feature = "paste")]
- #[structopt(short = "p", long = "paste")]
- paste: bool,
-
/// Disable ANSI colours in output
#[structopt(short = "C", long = "no-colour")]
no_colour: bool,
hash: Option<String>,
}
-impl Opt {
- fn get_paste(&self) -> bool {
- #[cfg(feature = "paste")] {
- return self.paste;
- }
- #[cfg(not(feature = "paste"))] {
- return false;
- }
- }
-}
-
/// Types of supported digest algorithm
#[derive(Debug, PartialEq, Copy, Clone)]
pub enum Algorithm {
}
/// The method by which one or more hashes were supplied to verify the calculated digest
+#[derive(Debug, PartialEq)]
pub enum VerificationSource {
CommandArgument,
- Clipboard,
- RawFile(PathBuf),
- DigestsFile(PathBuf),
+ RawFile(String),
+ DigestsFile(String),
}
/// A complete standalone hash result
}
impl Hash {
- pub fn new(alg: Algorithm, bytes: Vec<u8>, path: &PathBuf) -> Self {
+ pub fn new(alg: Algorithm, bytes: Vec<u8>, path: &Path) -> Self {
// Taking the filename component should always work?
// If not, just fall back to the full path
let filename = match path.file_name() {
}
/// A possible hash to match against. The algorithm is assumed.
+#[derive(Debug, PartialEq)]
pub struct CandidateHash {
bytes: Vec<u8>,
filename: Option<String>,
/// A list of candidate hashes that our input could potentially match. At this point it is
/// assumed that we will be verifying a digest of a particular, single algorithm.
+#[derive(Debug, PartialEq)]
pub struct CandidateHashes {
alg: Algorithm,
hashes: Vec<CandidateHash>,
}
/// Summary of an atetmpt to match the calculated digest against candidates
+#[derive(PartialEq)]
pub enum MatchLevel {
Ok,
Maybe,
fn main() {
hashgood().unwrap_or_else(|e| {
eprintln!("Error: {}", e);
- process::exit(1);
+ process::exit(EXIT_ERR);
});
}
fn hashgood() -> Result<(), Box<dyn Error>> {
let opt = get_verified_options()?;
let candidates = verify::get_candidate_hashes(&opt)?;
- let input = calculate::get_input_reader(&opt.input)?;
+ let input = calculate::get_input_reader(opt.input.as_path())?;
if let Some(c) = candidates {
// If we have a candidate hash of a particular type, use that specific algorithm
let hashes = calculate::create_digests(&[c.alg], input)?;
if c.alg == alg {
let hash = Hash::new(alg, bytes, &opt.input);
let verification = verify::verify_hash(&hash, &c);
+ let successful_match = verification.match_level == MatchLevel::Ok;
display::print_hash(
&hash,
verification.comparison_hash,
)?;
display::print_messages(verification.messages, opt.no_colour)?;
display::print_match_level(verification.match_level, opt.no_colour)?;
+ if !successful_match {
+ process::exit(EXIT_MISMATCH);
+ }
}
}
} else {
/// Parse the command line options and check for ambiguous or inconsistent settings
fn get_verified_options() -> Result<Opt, String> {
let opt = Opt::from_args();
- let hash_methods =
- opt.hash.is_some() as i32 + opt.get_paste() as i32 + opt.hash_file.is_some() as i32;
+ let hash_methods = opt.hash.is_some() as i32 + opt.hash_file.is_some() as i32;
if hash_methods > 1 {
if opt.hash.is_some() {
eprintln!("* specified as command line argument");
}
- if opt.get_paste() {
- eprintln!("* paste from clipboard (-p)")
- }
if opt.hash_file.is_some() {
eprintln!("* check hash from file (-c)")
}