sui_rpc_loadgen/payload/
mod.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
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
// Copyright (c) Mysten Labs, Inc.
// SPDX-License-Identifier: Apache-2.0

mod checkpoint_utils;
mod get_all_balances;
mod get_checkpoints;
mod get_object;
mod get_reference_gas_price;
mod multi_get_objects;
mod multi_get_transaction_blocks;
mod pay_sui;
mod query_transactions;
mod rpc_command_processor;
mod validation;
use strum_macros::EnumString;

use anyhow::Result;
use async_trait::async_trait;
use core::default::Default;
use std::time::Duration;
use sui_types::{
    base_types::SuiAddress, digests::TransactionDigest,
    messages_checkpoint::CheckpointSequenceNumber,
};

use crate::load_test::LoadTestConfig;
pub use rpc_command_processor::{
    load_addresses_from_file, load_digests_from_file, load_objects_from_file, RpcCommandProcessor,
};
use sui_types::base_types::ObjectID;

#[derive(Default, Clone)]
pub struct SignerInfo {
    pub encoded_keypair: String,
    /// Different thread should use different gas_payment to avoid equivocation
    pub gas_payment: Option<Vec<ObjectID>>,
    pub gas_budget: Option<u64>,
}

impl SignerInfo {
    pub fn new(encoded_keypair: String) -> Self {
        Self {
            encoded_keypair,
            gas_payment: None,
            gas_budget: None,
        }
    }
}

#[derive(Clone, Default)]
pub struct Payload {
    pub commands: Vec<Command>,
    pub signer_info: Option<SignerInfo>,
}

#[derive(Default, Clone)]
pub struct Command {
    pub data: CommandData,
    /// 0 means the command will be run once. Default to be 0
    pub repeat_n_times: usize,
    /// how long to wait between the start of two subsequent repeats
    /// If the previous command takes longer than `repeat_interval` to finish, the next command
    /// will run as soon as the previous command finishes
    /// Default to be 0
    pub repeat_interval: Duration,
}

impl Command {
    pub fn new_dry_run() -> Self {
        Self {
            data: CommandData::DryRun(DryRun {}),
            ..Default::default()
        }
    }

    pub fn new_pay_sui() -> Self {
        Self {
            data: CommandData::PaySui(PaySui {}),
            ..Default::default()
        }
    }

    pub fn new_get_checkpoints(
        start: CheckpointSequenceNumber,
        end: Option<CheckpointSequenceNumber>,
        verify_transactions: bool,
        verify_objects: bool,
        record: bool,
    ) -> Self {
        Self {
            data: CommandData::GetCheckpoints(GetCheckpoints {
                start,
                end,
                verify_transactions,
                verify_objects,
                record,
            }),
            ..Default::default()
        }
    }

    pub fn new_query_transaction_blocks(
        address_type: AddressQueryType,
        addresses: Vec<SuiAddress>,
    ) -> Self {
        let query_transactions = QueryTransactionBlocks {
            address_type,
            addresses,
        };
        Self {
            data: CommandData::QueryTransactionBlocks(query_transactions),
            ..Default::default()
        }
    }

    pub fn new_multi_get_transaction_blocks(digests: Vec<TransactionDigest>) -> Self {
        let multi_get_transaction_blocks = MultiGetTransactionBlocks { digests };
        Self {
            data: CommandData::MultiGetTransactionBlocks(multi_get_transaction_blocks),
            ..Default::default()
        }
    }

    pub fn new_multi_get_objects(object_ids: Vec<ObjectID>) -> Self {
        let multi_get_objects = MultiGetObjects { object_ids };
        Self {
            data: CommandData::MultiGetObjects(multi_get_objects),
            ..Default::default()
        }
    }

    pub fn new_get_object(object_ids: Vec<ObjectID>, chunk_size: usize) -> Self {
        let get_object = GetObject {
            object_ids,
            chunk_size,
        };
        Self {
            data: CommandData::GetObject(get_object),
            ..Default::default()
        }
    }

    pub fn new_get_all_balances(addresses: Vec<SuiAddress>, chunk_size: usize) -> Self {
        let get_all_balances = GetAllBalances {
            addresses,
            chunk_size,
        };
        Self {
            data: CommandData::GetAllBalances(get_all_balances),
            ..Default::default()
        }
    }

    pub fn new_get_reference_gas_price(num_repeats: usize) -> Self {
        let get_reference_gas_price = GetReferenceGasPrice { num_repeats };
        Self {
            data: CommandData::GetReferenceGasPrice(get_reference_gas_price),
            ..Default::default()
        }
    }

    pub fn with_repeat_n_times(mut self, num: usize) -> Self {
        self.repeat_n_times = num;
        self
    }

    pub fn with_repeat_interval(mut self, duration: Duration) -> Self {
        self.repeat_interval = duration;
        self
    }
}

#[derive(Clone)]
#[allow(dead_code)]
pub enum CommandData {
    DryRun(DryRun),
    GetCheckpoints(GetCheckpoints),
    PaySui(PaySui),
    QueryTransactionBlocks(QueryTransactionBlocks),
    MultiGetTransactionBlocks(MultiGetTransactionBlocks),
    MultiGetObjects(MultiGetObjects),
    GetObject(GetObject),
    GetAllBalances(GetAllBalances),
    GetReferenceGasPrice(GetReferenceGasPrice),
}

impl Default for CommandData {
    fn default() -> Self {
        CommandData::DryRun(DryRun {})
    }
}

#[derive(Clone)]
pub struct DryRun {}

#[derive(Clone, Default)]
pub struct GetCheckpoints {
    /// Default to start from 0
    pub start: CheckpointSequenceNumber,
    /// If None, use `getLatestCheckpointSequenceNumber`
    pub end: Option<CheckpointSequenceNumber>,
    pub verify_transactions: bool,
    pub verify_objects: bool,
    pub record: bool,
}

#[derive(Clone)]
pub struct PaySui {}

#[derive(Clone, Default)]
pub struct QueryTransactionBlocks {
    pub address_type: AddressQueryType,
    pub addresses: Vec<SuiAddress>,
}

#[derive(Clone)]
pub struct MultiGetTransactionBlocks {
    pub digests: Vec<TransactionDigest>,
}

#[derive(Clone, EnumString, Default)]
#[strum(serialize_all = "lowercase")]
pub enum AddressQueryType {
    #[default]
    From,
    To,
    Both,
}

#[derive(Clone)]
pub struct MultiGetObjects {
    pub object_ids: Vec<ObjectID>,
}

#[derive(Clone)]
pub struct GetObject {
    pub object_ids: Vec<ObjectID>,
    pub chunk_size: usize,
}

#[derive(Clone)]
pub struct GetAllBalances {
    pub addresses: Vec<SuiAddress>,
    pub chunk_size: usize,
}

#[derive(Clone)]
pub struct GetReferenceGasPrice {
    num_repeats: usize,
}

#[async_trait]
pub trait Processor {
    /// process commands in order
    async fn apply(&self, payload: &Payload) -> Result<()>;

    /// prepare payload for each thread according to LoadTestConfig
    async fn prepare(&self, config: &LoadTestConfig) -> Result<Vec<Payload>>;

    /// write results to file based on LoadTestConfig
    fn dump_cache_to_file(&self, config: &LoadTestConfig);
}

/// all payload should implement this trait
#[async_trait]
pub trait ProcessPayload<'a, T> {
    async fn process(&'a self, op: T, signer_info: &Option<SignerInfo>) -> Result<()>;
}