Merge pull request #54 from garious/imperative-genesis
Boot genesis block helper
This commit is contained in:
@ -6,10 +6,9 @@ use hash::Hash;
|
|||||||
use entry::Entry;
|
use entry::Entry;
|
||||||
use event::Event;
|
use event::Event;
|
||||||
use transaction::Transaction;
|
use transaction::Transaction;
|
||||||
use signature::{PublicKey, Signature};
|
use signature::{KeyPair, PublicKey, Signature};
|
||||||
use genesis::Genesis;
|
use genesis::Genesis;
|
||||||
use historian::{reserve_signature, Historian};
|
use historian::{reserve_signature, Historian};
|
||||||
use ring::signature::Ed25519KeyPair;
|
|
||||||
use std::sync::mpsc::SendError;
|
use std::sync::mpsc::SendError;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use std::result;
|
use std::result;
|
||||||
@ -131,7 +130,7 @@ impl Accountant {
|
|||||||
pub fn transfer(
|
pub fn transfer(
|
||||||
self: &mut Self,
|
self: &mut Self,
|
||||||
n: i64,
|
n: i64,
|
||||||
keypair: &Ed25519KeyPair,
|
keypair: &KeyPair,
|
||||||
to: PublicKey,
|
to: PublicKey,
|
||||||
) -> Result<Signature> {
|
) -> Result<Signature> {
|
||||||
let tr = Transaction::new(keypair, to, n, self.last_id);
|
let tr = Transaction::new(keypair, to, n, self.last_id);
|
||||||
@ -149,14 +148,15 @@ mod tests {
|
|||||||
use super::*;
|
use super::*;
|
||||||
use signature::{generate_keypair, get_pubkey};
|
use signature::{generate_keypair, get_pubkey};
|
||||||
use logger::ExitReason;
|
use logger::ExitReason;
|
||||||
use genesis::Creator;
|
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_accountant() {
|
fn test_accountant() {
|
||||||
let bob = Creator::new(1_000);
|
let alice = Genesis::new(10_000);
|
||||||
let bob_pubkey = bob.pubkey;
|
let bob_pubkey = get_pubkey(&generate_keypair());
|
||||||
let alice = Genesis::new(10_000, vec![bob]);
|
|
||||||
let mut acc = Accountant::new(&alice, Some(2));
|
let mut acc = Accountant::new(&alice, Some(2));
|
||||||
|
acc.transfer(1_000, &alice.get_keypair(), bob_pubkey)
|
||||||
|
.unwrap();
|
||||||
|
assert_eq!(acc.get_balance(&bob_pubkey).unwrap(), 1_000);
|
||||||
|
|
||||||
acc.transfer(500, &alice.get_keypair(), bob_pubkey).unwrap();
|
acc.transfer(500, &alice.get_keypair(), bob_pubkey).unwrap();
|
||||||
assert_eq!(acc.get_balance(&bob_pubkey).unwrap(), 1_500);
|
assert_eq!(acc.get_balance(&bob_pubkey).unwrap(), 1_500);
|
||||||
@ -170,10 +170,11 @@ mod tests {
|
|||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_invalid_transfer() {
|
fn test_invalid_transfer() {
|
||||||
let bob = Creator::new(1_000);
|
let alice = Genesis::new(11_000);
|
||||||
let bob_pubkey = bob.pubkey;
|
|
||||||
let alice = Genesis::new(11_000, vec![bob]);
|
|
||||||
let mut acc = Accountant::new(&alice, Some(2));
|
let mut acc = Accountant::new(&alice, Some(2));
|
||||||
|
let bob_pubkey = get_pubkey(&generate_keypair());
|
||||||
|
acc.transfer(1_000, &alice.get_keypair(), bob_pubkey)
|
||||||
|
.unwrap();
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
acc.transfer(10_001, &alice.get_keypair(), bob_pubkey),
|
acc.transfer(10_001, &alice.get_keypair(), bob_pubkey),
|
||||||
Err(AccountingError::InsufficientFunds)
|
Err(AccountingError::InsufficientFunds)
|
||||||
@ -192,11 +193,10 @@ mod tests {
|
|||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_transfer_to_newb() {
|
fn test_transfer_to_newb() {
|
||||||
let alice = Genesis::new(10_000, vec![]);
|
let alice = Genesis::new(10_000);
|
||||||
let mut acc = Accountant::new(&alice, Some(2));
|
let mut acc = Accountant::new(&alice, Some(2));
|
||||||
let alice_keypair = alice.get_keypair();
|
let alice_keypair = alice.get_keypair();
|
||||||
let bob_keypair = generate_keypair();
|
let bob_pubkey = get_pubkey(&generate_keypair());
|
||||||
let bob_pubkey = get_pubkey(&bob_keypair);
|
|
||||||
acc.transfer(500, &alice_keypair, bob_pubkey).unwrap();
|
acc.transfer(500, &alice_keypair, bob_pubkey).unwrap();
|
||||||
assert_eq!(acc.get_balance(&bob_pubkey).unwrap(), 500);
|
assert_eq!(acc.get_balance(&bob_pubkey).unwrap(), 500);
|
||||||
|
|
||||||
|
@ -6,10 +6,9 @@ use std::net::UdpSocket;
|
|||||||
use std::io;
|
use std::io;
|
||||||
use bincode::{deserialize, serialize};
|
use bincode::{deserialize, serialize};
|
||||||
use transaction::Transaction;
|
use transaction::Transaction;
|
||||||
use signature::{PublicKey, Signature};
|
use signature::{KeyPair, PublicKey, Signature};
|
||||||
use hash::Hash;
|
use hash::Hash;
|
||||||
use entry::Entry;
|
use entry::Entry;
|
||||||
use ring::signature::Ed25519KeyPair;
|
|
||||||
use accountant_skel::{Request, Response};
|
use accountant_skel::{Request, Response};
|
||||||
|
|
||||||
pub struct AccountantStub {
|
pub struct AccountantStub {
|
||||||
@ -36,7 +35,7 @@ impl AccountantStub {
|
|||||||
pub fn transfer(
|
pub fn transfer(
|
||||||
&self,
|
&self,
|
||||||
n: i64,
|
n: i64,
|
||||||
keypair: &Ed25519KeyPair,
|
keypair: &KeyPair,
|
||||||
to: PublicKey,
|
to: PublicKey,
|
||||||
last_id: &Hash,
|
last_id: &Hash,
|
||||||
) -> io::Result<Signature> {
|
) -> io::Result<Signature> {
|
||||||
@ -116,16 +115,16 @@ mod tests {
|
|||||||
use accountant_skel::AccountantSkel;
|
use accountant_skel::AccountantSkel;
|
||||||
use std::thread::{sleep, spawn};
|
use std::thread::{sleep, spawn};
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
use genesis::{Creator, Genesis};
|
use genesis::Genesis;
|
||||||
|
use signature::{generate_keypair, get_pubkey};
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_accountant_stub() {
|
fn test_accountant_stub() {
|
||||||
let addr = "127.0.0.1:9000";
|
let addr = "127.0.0.1:9000";
|
||||||
let send_addr = "127.0.0.1:9001";
|
let send_addr = "127.0.0.1:9001";
|
||||||
let bob = Creator::new(1_000);
|
let alice = Genesis::new(10_000);
|
||||||
let bob_pubkey = bob.pubkey;
|
|
||||||
let alice = Genesis::new(10_000, vec![bob]);
|
|
||||||
let acc = Accountant::new(&alice, None);
|
let acc = Accountant::new(&alice, None);
|
||||||
|
let bob_pubkey = get_pubkey(&generate_keypair());
|
||||||
spawn(move || AccountantSkel::new(acc).serve(addr).unwrap());
|
spawn(move || AccountantSkel::new(acc).serve(addr).unwrap());
|
||||||
sleep(Duration::from_millis(30));
|
sleep(Duration::from_millis(30));
|
||||||
|
|
||||||
@ -135,6 +134,6 @@ mod tests {
|
|||||||
let sig = acc.transfer(500, &alice.get_keypair(), bob_pubkey, &last_id)
|
let sig = acc.transfer(500, &alice.get_keypair(), bob_pubkey, &last_id)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
acc.wait_on_signature(&sig).unwrap();
|
acc.wait_on_signature(&sig).unwrap();
|
||||||
assert_eq!(acc.get_balance(&bob_pubkey).unwrap().unwrap(), 1_500);
|
assert_eq!(acc.get_balance(&bob_pubkey).unwrap().unwrap(), 500);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,19 +1,28 @@
|
|||||||
extern crate serde_json;
|
extern crate serde_json;
|
||||||
extern crate silk;
|
extern crate silk;
|
||||||
|
|
||||||
use silk::genesis::{Creator, Genesis};
|
use silk::genesis::Genesis;
|
||||||
use silk::signature::{generate_keypair, get_pubkey};
|
use silk::event::Event;
|
||||||
|
use silk::transaction::Transaction;
|
||||||
|
use silk::log::create_entries;
|
||||||
|
use silk::signature::{generate_keypair, get_pubkey, KeyPair, PublicKey};
|
||||||
|
use silk::hash::Hash;
|
||||||
|
|
||||||
|
fn transfer(from: &KeyPair, (to, tokens): (PublicKey, i64), last_id: Hash) -> Event {
|
||||||
|
Event::Transaction(Transaction::new(&from, to, tokens, last_id))
|
||||||
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let alice = Creator {
|
let alice = (get_pubkey(&generate_keypair()), 200);
|
||||||
tokens: 200,
|
let bob = (get_pubkey(&generate_keypair()), 100);
|
||||||
pubkey: get_pubkey(&generate_keypair()),
|
|
||||||
};
|
let gen = Genesis::new(500);
|
||||||
let bob = Creator {
|
let from = gen.get_keypair();
|
||||||
tokens: 100,
|
let seed = gen.get_seed();
|
||||||
pubkey: get_pubkey(&generate_keypair()),
|
let mut events = gen.create_events();
|
||||||
};
|
events.push(transfer(&from, alice, seed));
|
||||||
let creators = vec![alice, bob];
|
events.push(transfer(&from, bob, seed));
|
||||||
let gen = Genesis::new(500, creators);
|
|
||||||
|
create_entries(&seed, events);
|
||||||
println!("{}", serde_json::to_string(&gen).unwrap());
|
println!("{}", serde_json::to_string(&gen).unwrap());
|
||||||
}
|
}
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
use event::Event;
|
use event::Event;
|
||||||
use transaction::Transaction;
|
use transaction::Transaction;
|
||||||
use signature::{generate_keypair, get_pubkey, PublicKey};
|
use signature::{get_pubkey, PublicKey};
|
||||||
use entry::Entry;
|
use entry::Entry;
|
||||||
use log::create_entries;
|
use log::create_entries;
|
||||||
use hash::{hash, Hash};
|
use hash::{hash, Hash};
|
||||||
@ -10,37 +10,17 @@ use ring::rand::SystemRandom;
|
|||||||
use ring::signature::Ed25519KeyPair;
|
use ring::signature::Ed25519KeyPair;
|
||||||
use untrusted::Input;
|
use untrusted::Input;
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, Debug)]
|
|
||||||
pub struct Creator {
|
|
||||||
pub pubkey: PublicKey,
|
|
||||||
pub tokens: i64,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Creator {
|
|
||||||
pub fn new(tokens: i64) -> Self {
|
|
||||||
Creator {
|
|
||||||
pubkey: get_pubkey(&generate_keypair()),
|
|
||||||
tokens,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, Debug)]
|
#[derive(Serialize, Deserialize, Debug)]
|
||||||
pub struct Genesis {
|
pub struct Genesis {
|
||||||
pub pkcs8: Vec<u8>,
|
pub pkcs8: Vec<u8>,
|
||||||
pub tokens: i64,
|
pub tokens: i64,
|
||||||
pub creators: Vec<Creator>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Genesis {
|
impl Genesis {
|
||||||
pub fn new(tokens: i64, creators: Vec<Creator>) -> Self {
|
pub fn new(tokens: i64) -> Self {
|
||||||
let rnd = SystemRandom::new();
|
let rnd = SystemRandom::new();
|
||||||
let pkcs8 = Ed25519KeyPair::generate_pkcs8(&rnd).unwrap().to_vec();
|
let pkcs8 = Ed25519KeyPair::generate_pkcs8(&rnd).unwrap().to_vec();
|
||||||
Genesis {
|
Genesis { pkcs8, tokens }
|
||||||
pkcs8,
|
|
||||||
tokens,
|
|
||||||
creators,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_seed(&self) -> Hash {
|
pub fn get_seed(&self) -> Hash {
|
||||||
@ -55,23 +35,14 @@ impl Genesis {
|
|||||||
get_pubkey(&self.get_keypair())
|
get_pubkey(&self.get_keypair())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn create_transaction(&self, asset: i64, to: &PublicKey) -> Event {
|
|
||||||
let tr = Transaction::new(&self.get_keypair(), *to, asset, self.get_seed());
|
|
||||||
Event::Transaction(tr)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn create_events(&self) -> Vec<Event> {
|
pub fn create_events(&self) -> Vec<Event> {
|
||||||
let pubkey = self.get_pubkey();
|
let tr = Transaction::new(
|
||||||
let event0 = Event::Tick;
|
&self.get_keypair(),
|
||||||
let event1 = self.create_transaction(self.tokens, &pubkey);
|
self.get_pubkey(),
|
||||||
let mut events = vec![event0, event1];
|
self.tokens,
|
||||||
|
self.get_seed(),
|
||||||
for x in &self.creators {
|
);
|
||||||
let tx = self.create_transaction(x.tokens, &x.pubkey);
|
vec![Event::Tick, Event::Transaction(tr)]
|
||||||
events.push(tx);
|
|
||||||
}
|
|
||||||
|
|
||||||
events
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn create_entries(&self) -> Vec<Entry> {
|
pub fn create_entries(&self) -> Vec<Entry> {
|
||||||
@ -86,35 +57,19 @@ mod tests {
|
|||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_create_events() {
|
fn test_create_events() {
|
||||||
let mut events = Genesis::new(100, vec![]).create_events().into_iter();
|
let mut events = Genesis::new(100).create_events().into_iter();
|
||||||
assert_eq!(events.next().unwrap(), Event::Tick);
|
assert_eq!(events.next().unwrap(), Event::Tick);
|
||||||
if let Event::Transaction(Transaction { from, to, .. }) = events.next().unwrap() {
|
if let Event::Transaction(tr) = events.next().unwrap() {
|
||||||
assert_eq!(from, to);
|
assert_eq!(tr.from, tr.to);
|
||||||
} else {
|
} else {
|
||||||
assert!(false);
|
assert!(false);
|
||||||
}
|
}
|
||||||
assert_eq!(events.next(), None);
|
assert_eq!(events.next(), None);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_create_creator() {
|
|
||||||
assert_eq!(
|
|
||||||
Genesis::new(100, vec![Creator::new(42)])
|
|
||||||
.create_events()
|
|
||||||
.len(),
|
|
||||||
3
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_verify_entries() {
|
fn test_verify_entries() {
|
||||||
let entries = Genesis::new(100, vec![]).create_entries();
|
let entries = Genesis::new(100).create_entries();
|
||||||
assert!(verify_slice(&entries, &entries[0].id));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_verify_entries_with_transactions() {
|
|
||||||
let entries = Genesis::new(100, vec![Creator::new(42)]).create_entries();
|
|
||||||
assert!(verify_slice(&entries, &entries[0].id));
|
assert!(verify_slice(&entries, &entries[0].id));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,18 +6,19 @@ use ring::signature::Ed25519KeyPair;
|
|||||||
use ring::{rand, signature};
|
use ring::{rand, signature};
|
||||||
use untrusted;
|
use untrusted;
|
||||||
|
|
||||||
|
pub type KeyPair = Ed25519KeyPair;
|
||||||
pub type PublicKey = GenericArray<u8, U32>;
|
pub type PublicKey = GenericArray<u8, U32>;
|
||||||
pub type Signature = GenericArray<u8, U64>;
|
pub type Signature = GenericArray<u8, U64>;
|
||||||
|
|
||||||
/// Return a new ED25519 keypair
|
/// Return a new ED25519 keypair
|
||||||
pub fn generate_keypair() -> Ed25519KeyPair {
|
pub fn generate_keypair() -> KeyPair {
|
||||||
let rng = rand::SystemRandom::new();
|
let rng = rand::SystemRandom::new();
|
||||||
let pkcs8_bytes = signature::Ed25519KeyPair::generate_pkcs8(&rng).unwrap();
|
let pkcs8_bytes = signature::Ed25519KeyPair::generate_pkcs8(&rng).unwrap();
|
||||||
signature::Ed25519KeyPair::from_pkcs8(untrusted::Input::from(&pkcs8_bytes)).unwrap()
|
signature::Ed25519KeyPair::from_pkcs8(untrusted::Input::from(&pkcs8_bytes)).unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Return the public key for the given keypair
|
/// Return the public key for the given keypair
|
||||||
pub fn get_pubkey(keypair: &Ed25519KeyPair) -> PublicKey {
|
pub fn get_pubkey(keypair: &KeyPair) -> PublicKey {
|
||||||
GenericArray::clone_from_slice(keypair.public_key_bytes())
|
GenericArray::clone_from_slice(keypair.public_key_bytes())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
//! The `transaction` crate provides functionality for creating log transactions.
|
//! The `transaction` crate provides functionality for creating log transactions.
|
||||||
|
|
||||||
use signature::{get_pubkey, verify_signature, PublicKey, Signature};
|
use signature::{get_pubkey, verify_signature, KeyPair, PublicKey, Signature};
|
||||||
use ring::signature::Ed25519KeyPair;
|
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use bincode::serialize;
|
use bincode::serialize;
|
||||||
use hash::Hash;
|
use hash::Hash;
|
||||||
@ -16,7 +15,7 @@ pub struct Transaction<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl<T: Serialize> Transaction<T> {
|
impl<T: Serialize> Transaction<T> {
|
||||||
pub fn new(from_keypair: &Ed25519KeyPair, to: PublicKey, asset: T, last_id: Hash) -> Self {
|
pub fn new(from_keypair: &KeyPair, to: PublicKey, asset: T, last_id: Hash) -> Self {
|
||||||
let mut tr = Transaction {
|
let mut tr = Transaction {
|
||||||
from: get_pubkey(&from_keypair),
|
from: get_pubkey(&from_keypair),
|
||||||
to,
|
to,
|
||||||
@ -32,7 +31,7 @@ impl<T: Serialize> Transaction<T> {
|
|||||||
serialize(&(&self.from, &self.to, &self.asset, &self.last_id)).unwrap()
|
serialize(&(&self.from, &self.to, &self.asset, &self.last_id)).unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn sign(&mut self, keypair: &Ed25519KeyPair) {
|
pub fn sign(&mut self, keypair: &KeyPair) {
|
||||||
let sign_data = self.get_sign_data();
|
let sign_data = self.get_sign_data();
|
||||||
self.sig = Signature::clone_from_slice(keypair.sign(&sign_data).as_ref());
|
self.sig = Signature::clone_from_slice(keypair.sign(&sign_data).as_ref());
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user