This repository contains a prototypical Rust implementation of (synchronized) signatures based on tweakable hash functions and incomparable encodings. The code has not been audited and is not meant to be used in production. It is a playground to explore and benchmark these signatures. Use it at your own risk.
Note: Rust version >= 1.87 is required.
If you want to use this library, the main interface is that of a (synchronized) signature scheme, which is defined in the Signature trait. Here is a summary:
- A function
key_gen
to generate keys. - A function
sign
to sign messages using the secret key with respect to an epoch. - A function
verify
to verify signatures for a given message, public key, and epoch.
Importantly, each pair of secret key and epoch must not be used twice as input to sign
.
For a signature scheme T: SignatureScheme
, an example to use this interface may be as follows:
// generate keys (assume we have an rng)
let (pk, sk) = T::key_gen(&mut rng, 0, T::LIFETIME as usize);
// sign a random message for a random epoch
let message = rng.random();
let epoch = rng.random_range(0..activation_duration) as u32;
let sig = S::sign(&mut rng, &sk, epoch, &message);
// verify the signature
let is_valid = S::verify(&pk, epoch, &message, &sig);
See also function test_signature_scheme_correctness
in this file.
The code implements a generic framework from this paper, which builds XMSS-like hash-based signatures from a primitive called incomparable encodings.
Hardcoded instantiations of this generic framework (using SHA3 or Poseidon2) are defined in hashsig::signature::generalized_xmss
.
The parameters have been chosen based on the analysis in the paper using Python scripts. Details are as follows:
Submodule | Paper / Documentation | Parameters Set With |
---|---|---|
instantiations_sha::* |
original paper | this repository |
instantiations_poseidon::* |
original paper | this repository |
instantiations_poseidon_top_level::* |
this document, inspired by this | this repository |
Instantiations for different key lifetimes and different encodings are given in these modules.
Run the tests with
cargo test
By default, this will exclude some of the tests. In particular, correctness tests for real instantiations take quite long and are excluded. If you want to run all tests, you can use
cargo test --release --features slow-tests
Removing the --release
is also an option but tests will take even longer.
Benchmarks are provided using criterion. They take a while, as key generation is expensive, and as a large number of schemes are benchmarked. Run them with
cargo bench
The schemes that are benchmarked are hardcoded instantiations of the generic framework, which are defined in hashsig::signature::generalized_xmss
.
The parameters of these instantiations have been chosen carefully with the aim to achieve a desired security level.
By default, key generation is not benchmarked. There are two options to benchmark it:
- add the option
--features with-gen-benches-sha
or--features with-gen-benches-poseidon
or--features with-gen-benches-poseidon-top-level
tocargo bench
. Note that this will make benchmarks very slow, as key generation will be repeated within the benchmarks. Especially for Poseidon, this is not recommended. - use code similar to the one provided in
src/bin/main.rs
and run it withcargo run --release
.
If criterion only generates json files, one way to extract all means for all benchmarks easily (without re-running criterion) is to run
python3 benchmark-mean.py target
Confidence intervals can also be shown via
python3 benchmark-mean.py target --intervals
Apache Version 2.0.