Sha256 based proof of work over a typed piece of data.
Any type that implementes serde::Deserialize can be tagged with a proof of work.
Prove we did work targeting a phrase.
```rust use pow::Pow;
// very easy mode let difficulty = u128::maxvalue() - u128::maxvalue() / 2;
let phrase = b"Phrase to tag.".tovec(); let pw = Pow::provework(&phrase, difficulty).unwrap(); assert!(pw.score(&phrase).unwrap() >= difficulty); ```
Prove more difficult work. This time targeting a time.
```rust // more diffcult, takes around 100000 hashes to generate proof let difficulty = u128::maxvalue() - u128::maxvalue() / 100000;
let now: u64 = getunixtimeseconds(); let pw = Pow::provework(&now, difficulty).unwrap(); assert!(pw.score(&now).unwrap() >= difficulty); ```
Define a blockchain block.
rust
struct Block<T> {
prev: [u8; 32], // hash of last block
payload: T, // generic data
proof_of_work: Pow<([u8; 32], T)>,
}
To score a proof of work for a given (target, Pow) pair: Sha256 is calculated over the concatenation SALT + target + Pow. The first 16 bytes of the hash are interpreted as a 128 bit unsigned integer. That integer is the score. A constant, SALT, is used as prefix to prevent pow reuse from other systems such as proof of work blockchains.
In other words:
rust
fn score<T: Serialize>(target: &T, pow_tag: &Pow<T>) -> u128 {
let bytes = serialize(&SALT) + serialize(target) + serialize(pow_tag);
let hash = sha256(&bytes);
deserialize(&hash[..16])
}
It shouldn't matter to users of this library, but the bincode crate is used for cheap deterministic serialization. All values are serialized using network byte order.
Given a minimum score m. A Pow p satisfies the minimum score for target t iff score(t, p) >= m.
Difficulty settings are usually best adjusted dynamically a la bitcoin.
To manually select a difficulty, choose the average number of hashes required.
rust
fn difficulty(average: u128) -> u128 {
debug_assert_ne!(average, 0, "It is impossible to prove work in zero attempts.");
let m = u128::max_value();
m - m / average
}
Conversely, to calculate probable number of hashes required to satisfy a given minimum difficulty.
rust
fn average(difficulty: u128) -> u128 {
let m = u128::max_value();
if difficulty == m {
return m;
}
m / (m - difficulty)
}