Transaction data associated with polls.
This commit is contained in:
parent
9db4720162
commit
9741d48496
@ -16,6 +16,8 @@
|
|||||||
|
|
||||||
//! Indexes all rpc poll requests.
|
//! Indexes all rpc poll requests.
|
||||||
|
|
||||||
|
use util::hash::H256;
|
||||||
|
use std::collections::HashMap;
|
||||||
use transient_hashmap::{TransientHashMap, Timer, StandardTimer};
|
use transient_hashmap::{TransientHashMap, Timer, StandardTimer};
|
||||||
|
|
||||||
/// Lifetime of poll (in seconds).
|
/// Lifetime of poll (in seconds).
|
||||||
@ -43,7 +45,8 @@ impl<F> Clone for PollInfo<F> where F: Clone {
|
|||||||
/// Lazily garbage collects unused polls info.
|
/// Lazily garbage collects unused polls info.
|
||||||
pub struct PollManager<F, T = StandardTimer> where T: Timer {
|
pub struct PollManager<F, T = StandardTimer> where T: Timer {
|
||||||
polls: TransientHashMap<PollId, PollInfo<F>, T>,
|
polls: TransientHashMap<PollId, PollInfo<F>, T>,
|
||||||
next_available_id: PollId
|
transactions_data: HashMap<PollId, Vec<H256>>,
|
||||||
|
next_available_id: PollId,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<F> PollManager<F, StandardTimer> {
|
impl<F> PollManager<F, StandardTimer> {
|
||||||
@ -57,15 +60,25 @@ impl<F, T> PollManager<F, T> where T: Timer {
|
|||||||
pub fn new_with_timer(timer: T) -> Self {
|
pub fn new_with_timer(timer: T) -> Self {
|
||||||
PollManager {
|
PollManager {
|
||||||
polls: TransientHashMap::new_with_timer(POLL_LIFETIME, timer),
|
polls: TransientHashMap::new_with_timer(POLL_LIFETIME, timer),
|
||||||
|
transactions_data: HashMap::new(),
|
||||||
next_available_id: 0,
|
next_available_id: 0,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn prune(&mut self) {
|
||||||
|
self.polls.prune();
|
||||||
|
// self.polls.prune()
|
||||||
|
// .into_iter()
|
||||||
|
// .map(|key| {
|
||||||
|
// self.transactions_data.remove(key);
|
||||||
|
// });
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns id which can be used for new poll.
|
/// Returns id which can be used for new poll.
|
||||||
///
|
///
|
||||||
/// Stores information when last poll happend.
|
/// Stores information when last poll happend.
|
||||||
pub fn create_poll(&mut self, filter: F, block: BlockNumber) -> PollId {
|
pub fn create_poll(&mut self, filter: F, block: BlockNumber) -> PollId {
|
||||||
self.polls.prune();
|
self.prune();
|
||||||
let id = self.next_available_id;
|
let id = self.next_available_id;
|
||||||
self.next_available_id += 1;
|
self.next_available_id += 1;
|
||||||
self.polls.insert(id, PollInfo {
|
self.polls.insert(id, PollInfo {
|
||||||
@ -77,7 +90,7 @@ impl<F, T> PollManager<F, T> where T: Timer {
|
|||||||
|
|
||||||
/// Updates information when last poll happend.
|
/// Updates information when last poll happend.
|
||||||
pub fn update_poll(&mut self, id: &PollId, block: BlockNumber) {
|
pub fn update_poll(&mut self, id: &PollId, block: BlockNumber) {
|
||||||
self.polls.prune();
|
self.prune();
|
||||||
if let Some(info) = self.polls.get_mut(id) {
|
if let Some(info) = self.polls.get_mut(id) {
|
||||||
info.block_number = block;
|
info.block_number = block;
|
||||||
}
|
}
|
||||||
@ -85,13 +98,27 @@ impl<F, T> PollManager<F, T> where T: Timer {
|
|||||||
|
|
||||||
/// Returns number of block when last poll happend.
|
/// Returns number of block when last poll happend.
|
||||||
pub fn get_poll_info(&mut self, id: &PollId) -> Option<&PollInfo<F>> {
|
pub fn get_poll_info(&mut self, id: &PollId) -> Option<&PollInfo<F>> {
|
||||||
self.polls.prune();
|
self.prune();
|
||||||
self.polls.get(id)
|
self.polls.get(id)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn set_poll_transactions(&mut self, id: &PollId, transactions: Vec<H256>) {
|
||||||
|
self.prune();
|
||||||
|
if self.polls.get(id).is_some() {
|
||||||
|
self.transactions_data.insert(*id, transactions);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns last transactions hashes for given poll.
|
||||||
|
pub fn poll_transactions(&mut self, id: &PollId) -> Option<&Vec<H256>> {
|
||||||
|
self.prune();
|
||||||
|
self.transactions_data.get(id)
|
||||||
|
}
|
||||||
|
|
||||||
/// Removes poll info.
|
/// Removes poll info.
|
||||||
pub fn remove_poll(&mut self, id: &PollId) {
|
pub fn remove_poll(&mut self, id: &PollId) {
|
||||||
self.polls.remove(id);
|
self.polls.remove(id);
|
||||||
|
self.transactions_data.remove(id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -100,6 +127,7 @@ mod tests {
|
|||||||
use std::cell::RefCell;
|
use std::cell::RefCell;
|
||||||
use transient_hashmap::Timer;
|
use transient_hashmap::Timer;
|
||||||
use v1::helpers::PollManager;
|
use v1::helpers::PollManager;
|
||||||
|
use util::hash::H256;
|
||||||
|
|
||||||
struct TestTimer<'a> {
|
struct TestTimer<'a> {
|
||||||
time: &'a RefCell<i64>,
|
time: &'a RefCell<i64>,
|
||||||
@ -141,4 +169,72 @@ mod tests {
|
|||||||
indexer.remove_poll(&1);
|
indexer.remove_poll(&1);
|
||||||
assert!(indexer.get_poll_info(&1).is_none());
|
assert!(indexer.get_poll_info(&1).is_none());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn should_return_poll_transactions_hashes() {
|
||||||
|
// given
|
||||||
|
let mut indexer = PollManager::new();
|
||||||
|
let poll_id = indexer.create_poll(false, 20);
|
||||||
|
assert!(indexer.poll_transactions(&poll_id).is_none());
|
||||||
|
let transactions = vec![H256::from(1), H256::from(2)];
|
||||||
|
|
||||||
|
// when
|
||||||
|
indexer.set_poll_transactions(&poll_id, transactions.clone());
|
||||||
|
|
||||||
|
// then
|
||||||
|
let txs = indexer.poll_transactions(&poll_id);
|
||||||
|
assert_eq!(txs.unwrap(), &transactions);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn should_remove_transaction_data_when_poll_timed_out() {
|
||||||
|
// given
|
||||||
|
let time = RefCell::new(0);
|
||||||
|
let timer = TestTimer {
|
||||||
|
time: &time,
|
||||||
|
};
|
||||||
|
let mut indexer = PollManager::new_with_timer(timer);
|
||||||
|
let poll_id = indexer.create_poll(false, 20);
|
||||||
|
let transactions = vec![H256::from(1), H256::from(2)];
|
||||||
|
indexer.set_poll_transactions(&poll_id, transactions.clone());
|
||||||
|
assert!(indexer.poll_transactions(&poll_id).is_some());
|
||||||
|
|
||||||
|
// when
|
||||||
|
*time.borrow_mut() = 75;
|
||||||
|
indexer.prune();
|
||||||
|
|
||||||
|
// then
|
||||||
|
assert!(indexer.poll_transactions(&poll_id).is_none());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn should_remove_transaction_data_when_poll_is_removed() {
|
||||||
|
// given
|
||||||
|
let mut indexer = PollManager::new();
|
||||||
|
let poll_id = indexer.create_poll(false, 20);
|
||||||
|
let transactions = vec![H256::from(1), H256::from(2)];
|
||||||
|
|
||||||
|
// when
|
||||||
|
indexer.set_poll_transactions(&poll_id, transactions.clone());
|
||||||
|
assert!(indexer.poll_transactions(&poll_id).is_some());
|
||||||
|
indexer.remove_poll(&poll_id);
|
||||||
|
|
||||||
|
// then
|
||||||
|
assert!(indexer.poll_transactions(&poll_id).is_none());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn should_ignore_transactions_for_invalid_poll_id() {
|
||||||
|
// given
|
||||||
|
let mut indexer = PollManager::<()>::new();
|
||||||
|
let transactions = vec![H256::from(1), H256::from(2)];
|
||||||
|
|
||||||
|
// when
|
||||||
|
indexer.set_poll_transactions(&5, transactions.clone());
|
||||||
|
|
||||||
|
// then
|
||||||
|
assert!(indexer.poll_transactions(&5).is_none());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user