2017-05-19 18:22:08 +03:00
|
|
|
// Copyright 2017 The Grin Developers
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
//! Base types for the transaction pool's Directed Acyclic Graphs
|
|
|
|
|
|
|
|
use std::vec::Vec;
|
2017-10-18 23:42:51 +03:00
|
|
|
use std::collections::{HashMap, HashSet};
|
2017-05-19 18:22:08 +03:00
|
|
|
|
2017-11-01 02:20:55 +03:00
|
|
|
use util::secp::pedersen::Commitment;
|
2017-05-19 18:22:08 +03:00
|
|
|
|
|
|
|
use time;
|
|
|
|
|
|
|
|
use std::fmt;
|
|
|
|
|
|
|
|
use core::core;
|
2017-09-28 02:46:32 +03:00
|
|
|
use core::core::hash::Hashed;
|
2017-05-19 18:22:08 +03:00
|
|
|
|
|
|
|
/// An entry in the transaction pool.
|
|
|
|
/// These are the vertices of both of the graph structures
|
2017-10-18 23:42:51 +03:00
|
|
|
#[derive(Debug, PartialEq, Clone)]
|
2017-05-19 18:22:08 +03:00
|
|
|
pub struct PoolEntry {
|
2017-09-29 21:44:25 +03:00
|
|
|
// Core data
|
|
|
|
/// Unique identifier of this pool entry and the corresponding transaction
|
|
|
|
pub transaction_hash: core::hash::Hash,
|
|
|
|
|
|
|
|
// Metadata
|
|
|
|
/// Size estimate
|
|
|
|
pub size_estimate: u64,
|
|
|
|
/// Receive timestamp
|
|
|
|
pub receive_ts: time::Tm,
|
2017-05-19 18:22:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl PoolEntry {
|
2017-09-29 21:44:25 +03:00
|
|
|
/// Create new transaction pool entry
|
|
|
|
pub fn new(tx: &core::transaction::Transaction) -> PoolEntry {
|
|
|
|
PoolEntry {
|
|
|
|
transaction_hash: transaction_identifier(tx),
|
|
|
|
size_estimate: estimate_transaction_size(tx),
|
|
|
|
receive_ts: time::now_utc(),
|
|
|
|
}
|
|
|
|
}
|
2017-05-19 18:22:08 +03:00
|
|
|
}
|
|
|
|
|
2017-08-10 03:54:10 +03:00
|
|
|
/// TODO guessing this needs implementing
|
|
|
|
fn estimate_transaction_size(_tx: &core::transaction::Transaction) -> u64 {
|
2017-09-29 21:44:25 +03:00
|
|
|
0
|
2017-05-19 18:22:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// An edge connecting graph vertices.
|
|
|
|
/// For various use cases, one of either the source or destination may be
|
|
|
|
/// unpopulated
|
|
|
|
pub struct Edge {
|
2017-09-29 21:44:25 +03:00
|
|
|
// Source and Destination are the vertex id's, the transaction (kernel)
|
|
|
|
// hash.
|
|
|
|
source: Option<core::hash::Hash>,
|
|
|
|
destination: Option<core::hash::Hash>,
|
|
|
|
|
|
|
|
// Output is the output hash which this input/output pairing corresponds
|
|
|
|
// to.
|
|
|
|
output: Commitment,
|
2017-05-19 18:22:08 +03:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
impl Edge {
|
|
|
|
/// Create new edge
|
|
|
|
pub fn new(
|
|
|
|
source: Option<core::hash::Hash>,
|
|
|
|
destination: Option<core::hash::Hash>,
|
|
|
|
output: Commitment,
|
|
|
|
) -> Edge {
|
|
|
|
Edge {
|
|
|
|
source: source,
|
|
|
|
destination: destination,
|
|
|
|
output: output,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create new edge with a source
|
|
|
|
pub fn with_source(&self, src: Option<core::hash::Hash>) -> Edge {
|
|
|
|
Edge {
|
|
|
|
source: src,
|
|
|
|
destination: self.destination,
|
|
|
|
output: self.output,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create new edge with destination
|
|
|
|
pub fn with_destination(&self, dst: Option<core::hash::Hash>) -> Edge {
|
|
|
|
Edge {
|
|
|
|
source: self.source,
|
|
|
|
destination: dst,
|
|
|
|
output: self.output,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The output commitment of the edge
|
|
|
|
pub fn output_commitment(&self) -> Commitment {
|
|
|
|
self.output
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The destination hash of the edge
|
|
|
|
pub fn destination_hash(&self) -> Option<core::hash::Hash> {
|
|
|
|
self.destination
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The source hash of the edge
|
|
|
|
pub fn source_hash(&self) -> Option<core::hash::Hash> {
|
|
|
|
self.source
|
|
|
|
}
|
2017-05-19 18:22:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Debug for Edge {
|
2017-09-29 21:44:25 +03:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
write!(
|
|
|
|
f,
|
|
|
|
"Edge {{source: {:?}, destination: {:?}, commitment: {:?}}}",
|
|
|
|
self.source,
|
|
|
|
self.destination,
|
|
|
|
self.output
|
|
|
|
)
|
|
|
|
}
|
2017-05-19 18:22:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// The generic graph container. Both graphs, the pool and orphans, embed this
|
|
|
|
/// structure and add additional capability on top of it.
|
|
|
|
pub struct DirectedGraph {
|
2017-09-29 21:44:25 +03:00
|
|
|
edges: HashMap<Commitment, Edge>,
|
|
|
|
vertices: Vec<PoolEntry>,
|
2017-05-19 18:22:08 +03:00
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
// A small optimization: keeping roots (vertices with in-degree 0) in a
|
|
|
|
// separate list makes topological sort a bit faster. (This is true for
|
|
|
|
// Kahn's, not sure about other implementations)
|
|
|
|
roots: Vec<PoolEntry>,
|
2017-05-19 18:22:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl DirectedGraph {
|
2017-09-29 21:44:25 +03:00
|
|
|
/// Create an empty directed graph
|
|
|
|
pub fn empty() -> DirectedGraph {
|
|
|
|
DirectedGraph {
|
|
|
|
edges: HashMap::new(),
|
|
|
|
vertices: Vec::new(),
|
|
|
|
roots: Vec::new(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get an edge by its commitment
|
|
|
|
pub fn get_edge_by_commitment(&self, output_commitment: &Commitment) -> Option<&Edge> {
|
|
|
|
self.edges.get(output_commitment)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Remove an edge by its commitment
|
|
|
|
pub fn remove_edge_by_commitment(&mut self, output_commitment: &Commitment) -> Option<Edge> {
|
|
|
|
self.edges.remove(output_commitment)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Remove a vertex by its hash
|
|
|
|
pub fn remove_vertex(&mut self, tx_hash: core::hash::Hash) -> Option<PoolEntry> {
|
2017-11-01 02:32:33 +03:00
|
|
|
match self.roots
|
|
|
|
.iter()
|
|
|
|
.position(|x| x.transaction_hash == tx_hash)
|
|
|
|
{
|
2017-09-29 21:44:25 +03:00
|
|
|
Some(i) => Some(self.roots.swap_remove(i)),
|
2017-11-01 02:32:33 +03:00
|
|
|
None => match self.vertices
|
|
|
|
.iter()
|
|
|
|
.position(|x| x.transaction_hash == tx_hash)
|
|
|
|
{
|
|
|
|
Some(i) => Some(self.vertices.swap_remove(i)),
|
|
|
|
None => None,
|
|
|
|
},
|
2017-09-29 21:44:25 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-18 23:42:51 +03:00
|
|
|
/// Promote any non-root vertices to roots based on current edges.
|
2017-11-01 02:32:33 +03:00
|
|
|
/// For a given tx, if there are no edges with that tx as destination then
|
|
|
|
/// it is a root.
|
2017-10-18 23:42:51 +03:00
|
|
|
pub fn update_roots(&mut self) {
|
|
|
|
let mut new_vertices: Vec<PoolEntry> = vec![];
|
|
|
|
|
|
|
|
// first find the set of all destinations from the edges in the graph
|
2017-11-01 02:32:33 +03:00
|
|
|
// a root is a vertex that is not a destination of any edge
|
2017-10-18 23:42:51 +03:00
|
|
|
let destinations = self.edges
|
|
|
|
.values()
|
|
|
|
.filter_map(|edge| edge.destination)
|
|
|
|
.collect::<HashSet<_>>();
|
|
|
|
|
|
|
|
// now iterate over the current non-root vertices
|
2017-11-01 02:32:33 +03:00
|
|
|
// and check if it is now a root based on the set of edge destinations
|
2017-10-18 23:42:51 +03:00
|
|
|
for x in &self.vertices {
|
|
|
|
if destinations.contains(&x.transaction_hash) {
|
|
|
|
new_vertices.push(x.clone());
|
|
|
|
} else {
|
|
|
|
self.roots.push(x.clone());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// now update our vertices to reflect the updated list
|
|
|
|
self.vertices = new_vertices;
|
|
|
|
}
|
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
/// Adds a vertex and a set of incoming edges to the graph.
|
|
|
|
///
|
|
|
|
/// The PoolEntry at vertex is added to the graph; depending on the
|
|
|
|
/// number of incoming edges, the vertex is either added to the vertices
|
|
|
|
/// or to the roots.
|
|
|
|
///
|
|
|
|
/// Outgoing edges must not be included in edges; this method is designed
|
|
|
|
/// for adding vertices one at a time and only accepts incoming edges as
|
|
|
|
/// internal edges.
|
|
|
|
pub fn add_entry(&mut self, vertex: PoolEntry, mut edges: Vec<Edge>) {
|
|
|
|
if edges.len() == 0 {
|
|
|
|
self.roots.push(vertex);
|
|
|
|
} else {
|
|
|
|
self.vertices.push(vertex);
|
|
|
|
for edge in edges.drain(..) {
|
|
|
|
self.edges.insert(edge.output_commitment(), edge);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// add_vertex_only adds a vertex, meant to be complemented by add_edge_only
|
|
|
|
/// in cases where delivering a vector of edges is not feasible or efficient
|
|
|
|
pub fn add_vertex_only(&mut self, vertex: PoolEntry, is_root: bool) {
|
|
|
|
if is_root {
|
|
|
|
self.roots.push(vertex);
|
|
|
|
} else {
|
|
|
|
self.vertices.push(vertex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// add_edge_only adds an edge
|
|
|
|
pub fn add_edge_only(&mut self, edge: Edge) {
|
|
|
|
self.edges.insert(edge.output_commitment(), edge);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Number of vertices (root + internal)
|
|
|
|
pub fn len_vertices(&self) -> usize {
|
|
|
|
self.vertices.len() + self.roots.len()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Number of root vertices only
|
|
|
|
pub fn len_roots(&self) -> usize {
|
|
|
|
self.roots.len()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Number of edges
|
|
|
|
pub fn len_edges(&self) -> usize {
|
|
|
|
self.edges.len()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the current list of roots
|
|
|
|
pub fn get_roots(&self) -> Vec<core::hash::Hash> {
|
|
|
|
self.roots.iter().map(|x| x.transaction_hash).collect()
|
|
|
|
}
|
2017-10-18 23:47:37 +03:00
|
|
|
|
|
|
|
/// Get list of all vertices in this graph including the roots
|
|
|
|
pub fn get_vertices(&self) -> Vec<core::hash::Hash> {
|
|
|
|
let mut hashes = self.roots
|
|
|
|
.iter()
|
|
|
|
.map(|x| x.transaction_hash)
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
let non_root_hashes = self.vertices
|
|
|
|
.iter()
|
|
|
|
.map(|x| x.transaction_hash)
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
hashes.extend(&non_root_hashes);
|
2017-11-01 02:32:33 +03:00
|
|
|
return hashes;
|
2017-10-18 23:47:37 +03:00
|
|
|
}
|
2017-05-19 18:22:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Using transaction merkle_inputs_outputs to calculate a deterministic hash;
|
|
|
|
/// this hashing mechanism has some ambiguity issues especially around range
|
|
|
|
/// proofs and any extra data the kernel may cover, but it is used initially
|
|
|
|
/// for testing purposes.
|
|
|
|
pub fn transaction_identifier(tx: &core::transaction::Transaction) -> core::hash::Hash {
|
2017-09-29 21:44:25 +03:00
|
|
|
// core::transaction::merkle_inputs_outputs(&tx.inputs, &tx.outputs)
|
|
|
|
tx.hash()
|
2017-05-19 18:22:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2017-09-29 21:44:25 +03:00
|
|
|
use super::*;
|
2017-11-01 02:20:55 +03:00
|
|
|
use util::secp;
|
2017-10-06 00:40:46 +03:00
|
|
|
use keychain::Keychain;
|
2017-10-06 19:46:18 +03:00
|
|
|
use rand;
|
2017-10-17 00:23:10 +03:00
|
|
|
use core::core::SwitchCommitHash;
|
2017-05-19 18:22:08 +03:00
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
#[test]
|
|
|
|
fn test_add_entry() {
|
2017-10-06 00:40:46 +03:00
|
|
|
let keychain = Keychain::from_random_seed().unwrap();
|
2017-10-13 07:45:07 +03:00
|
|
|
let key_id1 = keychain.derive_key_id(1).unwrap();
|
|
|
|
let key_id2 = keychain.derive_key_id(2).unwrap();
|
|
|
|
let key_id3 = keychain.derive_key_id(3).unwrap();
|
2017-05-19 18:22:08 +03:00
|
|
|
|
2017-10-13 07:45:07 +03:00
|
|
|
let output_commit = keychain.commit(70, &key_id1).unwrap();
|
2017-10-17 00:23:10 +03:00
|
|
|
let switch_commit = keychain.switch_commit(&key_id1).unwrap();
|
|
|
|
let switch_commit_hash = SwitchCommitHash::from_switch_commit(switch_commit);
|
2017-09-29 21:44:25 +03:00
|
|
|
let inputs = vec![
|
2017-10-13 07:45:07 +03:00
|
|
|
core::transaction::Input(keychain.commit(50, &key_id2).unwrap()),
|
|
|
|
core::transaction::Input(keychain.commit(25, &key_id3).unwrap()),
|
2017-09-29 21:44:25 +03:00
|
|
|
];
|
2017-10-06 00:40:46 +03:00
|
|
|
let msg = secp::pedersen::ProofMessage::empty();
|
2017-09-29 21:44:25 +03:00
|
|
|
let outputs = vec![
|
|
|
|
core::transaction::Output {
|
|
|
|
features: core::transaction::DEFAULT_OUTPUT,
|
|
|
|
commit: output_commit,
|
2017-10-17 00:23:10 +03:00
|
|
|
switch_commit_hash: switch_commit_hash,
|
2017-11-01 02:32:33 +03:00
|
|
|
proof: keychain
|
|
|
|
.range_proof(100, &key_id1, output_commit, msg)
|
|
|
|
.unwrap(),
|
2017-09-29 21:44:25 +03:00
|
|
|
},
|
|
|
|
];
|
2017-10-11 21:12:01 +03:00
|
|
|
let test_transaction = core::transaction::Transaction::new(inputs, outputs, 5, 0);
|
2017-05-19 18:22:08 +03:00
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
let test_pool_entry = PoolEntry::new(&test_transaction);
|
2017-05-19 18:22:08 +03:00
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
let incoming_edge_1 = Edge::new(
|
|
|
|
Some(random_hash()),
|
|
|
|
Some(core::hash::ZERO_HASH),
|
|
|
|
output_commit,
|
|
|
|
);
|
2017-05-19 18:22:08 +03:00
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
let mut test_graph = DirectedGraph::empty();
|
2017-05-19 18:22:08 +03:00
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
test_graph.add_entry(test_pool_entry, vec![incoming_edge_1]);
|
2017-05-19 18:22:08 +03:00
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
assert_eq!(test_graph.vertices.len(), 1);
|
|
|
|
assert_eq!(test_graph.roots.len(), 0);
|
|
|
|
assert_eq!(test_graph.edges.len(), 1);
|
|
|
|
}
|
2017-05-19 18:22:08 +03:00
|
|
|
|
2017-10-03 03:02:31 +03:00
|
|
|
/// For testing/debugging: a random tx hash
|
|
|
|
fn random_hash() -> core::hash::Hash {
|
|
|
|
let hash_bytes: [u8; 32] = rand::random();
|
|
|
|
core::hash::Hash(hash_bytes)
|
|
|
|
}
|
2017-05-19 18:22:08 +03:00
|
|
|
}
|