Newer
Older
mod default_combine_data;
mod mock;
mod operator_provider;
mod tests;
mod timestamped_value;
pub use default_combine_data::DefaultCombineData;
use frame_support::{decl_error, decl_event, decl_module, decl_storage, ensure, traits::Time, Parameter};
use rstd::{prelude::*, vec};
use sp_runtime::{traits::Member, DispatchResult};
// FIXME: `pallet/frame-` prefix should be used for all pallet modules, but currently `frame_system`
// would cause compiling error in `decl_module!` and `construct_runtime!`
// #3295 https://github.com/paritytech/substrate/issues/3295
pub use orml_traits::{CombineData, DataProvider, OnNewData};
pub use timestamped_value::TimestampedValue;
type MomentOf<T> = <<T as Trait>::Time as Time>::Moment;
pub type TimestampedValueOf<T> = TimestampedValue<<T as Trait>::OracleValue, MomentOf<T>>;
pub trait Trait: frame_system::Trait {
type Event: From<Event<Self>> + Into<<Self as frame_system::Trait>::Event>;
type OnNewData: OnNewData<Self::OracleKey, Self::OracleValue>;
type OperatorProvider: OperatorProvider<Self::AccountId>;
type CombineData: CombineData<Self::OracleKey, TimestampedValueOf<Self>>;
type OracleKey: Parameter + Member;
type OracleValue: Parameter + Member + Ord;
}
decl_storage! {
trait Store for Module<T: Trait> as Oracle {
pub RawValues get(raw_values): double_map hasher(blake2_256) T::OracleKey, hasher(blake2_256) T::AccountId => Option<TimestampedValueOf<T>>;
pub HasUpdate get(has_update): map hasher(blake2_256) T::OracleKey => bool;
pub Values get(values): map hasher(blake2_256) T::OracleKey => Option<TimestampedValueOf<T>>;
NoPermission,
}
}
decl_module! {
pub struct Module<T: Trait> for enum Call where origin: T::Origin {
pub fn feed_value(origin, key: T::OracleKey, value: T::OracleValue) {
Self::_feed_values(who, vec![(key, value)])?;
pub fn feed_values(origin, values: Vec<(T::OracleKey, T::OracleValue)>) {
<T as Trait>::OracleKey,
<T as Trait>::OracleValue,
NewFeedData(AccountId, Vec<(OracleKey, OracleValue)>),
pub fn read_raw_values(key: &T::OracleKey) -> Vec<TimestampedValueOf<T>> {
T::OperatorProvider::operators()
.iter()
.filter_map(|x| <RawValues<T>>::get(key, x))
.collect()
}
/// Returns fresh combined value if has update, or latest combined value.
///
/// Note this will update values storage if has update.
pub fn get(key: &T::OracleKey) -> Option<TimestampedValueOf<T>> {
return Some(timestamped);
}
<Values<T>>::get(key)
/// Returns fresh combined value if has update, or latest combined value.
///
/// This is a no-op function which would not change storage.
pub fn get_no_op(key: &T::OracleKey) -> Option<TimestampedValueOf<T>> {
if Self::has_update(key) {
Self::combined(key)
} else {
Self::values(key)
}
}
fn combined(key: &T::OracleKey) -> Option<TimestampedValueOf<T>> {
let values = Self::read_raw_values(key);
T::CombineData::combine_data(key, values, Self::values(key))
}
impl<T: Trait> DataProvider<T::OracleKey, T::OracleValue> for Module<T> {
fn get(key: &T::OracleKey) -> Option<T::OracleValue> {
Self::get(key).map(|timestamped_value| timestamped_value.value)
}
}
fn _feed_values(who: T::AccountId, values: Vec<(T::OracleKey, T::OracleValue)>) -> DispatchResult {
ensure!(T::OperatorProvider::can_feed_data(&who), Error::<T>::NoPermission);
let now = T::Time::now();
for (key, value) in &values {
let timestamped = TimestampedValue {
value: value.clone(),
timestamp: now,
};
<RawValues<T>>::insert(&key, &who, timestamped);
<HasUpdate<T>>::insert(&key, true);
T::OnNewData::on_new_data(&key, &value);
}
Self::deposit_event(RawEvent::NewFeedData(who, values));