sui_core/authority/
transaction_deferral.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
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
// Copyright (c) Mysten Labs, Inc.
// SPDX-License-Identifier: Apache-2.0

use serde::{Deserialize, Serialize};
use sui_types::{base_types::ObjectID, messages_consensus::Round};

#[derive(Copy, Clone, Debug, Eq, PartialEq, PartialOrd, Ord, Serialize, Deserialize)]
pub enum DeferralKey {
    // For transactions deferred until new randomness is available (whether delayd due to
    // DKG, or skipped commits).
    Randomness {
        deferred_from_round: Round, // commit round, not randomness round
    },
    // ConsensusRound deferral key requires both the round to which the tx should be deferred (so that
    // we can efficiently load all txns that are now ready), and the round from which it has been
    // deferred (so that multiple rounds can efficiently defer to the same future round).
    ConsensusRound {
        future_round: Round,
        deferred_from_round: Round,
    },
}

impl DeferralKey {
    pub fn new_for_randomness(deferred_from_round: Round) -> Self {
        Self::Randomness {
            deferred_from_round,
        }
    }

    pub fn new_for_consensus_round(future_round: Round, deferred_from_round: Round) -> Self {
        Self::ConsensusRound {
            future_round,
            deferred_from_round,
        }
    }

    pub fn full_range_for_randomness() -> (Self, Self) {
        (
            Self::Randomness {
                deferred_from_round: 0,
            },
            Self::Randomness {
                deferred_from_round: u64::MAX,
            },
        )
    }

    // Returns a range of deferral keys that are deferred up to the given consensus round.
    pub fn range_for_up_to_consensus_round(consensus_round: Round) -> (Self, Self) {
        (
            Self::ConsensusRound {
                future_round: 0,
                deferred_from_round: 0,
            },
            Self::ConsensusRound {
                future_round: consensus_round.checked_add(1).unwrap(),
                deferred_from_round: 0,
            },
        )
    }

    pub fn deferred_from_round(&self) -> Round {
        match self {
            Self::Randomness {
                deferred_from_round,
            } => *deferred_from_round,
            Self::ConsensusRound {
                deferred_from_round,
                ..
            } => *deferred_from_round,
        }
    }
}

#[derive(Debug)]
pub enum DeferralReason {
    RandomnessNotReady,

    // The list of objects are congested objects.
    SharedObjectCongestion(Vec<ObjectID>),
}

pub fn transaction_deferral_within_limit(
    deferral_key: &DeferralKey,
    max_deferral_rounds_for_congestion_control: u64,
) -> bool {
    if let DeferralKey::ConsensusRound {
        future_round,
        deferred_from_round,
    } = deferral_key
    {
        return (future_round - deferred_from_round) <= max_deferral_rounds_for_congestion_control;
    }

    // TODO: drop transactions at the end of the queue if the queue is too long.

    true
}

#[cfg(test)]
mod object_cost_tests {
    use super::*;
    use typed_store::DBMapUtils;
    use typed_store::Map;
    use typed_store::{
        rocks::{DBMap, MetricConf},
        traits::{TableSummary, TypedStoreDebug},
    };

    #[tokio::test]
    async fn test_deferral_key_sort_order() {
        use rand::prelude::*;

        #[derive(DBMapUtils)]
        struct TestDB {
            deferred_certs: DBMap<DeferralKey, ()>,
        }

        // get a tempdir
        let tempdir = tempfile::tempdir().unwrap();

        let db = TestDB::open_tables_read_write(
            tempdir.path().to_owned(),
            MetricConf::new("test_db"),
            None,
            None,
        );

        for _ in 0..10000 {
            let future_round = rand::thread_rng().gen_range(0..u64::MAX);
            let current_round = rand::thread_rng().gen_range(0..u64::MAX);

            let key = DeferralKey::new_for_consensus_round(future_round, current_round);
            db.deferred_certs.insert(&key, &()).unwrap();
        }

        let mut previous_future_round = 0;
        for item in db.deferred_certs.safe_iter() {
            match item.unwrap().0 {
                DeferralKey::Randomness { .. } => (),
                DeferralKey::ConsensusRound { future_round, .. } => {
                    assert!(previous_future_round <= future_round);
                    previous_future_round = future_round;
                }
            }
        }
    }

    // Tests that fetching deferred transactions up to a given consensus rounds works as expected.
    #[tokio::test]
    async fn test_fetching_deferred_txs() {
        use rand::prelude::*;

        #[derive(DBMapUtils)]
        struct TestDB {
            deferred_certs: DBMap<DeferralKey, ()>,
        }

        // get a tempdir
        let tempdir = tempfile::tempdir().unwrap();

        let db = TestDB::open_tables_read_write(
            tempdir.path().to_owned(),
            MetricConf::new("test_db"),
            None,
            None,
        );

        // All future rounds are between 100 and 300.
        let min_future_round = 100;
        let max_future_round = 300;
        for _ in 0..10000 {
            let future_round = rand::thread_rng().gen_range(min_future_round..=max_future_round);
            let current_round = rand::thread_rng().gen_range(0..u64::MAX);

            db.deferred_certs
                .insert(
                    &DeferralKey::new_for_consensus_round(future_round, current_round),
                    &(),
                )
                .unwrap();
            // Add a randomness deferral txn to make sure that it won't show up when fetching deferred consensus round txs.
            db.deferred_certs
                .insert(&DeferralKey::new_for_randomness(current_round), &())
                .unwrap();
        }

        // Fetch all deferred transactions up to consensus round 200.
        let (min, max) = DeferralKey::range_for_up_to_consensus_round(200);
        let mut previous_future_round = 0;
        let mut result_count = 0;
        for result in db
            .deferred_certs
            .safe_iter_with_bounds(Some(min), Some(max))
        {
            let (key, _) = result.unwrap();
            match key {
                DeferralKey::Randomness { .. } => {
                    panic!("Should not receive randomness deferral txn.")
                }
                DeferralKey::ConsensusRound { future_round, .. } => {
                    assert!(previous_future_round <= future_round);
                    previous_future_round = future_round;
                    assert!(future_round <= 200);
                    result_count += 1;
                }
            }
        }
        assert!(result_count > 0);
    }
}