solana_message/versions/v0/
loaded.rs

1#[cfg(feature = "serde")]
2use serde_derive::{Deserialize, Serialize};
3use {
4    crate::{v0, AccountKeys},
5    solana_address::Address,
6    solana_sdk_ids::bpf_loader_upgradeable,
7    std::{borrow::Cow, collections::HashSet},
8};
9
10/// Combination of a version #0 message and its loaded addresses
11#[derive(Debug, Clone, Eq, PartialEq)]
12pub struct LoadedMessage<'a> {
13    /// Message which loaded a collection of lookup table addresses
14    pub message: Cow<'a, v0::Message>,
15    /// Addresses loaded with on-chain address lookup tables
16    pub loaded_addresses: Cow<'a, LoadedAddresses>,
17    /// List of boolean with same length as account_keys(), each boolean value indicates if
18    /// corresponding account key is writable or not.
19    pub is_writable_account_cache: Vec<bool>,
20}
21
22/// Collection of addresses loaded from on-chain lookup tables, split
23/// by readonly and writable.
24#[derive(Clone, Default, Debug, PartialEq, Eq)]
25#[cfg_attr(feature = "serde", derive(Deserialize, Serialize))]
26pub struct LoadedAddresses {
27    /// List of addresses for writable loaded accounts
28    pub writable: Vec<Address>,
29    /// List of addresses for read-only loaded accounts
30    pub readonly: Vec<Address>,
31}
32
33impl FromIterator<LoadedAddresses> for LoadedAddresses {
34    fn from_iter<T: IntoIterator<Item = LoadedAddresses>>(iter: T) -> Self {
35        let (writable, readonly): (Vec<Vec<Address>>, Vec<Vec<Address>>) = iter
36            .into_iter()
37            .map(|addresses| (addresses.writable, addresses.readonly))
38            .unzip();
39        LoadedAddresses {
40            writable: writable.into_iter().flatten().collect(),
41            readonly: readonly.into_iter().flatten().collect(),
42        }
43    }
44}
45
46impl LoadedAddresses {
47    /// Checks if there are no writable or readonly addresses
48    pub fn is_empty(&self) -> bool {
49        self.len() == 0
50    }
51
52    /// Combined length of loaded writable and readonly addresses
53    pub fn len(&self) -> usize {
54        self.writable.len().saturating_add(self.readonly.len())
55    }
56}
57
58impl<'a> LoadedMessage<'a> {
59    pub fn new(
60        message: v0::Message,
61        loaded_addresses: LoadedAddresses,
62        reserved_account_keys: &HashSet<Address>,
63    ) -> Self {
64        let mut loaded_message = Self {
65            message: Cow::Owned(message),
66            loaded_addresses: Cow::Owned(loaded_addresses),
67            is_writable_account_cache: Vec::default(),
68        };
69        loaded_message.set_is_writable_account_cache(reserved_account_keys);
70        loaded_message
71    }
72
73    pub fn new_borrowed(
74        message: &'a v0::Message,
75        loaded_addresses: &'a LoadedAddresses,
76        reserved_account_keys: &HashSet<Address>,
77    ) -> Self {
78        let mut loaded_message = Self {
79            message: Cow::Borrowed(message),
80            loaded_addresses: Cow::Borrowed(loaded_addresses),
81            is_writable_account_cache: Vec::default(),
82        };
83        loaded_message.set_is_writable_account_cache(reserved_account_keys);
84        loaded_message
85    }
86
87    fn set_is_writable_account_cache(&mut self, reserved_account_keys: &HashSet<Address>) {
88        let is_writable_account_cache = self
89            .account_keys()
90            .iter()
91            .enumerate()
92            .map(|(i, _key)| self.is_writable_internal(i, reserved_account_keys))
93            .collect::<Vec<_>>();
94        let _ = std::mem::replace(
95            &mut self.is_writable_account_cache,
96            is_writable_account_cache,
97        );
98    }
99
100    /// Returns the full list of static and dynamic account keys that are loaded for this message.
101    pub fn account_keys(&self) -> AccountKeys<'_> {
102        AccountKeys::new(&self.message.account_keys, Some(&self.loaded_addresses))
103    }
104
105    /// Returns the list of static account keys that are loaded for this message.
106    pub fn static_account_keys(&self) -> &[Address] {
107        &self.message.account_keys
108    }
109
110    /// Returns true if any account keys are duplicates
111    pub fn has_duplicates(&self) -> bool {
112        let mut uniq = HashSet::new();
113        self.account_keys().iter().any(|x| !uniq.insert(x))
114    }
115
116    /// Returns true if the account at the specified index was requested to be
117    /// writable.  This method should not be used directly.
118    fn is_writable_index(&self, key_index: usize) -> bool {
119        let header = &self.message.header;
120        let num_account_keys = self.message.account_keys.len();
121        let num_signed_accounts = usize::from(header.num_required_signatures);
122        if key_index >= num_account_keys {
123            let loaded_addresses_index = key_index.saturating_sub(num_account_keys);
124            loaded_addresses_index < self.loaded_addresses.writable.len()
125        } else if key_index >= num_signed_accounts {
126            let num_unsigned_accounts = num_account_keys.saturating_sub(num_signed_accounts);
127            let num_writable_unsigned_accounts = num_unsigned_accounts
128                .saturating_sub(usize::from(header.num_readonly_unsigned_accounts));
129            let unsigned_account_index = key_index.saturating_sub(num_signed_accounts);
130            unsigned_account_index < num_writable_unsigned_accounts
131        } else {
132            let num_writable_signed_accounts = num_signed_accounts
133                .saturating_sub(usize::from(header.num_readonly_signed_accounts));
134            key_index < num_writable_signed_accounts
135        }
136    }
137
138    /// Returns true if the account at the specified index was loaded as writable
139    fn is_writable_internal(
140        &self,
141        key_index: usize,
142        reserved_account_keys: &HashSet<Address>,
143    ) -> bool {
144        if self.is_writable_index(key_index) {
145            if let Some(key) = self.account_keys().get(key_index) {
146                return !(reserved_account_keys.contains(key) || self.demote_program_id(key_index));
147            }
148        }
149        false
150    }
151
152    pub fn is_writable(&self, key_index: usize) -> bool {
153        *self
154            .is_writable_account_cache
155            .get(key_index)
156            .unwrap_or(&false)
157    }
158
159    pub fn is_signer(&self, i: usize) -> bool {
160        i < self.message.header.num_required_signatures as usize
161    }
162
163    pub fn demote_program_id(&self, i: usize) -> bool {
164        self.is_key_called_as_program(i) && !self.is_upgradeable_loader_present()
165    }
166
167    /// Returns true if the account at the specified index is called as a program by an instruction
168    pub fn is_key_called_as_program(&self, key_index: usize) -> bool {
169        if let Ok(key_index) = u8::try_from(key_index) {
170            self.message
171                .instructions
172                .iter()
173                .any(|ix| ix.program_id_index == key_index)
174        } else {
175            false
176        }
177    }
178
179    /// Returns true if any account is the bpf upgradeable loader
180    pub fn is_upgradeable_loader_present(&self) -> bool {
181        self.account_keys()
182            .iter()
183            .any(|&key| key == bpf_loader_upgradeable::id())
184    }
185}
186
187#[cfg(test)]
188mod tests {
189    use {
190        super::*,
191        crate::{compiled_instruction::CompiledInstruction, MessageHeader},
192        itertools::Itertools,
193        solana_sdk_ids::{system_program, sysvar},
194    };
195
196    fn check_test_loaded_message() -> (LoadedMessage<'static>, [Address; 6]) {
197        let key0 = Address::new_unique();
198        let key1 = Address::new_unique();
199        let key2 = Address::new_unique();
200        let key3 = Address::new_unique();
201        let key4 = Address::new_unique();
202        let key5 = Address::new_unique();
203
204        let message = LoadedMessage::new(
205            v0::Message {
206                header: MessageHeader {
207                    num_required_signatures: 2,
208                    num_readonly_signed_accounts: 1,
209                    num_readonly_unsigned_accounts: 1,
210                },
211                account_keys: vec![key0, key1, key2, key3],
212                ..v0::Message::default()
213            },
214            LoadedAddresses {
215                writable: vec![key4],
216                readonly: vec![key5],
217            },
218            &HashSet::default(),
219        );
220
221        (message, [key0, key1, key2, key3, key4, key5])
222    }
223
224    #[test]
225    fn test_has_duplicates() {
226        let message = check_test_loaded_message().0;
227
228        assert!(!message.has_duplicates());
229    }
230
231    #[test]
232    fn test_has_duplicates_with_dupe_keys() {
233        let create_message_with_dupe_keys = |mut keys: Vec<Address>| {
234            LoadedMessage::new(
235                v0::Message {
236                    account_keys: keys.split_off(2),
237                    ..v0::Message::default()
238                },
239                LoadedAddresses {
240                    writable: keys.split_off(2),
241                    readonly: keys,
242                },
243                &HashSet::default(),
244            )
245        };
246
247        let key0 = Address::new_unique();
248        let key1 = Address::new_unique();
249        let key2 = Address::new_unique();
250        let key3 = Address::new_unique();
251        let dupe_key = Address::new_unique();
252
253        let keys = vec![key0, key1, key2, key3, dupe_key, dupe_key];
254        let keys_len = keys.len();
255        for keys in keys.into_iter().permutations(keys_len).unique() {
256            let message = create_message_with_dupe_keys(keys);
257            assert!(message.has_duplicates());
258        }
259    }
260
261    #[test]
262    fn test_is_writable_index() {
263        let message = check_test_loaded_message().0;
264
265        assert!(message.is_writable_index(0));
266        assert!(!message.is_writable_index(1));
267        assert!(message.is_writable_index(2));
268        assert!(!message.is_writable_index(3));
269        assert!(message.is_writable_index(4));
270        assert!(!message.is_writable_index(5));
271    }
272
273    #[test]
274    fn test_is_writable() {
275        let reserved_account_keys = HashSet::from_iter([sysvar::clock::id(), system_program::id()]);
276        let create_message_with_keys = |keys: Vec<Address>| {
277            LoadedMessage::new(
278                v0::Message {
279                    header: MessageHeader {
280                        num_required_signatures: 1,
281                        num_readonly_signed_accounts: 0,
282                        num_readonly_unsigned_accounts: 1,
283                    },
284                    account_keys: keys[..2].to_vec(),
285                    ..v0::Message::default()
286                },
287                LoadedAddresses {
288                    writable: keys[2..=2].to_vec(),
289                    readonly: keys[3..].to_vec(),
290                },
291                &reserved_account_keys,
292            )
293        };
294
295        let key0 = Address::new_unique();
296        let key1 = Address::new_unique();
297        let key2 = Address::new_unique();
298        {
299            let message = create_message_with_keys(vec![sysvar::clock::id(), key0, key1, key2]);
300            assert!(message.is_writable_index(0));
301            assert!(!message.is_writable(0));
302        }
303
304        {
305            let message = create_message_with_keys(vec![system_program::id(), key0, key1, key2]);
306            assert!(message.is_writable_index(0));
307            assert!(!message.is_writable(0));
308        }
309
310        {
311            let message = create_message_with_keys(vec![key0, key1, system_program::id(), key2]);
312            assert!(message.is_writable_index(2));
313            assert!(!message.is_writable(2));
314        }
315    }
316
317    #[test]
318    fn test_demote_writable_program() {
319        let key0 = Address::new_unique();
320        let key1 = Address::new_unique();
321        let key2 = Address::new_unique();
322        let message = LoadedMessage::new(
323            v0::Message {
324                header: MessageHeader {
325                    num_required_signatures: 1,
326                    num_readonly_signed_accounts: 0,
327                    num_readonly_unsigned_accounts: 0,
328                },
329                account_keys: vec![key0],
330                instructions: vec![CompiledInstruction {
331                    program_id_index: 2,
332                    accounts: vec![1],
333                    data: vec![],
334                }],
335                ..v0::Message::default()
336            },
337            LoadedAddresses {
338                writable: vec![key1, key2],
339                readonly: vec![],
340            },
341            &HashSet::default(),
342        );
343
344        assert!(message.is_writable_index(2));
345        assert!(!message.is_writable(2));
346    }
347}