af_oracle/event_ext/
apply.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
use af_sui_types::ObjectId;
use sui_framework_sdk::object::{ID, UID};

use crate::events;
use crate::oracle::PriceFeed;

/// Intended for defining how to update off-chain data with oracle event information.
///
/// This should work under the following assumptions.
///
/// # Assumptions
/// 1. The database is in sync with the on-chain protocol state at least up until the beginning of
///    the event's transaction.
pub trait Apply<T> {
    /// Use the contents of `self` to (possibly) mutate data in `target`.
    fn apply(self, target: &mut T);
}

/// General interface to help apply [`PriceFeedStorage`](crate::oracle::PriceFeedStorage)-related
/// events to a database.
pub trait PriceFeedStorage {
    fn insert_price_feed(&mut self, source_wrapper_id: ID, feed: PriceFeed);

    fn get_price_feed_mut(&mut self, source_wrapper_id: &ID) -> Option<&mut PriceFeed>;

    fn remove_price_feed(&mut self, source_wrapper_id: &ID) -> Option<PriceFeed>;
}

impl<T: PriceFeedStorage> Apply<T> for events::CreatedPriceFeed {
    fn apply(self, target: &mut T) {
        let Self {
            source_wrapper_id,
            price,
            timestamp,
            time_tolerance,
            ..
        } = self;
        let feed = PriceFeed {
            // The event doesn't have information about the object id of the feed. Thus, we
            // initialize it as zero to make it very clear to applications using this that the
            // object id here is not a real one. This shouldn't be a problem since all interactions
            // with price feeds in AfOracle are done through the `PriceFeedStorage`.
            id: UID {
                id: ID {
                    bytes: ObjectId::ZERO,
                },
            },
            price,
            timestamp,
            time_tolerance,
        };
        target.insert_price_feed(source_wrapper_id, feed);
    }
}

impl<T: PriceFeedStorage> Apply<T> for events::UpdatedPriceFeed {
    fn apply(self, target: &mut T) {
        let Self {
            source_wrapper_id,
            new_price,
            new_timestamp,
            ..
        } = self;
        // Under Assumption 1 of [`Apply`], if a price feed is not in the database at this point,
        // it's because we're applying and old event relative to the current state and the price
        // feed was removed later.
        if let Some(feed) = target.get_price_feed_mut(&source_wrapper_id) {
            feed.price = new_price;
            feed.timestamp = new_timestamp;
        }
    }
}

impl<T: PriceFeedStorage> Apply<T> for events::RemovedPriceFeed {
    fn apply(self, target: &mut T) {
        let Self {
            source_wrapper_id, ..
        } = self;
        target.remove_price_feed(&source_wrapper_id);
    }
}

impl<T: PriceFeedStorage> Apply<T> for events::UpdatedPriceFeedTimeTolerance {
    fn apply(self, target: &mut T) {
        let Self {
            source_wrapper_id,
            new_time_tolerance,
            ..
        } = self;
        // Under Assumption 1 of [`Apply`], if a price feed is not in the database at this point,
        // it's because we're applying and old event relative to the current state and the price
        // feed was removed later.
        if let Some(feed) = target.get_price_feed_mut(&source_wrapper_id) {
            feed.time_tolerance = new_time_tolerance;
        }
    }
}