Skip to content
Snippets Groups Projects
Commit 5465143a authored by STEVAN Antoine's avatar STEVAN Antoine :crab:
Browse files

add some more documentation (dragoon/komodo!159)

- 1c9077f6 add documentation to the errors
- c388cbbb better FEC doc
- 8d58278d doc of `zk.nb_elements_in_setup`
parent dad69f2c
No related branches found
No related tags found
No related merge requests found
...@@ -3,30 +3,49 @@ ...@@ -3,30 +3,49 @@
//! there are a few linear algebra errors and some related to ZK. //! there are a few linear algebra errors and some related to ZK.
use thiserror::Error; use thiserror::Error;
/// An error that Komodo could end up producing.
///
/// There are a few families of errors in Komodo:
/// - related to _linear algebra_
/// - related to FEC
/// - related to proving the shards
#[derive(Clone, Debug, Error, PartialEq)] #[derive(Clone, Debug, Error, PartialEq)]
pub enum KomodoError { pub enum KomodoError {
/// `{0}` is a custom error message when a matrix is invalid.
#[error("Invalid matrix elements: {0}")] #[error("Invalid matrix elements: {0}")]
InvalidMatrixElements(String), InvalidMatrixElements(String),
/// `{0}` and `{1}` are the shape of the rectangular matrix.
#[error("Matrix is not a square, ({0} x {1})")] #[error("Matrix is not a square, ({0} x {1})")]
NonSquareMatrix(usize, usize), NonSquareMatrix(usize, usize),
/// `{0}` is the ID of the row where the matrix inversion failed.
#[error("Matrix is not invertible at row {0}")] #[error("Matrix is not invertible at row {0}")]
NonInvertibleMatrix(usize), NonInvertibleMatrix(usize),
/// `{0}` and `{1}` are the shape of the left matrix and `{2}` and `{3}` are the shape of the
/// right matrix.
#[error("Matrices don't have compatible shapes: ({0} x {1}) and ({2} x {3})")] #[error("Matrices don't have compatible shapes: ({0} x {1}) and ({2} x {3})")]
IncompatibleMatrixShapes(usize, usize, usize, usize), IncompatibleMatrixShapes(usize, usize, usize, usize),
/// `{0}` and `{1}` are the IDs of the non-distinct _Vandermonde_ points and `{2}` is the list
/// of all the _Vandermonde_ points.
#[error( #[error(
"Seed points of a Vandermonde matrix should be distinct: {0} and {1} are the same ({2})" "Seed points of a Vandermonde matrix should be distinct: {0} and {1} are the same ({2})"
)] )]
InvalidVandermonde(usize, usize, String), InvalidVandermonde(usize, usize, String),
/// `{0}` is the actual number of shards and `{1}` is the expected amount.
#[error("Expected at least {1} shards, got {0}")] #[error("Expected at least {1} shards, got {0}")]
TooFewShards(usize, usize), TooFewShards(usize, usize),
/// `{0}` is a custom error message when shards are incompatible.
#[error("Shards are incompatible: {0}")] #[error("Shards are incompatible: {0}")]
IncompatibleShards(String), IncompatibleShards(String),
/// `{0}` is a custom error message when blocks are incompatible.
#[error("Blocks are incompatible: {0}")] #[error("Blocks are incompatible: {0}")]
IncompatibleBlocks(String), IncompatibleBlocks(String),
#[error("Degree is zero")] #[error("Degree is zero")]
DegreeIsZero, DegreeIsZero,
/// `{0}` is the supported degree of the trusted setup and `{1}` is the actual requested
/// polynomnial degree
#[error("too many coefficients: max is {0}, found {0}")] #[error("too many coefficients: max is {0}, found {0}")]
TooFewPowersInTrustedSetup(usize, usize), TooFewPowersInTrustedSetup(usize, usize),
/// `{0}` is a custom error message.
#[error("Another error: {0}")] #[error("Another error: {0}")]
Other(String), Other(String),
} }
//! a module to encode, recode and decode shards of data with FEC methods //! a module to encode, recode and decode shards of data with FEC methods.
use ark_ff::PrimeField; use ark_ff::PrimeField;
use ark_serialize::{CanonicalDeserialize, CanonicalSerialize}; use ark_serialize::{CanonicalDeserialize, CanonicalSerialize};
...@@ -8,24 +8,34 @@ use rs_merkle::{algorithms::Sha256, Hasher}; ...@@ -8,24 +8,34 @@ use rs_merkle::{algorithms::Sha256, Hasher};
use crate::{error::KomodoError, field, linalg::Matrix}; use crate::{error::KomodoError, field, linalg::Matrix};
/// representation of a FEC shard of data /// representation of a FEC shard of data.
///
/// - `k` is the code parameter, required to decode
/// - the _linear combination_ tells the decoded how the shard was constructed,
/// with respect to the original source shards => this effectively allows
/// support for _recoding_
/// - the hash and the size represent the original data
#[derive(Debug, Default, Clone, PartialEq, CanonicalSerialize, CanonicalDeserialize)] #[derive(Debug, Default, Clone, PartialEq, CanonicalSerialize, CanonicalDeserialize)]
pub struct Shard<F: PrimeField> { pub struct Shard<F: PrimeField> {
/// the code parameter, required to decode
pub k: u32, pub k: u32,
/// tells the decoder how the shard was constructed with respect to the original source shards
///
/// this effectively allows support for _recoding_.
///
/// If we denote the $k$ source shards by $(s_i)_{0 \le i \lt k}$, the linear combination by $k$
/// coefficients $(\alpha_i)_{0 \le i \lt k}$ and $s$ the shard itself, then
///
/// $$ s = \sum\limits_{i = 0}^{k - 1} \alpha_i s_i$$
pub linear_combination: Vec<F>, pub linear_combination: Vec<F>,
/// the hash of the original data, used for validation
pub hash: Vec<u8>, pub hash: Vec<u8>,
/// the shard itself
pub data: Vec<F>, pub data: Vec<F>,
/// the size of the original data, used for padding
pub size: usize, pub size: usize,
} }
impl<F: PrimeField> Shard<F> { impl<F: PrimeField> Shard<F> {
/// compute the linear combination between two [`Shard`]s /// compute the linear combination between two [`Shard`]s
///
/// if we denote the [`Shard`] itself and the other [`Shard`] by $s$ and $o$ respectively, the
/// output is
/// $$ \alpha s + \beta o $$
pub fn recode_with(&self, alpha: F, other: &Self, beta: F) -> Self { pub fn recode_with(&self, alpha: F, other: &Self, beta: F) -> Self {
if alpha.is_zero() { if alpha.is_zero() {
return other.clone(); return other.clone();
...@@ -56,10 +66,16 @@ impl<F: PrimeField> Shard<F> { ...@@ -56,10 +66,16 @@ impl<F: PrimeField> Shard<F> {
/// compute the linear combination between an arbitrary number of [`Shard`]s /// compute the linear combination between an arbitrary number of [`Shard`]s
/// ///
/// > **Note** /// > **Note**
/// >
/// > this is basically a multi-[`Shard`] wrapper around [`Shard::recode_with`] /// > this is basically a multi-[`Shard`] wrapper around [`Shard::recode_with`]
/// > /// >
/// > returns [`None`] if number of shards is not the same as the number of /// > returns [`None`] if the number of shards is not the same as the number of
/// > coefficients or if no shards are provided. /// > coefficients or if no shards are provided.
///
/// if the shards are the $(s_i)_{1 \le i \le n}$ and the coefficients the
/// $(\alpha_i)_{0 \le i \le n}$, then the output will be
///
/// $$ \sum\limits_{i = 1}^{n} \alpha_i s_i$$
pub fn recode_with_coeffs<F: PrimeField>(shards: &[Shard<F>], coeffs: &[F]) -> Option<Shard<F>> { pub fn recode_with_coeffs<F: PrimeField>(shards: &[Shard<F>], coeffs: &[F]) -> Option<Shard<F>> {
if shards.len() != coeffs.len() { if shards.len() != coeffs.len() {
return None; return None;
...@@ -86,6 +102,7 @@ pub fn recode_with_coeffs<F: PrimeField>(shards: &[Shard<F>], coeffs: &[F]) -> O ...@@ -86,6 +102,7 @@ pub fn recode_with_coeffs<F: PrimeField>(shards: &[Shard<F>], coeffs: &[F]) -> O
/// same or the hash of the data is different, an error will be returned. /// same or the hash of the data is different, an error will be returned.
/// ///
/// > **Note** /// > **Note**
/// >
/// > this is a wrapper around [`recode_with_coeffs`]. /// > this is a wrapper around [`recode_with_coeffs`].
pub fn recode_random<F: PrimeField>( pub fn recode_random<F: PrimeField>(
shards: &[Shard<F>], shards: &[Shard<F>],
...@@ -119,8 +136,12 @@ pub fn recode_random<F: PrimeField>( ...@@ -119,8 +136,12 @@ pub fn recode_random<F: PrimeField>(
/// applies a given encoding matrix to some data to generate encoded shards /// applies a given encoding matrix to some data to generate encoded shards
/// ///
/// > **Note** /// > **Note**
/// >
/// > the input data and the encoding matrix should have compatible shapes, /// > the input data and the encoding matrix should have compatible shapes,
/// > otherwise, an error might be thrown to the caller. /// > otherwise, an error might be thrown to the caller.
///
/// Padding might be applied depending on the size of the data compared to the size of the encoding
/// matrix.
pub fn encode<F: PrimeField>( pub fn encode<F: PrimeField>(
data: &[u8], data: &[u8],
encoding_mat: &Matrix<F>, encoding_mat: &Matrix<F>,
...@@ -152,10 +173,10 @@ pub fn encode<F: PrimeField>( ...@@ -152,10 +173,10 @@ pub fn encode<F: PrimeField>(
.collect()) .collect())
} }
/// reconstruct the original data from a set of encoded, possibly recoded, /// reconstruct the original data from a set of encoded, possibly recoded, shards
/// shards
/// ///
/// > **Note** /// > **Note**
/// >
/// > this function might fail in a variety of cases /// > this function might fail in a variety of cases
/// > - if there are too few shards /// > - if there are too few shards
/// > - if there are linear dependencies between shards /// > - if there are linear dependencies between shards
...@@ -224,6 +245,7 @@ mod tests { ...@@ -224,6 +245,7 @@ mod tests {
/// `contains_one_of(x, set)` is true iif `x` fully contains one of the lists from `set` /// `contains_one_of(x, set)` is true iif `x` fully contains one of the lists from `set`
/// ///
/// > **Note** /// > **Note**
/// >
/// > see [`containment`] for some example /// > see [`containment`] for some example
fn contains_one_of(x: &[usize], set: &[Vec<usize>]) -> bool { fn contains_one_of(x: &[usize], set: &[Vec<usize>]) -> bool {
set.iter().any(|y| y.iter().all(|z| x.contains(z))) set.iter().any(|y| y.iter().all(|z| x.contains(z)))
......
...@@ -158,8 +158,8 @@ where ...@@ -158,8 +158,8 @@ where
Ok(commits) Ok(commits)
} }
// compute the number of elements that a _trusted setup_ should have for data of /// compute the number of elements that a _trusted setup_ should have for data of
// a certain expected size /// a certain expected size
pub fn nb_elements_in_setup<F: PrimeField>(nb_bytes: usize) -> usize { pub fn nb_elements_in_setup<F: PrimeField>(nb_bytes: usize) -> usize {
let bytes_per_element = (F::MODULUS_BIT_SIZE as usize) / 8; let bytes_per_element = (F::MODULUS_BIT_SIZE as usize) / 8;
nb_bytes / bytes_per_element nb_bytes / bytes_per_element
......
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment