diff --git a/src/field.rs b/src/field.rs index 95f87b3..aa2d7eb 100644 --- a/src/field.rs +++ b/src/field.rs @@ -58,7 +58,7 @@ const EXP_TABLE: [u8; 512] = [ 0x58, 0xb0, 0x7d, 0xfa, 0xe9, 0xcf, 0x83, 0x1b, 0x36, 0x6c, 0xd8, 0xad, 0x47, 0x8e, 0x01, 0x02, ]; -#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)] +#[derive(Debug, PartialEq, Copy, Clone)] pub struct GF256(pub u8); #[allow(clippy::suspicious_arithmetic_impl)] diff --git a/src/lib.rs b/src/lib.rs index f3ece9b..e5b9b50 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -64,7 +64,7 @@ impl Sharks { math::get_evaluator(polys) } - /// Given a slice of shares, recovers the original secret. + /// Given a collection of shares, recovers the original secret. /// If the number of distinct shares is less than the minimum threshold an `Err` is returned, /// otherwise an `Ok` containing the secret. /// @@ -82,18 +82,28 @@ impl Sharks { /// secret = sharks.recover(&shares); /// // Not enough shares to recover secret /// assert!(secret.is_err()); - pub fn recover<'a, I, J>(&self, shares: I) -> Result, &str> + pub fn recover<'a, T>(&self, shares: T) -> Result, &str> where - I: IntoIterator, - J: Iterator + Clone, + T: IntoIterator, + T::IntoIter: Iterator, { - let shares = shares.into_iter(); - let shares_x: HashSet = shares.clone().map(|s| s.x.0).collect(); + let (keys, shares) = shares + .into_iter() + .map(|s| { + ( + s.x.0, + Share { + x: s.x, + y: s.y.clone(), + }, + ) + }) + .unzip::, Vec>(); - if shares_x.len() < self.0 as usize { + if keys.len() < self.0 as usize { Err("Not enough shares to recover original secret") } else { - Ok(math::interpolate(shares)) + Ok(math::interpolate(shares.as_slice())) } } } diff --git a/src/math.rs b/src/math.rs index 70a0e2a..a760cb0 100644 --- a/src/math.rs +++ b/src/math.rs @@ -9,25 +9,14 @@ use super::share::Share; // The expected `shares` argument format is the same as the output by the `get_evaluatorĀ“ function. // Where each (key, value) pair corresponds to one share, where the key is the `x` and the value is a vector of `y`, // where each element corresponds to one of the secret's byte chunks. -pub fn interpolate<'a, I, J>(shares: I) -> Vec -where - I: IntoIterator, - J: Iterator + Clone, -{ - let shares = shares.into_iter(); - (0..shares - .clone() - .peekable() - .peek() - .expect("There should be at least one share") - .y - .len()) +pub fn interpolate(shares: &[Share]) -> Vec { + (0..shares[0].y.len()) .map(|s| { shares - .clone() + .iter() .map(|s_i| { shares - .clone() + .iter() .filter(|s_j| s_j.x != s_i.x) .map(|s_j| s_j.x / (s_j.x - s_i.x)) .product::()