sui_protocol_config/
lib.rs

1// Copyright (c) Mysten Labs, Inc.
2// SPDX-License-Identifier: Apache-2.0
3
4use std::{
5    collections::{BTreeMap, BTreeSet},
6    sync::{
7        Arc, LazyLock,
8        atomic::{AtomicBool, Ordering},
9    },
10};
11
12#[cfg(msim)]
13use std::cell::RefCell;
14#[cfg(not(msim))]
15use std::sync::Mutex;
16
17use clap::*;
18use fastcrypto::encoding::{Base58, Encoding, Hex};
19use move_binary_format::{
20    binary_config::{BinaryConfig, TableConfig},
21    file_format_common::VERSION_1,
22};
23use move_core_types::account_address::AccountAddress;
24use move_vm_config::verifier::VerifierConfig;
25use mysten_common::in_integration_test;
26use serde::{Deserialize, Serialize};
27use serde_with::skip_serializing_none;
28use sui_protocol_config_macros::{
29    ProtocolConfigAccessors, ProtocolConfigFeatureFlagsGetters, ProtocolConfigOverride,
30};
31use tracing::{info, warn};
32
33/// The minimum and maximum protocol versions supported by this build.
34const MIN_PROTOCOL_VERSION: u64 = 1;
35const MAX_PROTOCOL_VERSION: u64 = 125;
36
37const TESTNET_USDC: &str =
38    "0xa1ec7fc00a6f40db9693ad1415d0c193ad3906494428cf252621037bd7117e29::usdc::USDC";
39
40const MAINNET_USDC: &str =
41    "0xdba34672e30cb065b1f93e3ab55318768fd6fef66c15942c9f7cb846e2f900e7::usdc::USDC";
42const MAINNET_USDSUI: &str =
43    "0x44f838219cf67b058f3b37907b655f226153c18e33dfcd0da559a844fea9b1c1::usdsui::USDSUI";
44const MAINNET_SUI_USDE: &str =
45    "0x41d587e5336f1c86cad50d38a7136db99333bb9bda91cea4ba69115defeb1402::sui_usde::SUI_USDE";
46const MAINNET_USDY: &str =
47    "0x960b531667636f39e85867775f52f6b1f220a058c4de786905bdf761e06a56bb::usdy::USDY";
48const MAINNET_FDUSD: &str =
49    "0xf16e6b723f242ec745dfd7634ad072c42d5c1d9ac9d62a39c381303eaa57693a::fdusd::FDUSD";
50const MAINNET_AUSD: &str =
51    "0x2053d08c1e2bd02791056171aab0fd12bd7cd7efad2ab8f6b9c8902f14df2ff2::ausd::AUSD";
52const MAINNET_USDB: &str =
53    "0xe14726c336e81b32328e92afc37345d159f5b550b09fa92bd43640cfdd0a0cfd::usdb::USDB";
54
55// Record history of protocol version allocations here:
56//
57// Version 1: Original version.
58// Version 2: Framework changes, including advancing epoch_start_time in safemode.
59// Version 3: gas model v2, including all sui conservation fixes. Fix for loaded child object
60//            changes, enable package upgrades, add limits on `max_size_written_objects`,
61//            `max_size_written_objects_system_tx`
62// Version 4: New reward slashing rate. Framework changes to skip stake susbidy when the epoch
63//            length is short.
64// Version 5: Package upgrade compatibility error fix. New gas cost table. New scoring decision
65//            mechanism that includes up to f scoring authorities.
66// Version 6: Change to how bytes are charged in the gas meter, increase buffer stake to 0.5f
67// Version 7: Disallow adding new abilities to types during package upgrades,
68//            disable_invariant_violation_check_in_swap_loc,
69//            disable init functions becoming entry,
70//            hash module bytes individually before computing package digest.
71// Version 8: Disallow changing abilities and type constraints for type parameters in structs
72//            during upgrades.
73// Version 9: Limit the length of Move idenfitiers to 128.
74//            Disallow extraneous module bytes,
75//            advance_to_highest_supported_protocol_version,
76// Version 10:increase bytecode verifier `max_verifier_meter_ticks_per_function` and
77//            `max_meter_ticks_per_module` limits each from 6_000_000 to 16_000_000. sui-system
78//            framework changes.
79// Version 11: Introduce `std::type_name::get_with_original_ids` to the system frameworks. Bound max depth of values within the VM.
80// Version 12: Changes to deepbook in framework to add API for querying marketplace.
81//             Change NW Batch to use versioned metadata field.
82//             Changes to sui-system package to add PTB-friendly unstake function, and minor cleanup.
83// Version 13: System package change deprecating `0xdee9::clob` and `0xdee9::custodian`, replaced by
84//             `0xdee9::clob_v2` and `0xdee9::custodian_v2`.
85// Version 14: Introduce a config variable to allow charging of computation to be either
86//             bucket base or rounding up. The presence of `gas_rounding_step` (or `None`)
87//             decides whether rounding is applied or not.
88// Version 15: Add reordering of user transactions by gas price after consensus.
89//             Add `sui::table_vec::drop` to the framework via a system package upgrade.
90// Version 16: Enabled simplified_unwrap_then_delete feature flag, which allows the execution engine
91//             to no longer consult the object store when generating unwrapped_then_deleted in the
92//             effects; this also allows us to stop including wrapped tombstones in accumulator.
93//             Add self-matching prevention for deepbook.
94// Version 17: Enable upgraded multisig support.
95// Version 18: Introduce execution layer versioning, preserve all existing behaviour in v0.
96//             Gas minimum charges moved to be a multiplier over the reference gas price. In this
97//             protocol version the multiplier is the same as the lowest bucket of computation
98//             such that the minimum transaction cost is the same as the minimum computation
99//             bucket.
100//             Add a feature flag to indicate the changes semantics of `base_tx_cost_fixed`.
101// Version 19: Changes to sui-system package to enable liquid staking.
102//             Add limit for total size of events.
103//             Increase limit for number of events emitted to 1024.
104// Version 20: Enables the flag `narwhal_new_leader_election_schedule` for the new narwhal leader
105//             schedule algorithm for enhanced fault tolerance and sets the bad node stake threshold
106//             value. Both values are set for all the environments except mainnet.
107// Version 21: ZKLogin known providers.
108// Version 22: Child object format change.
109// Version 23: Enabling the flag `narwhal_new_leader_election_schedule` for the new narwhal leader
110//             schedule algorithm for enhanced fault tolerance and sets the bad node stake threshold
111//             value for mainnet.
112// Version 24: Re-enable simple gas conservation checks.
113//             Package publish/upgrade number in a single transaction limited.
114//             JWK / authenticator state flags.
115// Version 25: Add sui::table_vec::swap and sui::table_vec::swap_remove to system packages.
116// Version 26: New gas model version.
117//             Add support for receiving objects off of other objects in devnet only.
118// Version 28: Add sui::zklogin::verify_zklogin_id and related functions to sui framework.
119//             Enable transaction effects v2 in devnet.
120// Version 29: Add verify_legacy_zklogin_address flag to sui framework, this add ability to verify
121//             transactions from a legacy zklogin address.
122// Version 30: Enable Narwhal CertificateV2
123//             Add support for random beacon.
124//             Enable transaction effects v2 in testnet.
125//             Deprecate supported oauth providers from protocol config and rely on node config
126//             instead.
127//             In execution, has_public_transfer is recomputed when loading the object.
128//             Add support for shared obj deletion and receiving objects off of other objects in devnet only.
129// Version 31: Add support for shared object deletion in devnet only.
130//             Add support for getting object ID referenced by receiving object in sui framework.
131//             Create new execution layer version, and preserve previous behavior in v1.
132//             Update semantics of `sui::transfer::receive` and add `sui::transfer::public_receive`.
133// Version 32: Add delete functions for VerifiedID and VerifiedIssuer.
134//             Add sui::token module to sui framework.
135//             Enable transfer to object in testnet.
136//             Enable Narwhal CertificateV2 on mainnet
137//             Make critbit tree and order getters public in deepbook.
138// Version 33: Add support for `receiving_object_id` function in framework
139//             Hardened OTW check.
140//             Enable transfer-to-object in mainnet.
141//             Enable shared object deletion in testnet.
142//             Enable effects v2 in mainnet.
143// Version 34: Framework changes for random beacon.
144// Version 35: Add poseidon hash function.
145//             Enable coin deny list.
146// Version 36: Enable group operations native functions in devnet.
147//             Enable shared object deletion in mainnet.
148//             Set the consensus accepted transaction size and the included transactions size in the proposed block.
149// Version 37: Reject entry functions with mutable Random.
150// Version 38: Introduce limits for binary tables size.
151// Version 39: Allow skipped epochs for randomness updates.
152//             Extra version to fix `test_upgrade_compatibility` simtest.
153// Version 40:
154// Version 41: Enable group operations native functions in testnet and mainnet (without msm).
155// Version 42: Migrate sui framework and related code to Move 2024
156// Version 43: Introduce the upper bound delta config for a zklogin signature's max epoch.
157//             Introduce an explicit parameter for the tick limit per package (previously this was
158//             represented by the parameter for the tick limit per module).
159// Version 44: Enable consensus fork detection on mainnet.
160//             Switch between Narwhal and Mysticeti consensus in tests, devnet and testnet.
161// Version 45: Use tonic networking for Mysticeti consensus.
162//             Set min Move binary format version to 6.
163//             Enable transactions to be signed with zkLogin inside multisig signature.
164//             Add native bridge.
165//             Enable native bridge in devnet
166//             Enable Leader Scoring & Schedule Change for Mysticeti consensus on testnet.
167// Version 46: Enable native bridge in testnet
168//             Enable resharing at the same initial shared version.
169// Version 47: Deepbook changes (framework update)
170// Version 48: Use tonic networking for Mysticeti.
171//             Resolve Move abort locations to the package id instead of the runtime module ID.
172//             Enable random beacon in testnet.
173//             Use new VM when verifying framework packages.
174// Version 49: Enable Move enums on devnet.
175//             Enable VDF in devnet
176//             Enable consensus commit prologue V3 in devnet.
177//             Run Mysticeti consensus by default.
178// Version 50: Add update_node_url to native bridge,
179//             New Move stdlib integer modules
180//             Enable checkpoint batching in testnet.
181//             Prepose consensus commit prologue in checkpoints.
182//             Set number of leaders per round for Mysticeti commits.
183// Version 51: Switch to DKG V1.
184//             Enable deny list v2 on devnet.
185// Version 52: Emit `CommitteeMemberUrlUpdateEvent` when updating bridge node url.
186//             std::config native functions.
187//             Modified sui-system package to enable withdrawal of stake before it becomes active.
188//             Enable soft bundle in devnet and testnet.
189//             Core macro visibility in sui core framework.
190//             Enable checkpoint batching in mainnet.
191//             Enable Mysticeti on mainnet.
192//             Enable Leader Scoring & Schedule Change for Mysticeti consensus on mainnet.
193//             Turn on count based shared object congestion control in devnet.
194//             Enable consensus commit prologue V3 in testnet.
195//             Enable enums on testnet.
196//             Add support for passkey in devnet.
197//             Enable deny list v2 on testnet and mainnet.
198// Version 53: Add feature flag to decide whether to attempt to finalize bridge committee
199//             Enable consensus commit prologue V3 on testnet.
200//             Turn on shared object congestion control in testnet.
201//             Update stdlib natives costs
202// Version 54: Enable random beacon on mainnet.
203//             Enable soft bundle on mainnet.
204// Version 55: Enable enums on mainnet.
205//             Rethrow serialization type layout errors instead of converting them.
206// Version 56: Enable bridge on mainnet.
207//             Note: do not use version 56 for any new features.
208// Version 57: Reduce minimum number of random beacon shares.
209// Version 58: Optimize boolean binops
210//             Finalize bridge committee on mainnet.
211//             Switch to distributed vote scoring in consensus in devnet
212// Version 59: Enable round prober in consensus.
213// Version 60: Validation of public inputs for Groth16 verification.
214//             Enable configuration of maximum number of type nodes in a type layout.
215// Version 61: Switch to distributed vote scoring in consensus in testnet
216//             Further reduce minimum number of random beacon shares.
217//             Add feature flag for Mysticeti fastpath.
218// Version 62: Makes the event's sending module package upgrade-aware.
219// Version 63: Enable gas based congestion control in consensus commit.
220// Version 64: Revert congestion control change.
221// Version 65: Enable distributed vote scoring in mainnet.
222// Version 66: Revert distributed vote scoring in mainnet.
223//             Framework fix for fungible staking book-keeping.
224// Version 67: Re-enable distributed vote scoring in mainnet.
225// Version 68: Add G1Uncompressed group to group ops.
226//             Update to Move stdlib.
227//             Enable gas based congestion control with overage.
228//             Further reduce minimum number of random beacon shares.
229//             Disallow adding new modules in `deps-only` packages.
230// Version 69: Sets number of rounds allowed for fastpath voting in consensus.
231//             Enable smart ancestor selection in devnet.
232//             Enable G1Uncompressed group in testnet.
233// Version 70: Enable smart ancestor selection in testnet.
234//             Enable probing for accepted rounds in round prober in testnet
235//             Add new gas model version to update charging of native functions.
236//             Add std::uq64_64 module to Move stdlib.
237//             Improve gas/wall time efficiency of some Move stdlib vector functions
238// Version 71: [SIP-45] Enable consensus amplification.
239// Version 72: Fix issue where `convert_type_argument_error` wasn't being used in all cases.
240//             Max gas budget moved to 50_000 SUI
241//             Max gas price moved to 50 SUI
242//             Variants as type nodes.
243// Version 73: Enable new marker table version.
244//             Enable consensus garbage collection and new commit rule for devnet.
245//             Enable zstd compression for consensus tonic network in testnet.
246//             Enable smart ancestor selection in mainnet.
247//             Enable probing for accepted rounds in round prober in mainnet
248// Version 74: Enable load_nitro_attestation move function in sui framework in devnet.
249//             Enable all gas costs for load_nitro_attestation.
250//             Enable zstd compression for consensus tonic network in mainnet.
251//             Enable the new commit rule for devnet.
252// Version 75: Enable passkey auth in testnet.
253// Version 76: Deprecate Deepbook V2 order placement and deposit.
254//             Removes unnecessary child object mutations
255//             Enable passkey auth in multisig for testnet.
256// Version 77: Enable uncompressed point group ops on mainnet.
257//             Enable consensus garbage collection for testnet
258//             Enable the new consensus commit rule for testnet.
259// Version 78: Make `TxContext` Move API native
260//             Enable execution time estimate mode for congestion control on testnet.
261// Version 79: Enable median based commit timestamp in consensus on testnet.
262//             Increase threshold for bad nodes that won't be considered leaders in consensus in testnet
263//             Enable load_nitro_attestation move function in sui framework in testnet.
264//             Enable consensus garbage collection for mainnet
265//             Enable the new consensus commit rule for mainnet.
266// Version 80: Bound size of values created in the adapter.
267// Version 81: Enable median based commit timestamp in consensus on mainnet.
268//             Enforce checkpoint timestamps are non-decreasing for testnet and mainnet.
269//             Increase threshold for bad nodes that won't be considered leaders in consensus in mainnet
270// Version 82: Relax bounding of size of values created in the adapter.
271// Version 83: Resolve `TypeInput` IDs to defining ID when converting to `TypeTag`s in the adapter.
272//             Enable execution time estimate mode for congestion control on mainnet.
273//             Enable nitro attestation upgraded parsing and mainnet.
274// Version 84: Limit number of stored execution time observations between epochs.
275// Version 85: Enable party transfer in devnet.
276// Version 86: Use type tags in the object runtime and adapter instead of `Type`s.
277//             Make variant count limit explicit in protocol config.
278//             Enable party transfer in testnet.
279// Version 87: Enable better type resolution errors in the adapter.
280// Version 88: Update `sui-system` package to use `calculate_rewards` function.
281//             Define the cost for the native Move function `rgp`.
282//             Ignore execution time observations after validator stops accepting certs.
283// Version 89: Add additional signature checks
284//             Add additional linkage checks
285// Version 90: Standard library improvements.
286//             Enable `debug_fatal` on Move invariant violations.
287//             Enable passkey and passkey inside multisig for mainnet.
288// Version 91: Minor changes in Sui Framework. Include CheckpointDigest in consensus dedup key for checkpoint signatures (V2).
289// Version 92: Disable checking shared object transfer restrictions per command = false
290// Version 93: Enable CheckpointDigest in consensus dedup key for checkpoint signatures.
291// Version 94: Decrease stored observations limit by 10% to stay within system object size limit.
292//             Enable party transfer on mainnet.
293// Version 95: Change type name id base cost to 52, increase max transactions per checkpoint to 20000.
294// Version 96: Enable authority capabilities v2.
295//             Fix bug where MFP transaction shared inputs' debts were not loaded
296//             Create Coin Registry object
297//             Enable checkpoint artifacts digest in devnet.
298// Version 97: Enable additional borrow checks
299// Version 98: Add authenticated event streams support via emit_authenticated function.
300//             Add better error messages to the loader.
301// Version 99: Enable new commit handler.
302// Version 100: Framework update
303// Version 101: Framework update
304//              Set max updates per settlement txn to 100.
305// Version 103: Framework update: internal Coin methods
306// Version 104: Framework update: CoinRegistry follow up for Coin methods
307//              Enable all non-zero PCRs parsing for nitro attestation native function in Devnet and Testnet.
308// Version 105: Framework update: address aliases
309//              Enable multi-epoch transaction expiration.
310//              Enable always include required PCRs (0-4 & 8) parsing even if they are zeros for
311//              nitro attestation native function in Devnet and Testnet.
312// Version 106: Framework update: accumulator storage fund calculations
313//              Enable address balances on devnet
314// Version 108: Enable new digit based gas rounding.
315//              Support TxContext in all parameter positions.
316//              Disable entry point signature check.
317//              Enable address aliases on testnet.
318//              Enable poseidon_bn254 on mainnet.
319// Version 109: Update where we set bounds for some binary tables to be a bit more idiomatic.
320// Version 110: Enable parsing on all nonzero custom pcrs in nitro attestation parsing native
321//              function on mainnet.
322//              split_checkpoints_in_consensus_handler in devnet
323//              Enable additional validation on zkLogin public identifier.
324// Version 111: Validator metadata
325// Version 112: Enable Ristretto255 in devnet.
326// Version 113: Validate gas price >= RGP at signing for address balance gas payments.
327// Version 114: Gate seeded test overrides for checkpoint tx limit behind feature flag.
328// Version 115: Gasless transaction drop safety.
329//              Enable address aliases on mainnet.
330//              Relax ValidDuring requirement for transactions with owned inputs.
331// Version 116: Enable Display Registry.
332//              Disable defer_unpaid_amplification (debugging).
333// Version 117: Update Sui System metadata handling.
334// Version 118: Adds `transfer_migration_cap` to display registry
335// Version 119: Enable the new VM.
336// Version 120: Disallow unused jump tables
337// Version 121: Re-enable defer_unpaid_amplification (devnet + testnet).
338// Version 122: Framework update: vector::empty is deprecated.
339//              Enable bulletproofs verification on devnet.
340//              Enable defer_unpaid_amplification on mainnet.
341// Version 123: Gas accounting refresh (gas_model v13).
342// Version 124: Add timestamp_based_epoch_close feature flag and enable in tests.
343//              Fix native call double-pop in gas meter stack height tracking (gas_model v14).
344//              Limit public inputs in groth16::prepare_verifying_key.
345//              Enable address balances, free tier (gasless), and coin reservations on mainnet.
346//              Enables enable_accumulators, enable_address_balance_gas_payments,
347//              enable_authenticated_event_streams, enable_coin_reservation_obj_refs,
348//              enable_object_funds_withdraw, convert_withdrawal_compatibility_ptb_arguments,
349//              split_checkpoints_in_consensus_handler, include_checkpoint_artifacts_digest_in_summary,
350//              and enable_gasless on mainnet to bring it in line with testnet.
351//              Configure mainnet gasless allowlist with stablecoin types and $0.01 minimum
352//              transfer per stable.
353// Version 125: Enable granular_post_execution_checks.
354//              Enable timestamp_based_epoch_close on testnet.
355
356#[derive(Copy, Clone, Debug, Hash, Serialize, Deserialize, PartialEq, Eq, PartialOrd, Ord)]
357pub struct ProtocolVersion(u64);
358
359impl ProtocolVersion {
360    // The minimum and maximum protocol version supported by this binary. Counterintuitively, this constant may
361    // change over time as support for old protocol versions is removed from the source. This
362    // ensures that when a new network (such as a testnet) is created, its genesis committee will
363    // use a protocol version that is actually supported by the binary.
364    pub const MIN: Self = Self(MIN_PROTOCOL_VERSION);
365
366    pub const MAX: Self = Self(MAX_PROTOCOL_VERSION);
367
368    #[cfg(not(msim))]
369    pub const MAX_ALLOWED: Self = Self::MAX;
370
371    // We create one additional "fake" version in simulator builds so that we can test upgrades.
372    #[cfg(msim)]
373    pub const MAX_ALLOWED: Self = Self(MAX_PROTOCOL_VERSION + 1);
374
375    pub fn new(v: u64) -> Self {
376        Self(v)
377    }
378
379    pub const fn as_u64(&self) -> u64 {
380        self.0
381    }
382
383    // For serde deserialization - we don't define a Default impl because there isn't a single
384    // universally appropriate default value.
385    pub fn max() -> Self {
386        Self::MAX
387    }
388
389    pub fn prev(self) -> Self {
390        Self(self.0.checked_sub(1).unwrap())
391    }
392}
393
394impl From<u64> for ProtocolVersion {
395    fn from(v: u64) -> Self {
396        Self::new(v)
397    }
398}
399
400impl std::ops::Sub<u64> for ProtocolVersion {
401    type Output = Self;
402    fn sub(self, rhs: u64) -> Self::Output {
403        Self::new(self.0 - rhs)
404    }
405}
406
407impl std::ops::Add<u64> for ProtocolVersion {
408    type Output = Self;
409    fn add(self, rhs: u64) -> Self::Output {
410        Self::new(self.0 + rhs)
411    }
412}
413
414#[derive(
415    Clone, Serialize, Deserialize, Debug, Default, PartialEq, Copy, PartialOrd, Ord, Eq, ValueEnum,
416)]
417pub enum Chain {
418    Mainnet,
419    Testnet,
420    #[default]
421    Unknown,
422}
423
424impl Chain {
425    pub fn as_str(self) -> &'static str {
426        match self {
427            Chain::Mainnet => "mainnet",
428            Chain::Testnet => "testnet",
429            Chain::Unknown => "unknown",
430        }
431    }
432}
433
434pub struct Error(pub String);
435
436// TODO: There are quite a few non boolean values in the feature flags. We should move them out.
437/// Records on/off feature flags that may vary at each protocol version.
438#[derive(Default, Clone, Serialize, Deserialize, Debug, ProtocolConfigFeatureFlagsGetters)]
439struct FeatureFlags {
440    // Add feature flags here, e.g.:
441    // new_protocol_feature: bool,
442    #[serde(skip_serializing_if = "is_false")]
443    package_upgrades: bool,
444    // If true, validators will commit to the root state digest
445    // in end of epoch checkpoint proposals
446    #[serde(skip_serializing_if = "is_false")]
447    commit_root_state_digest: bool,
448    // Pass epoch start time to advance_epoch safe mode function.
449    #[serde(skip_serializing_if = "is_false")]
450    advance_epoch_start_time_in_safe_mode: bool,
451    // If true, apply the fix to correctly capturing loaded child object versions in execution's
452    // object runtime.
453    #[serde(skip_serializing_if = "is_false")]
454    loaded_child_objects_fixed: bool,
455    // If true, treat missing types in the upgraded modules when creating an upgraded package as a
456    // compatibility error.
457    #[serde(skip_serializing_if = "is_false")]
458    missing_type_is_compatibility_error: bool,
459    // If true, then the scoring decision mechanism will not get disabled when we do have more than
460    // f low scoring authorities, but it will simply flag as low scoring only up to f authorities.
461    #[serde(skip_serializing_if = "is_false")]
462    scoring_decision_with_validity_cutoff: bool,
463
464    // DEPRECATED: this was an ephemeral feature flag only used by consensus handler, which has now
465    // been deployed everywhere.
466    #[serde(skip_serializing_if = "is_false")]
467    consensus_order_end_of_epoch_last: bool,
468
469    // Disallow adding abilities to types during package upgrades.
470    #[serde(skip_serializing_if = "is_false")]
471    disallow_adding_abilities_on_upgrade: bool,
472    // Disables unnecessary invariant check in the Move VM when swapping the value out of a local
473    #[serde(skip_serializing_if = "is_false")]
474    disable_invariant_violation_check_in_swap_loc: bool,
475    // advance to highest supported protocol version at epoch change, instead of the next consecutive
476    // protocol version.
477    #[serde(skip_serializing_if = "is_false")]
478    advance_to_highest_supported_protocol_version: bool,
479    // If true, disallow entry modifiers on entry functions
480    #[serde(skip_serializing_if = "is_false")]
481    ban_entry_init: bool,
482    // If true, hash module bytes individually when calculating package digests for upgrades
483    #[serde(skip_serializing_if = "is_false")]
484    package_digest_hash_module: bool,
485    // If true, disallow changing struct type parameters during package upgrades
486    #[serde(skip_serializing_if = "is_false")]
487    disallow_change_struct_type_params_on_upgrade: bool,
488    // If true, checks no extra bytes in a compiled module
489    #[serde(skip_serializing_if = "is_false")]
490    no_extraneous_module_bytes: bool,
491    // If true, then use the versioned metadata format in narwhal entities.
492    #[serde(skip_serializing_if = "is_false")]
493    narwhal_versioned_metadata: bool,
494
495    // Enable zklogin auth
496    #[serde(skip_serializing_if = "is_false")]
497    zklogin_auth: bool,
498    // How we order transactions coming out of consensus before sending to execution.
499    #[serde(skip_serializing_if = "ConsensusTransactionOrdering::is_none")]
500    consensus_transaction_ordering: ConsensusTransactionOrdering,
501
502    // Previously, the unwrapped_then_deleted field in TransactionEffects makes a distinction between
503    // whether an object has existed in the store previously (i.e. whether there is a tombstone).
504    // Such dependency makes effects generation inefficient, and requires us to include wrapped
505    // tombstone in state root hash.
506    // To prepare for effects V2, with this flag set to true, we simplify the definition of
507    // unwrapped_then_deleted to always include unwrapped then deleted objects,
508    // regardless of their previous state in the store.
509    #[serde(skip_serializing_if = "is_false")]
510    simplified_unwrap_then_delete: bool,
511    // Enable upgraded multisig support
512    #[serde(skip_serializing_if = "is_false")]
513    upgraded_multisig_supported: bool,
514    // If true minimum txn charge is a multiplier of the gas price
515    #[serde(skip_serializing_if = "is_false")]
516    txn_base_cost_as_multiplier: bool,
517
518    // If true, the ability to delete shared objects is in effect
519    #[serde(skip_serializing_if = "is_false")]
520    shared_object_deletion: bool,
521
522    // If true, then the new algorithm for the leader election schedule will be used
523    #[serde(skip_serializing_if = "is_false")]
524    narwhal_new_leader_election_schedule: bool,
525
526    // A list of supported OIDC providers that can be used for zklogin.
527    #[serde(skip_serializing_if = "is_empty")]
528    zklogin_supported_providers: BTreeSet<String>,
529
530    // If true, use the new child object format
531    #[serde(skip_serializing_if = "is_false")]
532    loaded_child_object_format: bool,
533
534    #[serde(skip_serializing_if = "is_false")]
535    enable_jwk_consensus_updates: bool,
536
537    #[serde(skip_serializing_if = "is_false")]
538    end_of_epoch_transaction_supported: bool,
539
540    // Perform simple conservation checks keeping into account out of gas scenarios
541    // while charging for storage.
542    #[serde(skip_serializing_if = "is_false")]
543    simple_conservation_checks: bool,
544
545    // If true, use the new child object format type logging
546    #[serde(skip_serializing_if = "is_false")]
547    loaded_child_object_format_type: bool,
548
549    // Enable receiving sent objects
550    #[serde(skip_serializing_if = "is_false")]
551    receive_objects: bool,
552
553    // If true, include CheckpointDigest in consensus dedup key for checkpoint signatures (V2).
554    #[serde(skip_serializing_if = "is_false")]
555    consensus_checkpoint_signature_key_includes_digest: bool,
556
557    // Enable random beacon protocol
558    #[serde(skip_serializing_if = "is_false")]
559    random_beacon: bool,
560
561    // Enable bridge protocol
562    #[serde(skip_serializing_if = "is_false")]
563    bridge: bool,
564
565    #[serde(skip_serializing_if = "is_false")]
566    enable_effects_v2: bool,
567
568    // If true, then use CertificateV2 in narwhal.
569    #[serde(skip_serializing_if = "is_false")]
570    narwhal_certificate_v2: bool,
571
572    // If true, allow verify with legacy zklogin address
573    #[serde(skip_serializing_if = "is_false")]
574    verify_legacy_zklogin_address: bool,
575
576    // Enable throughput aware consensus submission
577    #[serde(skip_serializing_if = "is_false")]
578    throughput_aware_consensus_submission: bool,
579
580    // If true, recompute has_public_transfer from the type instead of what is stored in the object
581    #[serde(skip_serializing_if = "is_false")]
582    recompute_has_public_transfer_in_execution: bool,
583
584    // If true, multisig containing zkLogin sig is accepted.
585    #[serde(skip_serializing_if = "is_false")]
586    accept_zklogin_in_multisig: bool,
587
588    // If true, multisig containing passkey sig is accepted.
589    #[serde(skip_serializing_if = "is_false")]
590    accept_passkey_in_multisig: bool,
591
592    // If true, additional zkLogin public identifier structure is validated.
593    #[serde(skip_serializing_if = "is_false")]
594    validate_zklogin_public_identifier: bool,
595
596    // If true, consensus prologue transaction also includes the consensus output digest.
597    // It can be used to detect consensus output folk.
598    #[serde(skip_serializing_if = "is_false")]
599    include_consensus_digest_in_prologue: bool,
600
601    // If true, use the hardened OTW check
602    #[serde(skip_serializing_if = "is_false")]
603    hardened_otw_check: bool,
604
605    // If true allow calling receiving_object_id function
606    #[serde(skip_serializing_if = "is_false")]
607    allow_receiving_object_id: bool,
608
609    // Enable the poseidon hash function
610    #[serde(skip_serializing_if = "is_false")]
611    enable_poseidon: bool,
612
613    // If true, enable the coin deny list.
614    #[serde(skip_serializing_if = "is_false")]
615    enable_coin_deny_list: bool,
616
617    // Enable native functions for group operations.
618    #[serde(skip_serializing_if = "is_false")]
619    enable_group_ops_native_functions: bool,
620
621    // Enable native function for msm.
622    #[serde(skip_serializing_if = "is_false")]
623    enable_group_ops_native_function_msm: bool,
624
625    // Enable group operations for Ristretto255
626    #[serde(skip_serializing_if = "is_false")]
627    enable_ristretto255_group_ops: bool,
628
629    // Enable native functions for group operations.
630    #[serde(skip_serializing_if = "is_false")]
631    enable_verify_bulletproofs_ristretto255: bool,
632
633    // Enable nitro attestation.
634    #[serde(skip_serializing_if = "is_false")]
635    enable_nitro_attestation: bool,
636
637    // Enable upgraded parsing of nitro attestation that interprets pcrs as a map.
638    #[serde(skip_serializing_if = "is_false")]
639    enable_nitro_attestation_upgraded_parsing: bool,
640
641    // Enable upgraded parsing of nitro attestation containing all nonzero PCRs.
642    #[serde(skip_serializing_if = "is_false")]
643    enable_nitro_attestation_all_nonzero_pcrs_parsing: bool,
644
645    // Enable upgraded parsing of nitro attestation to always include required PCRs, even when all zeros.
646    #[serde(skip_serializing_if = "is_false")]
647    enable_nitro_attestation_always_include_required_pcrs_parsing: bool,
648
649    // Reject functions with mutable Random.
650    #[serde(skip_serializing_if = "is_false")]
651    reject_mutable_random_on_entry_functions: bool,
652
653    // Controls the behavior of per object congestion control in consensus handler.
654    #[serde(skip_serializing_if = "PerObjectCongestionControlMode::is_none")]
655    per_object_congestion_control_mode: PerObjectCongestionControlMode,
656
657    // The consensus protocol to be used for the epoch.
658    #[serde(skip_serializing_if = "ConsensusChoice::is_narwhal")]
659    consensus_choice: ConsensusChoice,
660
661    // Consensus network to use.
662    #[serde(skip_serializing_if = "ConsensusNetwork::is_anemo")]
663    consensus_network: ConsensusNetwork,
664
665    // If true, use the correct (<=) comparison for max_gas_payment_objects instead of (<)
666    #[serde(skip_serializing_if = "is_false")]
667    correct_gas_payment_limit_check: bool,
668
669    // Set the upper bound allowed for max_epoch in zklogin signature.
670    #[serde(skip_serializing_if = "Option::is_none")]
671    zklogin_max_epoch_upper_bound_delta: Option<u64>,
672
673    // Controls leader scoring & schedule change in Mysticeti consensus.
674    #[serde(skip_serializing_if = "is_false")]
675    mysticeti_leader_scoring_and_schedule: bool,
676
677    // Enable resharing of shared objects using the same initial shared version
678    #[serde(skip_serializing_if = "is_false")]
679    reshare_at_same_initial_version: bool,
680
681    // Resolve Move abort locations to the package id instead of the runtime module ID.
682    #[serde(skip_serializing_if = "is_false")]
683    resolve_abort_locations_to_package_id: bool,
684
685    // Enables the use of the Mysticeti committed sub dag digest to the `ConsensusCommitInfo` in checkpoints.
686    // When disabled the default digest is used instead. It's important to have this guarded behind
687    // a flag as it will lead to checkpoint forks.
688    #[serde(skip_serializing_if = "is_false")]
689    mysticeti_use_committed_subdag_digest: bool,
690
691    // Enable VDF
692    #[serde(skip_serializing_if = "is_false")]
693    enable_vdf: bool,
694
695    // Controls whether consensus handler should record consensus determined shared object version
696    // assignments in consensus commit prologue transaction.
697    // The purpose of doing this is to enable replaying transaction without transaction effects.
698    // V2 also records initial shared versions for consensus objects.
699    #[serde(skip_serializing_if = "is_false")]
700    record_consensus_determined_version_assignments_in_prologue: bool,
701    #[serde(skip_serializing_if = "is_false")]
702    record_consensus_determined_version_assignments_in_prologue_v2: bool,
703
704    // Run verification of framework upgrades using a new/fresh VM.
705    #[serde(skip_serializing_if = "is_false")]
706    fresh_vm_on_framework_upgrade: bool,
707
708    // When set to true, the consensus commit prologue transaction will be placed first
709    // in a consensus commit in checkpoints.
710    // If a checkpoint contains multiple consensus commit, say [cm1][cm2]. The each commit's
711    // consensus commit prologue will be the first transaction in each segment:
712    //     [ccp1, rest cm1][ccp2, rest cm2]
713    // The reason to prepose the prologue transaction is to provide information for transaction
714    // cancellation.
715    #[serde(skip_serializing_if = "is_false")]
716    prepend_prologue_tx_in_consensus_commit_in_checkpoints: bool,
717
718    // Set number of leaders per round for Mysticeti commits.
719    #[serde(skip_serializing_if = "Option::is_none")]
720    mysticeti_num_leaders_per_round: Option<usize>,
721
722    // Enable Soft Bundle (SIP-19).
723    #[serde(skip_serializing_if = "is_false")]
724    soft_bundle: bool,
725
726    // If true, enable the coin deny list V2.
727    #[serde(skip_serializing_if = "is_false")]
728    enable_coin_deny_list_v2: bool,
729
730    // Enable passkey auth (SIP-9)
731    #[serde(skip_serializing_if = "is_false")]
732    passkey_auth: bool,
733
734    // Use AuthorityCapabilitiesV2
735    #[serde(skip_serializing_if = "is_false")]
736    authority_capabilities_v2: bool,
737
738    // Rethrow type layout errors during serialization instead of trying to convert them.
739    #[serde(skip_serializing_if = "is_false")]
740    rethrow_serialization_type_layout_errors: bool,
741
742    // Use distributed vote leader scoring strategy in consensus.
743    #[serde(skip_serializing_if = "is_false")]
744    consensus_distributed_vote_scoring_strategy: bool,
745
746    // Probe rounds received by peers from every authority.
747    #[serde(skip_serializing_if = "is_false")]
748    consensus_round_prober: bool,
749
750    // Validate identifier inputs separately
751    #[serde(skip_serializing_if = "is_false")]
752    validate_identifier_inputs: bool,
753
754    // Disallow self identifier
755    #[serde(skip_serializing_if = "is_false")]
756    disallow_self_identifier: bool,
757
758    // Enables Mysticeti fastpath.
759    #[serde(skip_serializing_if = "is_false")]
760    mysticeti_fastpath: bool,
761
762    // If true, disable pre-consensus locking for owned objects.
763    // All transactions go through consensus, and owned object conflict detection
764    // happens post-consensus via lock acquisition.
765    #[serde(skip_serializing_if = "is_false")]
766    disable_preconsensus_locking: bool,
767
768    // Makes the event's sending module version-aware.
769    #[serde(skip_serializing_if = "is_false")]
770    relocate_event_module: bool,
771
772    // Enable uncompressed group elements in BLS123-81 G1
773    #[serde(skip_serializing_if = "is_false")]
774    uncompressed_g1_group_elements: bool,
775
776    #[serde(skip_serializing_if = "is_false")]
777    disallow_new_modules_in_deps_only_packages: bool,
778
779    // Use smart ancestor selection in consensus.
780    #[serde(skip_serializing_if = "is_false")]
781    consensus_smart_ancestor_selection: bool,
782
783    // Probe accepted rounds in round prober.
784    #[serde(skip_serializing_if = "is_false")]
785    consensus_round_prober_probe_accepted_rounds: bool,
786
787    // Enable v2 native charging for natives.
788    #[serde(skip_serializing_if = "is_false")]
789    native_charging_v2: bool,
790
791    // Enables the new logic for collecting the subdag in the consensus linearizer. The new logic does not stop the recursion at the highest
792    // committed round for each authority, but allows to commit uncommitted blocks up to gc round (excluded) for that authority.
793    #[serde(skip_serializing_if = "is_false")]
794    consensus_linearize_subdag_v2: bool,
795
796    // Properly convert certain type argument errors in the execution layer.
797    #[serde(skip_serializing_if = "is_false")]
798    convert_type_argument_error: bool,
799
800    // Variants count as nodes
801    #[serde(skip_serializing_if = "is_false")]
802    variant_nodes: bool,
803
804    // If true, enable zstd compression for consensus tonic network.
805    #[serde(skip_serializing_if = "is_false")]
806    consensus_zstd_compression: bool,
807
808    // If true, enables the optimizations for child object mutations, removing unnecessary mutations
809    #[serde(skip_serializing_if = "is_false")]
810    minimize_child_object_mutations: bool,
811
812    // If true, record the additional state digest in the consensus commit prologue.
813    #[serde(skip_serializing_if = "is_false")]
814    record_additional_state_digest_in_prologue: bool,
815
816    // If true, enable `TxContext` Move API to go native.
817    #[serde(skip_serializing_if = "is_false")]
818    move_native_context: bool,
819
820    // If true, then it (1) will not enforce monotonicity checks for a block's ancestors and (2) calculates the commit's timestamp based on the
821    // weighted by stake median timestamp of the leader's ancestors.
822    #[serde(skip_serializing_if = "is_false")]
823    consensus_median_based_commit_timestamp: bool,
824
825    // If true, enables the normalization of PTB arguments but does not yet enable splatting
826    // `Result`s of length not equal to 1
827    #[serde(skip_serializing_if = "is_false")]
828    normalize_ptb_arguments: bool,
829
830    // If true, enabled batched block sync in consensus.
831    #[serde(skip_serializing_if = "is_false")]
832    consensus_batched_block_sync: bool,
833
834    // If true, enforces checkpoint timestamps are non-decreasing.
835    #[serde(skip_serializing_if = "is_false")]
836    enforce_checkpoint_timestamp_monotonicity: bool,
837
838    // If true, enables better errors and bounds for max ptb values
839    #[serde(skip_serializing_if = "is_false")]
840    max_ptb_value_size_v2: bool,
841
842    // If true, resolves all type input ids to be defining ID based in the adapter
843    #[serde(skip_serializing_if = "is_false")]
844    resolve_type_input_ids_to_defining_id: bool,
845
846    // Enable native function for party transfer
847    #[serde(skip_serializing_if = "is_false")]
848    enable_party_transfer: bool,
849
850    // Allow objects created or mutated in system transactions to exceed the max object size limit.
851    #[serde(skip_serializing_if = "is_false")]
852    allow_unbounded_system_objects: bool,
853
854    // Signifies the cut-over of using type tags instead of `Type`s in the object runtime.
855    #[serde(skip_serializing_if = "is_false")]
856    type_tags_in_object_runtime: bool,
857
858    // Enable accumulators
859    #[serde(skip_serializing_if = "is_false")]
860    enable_accumulators: bool,
861
862    // Enable coin reservation
863    #[serde(skip_serializing_if = "is_false")]
864    enable_coin_reservation_obj_refs: bool,
865
866    // If true, create the root accumulator object in the change epoch transaction.
867    // This must be enabled and shipped before `enable_accumulators` is set to true.
868    #[serde(skip_serializing_if = "is_false")]
869    create_root_accumulator_object: bool,
870
871    // Enable authenticated event streams
872    #[serde(skip_serializing_if = "is_false")]
873    enable_authenticated_event_streams: bool,
874
875    // Enable address balance gas payments
876    #[serde(skip_serializing_if = "is_false")]
877    enable_address_balance_gas_payments: bool,
878
879    // Validate gas price >= RGP at signing for address balance gas payments
880    #[serde(skip_serializing_if = "is_false")]
881    address_balance_gas_check_rgp_at_signing: bool,
882
883    #[serde(skip_serializing_if = "is_false")]
884    address_balance_gas_reject_gas_coin_arg: bool,
885
886    // Enable multi-epoch transaction expiration (max 1 epoch difference)
887    #[serde(skip_serializing_if = "is_false")]
888    enable_multi_epoch_transaction_expiration: bool,
889
890    // Relax ValidDuring expiration requirement for transactions with owned inputs
891    #[serde(skip_serializing_if = "is_false")]
892    relax_valid_during_for_owned_inputs: bool,
893
894    // Enable statically type checked ptb execution
895    #[serde(skip_serializing_if = "is_false")]
896    enable_ptb_execution_v2: bool,
897
898    // Provide better type resolution errors in the adapter.
899    #[serde(skip_serializing_if = "is_false")]
900    better_adapter_type_resolution_errors: bool,
901
902    // If true, record the time estimate processed in the consensus commit prologue.
903    #[serde(skip_serializing_if = "is_false")]
904    record_time_estimate_processed: bool,
905
906    // If true enable additional linkage checks.
907    #[serde(skip_serializing_if = "is_false")]
908    dependency_linkage_error: bool,
909
910    // If true enable additional multisig checks.
911    #[serde(skip_serializing_if = "is_false")]
912    additional_multisig_checks: bool,
913
914    // If true, ignore execution time observations after certs are closed.
915    #[serde(skip_serializing_if = "is_false")]
916    ignore_execution_time_observations_after_certs_closed: bool,
917
918    // If true use `debug_fatal` to report invariant violations.
919    // `debug_fatal` panics in debug builds and breaks tests/behavior based on older
920    // protocol versions (see make_vec_non_existent_type_v71.move)
921    #[serde(skip_serializing_if = "is_false")]
922    debug_fatal_on_move_invariant_violation: bool,
923
924    // DO NOT ENABLE THIS FOR PRODUCTION NETWORKS. used for testing only.
925    // Allow private accumulator entrypoints
926    #[serde(skip_serializing_if = "is_false")]
927    allow_private_accumulator_entrypoints: bool,
928
929    // If true, include indirect state in the additional consensus digest.
930    #[serde(skip_serializing_if = "is_false")]
931    additional_consensus_digest_indirect_state: bool,
932
933    // Check for `init` for new modules to a package on upgrade.
934    #[serde(skip_serializing_if = "is_false")]
935    check_for_init_during_upgrade: bool,
936
937    // Check shared object transfer restrictions per command.
938    #[serde(skip_serializing_if = "is_false")]
939    per_command_shared_object_transfer_rules: bool,
940
941    // Enable including checkpoint artifacts digest in the summary.
942    #[serde(skip_serializing_if = "is_false")]
943    include_checkpoint_artifacts_digest_in_summary: bool,
944
945    // If true, use MFP txns in load initial object debts.
946    #[serde(skip_serializing_if = "is_false")]
947    use_mfp_txns_in_load_initial_object_debts: bool,
948
949    // If true, cancel randomness-using txns when DKG has failed *before* doing other congestion checks.
950    #[serde(skip_serializing_if = "is_false")]
951    cancel_for_failed_dkg_early: bool,
952
953    // Enable coin registry protocol
954    #[serde(skip_serializing_if = "is_false")]
955    enable_coin_registry: bool,
956
957    // Use abstract size in the object runtime instead the legacy value size.
958    #[serde(skip_serializing_if = "is_false")]
959    abstract_size_in_object_runtime: bool,
960
961    // If true charge for loads into the cache (i.e., fetches from storage) in the object runtime.
962    #[serde(skip_serializing_if = "is_false")]
963    object_runtime_charge_cache_load_gas: bool,
964
965    // If true, perform additional borrow checks
966    #[serde(skip_serializing_if = "is_false")]
967    additional_borrow_checks: bool,
968
969    // If true, use the new commit handler.
970    #[serde(skip_serializing_if = "is_false")]
971    use_new_commit_handler: bool,
972
973    // If true return a better error message when we encounter a loader error.
974    #[serde(skip_serializing_if = "is_false")]
975    better_loader_errors: bool,
976
977    // If true generate layouts for dynamic fields
978    #[serde(skip_serializing_if = "is_false")]
979    generate_df_type_layouts: bool,
980
981    // If true, allow Move functions called in PTBs to return references
982    #[serde(skip_serializing_if = "is_false")]
983    allow_references_in_ptbs: bool,
984
985    // Enable display registry protocol
986    #[serde(skip_serializing_if = "is_false")]
987    enable_display_registry: bool,
988
989    // If true, enable private generics verifier v2
990    #[serde(skip_serializing_if = "is_false")]
991    private_generics_verifier_v2: bool,
992
993    // If true, deprecate global storage ops during Move module deserialization
994    #[serde(skip_serializing_if = "is_false")]
995    deprecate_global_storage_ops_during_deserialization: bool,
996
997    // If true, enable non-exclusive writes for user transactions.
998    // DO NOT ENABLE outside of the transaction test runner.
999    #[serde(skip_serializing_if = "is_false")]
1000    enable_non_exclusive_writes: bool,
1001
1002    // If true, deprecate global storage ops everywhere.
1003    #[serde(skip_serializing_if = "is_false")]
1004    deprecate_global_storage_ops: bool,
1005
1006    // If true, normalize depth formula to not be empty for zero depth.
1007    #[serde(skip_serializing_if = "is_false")]
1008    normalize_depth_formula: bool,
1009
1010    // If true, skip GC'ed accept votes in CommitFinalizer.
1011    #[serde(skip_serializing_if = "is_false")]
1012    consensus_skip_gced_accept_votes: bool,
1013
1014    // If true, include cancelled randomness txns in the consensus commit prologue.
1015    #[serde(skip_serializing_if = "is_false")]
1016    include_cancelled_randomness_txns_in_prologue: bool,
1017
1018    // Enables address aliases.
1019    #[serde(skip_serializing_if = "is_false")]
1020    address_aliases: bool,
1021
1022    // Corrects signature-to-signer mapping in CheckpointContentsV2.
1023    // TODO: remove old code and deprecate once in mainnet.
1024    #[serde(skip_serializing_if = "is_false")]
1025    fix_checkpoint_signature_mapping: bool,
1026
1027    // If true, enable object funds withdraw.
1028    #[serde(skip_serializing_if = "is_false")]
1029    enable_object_funds_withdraw: bool,
1030
1031    // If true, skip GC'ed blocks in direct finalization.
1032    #[serde(skip_serializing_if = "is_false")]
1033    consensus_skip_gced_blocks_in_direct_finalization: bool,
1034
1035    // If true, uses a new rounding mechanism for gas calculations, replacing the step-based one
1036    #[serde(skip_serializing_if = "is_false")]
1037    gas_rounding_halve_digits: bool,
1038
1039    // If true, enable tx contexts in all argument positions
1040    #[serde(skip_serializing_if = "is_false")]
1041    flexible_tx_context_positions: bool,
1042
1043    // If true, disable entry point signature check.
1044    #[serde(skip_serializing_if = "is_false")]
1045    disable_entry_point_signature_check: bool,
1046
1047    // If true, convert withdrawal compatibility PTB arguments to coins at the start of the PTB.
1048    #[serde(skip_serializing_if = "is_false")]
1049    convert_withdrawal_compatibility_ptb_arguments: bool,
1050
1051    // If true, additional restrictions for hot or not entry functions are enforced.
1052    #[serde(skip_serializing_if = "is_false")]
1053    restrict_hot_or_not_entry_functions: bool,
1054
1055    // If true, split checkpoints in consensus handler.
1056    #[serde(skip_serializing_if = "is_false")]
1057    split_checkpoints_in_consensus_handler: bool,
1058
1059    // If true, always accept committed system transactions.
1060    #[serde(skip_serializing_if = "is_false")]
1061    consensus_always_accept_system_transactions: bool,
1062
1063    // If true perform consistent verification of metadata
1064    #[serde(skip_serializing_if = "is_false")]
1065    validator_metadata_verify_v2: bool,
1066
1067    // If true, defer transactions with unpaid consensus amplification
1068    // (where duplicate count exceeds gas_price / RGP + 1)
1069    #[serde(skip_serializing_if = "is_false")]
1070    defer_unpaid_amplification: bool,
1071
1072    #[serde(skip_serializing_if = "is_false")]
1073    randomize_checkpoint_tx_limit_in_tests: bool,
1074
1075    // If true, mark the gas coin as uninitialized in drop safety when there is no gas coin.
1076    #[serde(skip_serializing_if = "is_false")]
1077    gasless_transaction_drop_safety: bool,
1078
1079    // When split-checkpoints enabled, merge randomness and non-randomness schedulables together.
1080    #[serde(skip_serializing_if = "is_false")]
1081    merge_randomness_into_checkpoint: bool,
1082
1083    // If true, use coin party owner information.
1084    #[serde(skip_serializing_if = "is_false")]
1085    use_coin_party_owner: bool,
1086
1087    #[serde(skip_serializing_if = "is_false")]
1088    enable_gasless: bool,
1089
1090    #[serde(skip_serializing_if = "is_false")]
1091    gasless_verify_remaining_balance: bool,
1092
1093    #[serde(skip_serializing_if = "is_false")]
1094    disallow_jump_orphans: bool,
1095
1096    // If true, return early on type mismatch in receive_object.
1097    #[serde(skip_serializing_if = "is_false")]
1098    early_return_receive_object_mismatched_type: bool,
1099
1100    // If true, use consensus commit timestamps to determine epoch close instead of EndOfPublish voting.
1101    // Each validator transitions from AcceptAllCerts to RejectAllCerts when the consensus commit
1102    // timestamp exceeds the reconfiguration timestamp. EndOfPublish quorum still works as a
1103    // fallback for manual epoch close.
1104    #[serde(skip_serializing_if = "is_false")]
1105    timestamp_based_epoch_close: bool,
1106
1107    // If true, groth16::prepare_verifying_key checks that the verifying key has no more than
1108    // MAX_PUBLIC_INPUTS public inputs.
1109    #[serde(skip_serializing_if = "is_false")]
1110    limit_groth16_pvk_inputs: bool,
1111
1112    // If true, the funds-accumulator address-balance change invariant
1113    // (`TemporaryStore::check_address_balance_changes`) is enforced as a consensus check —
1114    // violations abort the tx via the conservation-recovery flow. When false, the check still
1115    // runs but a violation panics so unexpected violations surface during rollout.
1116    #[serde(skip_serializing_if = "is_false")]
1117    enforce_address_balance_change_invariant: bool,
1118
1119    // Enables more granular post-execution checks.
1120    #[serde(skip_serializing_if = "is_false")]
1121    granular_post_execution_checks: bool,
1122}
1123
1124fn is_false(b: &bool) -> bool {
1125    !b
1126}
1127
1128fn is_empty(b: &BTreeSet<String>) -> bool {
1129    b.is_empty()
1130}
1131
1132fn is_zero(val: &u64) -> bool {
1133    *val == 0
1134}
1135
1136/// Ordering mechanism for transactions in one Narwhal consensus output.
1137#[derive(Default, Copy, Clone, PartialEq, Eq, Serialize, Deserialize, Debug)]
1138pub enum ConsensusTransactionOrdering {
1139    /// No ordering. Transactions are processed in the order they appear in the consensus output.
1140    #[default]
1141    None,
1142    /// Order transactions by gas price, highest first.
1143    ByGasPrice,
1144}
1145
1146impl ConsensusTransactionOrdering {
1147    pub fn is_none(&self) -> bool {
1148        matches!(self, ConsensusTransactionOrdering::None)
1149    }
1150}
1151
1152#[derive(Default, Copy, Clone, PartialEq, Eq, Serialize, Deserialize, Debug)]
1153pub struct ExecutionTimeEstimateParams {
1154    // Targeted per-object utilization as an integer percentage (1-100).
1155    pub target_utilization: u64,
1156    // Schedule up to this much extra work (in microseconds) per object,
1157    // but don't allow more than a single transaction to exceed this
1158    // burst limit.
1159    pub allowed_txn_cost_overage_burst_limit_us: u64,
1160
1161    // For separate budget for randomness-using tx, the above limits are
1162    // used with this integer-percentage scaling factor (1-100).
1163    pub randomness_scalar: u64,
1164
1165    // Absolute maximum allowed transaction duration estimate (in microseconds).
1166    pub max_estimate_us: u64,
1167
1168    // Number of the final checkpoints in an epoch whose observations should be
1169    // stored for use in the next epoch.
1170    pub stored_observations_num_included_checkpoints: u64,
1171
1172    // Absolute limit on the number of saved observations at end of epoch.
1173    pub stored_observations_limit: u64,
1174
1175    // Requires observations from at least this amount of stake in order to use
1176    // observation-based execution time estimates instead of the default.
1177    #[serde(skip_serializing_if = "is_zero")]
1178    pub stake_weighted_median_threshold: u64,
1179
1180    // For backwards compatibility with old behavior we use a zero default duration when adding
1181    // new execution time observation keys and a zero generation when loading stored observations.
1182    // This can be removed once set to "true" on mainnet.
1183    #[serde(skip_serializing_if = "is_false")]
1184    pub default_none_duration_for_new_keys: bool,
1185
1186    // Number of observations per chunk. When None, chunking is disabled.
1187    #[serde(skip_serializing_if = "Option::is_none")]
1188    pub observations_chunk_size: Option<u64>,
1189}
1190
1191// The config for per object congestion control in consensus handler.
1192#[derive(Default, Copy, Clone, PartialEq, Eq, Serialize, Deserialize, Debug)]
1193pub enum PerObjectCongestionControlMode {
1194    #[default]
1195    None, // Deprecated.
1196    TotalGasBudget,                                     // Deprecated.
1197    TotalTxCount,                                       // Deprecated.
1198    TotalGasBudgetWithCap,                              // Deprecated.
1199    ExecutionTimeEstimate(ExecutionTimeEstimateParams), // Use execution time estimate as execution cost.
1200}
1201
1202impl PerObjectCongestionControlMode {
1203    pub fn is_none(&self) -> bool {
1204        matches!(self, PerObjectCongestionControlMode::None)
1205    }
1206}
1207
1208// Configuration options for consensus algorithm.
1209#[derive(Default, Copy, Clone, PartialEq, Eq, Serialize, Deserialize, Debug)]
1210pub enum ConsensusChoice {
1211    #[default]
1212    Narwhal,
1213    SwapEachEpoch,
1214    Mysticeti,
1215}
1216
1217impl ConsensusChoice {
1218    pub fn is_narwhal(&self) -> bool {
1219        matches!(self, ConsensusChoice::Narwhal)
1220    }
1221}
1222
1223// Configuration options for consensus network.
1224#[derive(Default, Copy, Clone, PartialEq, Eq, Serialize, Deserialize, Debug)]
1225pub enum ConsensusNetwork {
1226    #[default]
1227    Anemo,
1228    Tonic,
1229}
1230
1231impl ConsensusNetwork {
1232    pub fn is_anemo(&self) -> bool {
1233        matches!(self, ConsensusNetwork::Anemo)
1234    }
1235}
1236
1237/// Constants that change the behavior of the protocol.
1238///
1239/// The value of each constant here must be fixed for a given protocol version. To change the value
1240/// of a constant, advance the protocol version, and add support for it in `get_for_version` under
1241/// the new version number.
1242/// (below).
1243///
1244/// To add a new field to this struct, use the following procedure:
1245/// - Advance the protocol version.
1246/// - Add the field as a private `Option<T>` to the struct.
1247/// - Initialize the field to `None` in prior protocol versions.
1248/// - Initialize the field to `Some(val)` for your new protocol version.
1249/// - Add a public getter that simply unwraps the field.
1250/// - Two public getters of the form `field(&self) -> field_type`
1251///     and `field_as_option(&self) -> Option<field_type>` will be automatically generated for you.
1252/// Example for a field: `new_constant: Option<u64>`
1253/// ```rust,ignore
1254///      pub fn new_constant(&self) -> u64 {
1255///         self.new_constant.expect(Self::CONSTANT_ERR_MSG)
1256///     }
1257///      pub fn new_constant_as_option(&self) -> Option<u64> {
1258///         self.new_constant.expect(Self::CONSTANT_ERR_MSG)
1259///     }
1260/// ```
1261/// With `pub fn new_constant(&self) -> u64`, if the constant is accessed in a protocol version
1262/// in which it is not defined, the validator will crash. (Crashing is necessary because
1263/// this type of error would almost always result in forking if not prevented here).
1264/// If you don't want the validator to crash, you can use the
1265/// `pub fn new_constant_as_option(&self) -> Option<u64>` getter, which will
1266/// return `None` if the field is not defined at that version.
1267/// - If you want a customized getter, you can add a method in the impl.
1268#[skip_serializing_none]
1269#[derive(Clone, Serialize, Debug, ProtocolConfigAccessors, ProtocolConfigOverride)]
1270pub struct ProtocolConfig {
1271    pub version: ProtocolVersion,
1272
1273    feature_flags: FeatureFlags,
1274
1275    // ==== Transaction input limits ====
1276    /// Maximum serialized size of a transaction (in bytes).
1277    max_tx_size_bytes: Option<u64>,
1278
1279    /// Maximum number of input objects to a transaction. Enforced by the transaction input checker
1280    max_input_objects: Option<u64>,
1281
1282    /// Max size of objects a transaction can write to disk after completion. Enforce by the Sui adapter.
1283    /// This is the sum of the serialized size of all objects written to disk.
1284    /// The max size of individual objects on the other hand is `max_move_object_size`.
1285    max_size_written_objects: Option<u64>,
1286    /// Max size of objects a system transaction can write to disk after completion. Enforce by the Sui adapter.
1287    /// Similar to `max_size_written_objects` but for system transactions.
1288    max_size_written_objects_system_tx: Option<u64>,
1289
1290    /// Maximum size of serialized transaction effects.
1291    max_serialized_tx_effects_size_bytes: Option<u64>,
1292
1293    /// Maximum size of serialized transaction effects for system transactions.
1294    max_serialized_tx_effects_size_bytes_system_tx: Option<u64>,
1295
1296    /// Maximum number of gas payment objects for a transaction.
1297    max_gas_payment_objects: Option<u32>,
1298
1299    /// Maximum number of modules in a Publish transaction.
1300    max_modules_in_publish: Option<u32>,
1301
1302    /// Maximum number of transitive dependencies in a package when publishing.
1303    max_package_dependencies: Option<u32>,
1304
1305    /// Maximum number of arguments in a move call or a ProgrammableTransaction's
1306    /// TransferObjects command.
1307    max_arguments: Option<u32>,
1308
1309    /// Maximum number of total type arguments, computed recursively.
1310    max_type_arguments: Option<u32>,
1311
1312    /// Maximum depth of an individual type argument.
1313    max_type_argument_depth: Option<u32>,
1314
1315    /// Maximum size of a Pure CallArg.
1316    max_pure_argument_size: Option<u32>,
1317
1318    /// Maximum number of Commands in a ProgrammableTransaction.
1319    max_programmable_tx_commands: Option<u32>,
1320
1321    // ==== Move VM, Move bytecode verifier, and execution limits ===
1322    /// Maximum Move bytecode version the VM understands. All older versions are accepted.
1323    move_binary_format_version: Option<u32>,
1324    min_move_binary_format_version: Option<u32>,
1325
1326    /// Configuration controlling binary tables size.
1327    binary_module_handles: Option<u16>,
1328    binary_struct_handles: Option<u16>,
1329    binary_function_handles: Option<u16>,
1330    binary_function_instantiations: Option<u16>,
1331    binary_signatures: Option<u16>,
1332    binary_constant_pool: Option<u16>,
1333    binary_identifiers: Option<u16>,
1334    binary_address_identifiers: Option<u16>,
1335    binary_struct_defs: Option<u16>,
1336    binary_struct_def_instantiations: Option<u16>,
1337    binary_function_defs: Option<u16>,
1338    binary_field_handles: Option<u16>,
1339    binary_field_instantiations: Option<u16>,
1340    binary_friend_decls: Option<u16>,
1341    binary_enum_defs: Option<u16>,
1342    binary_enum_def_instantiations: Option<u16>,
1343    binary_variant_handles: Option<u16>,
1344    binary_variant_instantiation_handles: Option<u16>,
1345
1346    /// Maximum size of the `contents` part of an object, in bytes. Enforced by the Sui adapter when effects are produced.
1347    max_move_object_size: Option<u64>,
1348
1349    // TODO: Option<increase to 500 KB. currently, publishing a package > 500 KB exceeds the max computation gas cost
1350    /// Maximum size of a Move package object, in bytes. Enforced by the Sui adapter at the end of a publish transaction.
1351    max_move_package_size: Option<u64>,
1352
1353    /// Max number of publish or upgrade commands allowed in a programmable transaction block.
1354    max_publish_or_upgrade_per_ptb: Option<u64>,
1355
1356    /// Maximum gas budget in MIST that a transaction can use.
1357    max_tx_gas: Option<u64>,
1358
1359    /// Maximum amount of the proposed gas price in MIST (defined in the transaction).
1360    max_gas_price: Option<u64>,
1361
1362    /// For aborted txns, we cap the gas price at a factor of RGP. This lowers risk of setting higher priority gas price
1363    /// if there's a chance the txn will abort.
1364    max_gas_price_rgp_factor_for_aborted_transactions: Option<u64>,
1365
1366    /// The max computation bucket for gas. This is the max that can be charged for computation.
1367    max_gas_computation_bucket: Option<u64>,
1368
1369    // Define the value used to round up computation gas charges
1370    gas_rounding_step: Option<u64>,
1371
1372    /// Maximum number of nested loops. Enforced by the Move bytecode verifier.
1373    max_loop_depth: Option<u64>,
1374
1375    /// Maximum number of type arguments that can be bound to generic type parameters. Enforced by the Move bytecode verifier.
1376    max_generic_instantiation_length: Option<u64>,
1377
1378    /// Maximum number of parameters that a Move function can have. Enforced by the Move bytecode verifier.
1379    max_function_parameters: Option<u64>,
1380
1381    /// Maximum number of basic blocks that a Move function can have. Enforced by the Move bytecode verifier.
1382    max_basic_blocks: Option<u64>,
1383
1384    /// Maximum stack size value. Enforced by the Move bytecode verifier.
1385    max_value_stack_size: Option<u64>,
1386
1387    /// Maximum number of "type nodes", a metric for how big a SignatureToken will be when expanded into a fully qualified type. Enforced by the Move bytecode verifier.
1388    max_type_nodes: Option<u64>,
1389
1390    /// Maximum number of push instructions in one function. Enforced by the Move bytecode verifier.
1391    max_push_size: Option<u64>,
1392
1393    /// Maximum number of struct definitions in a module. Enforced by the Move bytecode verifier.
1394    max_struct_definitions: Option<u64>,
1395
1396    /// Maximum number of function definitions in a module. Enforced by the Move bytecode verifier.
1397    max_function_definitions: Option<u64>,
1398
1399    /// Maximum number of fields allowed in a struct definition. Enforced by the Move bytecode verifier.
1400    max_fields_in_struct: Option<u64>,
1401
1402    /// Maximum dependency depth. Enforced by the Move linker when loading dependent modules.
1403    max_dependency_depth: Option<u64>,
1404
1405    /// Maximum number of Move events that a single transaction can emit. Enforced by the VM during execution.
1406    max_num_event_emit: Option<u64>,
1407
1408    /// Maximum number of new IDs that a single transaction can create. Enforced by the VM during execution.
1409    max_num_new_move_object_ids: Option<u64>,
1410
1411    /// Maximum number of new IDs that a single system transaction can create. Enforced by the VM during execution.
1412    max_num_new_move_object_ids_system_tx: Option<u64>,
1413
1414    /// Maximum number of IDs that a single transaction can delete. Enforced by the VM during execution.
1415    max_num_deleted_move_object_ids: Option<u64>,
1416
1417    /// Maximum number of IDs that a single system transaction can delete. Enforced by the VM during execution.
1418    max_num_deleted_move_object_ids_system_tx: Option<u64>,
1419
1420    /// Maximum number of IDs that a single transaction can transfer. Enforced by the VM during execution.
1421    max_num_transferred_move_object_ids: Option<u64>,
1422
1423    /// Maximum number of IDs that a single system transaction can transfer. Enforced by the VM during execution.
1424    max_num_transferred_move_object_ids_system_tx: Option<u64>,
1425
1426    /// Maximum size of a Move user event. Enforced by the VM during execution.
1427    max_event_emit_size: Option<u64>,
1428
1429    /// Maximum size of a Move user event. Enforced by the VM during execution.
1430    max_event_emit_size_total: Option<u64>,
1431
1432    /// Maximum length of a vector in Move. Enforced by the VM during execution, and for constants, by the verifier.
1433    max_move_vector_len: Option<u64>,
1434
1435    /// Maximum length of an `Identifier` in Move. Enforced by the bytecode verifier at signing.
1436    max_move_identifier_len: Option<u64>,
1437
1438    /// Maximum depth of a Move value within the VM.
1439    max_move_value_depth: Option<u64>,
1440
1441    /// Maximum number of variants in an enum. Enforced by the bytecode verifier at signing.
1442    max_move_enum_variants: Option<u64>,
1443
1444    /// Maximum number of back edges in Move function. Enforced by the bytecode verifier at signing.
1445    max_back_edges_per_function: Option<u64>,
1446
1447    /// Maximum number of back edges in Move module. Enforced by the bytecode verifier at signing.
1448    max_back_edges_per_module: Option<u64>,
1449
1450    /// Maximum number of meter `ticks` spent verifying a Move function. Enforced by the bytecode verifier at signing.
1451    max_verifier_meter_ticks_per_function: Option<u64>,
1452
1453    /// Maximum number of meter `ticks` spent verifying a Move module. Enforced by the bytecode verifier at signing.
1454    max_meter_ticks_per_module: Option<u64>,
1455
1456    /// Maximum number of meter `ticks` spent verifying a Move package. Enforced by the bytecode verifier at signing.
1457    max_meter_ticks_per_package: Option<u64>,
1458
1459    // === Object runtime internal operation limits ====
1460    // These affect dynamic fields
1461    /// Maximum number of cached objects in the object runtime ObjectStore. Enforced by object runtime during execution
1462    object_runtime_max_num_cached_objects: Option<u64>,
1463
1464    /// Maximum number of cached objects in the object runtime ObjectStore in system transaction. Enforced by object runtime during execution
1465    object_runtime_max_num_cached_objects_system_tx: Option<u64>,
1466
1467    /// Maximum number of stored objects accessed by object runtime ObjectStore. Enforced by object runtime during execution
1468    object_runtime_max_num_store_entries: Option<u64>,
1469
1470    /// Maximum number of stored objects accessed by object runtime ObjectStore in system transaction. Enforced by object runtime during execution
1471    object_runtime_max_num_store_entries_system_tx: Option<u64>,
1472
1473    // === Execution gas costs ====
1474    /// Base cost for any Sui transaction
1475    base_tx_cost_fixed: Option<u64>,
1476
1477    /// Additional cost for a transaction that publishes a package
1478    /// i.e., the base cost of such a transaction is base_tx_cost_fixed + package_publish_cost_fixed
1479    package_publish_cost_fixed: Option<u64>,
1480
1481    /// Cost per byte of a Move call transaction
1482    /// i.e., the cost of such a transaction is base_cost + (base_tx_cost_per_byte * size)
1483    base_tx_cost_per_byte: Option<u64>,
1484
1485    /// Cost per byte for a transaction that publishes a package
1486    package_publish_cost_per_byte: Option<u64>,
1487
1488    // Per-byte cost of reading an object during transaction execution
1489    obj_access_cost_read_per_byte: Option<u64>,
1490
1491    // Per-byte cost of writing an object during transaction execution
1492    obj_access_cost_mutate_per_byte: Option<u64>,
1493
1494    // Per-byte cost of deleting an object during transaction execution
1495    obj_access_cost_delete_per_byte: Option<u64>,
1496
1497    /// Per-byte cost charged for each input object to a transaction.
1498    /// Meant to approximate the cost of checking locks for each object
1499    // TODO: Option<I'm not sure that this cost makes sense. Checking locks is "free"
1500    // in the sense that an invalid tx that can never be committed/pay gas can
1501    // force validators to check an arbitrary number of locks. If those checks are
1502    // "free" for invalid transactions, why charge for them in valid transactions
1503    // TODO: Option<if we keep this, I think we probably want it to be a fixed cost rather
1504    // than a per-byte cost. checking an object lock should not require loading an
1505    // entire object, just consulting an ID -> tx digest map
1506    obj_access_cost_verify_per_byte: Option<u64>,
1507
1508    // Maximal nodes which are allowed when converting to a type layout.
1509    max_type_to_layout_nodes: Option<u64>,
1510
1511    // Maximal size in bytes that a PTB value can be
1512    max_ptb_value_size: Option<u64>,
1513
1514    // === Gas version. gas model ===
1515    /// Gas model version, what code we are using to charge gas
1516    gas_model_version: Option<u64>,
1517
1518    // === Storage gas costs ===
1519    /// Per-byte cost of storing an object in the Sui global object store. Some of this cost may be refundable if the object is later freed
1520    obj_data_cost_refundable: Option<u64>,
1521
1522    // Per-byte cost of storing an object in the Sui transaction log (e.g., in CertifiedTransactionEffects)
1523    // This depends on the size of various fields including the effects
1524    // TODO: Option<I don't fully understand this^ and more details would be useful
1525    obj_metadata_cost_non_refundable: Option<u64>,
1526
1527    // === Tokenomics ===
1528
1529    // TODO: Option<this should be changed to u64.
1530    /// Sender of a txn that touches an object will get this percent of the storage rebate back.
1531    /// In basis point.
1532    storage_rebate_rate: Option<u64>,
1533
1534    /// 5% of the storage fund's share of rewards are reinvested into the storage fund.
1535    /// In basis point.
1536    storage_fund_reinvest_rate: Option<u64>,
1537
1538    /// The share of rewards that will be slashed and redistributed is 50%.
1539    /// In basis point.
1540    reward_slashing_rate: Option<u64>,
1541
1542    /// Unit gas price, Mist per internal gas unit.
1543    storage_gas_price: Option<u64>,
1544
1545    /// Per-object storage cost for accumulator objects, used during end-of-epoch accounting.
1546    accumulator_object_storage_cost: Option<u64>,
1547
1548    // === Core Protocol ===
1549    /// Max number of transactions per checkpoint.
1550    /// Note that this is a protocol constant and not a config as validators must have this set to
1551    /// the same value, otherwise they *will* fork.
1552    max_transactions_per_checkpoint: Option<u64>,
1553
1554    /// Max size of a checkpoint in bytes.
1555    /// Note that this is a protocol constant and not a config as validators must have this set to
1556    /// the same value, otherwise they *will* fork.
1557    max_checkpoint_size_bytes: Option<u64>,
1558
1559    /// A protocol upgrade always requires 2f+1 stake to agree. We support a buffer of additional
1560    /// stake (as a fraction of f, expressed in basis points) that is required before an upgrade
1561    /// can happen automatically. 10000bps would indicate that complete unanimity is required (all
1562    /// 3f+1 must vote), while 0bps would indicate that 2f+1 is sufficient.
1563    buffer_stake_for_protocol_upgrade_bps: Option<u64>,
1564
1565    // === Native Function Costs ===
1566
1567    // `address` module
1568    // Cost params for the Move native function `address::from_bytes(bytes: vector<u8>)`
1569    address_from_bytes_cost_base: Option<u64>,
1570    // Cost params for the Move native function `address::to_u256(address): u256`
1571    address_to_u256_cost_base: Option<u64>,
1572    // Cost params for the Move native function `address::from_u256(u256): address`
1573    address_from_u256_cost_base: Option<u64>,
1574
1575    // `config` module
1576    // Cost params for the Move native function `read_setting_impl<Name: copy + drop + store,
1577    // SettingValue: key + store, SettingDataValue: store, Value: copy + drop + store,
1578    // >(config: address, name: address, current_epoch: u64): Option<Value>`
1579    config_read_setting_impl_cost_base: Option<u64>,
1580    config_read_setting_impl_cost_per_byte: Option<u64>,
1581
1582    // `dynamic_field` module
1583    // Cost params for the Move native function `hash_type_and_key<K: copy + drop + store>(parent: address, k: K): address`
1584    dynamic_field_hash_type_and_key_cost_base: Option<u64>,
1585    dynamic_field_hash_type_and_key_type_cost_per_byte: Option<u64>,
1586    dynamic_field_hash_type_and_key_value_cost_per_byte: Option<u64>,
1587    dynamic_field_hash_type_and_key_type_tag_cost_per_byte: Option<u64>,
1588    // Cost params for the Move native function `add_child_object<Child: key>(parent: address, child: Child)`
1589    dynamic_field_add_child_object_cost_base: Option<u64>,
1590    dynamic_field_add_child_object_type_cost_per_byte: Option<u64>,
1591    dynamic_field_add_child_object_value_cost_per_byte: Option<u64>,
1592    dynamic_field_add_child_object_struct_tag_cost_per_byte: Option<u64>,
1593    // Cost params for the Move native function `borrow_child_object_mut<Child: key>(parent: &mut UID, id: address): &mut Child`
1594    dynamic_field_borrow_child_object_cost_base: Option<u64>,
1595    dynamic_field_borrow_child_object_child_ref_cost_per_byte: Option<u64>,
1596    dynamic_field_borrow_child_object_type_cost_per_byte: Option<u64>,
1597    // Cost params for the Move native function `remove_child_object<Child: key>(parent: address, id: address): Child`
1598    dynamic_field_remove_child_object_cost_base: Option<u64>,
1599    dynamic_field_remove_child_object_child_cost_per_byte: Option<u64>,
1600    dynamic_field_remove_child_object_type_cost_per_byte: Option<u64>,
1601    // Cost params for the Move native function `has_child_object(parent: address, id: address): bool`
1602    dynamic_field_has_child_object_cost_base: Option<u64>,
1603    // Cost params for the Move native function `has_child_object_with_ty<Child: key>(parent: address, id: address): bool`
1604    dynamic_field_has_child_object_with_ty_cost_base: Option<u64>,
1605    dynamic_field_has_child_object_with_ty_type_cost_per_byte: Option<u64>,
1606    dynamic_field_has_child_object_with_ty_type_tag_cost_per_byte: Option<u64>,
1607
1608    // `event` module
1609    // Cost params for the Move native function `event::emit<T: copy + drop>(event: T)`
1610    event_emit_cost_base: Option<u64>,
1611    event_emit_value_size_derivation_cost_per_byte: Option<u64>,
1612    event_emit_tag_size_derivation_cost_per_byte: Option<u64>,
1613    event_emit_output_cost_per_byte: Option<u64>,
1614    event_emit_auth_stream_cost: Option<u64>,
1615
1616    //  `object` module
1617    // Cost params for the Move native function `borrow_uid<T: key>(obj: &T): &UID`
1618    object_borrow_uid_cost_base: Option<u64>,
1619    // Cost params for the Move native function `delete_impl(id: address)`
1620    object_delete_impl_cost_base: Option<u64>,
1621    // Cost params for the Move native function `record_new_uid(id: address)`
1622    object_record_new_uid_cost_base: Option<u64>,
1623
1624    // Transfer
1625    // Cost params for the Move native function `transfer_impl<T: key>(obj: T, recipient: address)`
1626    transfer_transfer_internal_cost_base: Option<u64>,
1627    // Cost params for the Move native function `party_transfer_impl<T: key>(obj: T, party_members: vector<address>)`
1628    transfer_party_transfer_internal_cost_base: Option<u64>,
1629    // Cost params for the Move native function `freeze_object<T: key>(obj: T)`
1630    transfer_freeze_object_cost_base: Option<u64>,
1631    // Cost params for the Move native function `share_object<T: key>(obj: T)`
1632    transfer_share_object_cost_base: Option<u64>,
1633    // Cost params for the Move native function
1634    // `receive_object<T: key>(p: &mut UID, recv: Receiving<T>T)`
1635    transfer_receive_object_cost_base: Option<u64>,
1636    transfer_receive_object_cost_per_byte: Option<u64>,
1637    transfer_receive_object_type_cost_per_byte: Option<u64>,
1638
1639    // TxContext
1640    // Cost params for the Move native function `transfer_impl<T: key>(obj: T, recipient: address)`
1641    tx_context_derive_id_cost_base: Option<u64>,
1642    tx_context_fresh_id_cost_base: Option<u64>,
1643    tx_context_sender_cost_base: Option<u64>,
1644    tx_context_epoch_cost_base: Option<u64>,
1645    tx_context_epoch_timestamp_ms_cost_base: Option<u64>,
1646    tx_context_sponsor_cost_base: Option<u64>,
1647    tx_context_rgp_cost_base: Option<u64>,
1648    tx_context_gas_price_cost_base: Option<u64>,
1649    tx_context_gas_budget_cost_base: Option<u64>,
1650    tx_context_ids_created_cost_base: Option<u64>,
1651    tx_context_replace_cost_base: Option<u64>,
1652
1653    // Types
1654    // Cost params for the Move native function `is_one_time_witness<T: drop>(_: &T): bool`
1655    types_is_one_time_witness_cost_base: Option<u64>,
1656    types_is_one_time_witness_type_tag_cost_per_byte: Option<u64>,
1657    types_is_one_time_witness_type_cost_per_byte: Option<u64>,
1658
1659    // Validator
1660    // Cost params for the Move native function `validate_metadata_bcs(metadata: vector<u8>)`
1661    validator_validate_metadata_cost_base: Option<u64>,
1662    validator_validate_metadata_data_cost_per_byte: Option<u64>,
1663
1664    // Crypto natives
1665    crypto_invalid_arguments_cost: Option<u64>,
1666    // bls12381::bls12381_min_sig_verify
1667    bls12381_bls12381_min_sig_verify_cost_base: Option<u64>,
1668    bls12381_bls12381_min_sig_verify_msg_cost_per_byte: Option<u64>,
1669    bls12381_bls12381_min_sig_verify_msg_cost_per_block: Option<u64>,
1670
1671    // bls12381::bls12381_min_pk_verify
1672    bls12381_bls12381_min_pk_verify_cost_base: Option<u64>,
1673    bls12381_bls12381_min_pk_verify_msg_cost_per_byte: Option<u64>,
1674    bls12381_bls12381_min_pk_verify_msg_cost_per_block: Option<u64>,
1675
1676    // ecdsa_k1::ecrecover
1677    ecdsa_k1_ecrecover_keccak256_cost_base: Option<u64>,
1678    ecdsa_k1_ecrecover_keccak256_msg_cost_per_byte: Option<u64>,
1679    ecdsa_k1_ecrecover_keccak256_msg_cost_per_block: Option<u64>,
1680    ecdsa_k1_ecrecover_sha256_cost_base: Option<u64>,
1681    ecdsa_k1_ecrecover_sha256_msg_cost_per_byte: Option<u64>,
1682    ecdsa_k1_ecrecover_sha256_msg_cost_per_block: Option<u64>,
1683
1684    // ecdsa_k1::decompress_pubkey
1685    ecdsa_k1_decompress_pubkey_cost_base: Option<u64>,
1686
1687    // ecdsa_k1::secp256k1_verify
1688    ecdsa_k1_secp256k1_verify_keccak256_cost_base: Option<u64>,
1689    ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_byte: Option<u64>,
1690    ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_block: Option<u64>,
1691    ecdsa_k1_secp256k1_verify_sha256_cost_base: Option<u64>,
1692    ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_byte: Option<u64>,
1693    ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_block: Option<u64>,
1694
1695    // ecdsa_r1::ecrecover
1696    ecdsa_r1_ecrecover_keccak256_cost_base: Option<u64>,
1697    ecdsa_r1_ecrecover_keccak256_msg_cost_per_byte: Option<u64>,
1698    ecdsa_r1_ecrecover_keccak256_msg_cost_per_block: Option<u64>,
1699    ecdsa_r1_ecrecover_sha256_cost_base: Option<u64>,
1700    ecdsa_r1_ecrecover_sha256_msg_cost_per_byte: Option<u64>,
1701    ecdsa_r1_ecrecover_sha256_msg_cost_per_block: Option<u64>,
1702
1703    // ecdsa_r1::secp256k1_verify
1704    ecdsa_r1_secp256r1_verify_keccak256_cost_base: Option<u64>,
1705    ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_byte: Option<u64>,
1706    ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_block: Option<u64>,
1707    ecdsa_r1_secp256r1_verify_sha256_cost_base: Option<u64>,
1708    ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_byte: Option<u64>,
1709    ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_block: Option<u64>,
1710
1711    // ecvrf::verify
1712    ecvrf_ecvrf_verify_cost_base: Option<u64>,
1713    ecvrf_ecvrf_verify_alpha_string_cost_per_byte: Option<u64>,
1714    ecvrf_ecvrf_verify_alpha_string_cost_per_block: Option<u64>,
1715
1716    // ed25519
1717    ed25519_ed25519_verify_cost_base: Option<u64>,
1718    ed25519_ed25519_verify_msg_cost_per_byte: Option<u64>,
1719    ed25519_ed25519_verify_msg_cost_per_block: Option<u64>,
1720
1721    // groth16::prepare_verifying_key
1722    groth16_prepare_verifying_key_bls12381_cost_base: Option<u64>,
1723    groth16_prepare_verifying_key_bn254_cost_base: Option<u64>,
1724
1725    // groth16::verify_groth16_proof_internal
1726    groth16_verify_groth16_proof_internal_bls12381_cost_base: Option<u64>,
1727    groth16_verify_groth16_proof_internal_bls12381_cost_per_public_input: Option<u64>,
1728    groth16_verify_groth16_proof_internal_bn254_cost_base: Option<u64>,
1729    groth16_verify_groth16_proof_internal_bn254_cost_per_public_input: Option<u64>,
1730    groth16_verify_groth16_proof_internal_public_input_cost_per_byte: Option<u64>,
1731
1732    // hash::blake2b256
1733    hash_blake2b256_cost_base: Option<u64>,
1734    hash_blake2b256_data_cost_per_byte: Option<u64>,
1735    hash_blake2b256_data_cost_per_block: Option<u64>,
1736
1737    // hash::keccak256
1738    hash_keccak256_cost_base: Option<u64>,
1739    hash_keccak256_data_cost_per_byte: Option<u64>,
1740    hash_keccak256_data_cost_per_block: Option<u64>,
1741
1742    // poseidon::poseidon_bn254
1743    poseidon_bn254_cost_base: Option<u64>,
1744    poseidon_bn254_cost_per_block: Option<u64>,
1745
1746    // group_ops
1747    group_ops_bls12381_decode_scalar_cost: Option<u64>,
1748    group_ops_bls12381_decode_g1_cost: Option<u64>,
1749    group_ops_bls12381_decode_g2_cost: Option<u64>,
1750    group_ops_bls12381_decode_gt_cost: Option<u64>,
1751    group_ops_bls12381_scalar_add_cost: Option<u64>,
1752    group_ops_bls12381_g1_add_cost: Option<u64>,
1753    group_ops_bls12381_g2_add_cost: Option<u64>,
1754    group_ops_bls12381_gt_add_cost: Option<u64>,
1755    group_ops_bls12381_scalar_sub_cost: Option<u64>,
1756    group_ops_bls12381_g1_sub_cost: Option<u64>,
1757    group_ops_bls12381_g2_sub_cost: Option<u64>,
1758    group_ops_bls12381_gt_sub_cost: Option<u64>,
1759    group_ops_bls12381_scalar_mul_cost: Option<u64>,
1760    group_ops_bls12381_g1_mul_cost: Option<u64>,
1761    group_ops_bls12381_g2_mul_cost: Option<u64>,
1762    group_ops_bls12381_gt_mul_cost: Option<u64>,
1763    group_ops_bls12381_scalar_div_cost: Option<u64>,
1764    group_ops_bls12381_g1_div_cost: Option<u64>,
1765    group_ops_bls12381_g2_div_cost: Option<u64>,
1766    group_ops_bls12381_gt_div_cost: Option<u64>,
1767    group_ops_bls12381_g1_hash_to_base_cost: Option<u64>,
1768    group_ops_bls12381_g2_hash_to_base_cost: Option<u64>,
1769    group_ops_bls12381_g1_hash_to_cost_per_byte: Option<u64>,
1770    group_ops_bls12381_g2_hash_to_cost_per_byte: Option<u64>,
1771    group_ops_bls12381_g1_msm_base_cost: Option<u64>,
1772    group_ops_bls12381_g2_msm_base_cost: Option<u64>,
1773    group_ops_bls12381_g1_msm_base_cost_per_input: Option<u64>,
1774    group_ops_bls12381_g2_msm_base_cost_per_input: Option<u64>,
1775    group_ops_bls12381_msm_max_len: Option<u32>,
1776    group_ops_bls12381_pairing_cost: Option<u64>,
1777    group_ops_bls12381_g1_to_uncompressed_g1_cost: Option<u64>,
1778    group_ops_bls12381_uncompressed_g1_to_g1_cost: Option<u64>,
1779    group_ops_bls12381_uncompressed_g1_sum_base_cost: Option<u64>,
1780    group_ops_bls12381_uncompressed_g1_sum_cost_per_term: Option<u64>,
1781    group_ops_bls12381_uncompressed_g1_sum_max_terms: Option<u64>,
1782
1783    group_ops_ristretto_decode_scalar_cost: Option<u64>,
1784    group_ops_ristretto_decode_point_cost: Option<u64>,
1785    group_ops_ristretto_scalar_add_cost: Option<u64>,
1786    group_ops_ristretto_point_add_cost: Option<u64>,
1787    group_ops_ristretto_scalar_sub_cost: Option<u64>,
1788    group_ops_ristretto_point_sub_cost: Option<u64>,
1789    group_ops_ristretto_scalar_mul_cost: Option<u64>,
1790    group_ops_ristretto_point_mul_cost: Option<u64>,
1791    group_ops_ristretto_scalar_div_cost: Option<u64>,
1792    group_ops_ristretto_point_div_cost: Option<u64>,
1793
1794    verify_bulletproofs_ristretto255_base_cost: Option<u64>,
1795    verify_bulletproofs_ristretto255_cost_per_bit_and_commitment: Option<u64>,
1796
1797    // hmac::hmac_sha3_256
1798    hmac_hmac_sha3_256_cost_base: Option<u64>,
1799    hmac_hmac_sha3_256_input_cost_per_byte: Option<u64>,
1800    hmac_hmac_sha3_256_input_cost_per_block: Option<u64>,
1801
1802    // zklogin::check_zklogin_id
1803    check_zklogin_id_cost_base: Option<u64>,
1804    // zklogin::check_zklogin_issuer
1805    check_zklogin_issuer_cost_base: Option<u64>,
1806
1807    vdf_verify_vdf_cost: Option<u64>,
1808    vdf_hash_to_input_cost: Option<u64>,
1809
1810    // nitro_attestation::load_nitro_attestation
1811    nitro_attestation_parse_base_cost: Option<u64>,
1812    nitro_attestation_parse_cost_per_byte: Option<u64>,
1813    nitro_attestation_verify_base_cost: Option<u64>,
1814    nitro_attestation_verify_cost_per_cert: Option<u64>,
1815
1816    // Stdlib costs
1817    bcs_per_byte_serialized_cost: Option<u64>,
1818    bcs_legacy_min_output_size_cost: Option<u64>,
1819    bcs_failure_cost: Option<u64>,
1820
1821    hash_sha2_256_base_cost: Option<u64>,
1822    hash_sha2_256_per_byte_cost: Option<u64>,
1823    hash_sha2_256_legacy_min_input_len_cost: Option<u64>,
1824    hash_sha3_256_base_cost: Option<u64>,
1825    hash_sha3_256_per_byte_cost: Option<u64>,
1826    hash_sha3_256_legacy_min_input_len_cost: Option<u64>,
1827    type_name_get_base_cost: Option<u64>,
1828    type_name_get_per_byte_cost: Option<u64>,
1829    type_name_id_base_cost: Option<u64>,
1830
1831    string_check_utf8_base_cost: Option<u64>,
1832    string_check_utf8_per_byte_cost: Option<u64>,
1833    string_is_char_boundary_base_cost: Option<u64>,
1834    string_sub_string_base_cost: Option<u64>,
1835    string_sub_string_per_byte_cost: Option<u64>,
1836    string_index_of_base_cost: Option<u64>,
1837    string_index_of_per_byte_pattern_cost: Option<u64>,
1838    string_index_of_per_byte_searched_cost: Option<u64>,
1839
1840    vector_empty_base_cost: Option<u64>,
1841    vector_length_base_cost: Option<u64>,
1842    vector_push_back_base_cost: Option<u64>,
1843    vector_push_back_legacy_per_abstract_memory_unit_cost: Option<u64>,
1844    vector_borrow_base_cost: Option<u64>,
1845    vector_pop_back_base_cost: Option<u64>,
1846    vector_destroy_empty_base_cost: Option<u64>,
1847    vector_swap_base_cost: Option<u64>,
1848    debug_print_base_cost: Option<u64>,
1849    debug_print_stack_trace_base_cost: Option<u64>,
1850
1851    // ==== Ephemeral (consensus only) params deleted ====
1852    //
1853    // Const params for consensus scoring decision
1854    // The scaling factor property for the MED outlier detection
1855    // scoring_decision_mad_divisor: Option<f64>,
1856    // The cutoff value for the MED outlier detection
1857    // scoring_decision_cutoff_value: Option<f64>,
1858    /// === Execution Version ===
1859    execution_version: Option<u64>,
1860
1861    // Dictates the threshold (percentage of stake) that is used to calculate the "bad" nodes to be
1862    // swapped when creating the consensus schedule. The values should be of the range [0 - 33]. Anything
1863    // above 33 (f) will not be allowed.
1864    consensus_bad_nodes_stake_threshold: Option<u64>,
1865
1866    max_jwk_votes_per_validator_per_epoch: Option<u64>,
1867    // The maximum age of a JWK in epochs before it is removed from the AuthenticatorState object.
1868    // Applied at the end of an epoch as a delta from the new epoch value, so setting this to 1
1869    // will cause the new epoch to start with JWKs from the previous epoch still valid.
1870    max_age_of_jwk_in_epochs: Option<u64>,
1871
1872    // === random beacon ===
1873    /// Maximum allowed precision loss when reducing voting weights for the random beacon
1874    /// protocol.
1875    random_beacon_reduction_allowed_delta: Option<u16>,
1876
1877    /// Minimum number of shares below which voting weights will not be reduced for the
1878    /// random beacon protocol.
1879    random_beacon_reduction_lower_bound: Option<u32>,
1880
1881    /// Consensus Round after which DKG should be aborted and randomness disabled for
1882    /// the epoch, if it hasn't already completed.
1883    random_beacon_dkg_timeout_round: Option<u32>,
1884
1885    /// Minimum interval between consecutive rounds of generated randomness.
1886    random_beacon_min_round_interval_ms: Option<u64>,
1887
1888    /// Version of the random beacon DKG protocol.
1889    /// 0 was deprecated (and currently not supported), 1 is the default version.
1890    random_beacon_dkg_version: Option<u64>,
1891
1892    /// The maximum serialised transaction size (in bytes) accepted by consensus. That should be bigger than the
1893    /// `max_tx_size_bytes` with some additional headroom.
1894    consensus_max_transaction_size_bytes: Option<u64>,
1895    /// The maximum size of transactions included in a consensus block.
1896    consensus_max_transactions_in_block_bytes: Option<u64>,
1897    /// The maximum number of transactions included in a consensus block.
1898    consensus_max_num_transactions_in_block: Option<u64>,
1899
1900    /// The maximum number of rounds where transaction voting is allowed.
1901    consensus_voting_rounds: Option<u32>,
1902
1903    /// DEPRECATED. Do not use.
1904    max_accumulated_txn_cost_per_object_in_narwhal_commit: Option<u64>,
1905
1906    /// The max number of consensus rounds a transaction can be deferred due to shared object congestion.
1907    /// Transactions will be cancelled after this many rounds.
1908    max_deferral_rounds_for_congestion_control: Option<u64>,
1909
1910    /// DEPRECATED. Do not use.
1911    max_txn_cost_overage_per_object_in_commit: Option<u64>,
1912
1913    /// DEPRECATED. Do not use.
1914    allowed_txn_cost_overage_burst_per_object_in_commit: Option<u64>,
1915
1916    /// Minimum interval of commit timestamps between consecutive checkpoints.
1917    min_checkpoint_interval_ms: Option<u64>,
1918
1919    /// Version number to use for version_specific_data in `CheckpointSummary`.
1920    checkpoint_summary_version_specific_data: Option<u64>,
1921
1922    /// The max number of transactions that can be included in a single Soft Bundle.
1923    max_soft_bundle_size: Option<u64>,
1924
1925    /// Whether to try to form bridge committee
1926    // Note: this is not a feature flag because we want to distinguish between
1927    // `None` and `Some(false)`, as committee was already finalized on Testnet.
1928    bridge_should_try_to_finalize_committee: Option<bool>,
1929
1930    /// The max accumulated txn execution cost per object in a mysticeti. Transactions
1931    /// in a commit will be deferred once their touch shared objects hit this limit,
1932    /// unless the selected congestion control mode allows overage.
1933    /// This config plays the same role as `max_accumulated_txn_cost_per_object_in_narwhal_commit`
1934    /// but for mysticeti commits due to that mysticeti has higher commit rate.
1935    max_accumulated_txn_cost_per_object_in_mysticeti_commit: Option<u64>,
1936
1937    /// As above, but separate per-commit budget for transactions that use randomness.
1938    /// If not configured, uses the setting for `max_accumulated_txn_cost_per_object_in_mysticeti_commit`.
1939    max_accumulated_randomness_txn_cost_per_object_in_mysticeti_commit: Option<u64>,
1940
1941    /// Configures the garbage collection depth for consensus. When is unset or `0` then the garbage collection
1942    /// is disabled.
1943    consensus_gc_depth: Option<u32>,
1944
1945    /// DEPRECATED. Do not use.
1946    gas_budget_based_txn_cost_cap_factor: Option<u64>,
1947
1948    /// DEPRECATED. Do not use.
1949    gas_budget_based_txn_cost_absolute_cap_commit_count: Option<u64>,
1950
1951    /// SIP-45: K in the formula `amplification_factor = max(0, gas_price / reference_gas_price - K)`.
1952    /// This is the threshold for activating consensus amplification.
1953    sip_45_consensus_amplification_threshold: Option<u64>,
1954
1955    /// DEPRECATED: this was an ephemeral feature flag only used in per-epoch tables, which has now
1956    /// been deployed everywhere.
1957    use_object_per_epoch_marker_table_v2: Option<bool>,
1958
1959    /// The number of commits to consider when computing a deterministic commit rate.
1960    consensus_commit_rate_estimation_window_size: Option<u32>,
1961
1962    /// A list of effective AliasedAddress.
1963    /// For each pair, `aliased` is allowed to act as `original` for any of the transaction digests
1964    /// listed in `tx_digests`
1965    #[serde(skip_serializing_if = "Vec::is_empty")]
1966    aliased_addresses: Vec<AliasedAddress>,
1967
1968    /// The base charge for each command in a programmable transaction. This is a fixed cost to
1969    /// account for the overhead of processing each command.
1970    translation_per_command_base_charge: Option<u64>,
1971
1972    /// The base charge for each input in a programmable transaction regardless of if it is used or
1973    /// not, or a pure/object/funds withdrawal input.
1974    translation_per_input_base_charge: Option<u64>,
1975
1976    /// The base charge for each byte of pure input in a programmable transaction.
1977    translation_pure_input_per_byte_charge: Option<u64>,
1978
1979    /// The multiplier for the number of type nodes when charging for type loading.
1980    /// This is multiplied by the number of type nodes to get the total cost.
1981    /// This should be a small number to avoid excessive gas costs for loading types.
1982    translation_per_type_node_charge: Option<u64>,
1983
1984    /// The multiplier for the number of type references when charging for type checking and reference
1985    /// checking.
1986    translation_per_reference_node_charge: Option<u64>,
1987
1988    /// The multiplier for each linkage entry when charging for linkage tables that we have
1989    /// created.
1990    translation_per_linkage_entry_charge: Option<u64>,
1991
1992    /// The maximum number of updates per settlement transaction.
1993    max_updates_per_settlement_txn: Option<u32>,
1994
1995    /// Maximum computation units allowed for a gasless transaction.
1996    gasless_max_computation_units: Option<u64>,
1997
1998    /// Allowed token types for gasless transactions, with minimum transfer sizes per token.
1999    gasless_allowed_token_types: Option<Vec<(String, u64)>>,
2000
2001    /// Maximum number of unused Pure inputs allowed in a gasless transaction.
2002    /// Object and FundsWithdrawal inputs must always be used.
2003    /// When None, there is no limit (effectively unlimited).
2004    gasless_max_unused_inputs: Option<u64>,
2005
2006    /// Maximum size in bytes of each Pure input in a gasless transaction.
2007    /// When None, there is no limit (effectively unlimited).
2008    gasless_max_pure_input_bytes: Option<u64>,
2009
2010    /// Max tps for gasless transactions. Unlimited when unset, zero when set to zero.
2011    gasless_max_tps: Option<u64>,
2012
2013    #[serde(skip_serializing_if = "Option::is_none")]
2014    #[skip_accessor]
2015    include_special_package_amendments: Option<Arc<Amendments>>,
2016
2017    /// Maximum serialized size in bytes of a gasless transaction (SenderSignedData).
2018    /// Bounds the persistent storage impact of each admitted gasless transaction.
2019    gasless_max_tx_size_bytes: Option<u64>,
2020}
2021
2022/// An aliased address.
2023#[derive(Clone, Serialize, Deserialize, Debug)]
2024pub struct AliasedAddress {
2025    /// The original address.
2026    pub original: [u8; 32],
2027    /// An aliased address which is allowed to act as the original address.
2028    pub aliased: [u8; 32],
2029    /// A list of transaction digests for which the aliasing is allowed to be in effect.
2030    pub allowed_tx_digests: Vec<[u8; 32]>,
2031}
2032
2033// feature flags
2034impl ProtocolConfig {
2035    // Add checks for feature flag support here, e.g.:
2036    // pub fn check_new_protocol_feature_supported(&self) -> Result<(), Error> {
2037    //     if self.feature_flags.new_protocol_feature_supported {
2038    //         Ok(())
2039    //     } else {
2040    //         Err(Error(format!(
2041    //             "new_protocol_feature is not supported at {:?}",
2042    //             self.version
2043    //         )))
2044    //     }
2045    // }
2046
2047    pub fn check_package_upgrades_supported(&self) -> Result<(), Error> {
2048        if self.feature_flags.package_upgrades {
2049            Ok(())
2050        } else {
2051            Err(Error(format!(
2052                "package upgrades are not supported at {:?}",
2053                self.version
2054            )))
2055        }
2056    }
2057
2058    pub fn allow_receiving_object_id(&self) -> bool {
2059        self.feature_flags.allow_receiving_object_id
2060    }
2061
2062    pub fn receiving_objects_supported(&self) -> bool {
2063        self.feature_flags.receive_objects
2064    }
2065
2066    pub fn package_upgrades_supported(&self) -> bool {
2067        self.feature_flags.package_upgrades
2068    }
2069
2070    pub fn check_commit_root_state_digest_supported(&self) -> bool {
2071        self.feature_flags.commit_root_state_digest
2072    }
2073
2074    pub fn get_advance_epoch_start_time_in_safe_mode(&self) -> bool {
2075        self.feature_flags.advance_epoch_start_time_in_safe_mode
2076    }
2077
2078    pub fn loaded_child_objects_fixed(&self) -> bool {
2079        self.feature_flags.loaded_child_objects_fixed
2080    }
2081
2082    pub fn missing_type_is_compatibility_error(&self) -> bool {
2083        self.feature_flags.missing_type_is_compatibility_error
2084    }
2085
2086    pub fn scoring_decision_with_validity_cutoff(&self) -> bool {
2087        self.feature_flags.scoring_decision_with_validity_cutoff
2088    }
2089
2090    pub fn narwhal_versioned_metadata(&self) -> bool {
2091        self.feature_flags.narwhal_versioned_metadata
2092    }
2093
2094    pub fn consensus_order_end_of_epoch_last(&self) -> bool {
2095        self.feature_flags.consensus_order_end_of_epoch_last
2096    }
2097
2098    pub fn disallow_adding_abilities_on_upgrade(&self) -> bool {
2099        self.feature_flags.disallow_adding_abilities_on_upgrade
2100    }
2101
2102    pub fn disable_invariant_violation_check_in_swap_loc(&self) -> bool {
2103        self.feature_flags
2104            .disable_invariant_violation_check_in_swap_loc
2105    }
2106
2107    pub fn advance_to_highest_supported_protocol_version(&self) -> bool {
2108        self.feature_flags
2109            .advance_to_highest_supported_protocol_version
2110    }
2111
2112    pub fn ban_entry_init(&self) -> bool {
2113        self.feature_flags.ban_entry_init
2114    }
2115
2116    pub fn package_digest_hash_module(&self) -> bool {
2117        self.feature_flags.package_digest_hash_module
2118    }
2119
2120    pub fn disallow_change_struct_type_params_on_upgrade(&self) -> bool {
2121        self.feature_flags
2122            .disallow_change_struct_type_params_on_upgrade
2123    }
2124
2125    pub fn no_extraneous_module_bytes(&self) -> bool {
2126        self.feature_flags.no_extraneous_module_bytes
2127    }
2128
2129    pub fn zklogin_auth(&self) -> bool {
2130        self.feature_flags.zklogin_auth
2131    }
2132
2133    pub fn zklogin_supported_providers(&self) -> &BTreeSet<String> {
2134        &self.feature_flags.zklogin_supported_providers
2135    }
2136
2137    pub fn consensus_transaction_ordering(&self) -> ConsensusTransactionOrdering {
2138        self.feature_flags.consensus_transaction_ordering
2139    }
2140
2141    pub fn simplified_unwrap_then_delete(&self) -> bool {
2142        self.feature_flags.simplified_unwrap_then_delete
2143    }
2144
2145    pub fn supports_upgraded_multisig(&self) -> bool {
2146        self.feature_flags.upgraded_multisig_supported
2147    }
2148
2149    pub fn txn_base_cost_as_multiplier(&self) -> bool {
2150        self.feature_flags.txn_base_cost_as_multiplier
2151    }
2152
2153    pub fn shared_object_deletion(&self) -> bool {
2154        self.feature_flags.shared_object_deletion
2155    }
2156
2157    pub fn narwhal_new_leader_election_schedule(&self) -> bool {
2158        self.feature_flags.narwhal_new_leader_election_schedule
2159    }
2160
2161    pub fn loaded_child_object_format(&self) -> bool {
2162        self.feature_flags.loaded_child_object_format
2163    }
2164
2165    pub fn enable_jwk_consensus_updates(&self) -> bool {
2166        let ret = self.feature_flags.enable_jwk_consensus_updates;
2167        if ret {
2168            // jwk updates required end-of-epoch transactions
2169            assert!(self.feature_flags.end_of_epoch_transaction_supported);
2170        }
2171        ret
2172    }
2173
2174    pub fn simple_conservation_checks(&self) -> bool {
2175        self.feature_flags.simple_conservation_checks
2176    }
2177
2178    pub fn loaded_child_object_format_type(&self) -> bool {
2179        self.feature_flags.loaded_child_object_format_type
2180    }
2181
2182    pub fn end_of_epoch_transaction_supported(&self) -> bool {
2183        let ret = self.feature_flags.end_of_epoch_transaction_supported;
2184        if !ret {
2185            // jwk updates required end-of-epoch transactions
2186            assert!(!self.feature_flags.enable_jwk_consensus_updates);
2187        }
2188        ret
2189    }
2190
2191    pub fn recompute_has_public_transfer_in_execution(&self) -> bool {
2192        self.feature_flags
2193            .recompute_has_public_transfer_in_execution
2194    }
2195
2196    // this function only exists for readability in the genesis code.
2197    pub fn create_authenticator_state_in_genesis(&self) -> bool {
2198        self.enable_jwk_consensus_updates()
2199    }
2200
2201    pub fn random_beacon(&self) -> bool {
2202        self.feature_flags.random_beacon
2203    }
2204
2205    pub fn dkg_version(&self) -> u64 {
2206        // Version 0 was deprecated and removed, the default is 1 if not set.
2207        self.random_beacon_dkg_version.unwrap_or(1)
2208    }
2209
2210    pub fn enable_bridge(&self) -> bool {
2211        let ret = self.feature_flags.bridge;
2212        if ret {
2213            // bridge required end-of-epoch transactions
2214            assert!(self.feature_flags.end_of_epoch_transaction_supported);
2215        }
2216        ret
2217    }
2218
2219    pub fn should_try_to_finalize_bridge_committee(&self) -> bool {
2220        if !self.enable_bridge() {
2221            return false;
2222        }
2223        // In the older protocol version, always try to finalize the committee.
2224        self.bridge_should_try_to_finalize_committee.unwrap_or(true)
2225    }
2226
2227    pub fn enable_effects_v2(&self) -> bool {
2228        self.feature_flags.enable_effects_v2
2229    }
2230
2231    pub fn narwhal_certificate_v2(&self) -> bool {
2232        self.feature_flags.narwhal_certificate_v2
2233    }
2234
2235    pub fn verify_legacy_zklogin_address(&self) -> bool {
2236        self.feature_flags.verify_legacy_zklogin_address
2237    }
2238
2239    pub fn accept_zklogin_in_multisig(&self) -> bool {
2240        self.feature_flags.accept_zklogin_in_multisig
2241    }
2242
2243    pub fn accept_passkey_in_multisig(&self) -> bool {
2244        self.feature_flags.accept_passkey_in_multisig
2245    }
2246
2247    pub fn validate_zklogin_public_identifier(&self) -> bool {
2248        self.feature_flags.validate_zklogin_public_identifier
2249    }
2250
2251    pub fn zklogin_max_epoch_upper_bound_delta(&self) -> Option<u64> {
2252        self.feature_flags.zklogin_max_epoch_upper_bound_delta
2253    }
2254
2255    pub fn throughput_aware_consensus_submission(&self) -> bool {
2256        self.feature_flags.throughput_aware_consensus_submission
2257    }
2258
2259    pub fn include_consensus_digest_in_prologue(&self) -> bool {
2260        self.feature_flags.include_consensus_digest_in_prologue
2261    }
2262
2263    pub fn record_consensus_determined_version_assignments_in_prologue(&self) -> bool {
2264        self.feature_flags
2265            .record_consensus_determined_version_assignments_in_prologue
2266    }
2267
2268    pub fn record_additional_state_digest_in_prologue(&self) -> bool {
2269        self.feature_flags
2270            .record_additional_state_digest_in_prologue
2271    }
2272
2273    pub fn record_consensus_determined_version_assignments_in_prologue_v2(&self) -> bool {
2274        self.feature_flags
2275            .record_consensus_determined_version_assignments_in_prologue_v2
2276    }
2277
2278    pub fn prepend_prologue_tx_in_consensus_commit_in_checkpoints(&self) -> bool {
2279        self.feature_flags
2280            .prepend_prologue_tx_in_consensus_commit_in_checkpoints
2281    }
2282
2283    pub fn hardened_otw_check(&self) -> bool {
2284        self.feature_flags.hardened_otw_check
2285    }
2286
2287    pub fn enable_poseidon(&self) -> bool {
2288        self.feature_flags.enable_poseidon
2289    }
2290
2291    pub fn enable_coin_deny_list_v1(&self) -> bool {
2292        self.feature_flags.enable_coin_deny_list
2293    }
2294
2295    pub fn enable_accumulators(&self) -> bool {
2296        self.feature_flags.enable_accumulators
2297    }
2298
2299    pub fn enable_coin_reservation_obj_refs(&self) -> bool {
2300        self.new_vm_enabled() && self.feature_flags.enable_coin_reservation_obj_refs
2301    }
2302
2303    pub fn create_root_accumulator_object(&self) -> bool {
2304        self.feature_flags.create_root_accumulator_object
2305    }
2306
2307    pub fn enable_address_balance_gas_payments(&self) -> bool {
2308        self.feature_flags.enable_address_balance_gas_payments
2309    }
2310
2311    pub fn address_balance_gas_check_rgp_at_signing(&self) -> bool {
2312        self.feature_flags.address_balance_gas_check_rgp_at_signing
2313    }
2314
2315    pub fn address_balance_gas_reject_gas_coin_arg(&self) -> bool {
2316        self.feature_flags.address_balance_gas_reject_gas_coin_arg
2317    }
2318
2319    pub fn enable_multi_epoch_transaction_expiration(&self) -> bool {
2320        self.feature_flags.enable_multi_epoch_transaction_expiration
2321    }
2322
2323    pub fn relax_valid_during_for_owned_inputs(&self) -> bool {
2324        self.feature_flags.relax_valid_during_for_owned_inputs
2325    }
2326
2327    pub fn enable_authenticated_event_streams(&self) -> bool {
2328        self.feature_flags.enable_authenticated_event_streams && self.enable_accumulators()
2329    }
2330
2331    pub fn enable_non_exclusive_writes(&self) -> bool {
2332        self.feature_flags.enable_non_exclusive_writes
2333    }
2334
2335    pub fn enable_coin_registry(&self) -> bool {
2336        self.feature_flags.enable_coin_registry
2337    }
2338
2339    pub fn enable_display_registry(&self) -> bool {
2340        self.feature_flags.enable_display_registry
2341    }
2342
2343    pub fn enable_coin_deny_list_v2(&self) -> bool {
2344        self.feature_flags.enable_coin_deny_list_v2
2345    }
2346
2347    pub fn enable_group_ops_native_functions(&self) -> bool {
2348        self.feature_flags.enable_group_ops_native_functions
2349    }
2350
2351    pub fn enable_group_ops_native_function_msm(&self) -> bool {
2352        self.feature_flags.enable_group_ops_native_function_msm
2353    }
2354
2355    pub fn enable_ristretto255_group_ops(&self) -> bool {
2356        self.feature_flags.enable_ristretto255_group_ops
2357    }
2358
2359    pub fn enable_verify_bulletproofs_ristretto255(&self) -> bool {
2360        self.feature_flags.enable_verify_bulletproofs_ristretto255
2361    }
2362
2363    pub fn reject_mutable_random_on_entry_functions(&self) -> bool {
2364        self.feature_flags.reject_mutable_random_on_entry_functions
2365    }
2366
2367    pub fn per_object_congestion_control_mode(&self) -> PerObjectCongestionControlMode {
2368        self.feature_flags.per_object_congestion_control_mode
2369    }
2370
2371    pub fn consensus_choice(&self) -> ConsensusChoice {
2372        self.feature_flags.consensus_choice
2373    }
2374
2375    pub fn consensus_network(&self) -> ConsensusNetwork {
2376        self.feature_flags.consensus_network
2377    }
2378
2379    pub fn correct_gas_payment_limit_check(&self) -> bool {
2380        self.feature_flags.correct_gas_payment_limit_check
2381    }
2382
2383    pub fn reshare_at_same_initial_version(&self) -> bool {
2384        self.feature_flags.reshare_at_same_initial_version
2385    }
2386
2387    pub fn resolve_abort_locations_to_package_id(&self) -> bool {
2388        self.feature_flags.resolve_abort_locations_to_package_id
2389    }
2390
2391    pub fn mysticeti_use_committed_subdag_digest(&self) -> bool {
2392        self.feature_flags.mysticeti_use_committed_subdag_digest
2393    }
2394
2395    pub fn enable_vdf(&self) -> bool {
2396        self.feature_flags.enable_vdf
2397    }
2398
2399    pub fn fresh_vm_on_framework_upgrade(&self) -> bool {
2400        self.feature_flags.fresh_vm_on_framework_upgrade
2401    }
2402
2403    pub fn mysticeti_num_leaders_per_round(&self) -> Option<usize> {
2404        self.feature_flags.mysticeti_num_leaders_per_round
2405    }
2406
2407    pub fn soft_bundle(&self) -> bool {
2408        self.feature_flags.soft_bundle
2409    }
2410
2411    pub fn passkey_auth(&self) -> bool {
2412        self.feature_flags.passkey_auth
2413    }
2414
2415    pub fn authority_capabilities_v2(&self) -> bool {
2416        self.feature_flags.authority_capabilities_v2
2417    }
2418
2419    pub fn max_transaction_size_bytes(&self) -> u64 {
2420        // Provide a default value if protocol config version is too low.
2421        self.consensus_max_transaction_size_bytes
2422            .unwrap_or(256 * 1024)
2423    }
2424
2425    pub fn max_transactions_in_block_bytes(&self) -> u64 {
2426        if cfg!(msim) {
2427            256 * 1024
2428        } else {
2429            self.consensus_max_transactions_in_block_bytes
2430                .unwrap_or(512 * 1024)
2431        }
2432    }
2433
2434    pub fn max_num_transactions_in_block(&self) -> u64 {
2435        if cfg!(msim) {
2436            8
2437        } else {
2438            self.consensus_max_num_transactions_in_block.unwrap_or(512)
2439        }
2440    }
2441
2442    pub fn rethrow_serialization_type_layout_errors(&self) -> bool {
2443        self.feature_flags.rethrow_serialization_type_layout_errors
2444    }
2445
2446    pub fn consensus_distributed_vote_scoring_strategy(&self) -> bool {
2447        self.feature_flags
2448            .consensus_distributed_vote_scoring_strategy
2449    }
2450
2451    pub fn consensus_round_prober(&self) -> bool {
2452        self.feature_flags.consensus_round_prober
2453    }
2454
2455    pub fn validate_identifier_inputs(&self) -> bool {
2456        self.feature_flags.validate_identifier_inputs
2457    }
2458
2459    pub fn gc_depth(&self) -> u32 {
2460        self.consensus_gc_depth.unwrap_or(0)
2461    }
2462
2463    pub fn mysticeti_fastpath(&self) -> bool {
2464        self.feature_flags.mysticeti_fastpath
2465    }
2466
2467    pub fn relocate_event_module(&self) -> bool {
2468        self.feature_flags.relocate_event_module
2469    }
2470
2471    pub fn uncompressed_g1_group_elements(&self) -> bool {
2472        self.feature_flags.uncompressed_g1_group_elements
2473    }
2474
2475    pub fn disallow_new_modules_in_deps_only_packages(&self) -> bool {
2476        self.feature_flags
2477            .disallow_new_modules_in_deps_only_packages
2478    }
2479
2480    pub fn consensus_smart_ancestor_selection(&self) -> bool {
2481        self.feature_flags.consensus_smart_ancestor_selection
2482    }
2483
2484    pub fn disable_preconsensus_locking(&self) -> bool {
2485        self.feature_flags.disable_preconsensus_locking
2486    }
2487
2488    pub fn consensus_round_prober_probe_accepted_rounds(&self) -> bool {
2489        self.feature_flags
2490            .consensus_round_prober_probe_accepted_rounds
2491    }
2492
2493    pub fn native_charging_v2(&self) -> bool {
2494        self.feature_flags.native_charging_v2
2495    }
2496
2497    pub fn consensus_linearize_subdag_v2(&self) -> bool {
2498        let res = self.feature_flags.consensus_linearize_subdag_v2;
2499        assert!(
2500            !res || self.gc_depth() > 0,
2501            "The consensus linearize sub dag V2 requires GC to be enabled"
2502        );
2503        res
2504    }
2505
2506    pub fn consensus_median_based_commit_timestamp(&self) -> bool {
2507        let res = self.feature_flags.consensus_median_based_commit_timestamp;
2508        assert!(
2509            !res || self.gc_depth() > 0,
2510            "The consensus median based commit timestamp requires GC to be enabled"
2511        );
2512        res
2513    }
2514
2515    pub fn consensus_batched_block_sync(&self) -> bool {
2516        self.feature_flags.consensus_batched_block_sync
2517    }
2518
2519    pub fn convert_type_argument_error(&self) -> bool {
2520        self.feature_flags.convert_type_argument_error
2521    }
2522
2523    pub fn variant_nodes(&self) -> bool {
2524        self.feature_flags.variant_nodes
2525    }
2526
2527    pub fn consensus_zstd_compression(&self) -> bool {
2528        self.feature_flags.consensus_zstd_compression
2529    }
2530
2531    pub fn enable_nitro_attestation(&self) -> bool {
2532        self.feature_flags.enable_nitro_attestation
2533    }
2534
2535    pub fn enable_nitro_attestation_upgraded_parsing(&self) -> bool {
2536        self.feature_flags.enable_nitro_attestation_upgraded_parsing
2537    }
2538
2539    pub fn enable_nitro_attestation_all_nonzero_pcrs_parsing(&self) -> bool {
2540        self.feature_flags
2541            .enable_nitro_attestation_all_nonzero_pcrs_parsing
2542    }
2543
2544    pub fn enable_nitro_attestation_always_include_required_pcrs_parsing(&self) -> bool {
2545        self.feature_flags
2546            .enable_nitro_attestation_always_include_required_pcrs_parsing
2547    }
2548
2549    pub fn get_consensus_commit_rate_estimation_window_size(&self) -> u32 {
2550        self.consensus_commit_rate_estimation_window_size
2551            .unwrap_or(0)
2552    }
2553
2554    pub fn consensus_num_requested_prior_commits_at_startup(&self) -> u32 {
2555        // Currently there is only one parameter driving this value. If there are multiple
2556        // things computed from prior consensus commits, this function must return the max
2557        // of all of them.
2558        let window_size = self.get_consensus_commit_rate_estimation_window_size();
2559        // Ensure we are not using past commits without recording a state digest in the prologue.
2560        assert!(window_size == 0 || self.record_additional_state_digest_in_prologue());
2561        window_size
2562    }
2563
2564    pub fn minimize_child_object_mutations(&self) -> bool {
2565        self.feature_flags.minimize_child_object_mutations
2566    }
2567
2568    pub fn move_native_context(&self) -> bool {
2569        self.feature_flags.move_native_context
2570    }
2571
2572    pub fn normalize_ptb_arguments(&self) -> bool {
2573        self.feature_flags.normalize_ptb_arguments
2574    }
2575
2576    pub fn enforce_checkpoint_timestamp_monotonicity(&self) -> bool {
2577        self.feature_flags.enforce_checkpoint_timestamp_monotonicity
2578    }
2579
2580    pub fn max_ptb_value_size_v2(&self) -> bool {
2581        self.feature_flags.max_ptb_value_size_v2
2582    }
2583
2584    pub fn resolve_type_input_ids_to_defining_id(&self) -> bool {
2585        self.feature_flags.resolve_type_input_ids_to_defining_id
2586    }
2587
2588    pub fn enable_party_transfer(&self) -> bool {
2589        self.feature_flags.enable_party_transfer
2590    }
2591
2592    pub fn allow_unbounded_system_objects(&self) -> bool {
2593        self.feature_flags.allow_unbounded_system_objects
2594    }
2595
2596    pub fn type_tags_in_object_runtime(&self) -> bool {
2597        self.feature_flags.type_tags_in_object_runtime
2598    }
2599
2600    pub fn enable_ptb_execution_v2(&self) -> bool {
2601        self.feature_flags.enable_ptb_execution_v2
2602    }
2603
2604    pub fn better_adapter_type_resolution_errors(&self) -> bool {
2605        self.feature_flags.better_adapter_type_resolution_errors
2606    }
2607
2608    pub fn record_time_estimate_processed(&self) -> bool {
2609        self.feature_flags.record_time_estimate_processed
2610    }
2611
2612    pub fn ignore_execution_time_observations_after_certs_closed(&self) -> bool {
2613        self.feature_flags
2614            .ignore_execution_time_observations_after_certs_closed
2615    }
2616
2617    pub fn dependency_linkage_error(&self) -> bool {
2618        self.feature_flags.dependency_linkage_error
2619    }
2620
2621    pub fn additional_multisig_checks(&self) -> bool {
2622        self.feature_flags.additional_multisig_checks
2623    }
2624
2625    pub fn debug_fatal_on_move_invariant_violation(&self) -> bool {
2626        self.feature_flags.debug_fatal_on_move_invariant_violation
2627    }
2628
2629    pub fn allow_private_accumulator_entrypoints(&self) -> bool {
2630        self.feature_flags.allow_private_accumulator_entrypoints
2631    }
2632
2633    pub fn additional_consensus_digest_indirect_state(&self) -> bool {
2634        self.feature_flags
2635            .additional_consensus_digest_indirect_state
2636    }
2637
2638    pub fn check_for_init_during_upgrade(&self) -> bool {
2639        self.feature_flags.check_for_init_during_upgrade
2640    }
2641
2642    pub fn per_command_shared_object_transfer_rules(&self) -> bool {
2643        self.feature_flags.per_command_shared_object_transfer_rules
2644    }
2645
2646    pub fn consensus_checkpoint_signature_key_includes_digest(&self) -> bool {
2647        self.feature_flags
2648            .consensus_checkpoint_signature_key_includes_digest
2649    }
2650
2651    pub fn include_checkpoint_artifacts_digest_in_summary(&self) -> bool {
2652        self.feature_flags
2653            .include_checkpoint_artifacts_digest_in_summary
2654    }
2655
2656    pub fn use_mfp_txns_in_load_initial_object_debts(&self) -> bool {
2657        self.feature_flags.use_mfp_txns_in_load_initial_object_debts
2658    }
2659
2660    pub fn cancel_for_failed_dkg_early(&self) -> bool {
2661        self.feature_flags.cancel_for_failed_dkg_early
2662    }
2663
2664    pub fn abstract_size_in_object_runtime(&self) -> bool {
2665        self.feature_flags.abstract_size_in_object_runtime
2666    }
2667
2668    pub fn object_runtime_charge_cache_load_gas(&self) -> bool {
2669        self.feature_flags.object_runtime_charge_cache_load_gas
2670    }
2671
2672    pub fn additional_borrow_checks(&self) -> bool {
2673        self.feature_flags.additional_borrow_checks
2674    }
2675
2676    pub fn use_new_commit_handler(&self) -> bool {
2677        self.feature_flags.use_new_commit_handler
2678    }
2679
2680    pub fn better_loader_errors(&self) -> bool {
2681        self.feature_flags.better_loader_errors
2682    }
2683
2684    pub fn generate_df_type_layouts(&self) -> bool {
2685        self.feature_flags.generate_df_type_layouts
2686    }
2687
2688    pub fn allow_references_in_ptbs(&self) -> bool {
2689        self.feature_flags.allow_references_in_ptbs
2690    }
2691
2692    pub fn private_generics_verifier_v2(&self) -> bool {
2693        self.feature_flags.private_generics_verifier_v2
2694    }
2695
2696    pub fn deprecate_global_storage_ops_during_deserialization(&self) -> bool {
2697        self.feature_flags
2698            .deprecate_global_storage_ops_during_deserialization
2699    }
2700
2701    pub fn enable_observation_chunking(&self) -> bool {
2702        matches!(self.feature_flags.per_object_congestion_control_mode,
2703            PerObjectCongestionControlMode::ExecutionTimeEstimate(ref params)
2704                if params.observations_chunk_size.is_some()
2705        )
2706    }
2707
2708    pub fn deprecate_global_storage_ops(&self) -> bool {
2709        self.feature_flags.deprecate_global_storage_ops
2710    }
2711
2712    pub fn normalize_depth_formula(&self) -> bool {
2713        self.feature_flags.normalize_depth_formula
2714    }
2715
2716    pub fn consensus_skip_gced_accept_votes(&self) -> bool {
2717        self.feature_flags.consensus_skip_gced_accept_votes
2718    }
2719
2720    pub fn include_cancelled_randomness_txns_in_prologue(&self) -> bool {
2721        self.feature_flags
2722            .include_cancelled_randomness_txns_in_prologue
2723    }
2724
2725    pub fn address_aliases(&self) -> bool {
2726        let address_aliases = self.feature_flags.address_aliases;
2727        assert!(
2728            !address_aliases || self.mysticeti_fastpath(),
2729            "Address aliases requires Mysticeti fastpath to be enabled"
2730        );
2731        if address_aliases {
2732            assert!(
2733                self.feature_flags.disable_preconsensus_locking,
2734                "Address aliases requires CertifiedTransaction to be disabled"
2735            );
2736        }
2737        address_aliases
2738    }
2739
2740    pub fn fix_checkpoint_signature_mapping(&self) -> bool {
2741        self.feature_flags.fix_checkpoint_signature_mapping
2742    }
2743
2744    pub fn enable_object_funds_withdraw(&self) -> bool {
2745        self.feature_flags.enable_object_funds_withdraw
2746    }
2747
2748    pub fn gas_rounding_halve_digits(&self) -> bool {
2749        self.feature_flags.gas_rounding_halve_digits
2750    }
2751
2752    pub fn flexible_tx_context_positions(&self) -> bool {
2753        self.feature_flags.flexible_tx_context_positions
2754    }
2755
2756    pub fn disable_entry_point_signature_check(&self) -> bool {
2757        self.feature_flags.disable_entry_point_signature_check
2758    }
2759
2760    pub fn consensus_skip_gced_blocks_in_direct_finalization(&self) -> bool {
2761        self.feature_flags
2762            .consensus_skip_gced_blocks_in_direct_finalization
2763    }
2764
2765    pub fn convert_withdrawal_compatibility_ptb_arguments(&self) -> bool {
2766        self.feature_flags
2767            .convert_withdrawal_compatibility_ptb_arguments
2768    }
2769
2770    pub fn restrict_hot_or_not_entry_functions(&self) -> bool {
2771        self.feature_flags.restrict_hot_or_not_entry_functions
2772    }
2773
2774    pub fn split_checkpoints_in_consensus_handler(&self) -> bool {
2775        self.feature_flags.split_checkpoints_in_consensus_handler
2776    }
2777
2778    pub fn consensus_always_accept_system_transactions(&self) -> bool {
2779        self.feature_flags
2780            .consensus_always_accept_system_transactions
2781    }
2782
2783    pub fn validator_metadata_verify_v2(&self) -> bool {
2784        self.feature_flags.validator_metadata_verify_v2
2785    }
2786
2787    pub fn defer_unpaid_amplification(&self) -> bool {
2788        self.feature_flags.defer_unpaid_amplification
2789    }
2790
2791    pub fn gasless_transaction_drop_safety(&self) -> bool {
2792        self.feature_flags.gasless_transaction_drop_safety
2793    }
2794
2795    pub fn new_vm_enabled(&self) -> bool {
2796        self.execution_version.is_some_and(|v| v >= 4)
2797    }
2798
2799    pub fn merge_randomness_into_checkpoint(&self) -> bool {
2800        self.feature_flags.merge_randomness_into_checkpoint
2801    }
2802
2803    pub fn use_coin_party_owner(&self) -> bool {
2804        self.feature_flags.use_coin_party_owner
2805    }
2806
2807    pub fn enable_gasless(&self) -> bool {
2808        self.feature_flags.enable_gasless
2809    }
2810
2811    pub fn gasless_verify_remaining_balance(&self) -> bool {
2812        self.feature_flags.gasless_verify_remaining_balance
2813    }
2814
2815    pub fn gasless_allowed_token_types(&self) -> &[(String, u64)] {
2816        debug_assert!(self.gasless_allowed_token_types.is_some());
2817        self.gasless_allowed_token_types.as_deref().unwrap_or(&[])
2818    }
2819
2820    pub fn get_gasless_max_unused_inputs(&self) -> u64 {
2821        self.gasless_max_unused_inputs.unwrap_or(u64::MAX)
2822    }
2823
2824    pub fn get_gasless_max_pure_input_bytes(&self) -> u64 {
2825        self.gasless_max_pure_input_bytes.unwrap_or(u64::MAX)
2826    }
2827
2828    pub fn get_gasless_max_tx_size_bytes(&self) -> u64 {
2829        self.gasless_max_tx_size_bytes.unwrap_or(u64::MAX)
2830    }
2831
2832    pub fn disallow_jump_orphans(&self) -> bool {
2833        self.feature_flags.disallow_jump_orphans
2834    }
2835
2836    pub fn early_return_receive_object_mismatched_type(&self) -> bool {
2837        self.feature_flags
2838            .early_return_receive_object_mismatched_type
2839    }
2840
2841    pub fn include_special_package_amendments_as_option(&self) -> &Option<Arc<Amendments>> {
2842        &self.include_special_package_amendments
2843    }
2844
2845    pub fn timestamp_based_epoch_close(&self) -> bool {
2846        self.feature_flags.timestamp_based_epoch_close
2847    }
2848
2849    pub fn limit_groth16_pvk_inputs(&self) -> bool {
2850        self.feature_flags.limit_groth16_pvk_inputs
2851    }
2852
2853    pub fn enforce_address_balance_change_invariant(&self) -> bool {
2854        self.feature_flags.enforce_address_balance_change_invariant
2855    }
2856
2857    pub fn granular_post_execution_checks(&self) -> bool {
2858        self.feature_flags.granular_post_execution_checks
2859    }
2860}
2861
2862#[cfg(not(msim))]
2863static POISON_VERSION_METHODS: AtomicBool = AtomicBool::new(false);
2864
2865// Use a thread local in sim tests for test isolation.
2866#[cfg(msim)]
2867thread_local! {
2868    static POISON_VERSION_METHODS: AtomicBool = AtomicBool::new(false);
2869}
2870
2871// Instantiations for each protocol version.
2872impl ProtocolConfig {
2873    /// Get the value ProtocolConfig that are in effect during the given protocol version.
2874    pub fn get_for_version(version: ProtocolVersion, chain: Chain) -> Self {
2875        // ProtocolVersion can be deserialized so we need to check it here as well.
2876        assert!(
2877            version >= ProtocolVersion::MIN,
2878            "Network protocol version is {:?}, but the minimum supported version by the binary is {:?}. Please upgrade the binary.",
2879            version,
2880            ProtocolVersion::MIN.0,
2881        );
2882        assert!(
2883            version <= ProtocolVersion::MAX_ALLOWED,
2884            "Network protocol version is {:?}, but the maximum supported version by the binary is {:?}. Please upgrade the binary.",
2885            version,
2886            ProtocolVersion::MAX_ALLOWED.0,
2887        );
2888
2889        let mut ret = Self::get_for_version_impl(version, chain);
2890        ret.version = version;
2891
2892        ret = Self::apply_config_override(version, ret);
2893
2894        if std::env::var("SUI_PROTOCOL_CONFIG_OVERRIDE_ENABLE").is_ok() {
2895            warn!(
2896                "overriding ProtocolConfig settings with custom settings; this may break non-local networks"
2897            );
2898            let overrides: ProtocolConfigOptional =
2899                serde_env::from_env_with_prefix("SUI_PROTOCOL_CONFIG_OVERRIDE")
2900                    .expect("failed to parse ProtocolConfig override env variables");
2901            overrides.apply_to(&mut ret);
2902        }
2903
2904        ret
2905    }
2906
2907    /// Get the value ProtocolConfig that are in effect during the given protocol version.
2908    /// Or none if the version is not supported.
2909    pub fn get_for_version_if_supported(version: ProtocolVersion, chain: Chain) -> Option<Self> {
2910        if version.0 >= ProtocolVersion::MIN.0 && version.0 <= ProtocolVersion::MAX_ALLOWED.0 {
2911            let mut ret = Self::get_for_version_impl(version, chain);
2912            ret.version = version;
2913            ret = Self::apply_config_override(version, ret);
2914            Some(ret)
2915        } else {
2916            None
2917        }
2918    }
2919
2920    #[cfg(not(msim))]
2921    pub fn poison_get_for_min_version() {
2922        POISON_VERSION_METHODS.store(true, Ordering::Relaxed);
2923    }
2924
2925    #[cfg(not(msim))]
2926    fn load_poison_get_for_min_version() -> bool {
2927        POISON_VERSION_METHODS.load(Ordering::Relaxed)
2928    }
2929
2930    #[cfg(msim)]
2931    pub fn poison_get_for_min_version() {
2932        POISON_VERSION_METHODS.with(|p| p.store(true, Ordering::Relaxed));
2933    }
2934
2935    #[cfg(msim)]
2936    fn load_poison_get_for_min_version() -> bool {
2937        POISON_VERSION_METHODS.with(|p| p.load(Ordering::Relaxed))
2938    }
2939
2940    /// Convenience to get the constants at the current minimum supported version.
2941    /// Mainly used by client code that may not yet be protocol-version aware.
2942    pub fn get_for_min_version() -> Self {
2943        if Self::load_poison_get_for_min_version() {
2944            panic!("get_for_min_version called on validator");
2945        }
2946        ProtocolConfig::get_for_version(ProtocolVersion::MIN, Chain::Unknown)
2947    }
2948
2949    /// CAREFUL! - You probably want to use `get_for_version` instead.
2950    ///
2951    /// Convenience to get the constants at the current maximum supported version.
2952    /// Mainly used by genesis. Note well that this function uses the max version
2953    /// supported locally by the node, which is not necessarily the current version
2954    /// of the network. ALSO, this function disregards chain specific config (by
2955    /// using Chain::Unknown), thereby potentially returning a protocol config that
2956    /// is incorrect for some feature flags. Definitely safe for testing and for
2957    /// protocol version 11 and prior.
2958    #[allow(non_snake_case)]
2959    pub fn get_for_max_version_UNSAFE() -> Self {
2960        if Self::load_poison_get_for_min_version() {
2961            panic!("get_for_max_version_UNSAFE called on validator");
2962        }
2963        ProtocolConfig::get_for_version(ProtocolVersion::MAX, Chain::Unknown)
2964    }
2965
2966    fn get_for_version_impl(version: ProtocolVersion, chain: Chain) -> Self {
2967        #[cfg(msim)]
2968        {
2969            // populate the fake simulator version # with a different base tx cost.
2970            if version == ProtocolVersion::MAX_ALLOWED {
2971                let mut config = Self::get_for_version_impl(version - 1, Chain::Unknown);
2972                config.base_tx_cost_fixed = Some(config.base_tx_cost_fixed() + 1000);
2973                return config;
2974            }
2975        }
2976
2977        // IMPORTANT: Never modify the value of any constant for a pre-existing protocol version.
2978        // To change the values here you must create a new protocol version with the new values!
2979        let mut cfg = Self {
2980            // will be overwritten before being returned
2981            version,
2982
2983            // All flags are disabled in V1
2984            feature_flags: Default::default(),
2985
2986            max_tx_size_bytes: Some(128 * 1024),
2987            // We need this number to be at least 100x less than `max_serialized_tx_effects_size_bytes`otherwise effects can be huge
2988            max_input_objects: Some(2048),
2989            max_serialized_tx_effects_size_bytes: Some(512 * 1024),
2990            max_serialized_tx_effects_size_bytes_system_tx: Some(512 * 1024 * 16),
2991            max_gas_payment_objects: Some(256),
2992            max_modules_in_publish: Some(128),
2993            max_package_dependencies: None,
2994            max_arguments: Some(512),
2995            max_type_arguments: Some(16),
2996            max_type_argument_depth: Some(16),
2997            max_pure_argument_size: Some(16 * 1024),
2998            max_programmable_tx_commands: Some(1024),
2999            move_binary_format_version: Some(6),
3000            min_move_binary_format_version: None,
3001            binary_module_handles: None,
3002            binary_struct_handles: None,
3003            binary_function_handles: None,
3004            binary_function_instantiations: None,
3005            binary_signatures: None,
3006            binary_constant_pool: None,
3007            binary_identifiers: None,
3008            binary_address_identifiers: None,
3009            binary_struct_defs: None,
3010            binary_struct_def_instantiations: None,
3011            binary_function_defs: None,
3012            binary_field_handles: None,
3013            binary_field_instantiations: None,
3014            binary_friend_decls: None,
3015            binary_enum_defs: None,
3016            binary_enum_def_instantiations: None,
3017            binary_variant_handles: None,
3018            binary_variant_instantiation_handles: None,
3019            max_move_object_size: Some(250 * 1024),
3020            max_move_package_size: Some(100 * 1024),
3021            max_publish_or_upgrade_per_ptb: None,
3022            max_tx_gas: Some(10_000_000_000),
3023            max_gas_price: Some(100_000),
3024            max_gas_price_rgp_factor_for_aborted_transactions: None,
3025            max_gas_computation_bucket: Some(5_000_000),
3026            max_loop_depth: Some(5),
3027            max_generic_instantiation_length: Some(32),
3028            max_function_parameters: Some(128),
3029            max_basic_blocks: Some(1024),
3030            max_value_stack_size: Some(1024),
3031            max_type_nodes: Some(256),
3032            max_push_size: Some(10000),
3033            max_struct_definitions: Some(200),
3034            max_function_definitions: Some(1000),
3035            max_fields_in_struct: Some(32),
3036            max_dependency_depth: Some(100),
3037            max_num_event_emit: Some(256),
3038            max_num_new_move_object_ids: Some(2048),
3039            max_num_new_move_object_ids_system_tx: Some(2048 * 16),
3040            max_num_deleted_move_object_ids: Some(2048),
3041            max_num_deleted_move_object_ids_system_tx: Some(2048 * 16),
3042            max_num_transferred_move_object_ids: Some(2048),
3043            max_num_transferred_move_object_ids_system_tx: Some(2048 * 16),
3044            max_event_emit_size: Some(250 * 1024),
3045            max_move_vector_len: Some(256 * 1024),
3046            max_type_to_layout_nodes: None,
3047            max_ptb_value_size: None,
3048
3049            max_back_edges_per_function: Some(10_000),
3050            max_back_edges_per_module: Some(10_000),
3051            max_verifier_meter_ticks_per_function: Some(6_000_000),
3052            max_meter_ticks_per_module: Some(6_000_000),
3053            max_meter_ticks_per_package: None,
3054
3055            object_runtime_max_num_cached_objects: Some(1000),
3056            object_runtime_max_num_cached_objects_system_tx: Some(1000 * 16),
3057            object_runtime_max_num_store_entries: Some(1000),
3058            object_runtime_max_num_store_entries_system_tx: Some(1000 * 16),
3059            base_tx_cost_fixed: Some(110_000),
3060            package_publish_cost_fixed: Some(1_000),
3061            base_tx_cost_per_byte: Some(0),
3062            package_publish_cost_per_byte: Some(80),
3063            obj_access_cost_read_per_byte: Some(15),
3064            obj_access_cost_mutate_per_byte: Some(40),
3065            obj_access_cost_delete_per_byte: Some(40),
3066            obj_access_cost_verify_per_byte: Some(200),
3067            obj_data_cost_refundable: Some(100),
3068            obj_metadata_cost_non_refundable: Some(50),
3069            gas_model_version: Some(1),
3070            storage_rebate_rate: Some(9900),
3071            storage_fund_reinvest_rate: Some(500),
3072            reward_slashing_rate: Some(5000),
3073            storage_gas_price: Some(1),
3074            accumulator_object_storage_cost: None,
3075            max_transactions_per_checkpoint: Some(10_000),
3076            max_checkpoint_size_bytes: Some(30 * 1024 * 1024),
3077
3078            // For now, perform upgrades with a bare quorum of validators.
3079            // MUSTFIX: This number should be increased to at least 2000 (20%) for mainnet.
3080            buffer_stake_for_protocol_upgrade_bps: Some(0),
3081
3082            // === Native Function Costs ===
3083            // `address` module
3084            // Cost params for the Move native function `address::from_bytes(bytes: vector<u8>)`
3085            address_from_bytes_cost_base: Some(52),
3086            // Cost params for the Move native function `address::to_u256(address): u256`
3087            address_to_u256_cost_base: Some(52),
3088            // Cost params for the Move native function `address::from_u256(u256): address`
3089            address_from_u256_cost_base: Some(52),
3090
3091            // `config` module
3092            // Cost params for the Move native function `read_setting_impl``
3093            config_read_setting_impl_cost_base: None,
3094            config_read_setting_impl_cost_per_byte: None,
3095
3096            // `dynamic_field` module
3097            // Cost params for the Move native function `hash_type_and_key<K: copy + drop + store>(parent: address, k: K): address`
3098            dynamic_field_hash_type_and_key_cost_base: Some(100),
3099            dynamic_field_hash_type_and_key_type_cost_per_byte: Some(2),
3100            dynamic_field_hash_type_and_key_value_cost_per_byte: Some(2),
3101            dynamic_field_hash_type_and_key_type_tag_cost_per_byte: Some(2),
3102            // Cost params for the Move native function `add_child_object<Child: key>(parent: address, child: Child)`
3103            dynamic_field_add_child_object_cost_base: Some(100),
3104            dynamic_field_add_child_object_type_cost_per_byte: Some(10),
3105            dynamic_field_add_child_object_value_cost_per_byte: Some(10),
3106            dynamic_field_add_child_object_struct_tag_cost_per_byte: Some(10),
3107            // Cost params for the Move native function `borrow_child_object_mut<Child: key>(parent: &mut UID, id: address): &mut Child`
3108            dynamic_field_borrow_child_object_cost_base: Some(100),
3109            dynamic_field_borrow_child_object_child_ref_cost_per_byte: Some(10),
3110            dynamic_field_borrow_child_object_type_cost_per_byte: Some(10),
3111            // Cost params for the Move native function `remove_child_object<Child: key>(parent: address, id: address): Child`
3112            dynamic_field_remove_child_object_cost_base: Some(100),
3113            dynamic_field_remove_child_object_child_cost_per_byte: Some(2),
3114            dynamic_field_remove_child_object_type_cost_per_byte: Some(2),
3115            // Cost params for the Move native function `has_child_object(parent: address, id: address): bool`
3116            dynamic_field_has_child_object_cost_base: Some(100),
3117            // Cost params for the Move native function `has_child_object_with_ty<Child: key>(parent: address, id: address): bool`
3118            dynamic_field_has_child_object_with_ty_cost_base: Some(100),
3119            dynamic_field_has_child_object_with_ty_type_cost_per_byte: Some(2),
3120            dynamic_field_has_child_object_with_ty_type_tag_cost_per_byte: Some(2),
3121
3122            // `event` module
3123            // Cost params for the Move native function `event::emit<T: copy + drop>(event: T)`
3124            event_emit_cost_base: Some(52),
3125            event_emit_value_size_derivation_cost_per_byte: Some(2),
3126            event_emit_tag_size_derivation_cost_per_byte: Some(5),
3127            event_emit_output_cost_per_byte: Some(10),
3128            event_emit_auth_stream_cost: None,
3129
3130            //  `object` module
3131            // Cost params for the Move native function `borrow_uid<T: key>(obj: &T): &UID`
3132            object_borrow_uid_cost_base: Some(52),
3133            // Cost params for the Move native function `delete_impl(id: address)`
3134            object_delete_impl_cost_base: Some(52),
3135            // Cost params for the Move native function `record_new_uid(id: address)`
3136            object_record_new_uid_cost_base: Some(52),
3137
3138            // `transfer` module
3139            // Cost params for the Move native function `transfer_impl<T: key>(obj: T, recipient: address)`
3140            transfer_transfer_internal_cost_base: Some(52),
3141            // Cost params for the Move native function `party_transfer_impl<T: key>(obj: T, party_members: vector<address>)`
3142            transfer_party_transfer_internal_cost_base: None,
3143            // Cost params for the Move native function `freeze_object<T: key>(obj: T)`
3144            transfer_freeze_object_cost_base: Some(52),
3145            // Cost params for the Move native function `share_object<T: key>(obj: T)`
3146            transfer_share_object_cost_base: Some(52),
3147            transfer_receive_object_cost_base: None,
3148            transfer_receive_object_type_cost_per_byte: None,
3149            transfer_receive_object_cost_per_byte: None,
3150
3151            // `tx_context` module
3152            // Cost params for the Move native function `transfer_impl<T: key>(obj: T, recipient: address)`
3153            tx_context_derive_id_cost_base: Some(52),
3154            tx_context_fresh_id_cost_base: None,
3155            tx_context_sender_cost_base: None,
3156            tx_context_epoch_cost_base: None,
3157            tx_context_epoch_timestamp_ms_cost_base: None,
3158            tx_context_sponsor_cost_base: None,
3159            tx_context_rgp_cost_base: None,
3160            tx_context_gas_price_cost_base: None,
3161            tx_context_gas_budget_cost_base: None,
3162            tx_context_ids_created_cost_base: None,
3163            tx_context_replace_cost_base: None,
3164
3165            // `types` module
3166            // Cost params for the Move native function `is_one_time_witness<T: drop>(_: &T): bool`
3167            types_is_one_time_witness_cost_base: Some(52),
3168            types_is_one_time_witness_type_tag_cost_per_byte: Some(2),
3169            types_is_one_time_witness_type_cost_per_byte: Some(2),
3170
3171            // `validator` module
3172            // Cost params for the Move native function `validate_metadata_bcs(metadata: vector<u8>)`
3173            validator_validate_metadata_cost_base: Some(52),
3174            validator_validate_metadata_data_cost_per_byte: Some(2),
3175
3176            // Crypto
3177            crypto_invalid_arguments_cost: Some(100),
3178            // bls12381::bls12381_min_pk_verify
3179            bls12381_bls12381_min_sig_verify_cost_base: Some(52),
3180            bls12381_bls12381_min_sig_verify_msg_cost_per_byte: Some(2),
3181            bls12381_bls12381_min_sig_verify_msg_cost_per_block: Some(2),
3182
3183            // bls12381::bls12381_min_pk_verify
3184            bls12381_bls12381_min_pk_verify_cost_base: Some(52),
3185            bls12381_bls12381_min_pk_verify_msg_cost_per_byte: Some(2),
3186            bls12381_bls12381_min_pk_verify_msg_cost_per_block: Some(2),
3187
3188            // ecdsa_k1::ecrecover
3189            ecdsa_k1_ecrecover_keccak256_cost_base: Some(52),
3190            ecdsa_k1_ecrecover_keccak256_msg_cost_per_byte: Some(2),
3191            ecdsa_k1_ecrecover_keccak256_msg_cost_per_block: Some(2),
3192            ecdsa_k1_ecrecover_sha256_cost_base: Some(52),
3193            ecdsa_k1_ecrecover_sha256_msg_cost_per_byte: Some(2),
3194            ecdsa_k1_ecrecover_sha256_msg_cost_per_block: Some(2),
3195
3196            // ecdsa_k1::decompress_pubkey
3197            ecdsa_k1_decompress_pubkey_cost_base: Some(52),
3198
3199            // ecdsa_k1::secp256k1_verify
3200            ecdsa_k1_secp256k1_verify_keccak256_cost_base: Some(52),
3201            ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_byte: Some(2),
3202            ecdsa_k1_secp256k1_verify_keccak256_msg_cost_per_block: Some(2),
3203            ecdsa_k1_secp256k1_verify_sha256_cost_base: Some(52),
3204            ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_byte: Some(2),
3205            ecdsa_k1_secp256k1_verify_sha256_msg_cost_per_block: Some(2),
3206
3207            // ecdsa_r1::ecrecover
3208            ecdsa_r1_ecrecover_keccak256_cost_base: Some(52),
3209            ecdsa_r1_ecrecover_keccak256_msg_cost_per_byte: Some(2),
3210            ecdsa_r1_ecrecover_keccak256_msg_cost_per_block: Some(2),
3211            ecdsa_r1_ecrecover_sha256_cost_base: Some(52),
3212            ecdsa_r1_ecrecover_sha256_msg_cost_per_byte: Some(2),
3213            ecdsa_r1_ecrecover_sha256_msg_cost_per_block: Some(2),
3214
3215            // ecdsa_r1::secp256k1_verify
3216            ecdsa_r1_secp256r1_verify_keccak256_cost_base: Some(52),
3217            ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_byte: Some(2),
3218            ecdsa_r1_secp256r1_verify_keccak256_msg_cost_per_block: Some(2),
3219            ecdsa_r1_secp256r1_verify_sha256_cost_base: Some(52),
3220            ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_byte: Some(2),
3221            ecdsa_r1_secp256r1_verify_sha256_msg_cost_per_block: Some(2),
3222
3223            // ecvrf::verify
3224            ecvrf_ecvrf_verify_cost_base: Some(52),
3225            ecvrf_ecvrf_verify_alpha_string_cost_per_byte: Some(2),
3226            ecvrf_ecvrf_verify_alpha_string_cost_per_block: Some(2),
3227
3228            // ed25519
3229            ed25519_ed25519_verify_cost_base: Some(52),
3230            ed25519_ed25519_verify_msg_cost_per_byte: Some(2),
3231            ed25519_ed25519_verify_msg_cost_per_block: Some(2),
3232
3233            // groth16::prepare_verifying_key
3234            groth16_prepare_verifying_key_bls12381_cost_base: Some(52),
3235            groth16_prepare_verifying_key_bn254_cost_base: Some(52),
3236
3237            // groth16::verify_groth16_proof_internal
3238            groth16_verify_groth16_proof_internal_bls12381_cost_base: Some(52),
3239            groth16_verify_groth16_proof_internal_bls12381_cost_per_public_input: Some(2),
3240            groth16_verify_groth16_proof_internal_bn254_cost_base: Some(52),
3241            groth16_verify_groth16_proof_internal_bn254_cost_per_public_input: Some(2),
3242            groth16_verify_groth16_proof_internal_public_input_cost_per_byte: Some(2),
3243
3244            // hash::blake2b256
3245            hash_blake2b256_cost_base: Some(52),
3246            hash_blake2b256_data_cost_per_byte: Some(2),
3247            hash_blake2b256_data_cost_per_block: Some(2),
3248
3249            // hash::keccak256
3250            hash_keccak256_cost_base: Some(52),
3251            hash_keccak256_data_cost_per_byte: Some(2),
3252            hash_keccak256_data_cost_per_block: Some(2),
3253
3254            poseidon_bn254_cost_base: None,
3255            poseidon_bn254_cost_per_block: None,
3256
3257            // hmac::hmac_sha3_256
3258            hmac_hmac_sha3_256_cost_base: Some(52),
3259            hmac_hmac_sha3_256_input_cost_per_byte: Some(2),
3260            hmac_hmac_sha3_256_input_cost_per_block: Some(2),
3261
3262            // group ops
3263            group_ops_bls12381_decode_scalar_cost: None,
3264            group_ops_bls12381_decode_g1_cost: None,
3265            group_ops_bls12381_decode_g2_cost: None,
3266            group_ops_bls12381_decode_gt_cost: None,
3267            group_ops_bls12381_scalar_add_cost: None,
3268            group_ops_bls12381_g1_add_cost: None,
3269            group_ops_bls12381_g2_add_cost: None,
3270            group_ops_bls12381_gt_add_cost: None,
3271            group_ops_bls12381_scalar_sub_cost: None,
3272            group_ops_bls12381_g1_sub_cost: None,
3273            group_ops_bls12381_g2_sub_cost: None,
3274            group_ops_bls12381_gt_sub_cost: None,
3275            group_ops_bls12381_scalar_mul_cost: None,
3276            group_ops_bls12381_g1_mul_cost: None,
3277            group_ops_bls12381_g2_mul_cost: None,
3278            group_ops_bls12381_gt_mul_cost: None,
3279            group_ops_bls12381_scalar_div_cost: None,
3280            group_ops_bls12381_g1_div_cost: None,
3281            group_ops_bls12381_g2_div_cost: None,
3282            group_ops_bls12381_gt_div_cost: None,
3283            group_ops_bls12381_g1_hash_to_base_cost: None,
3284            group_ops_bls12381_g2_hash_to_base_cost: None,
3285            group_ops_bls12381_g1_hash_to_cost_per_byte: None,
3286            group_ops_bls12381_g2_hash_to_cost_per_byte: None,
3287            group_ops_bls12381_g1_msm_base_cost: None,
3288            group_ops_bls12381_g2_msm_base_cost: None,
3289            group_ops_bls12381_g1_msm_base_cost_per_input: None,
3290            group_ops_bls12381_g2_msm_base_cost_per_input: None,
3291            group_ops_bls12381_msm_max_len: None,
3292            group_ops_bls12381_pairing_cost: None,
3293            group_ops_bls12381_g1_to_uncompressed_g1_cost: None,
3294            group_ops_bls12381_uncompressed_g1_to_g1_cost: None,
3295            group_ops_bls12381_uncompressed_g1_sum_base_cost: None,
3296            group_ops_bls12381_uncompressed_g1_sum_cost_per_term: None,
3297            group_ops_bls12381_uncompressed_g1_sum_max_terms: None,
3298
3299            group_ops_ristretto_decode_scalar_cost: None,
3300            group_ops_ristretto_decode_point_cost: None,
3301            group_ops_ristretto_scalar_add_cost: None,
3302            group_ops_ristretto_point_add_cost: None,
3303            group_ops_ristretto_scalar_sub_cost: None,
3304            group_ops_ristretto_point_sub_cost: None,
3305            group_ops_ristretto_scalar_mul_cost: None,
3306            group_ops_ristretto_point_mul_cost: None,
3307            group_ops_ristretto_scalar_div_cost: None,
3308            group_ops_ristretto_point_div_cost: None,
3309
3310            verify_bulletproofs_ristretto255_base_cost: None,
3311            verify_bulletproofs_ristretto255_cost_per_bit_and_commitment: None,
3312
3313            // zklogin::check_zklogin_id
3314            check_zklogin_id_cost_base: None,
3315            // zklogin::check_zklogin_issuer
3316            check_zklogin_issuer_cost_base: None,
3317
3318            vdf_verify_vdf_cost: None,
3319            vdf_hash_to_input_cost: None,
3320
3321            // nitro_attestation::verify_nitro_attestation
3322            nitro_attestation_parse_base_cost: None,
3323            nitro_attestation_parse_cost_per_byte: None,
3324            nitro_attestation_verify_base_cost: None,
3325            nitro_attestation_verify_cost_per_cert: None,
3326
3327            bcs_per_byte_serialized_cost: None,
3328            bcs_legacy_min_output_size_cost: None,
3329            bcs_failure_cost: None,
3330            hash_sha2_256_base_cost: None,
3331            hash_sha2_256_per_byte_cost: None,
3332            hash_sha2_256_legacy_min_input_len_cost: None,
3333            hash_sha3_256_base_cost: None,
3334            hash_sha3_256_per_byte_cost: None,
3335            hash_sha3_256_legacy_min_input_len_cost: None,
3336            type_name_get_base_cost: None,
3337            type_name_get_per_byte_cost: None,
3338            type_name_id_base_cost: None,
3339            string_check_utf8_base_cost: None,
3340            string_check_utf8_per_byte_cost: None,
3341            string_is_char_boundary_base_cost: None,
3342            string_sub_string_base_cost: None,
3343            string_sub_string_per_byte_cost: None,
3344            string_index_of_base_cost: None,
3345            string_index_of_per_byte_pattern_cost: None,
3346            string_index_of_per_byte_searched_cost: None,
3347            vector_empty_base_cost: None,
3348            vector_length_base_cost: None,
3349            vector_push_back_base_cost: None,
3350            vector_push_back_legacy_per_abstract_memory_unit_cost: None,
3351            vector_borrow_base_cost: None,
3352            vector_pop_back_base_cost: None,
3353            vector_destroy_empty_base_cost: None,
3354            vector_swap_base_cost: None,
3355            debug_print_base_cost: None,
3356            debug_print_stack_trace_base_cost: None,
3357
3358            max_size_written_objects: None,
3359            max_size_written_objects_system_tx: None,
3360
3361            // ==== Ephemeral (consensus only) params deleted ====
3362            // Const params for consensus scoring decision
3363            // scoring_decision_mad_divisor: None,
3364            // scoring_decision_cutoff_value: None,
3365
3366            // Limits the length of a Move identifier
3367            max_move_identifier_len: None,
3368            max_move_value_depth: None,
3369            max_move_enum_variants: None,
3370
3371            gas_rounding_step: None,
3372
3373            execution_version: None,
3374
3375            max_event_emit_size_total: None,
3376
3377            consensus_bad_nodes_stake_threshold: None,
3378
3379            max_jwk_votes_per_validator_per_epoch: None,
3380
3381            max_age_of_jwk_in_epochs: None,
3382
3383            random_beacon_reduction_allowed_delta: None,
3384
3385            random_beacon_reduction_lower_bound: None,
3386
3387            random_beacon_dkg_timeout_round: None,
3388
3389            random_beacon_min_round_interval_ms: None,
3390
3391            random_beacon_dkg_version: None,
3392
3393            consensus_max_transaction_size_bytes: None,
3394
3395            consensus_max_transactions_in_block_bytes: None,
3396
3397            consensus_max_num_transactions_in_block: None,
3398
3399            consensus_voting_rounds: None,
3400
3401            max_accumulated_txn_cost_per_object_in_narwhal_commit: None,
3402
3403            max_deferral_rounds_for_congestion_control: None,
3404
3405            max_txn_cost_overage_per_object_in_commit: None,
3406
3407            allowed_txn_cost_overage_burst_per_object_in_commit: None,
3408
3409            min_checkpoint_interval_ms: None,
3410
3411            checkpoint_summary_version_specific_data: None,
3412
3413            max_soft_bundle_size: None,
3414
3415            bridge_should_try_to_finalize_committee: None,
3416
3417            max_accumulated_txn_cost_per_object_in_mysticeti_commit: None,
3418
3419            max_accumulated_randomness_txn_cost_per_object_in_mysticeti_commit: None,
3420
3421            consensus_gc_depth: None,
3422
3423            gas_budget_based_txn_cost_cap_factor: None,
3424
3425            gas_budget_based_txn_cost_absolute_cap_commit_count: None,
3426
3427            sip_45_consensus_amplification_threshold: None,
3428
3429            use_object_per_epoch_marker_table_v2: None,
3430
3431            consensus_commit_rate_estimation_window_size: None,
3432
3433            aliased_addresses: vec![],
3434
3435            translation_per_command_base_charge: None,
3436            translation_per_input_base_charge: None,
3437            translation_pure_input_per_byte_charge: None,
3438            translation_per_type_node_charge: None,
3439            translation_per_reference_node_charge: None,
3440            translation_per_linkage_entry_charge: None,
3441
3442            max_updates_per_settlement_txn: None,
3443
3444            gasless_max_computation_units: None,
3445            gasless_allowed_token_types: None,
3446            gasless_max_unused_inputs: None,
3447            gasless_max_pure_input_bytes: None,
3448            gasless_max_tps: None,
3449            include_special_package_amendments: None,
3450            gasless_max_tx_size_bytes: None,
3451            // When adding a new constant, set it to None in the earliest version, like this:
3452            // new_constant: None,
3453        };
3454        for cur in 2..=version.0 {
3455            match cur {
3456                1 => unreachable!(),
3457                2 => {
3458                    cfg.feature_flags.advance_epoch_start_time_in_safe_mode = true;
3459                }
3460                3 => {
3461                    // changes for gas model
3462                    cfg.gas_model_version = Some(2);
3463                    // max gas budget is in MIST and an absolute value 50SUI
3464                    cfg.max_tx_gas = Some(50_000_000_000);
3465                    // min gas budget is in MIST and an absolute value 2000MIST or 0.000002SUI
3466                    cfg.base_tx_cost_fixed = Some(2_000);
3467                    // storage gas price multiplier
3468                    cfg.storage_gas_price = Some(76);
3469                    cfg.feature_flags.loaded_child_objects_fixed = true;
3470                    // max size of written objects during a TXn
3471                    // this is a sum of all objects written during a TXn
3472                    cfg.max_size_written_objects = Some(5 * 1000 * 1000);
3473                    // max size of written objects during a system TXn to allow for larger writes
3474                    // akin to `max_size_written_objects` but for system TXns
3475                    cfg.max_size_written_objects_system_tx = Some(50 * 1000 * 1000);
3476                    cfg.feature_flags.package_upgrades = true;
3477                }
3478                // This is the first protocol version currently possible.
3479                // Mainnet starts with version 4. Previous versions are pre mainnet and have
3480                // all been wiped out.
3481                // Every other chain is after version 4.
3482                4 => {
3483                    // Change reward slashing rate to 100%.
3484                    cfg.reward_slashing_rate = Some(10000);
3485                    // protect old and new lookup for object version
3486                    cfg.gas_model_version = Some(3);
3487                }
3488                5 => {
3489                    cfg.feature_flags.missing_type_is_compatibility_error = true;
3490                    cfg.gas_model_version = Some(4);
3491                    cfg.feature_flags.scoring_decision_with_validity_cutoff = true;
3492                    // ==== Ephemeral (consensus only) params deleted ====
3493                    // cfg.scoring_decision_mad_divisor = Some(2.3);
3494                    // cfg.scoring_decision_cutoff_value = Some(2.5);
3495                }
3496                6 => {
3497                    cfg.gas_model_version = Some(5);
3498                    cfg.buffer_stake_for_protocol_upgrade_bps = Some(5000);
3499                    cfg.feature_flags.consensus_order_end_of_epoch_last = true;
3500                }
3501                7 => {
3502                    cfg.feature_flags.disallow_adding_abilities_on_upgrade = true;
3503                    cfg.feature_flags
3504                        .disable_invariant_violation_check_in_swap_loc = true;
3505                    cfg.feature_flags.ban_entry_init = true;
3506                    cfg.feature_flags.package_digest_hash_module = true;
3507                }
3508                8 => {
3509                    cfg.feature_flags
3510                        .disallow_change_struct_type_params_on_upgrade = true;
3511                }
3512                9 => {
3513                    // Limits the length of a Move identifier
3514                    cfg.max_move_identifier_len = Some(128);
3515                    cfg.feature_flags.no_extraneous_module_bytes = true;
3516                    cfg.feature_flags
3517                        .advance_to_highest_supported_protocol_version = true;
3518                }
3519                10 => {
3520                    cfg.max_verifier_meter_ticks_per_function = Some(16_000_000);
3521                    cfg.max_meter_ticks_per_module = Some(16_000_000);
3522                }
3523                11 => {
3524                    cfg.max_move_value_depth = Some(128);
3525                }
3526                12 => {
3527                    cfg.feature_flags.narwhal_versioned_metadata = true;
3528                    if chain != Chain::Mainnet {
3529                        cfg.feature_flags.commit_root_state_digest = true;
3530                    }
3531
3532                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3533                        cfg.feature_flags.zklogin_auth = true;
3534                    }
3535                }
3536                13 => {}
3537                14 => {
3538                    cfg.gas_rounding_step = Some(1_000);
3539                    cfg.gas_model_version = Some(6);
3540                }
3541                15 => {
3542                    cfg.feature_flags.consensus_transaction_ordering =
3543                        ConsensusTransactionOrdering::ByGasPrice;
3544                }
3545                16 => {
3546                    cfg.feature_flags.simplified_unwrap_then_delete = true;
3547                }
3548                17 => {
3549                    cfg.feature_flags.upgraded_multisig_supported = true;
3550                }
3551                18 => {
3552                    cfg.execution_version = Some(1);
3553                    // Following flags are implied by this execution version.  Once support for earlier
3554                    // protocol versions is dropped, these flags can be removed:
3555                    // cfg.feature_flags.package_upgrades = true;
3556                    // cfg.feature_flags.disallow_adding_abilities_on_upgrade = true;
3557                    // cfg.feature_flags.disallow_change_struct_type_params_on_upgrade = true;
3558                    // cfg.feature_flags.loaded_child_objects_fixed = true;
3559                    // cfg.feature_flags.ban_entry_init = true;
3560                    // cfg.feature_flags.pack_digest_hash_modules = true;
3561                    cfg.feature_flags.txn_base_cost_as_multiplier = true;
3562                    // this is a multiplier of the gas price
3563                    cfg.base_tx_cost_fixed = Some(1_000);
3564                }
3565                19 => {
3566                    cfg.max_num_event_emit = Some(1024);
3567                    // We maintain the same total size limit for events, but increase the number of
3568                    // events that can be emitted.
3569                    cfg.max_event_emit_size_total = Some(
3570                        256 /* former event count limit */ * 250 * 1024, /* size limit per event */
3571                    );
3572                }
3573                20 => {
3574                    cfg.feature_flags.commit_root_state_digest = true;
3575
3576                    if chain != Chain::Mainnet {
3577                        cfg.feature_flags.narwhal_new_leader_election_schedule = true;
3578                        cfg.consensus_bad_nodes_stake_threshold = Some(20);
3579                    }
3580                }
3581
3582                21 => {
3583                    if chain != Chain::Mainnet {
3584                        cfg.feature_flags.zklogin_supported_providers = BTreeSet::from([
3585                            "Google".to_string(),
3586                            "Facebook".to_string(),
3587                            "Twitch".to_string(),
3588                        ]);
3589                    }
3590                }
3591                22 => {
3592                    cfg.feature_flags.loaded_child_object_format = true;
3593                }
3594                23 => {
3595                    cfg.feature_flags.loaded_child_object_format_type = true;
3596                    cfg.feature_flags.narwhal_new_leader_election_schedule = true;
3597                    // Taking a baby step approach, we consider only 20% by stake as bad nodes so we
3598                    // have a 80% by stake of nodes participating in the leader committee. That allow
3599                    // us for more redundancy in case we have validators under performing - since the
3600                    // responsibility is shared amongst more nodes. We can increase that once we do have
3601                    // higher confidence.
3602                    cfg.consensus_bad_nodes_stake_threshold = Some(20);
3603                }
3604                24 => {
3605                    cfg.feature_flags.simple_conservation_checks = true;
3606                    cfg.max_publish_or_upgrade_per_ptb = Some(5);
3607
3608                    cfg.feature_flags.end_of_epoch_transaction_supported = true;
3609
3610                    if chain != Chain::Mainnet {
3611                        cfg.feature_flags.enable_jwk_consensus_updates = true;
3612                        // Max of 10 votes per hour
3613                        cfg.max_jwk_votes_per_validator_per_epoch = Some(240);
3614                        cfg.max_age_of_jwk_in_epochs = Some(1);
3615                    }
3616                }
3617                25 => {
3618                    // Enable zkLogin for all providers in all networks.
3619                    cfg.feature_flags.zklogin_supported_providers = BTreeSet::from([
3620                        "Google".to_string(),
3621                        "Facebook".to_string(),
3622                        "Twitch".to_string(),
3623                    ]);
3624                    cfg.feature_flags.zklogin_auth = true;
3625
3626                    // Enable jwk consensus updates
3627                    cfg.feature_flags.enable_jwk_consensus_updates = true;
3628                    cfg.max_jwk_votes_per_validator_per_epoch = Some(240);
3629                    cfg.max_age_of_jwk_in_epochs = Some(1);
3630                }
3631                26 => {
3632                    cfg.gas_model_version = Some(7);
3633                    // Only enable receiving objects in devnet
3634                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3635                        cfg.transfer_receive_object_cost_base = Some(52);
3636                        cfg.feature_flags.receive_objects = true;
3637                    }
3638                }
3639                27 => {
3640                    cfg.gas_model_version = Some(8);
3641                }
3642                28 => {
3643                    // zklogin::check_zklogin_id
3644                    cfg.check_zklogin_id_cost_base = Some(200);
3645                    // zklogin::check_zklogin_issuer
3646                    cfg.check_zklogin_issuer_cost_base = Some(200);
3647
3648                    // Only enable effects v2 on devnet.
3649                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3650                        cfg.feature_flags.enable_effects_v2 = true;
3651                    }
3652                }
3653                29 => {
3654                    cfg.feature_flags.verify_legacy_zklogin_address = true;
3655                }
3656                30 => {
3657                    // Only enable nw certificate v2 on testnet.
3658                    if chain != Chain::Mainnet {
3659                        cfg.feature_flags.narwhal_certificate_v2 = true;
3660                    }
3661
3662                    cfg.random_beacon_reduction_allowed_delta = Some(800);
3663                    // Only enable effects v2 on devnet and testnet.
3664                    if chain != Chain::Mainnet {
3665                        cfg.feature_flags.enable_effects_v2 = true;
3666                    }
3667
3668                    // zklogin_supported_providers config is deprecated, zklogin
3669                    // signature verifier will use the fetched jwk map to determine
3670                    // whether the provider is supported based on node config.
3671                    cfg.feature_flags.zklogin_supported_providers = BTreeSet::default();
3672
3673                    cfg.feature_flags.recompute_has_public_transfer_in_execution = true;
3674                }
3675                31 => {
3676                    cfg.execution_version = Some(2);
3677                    // Only enable shared object deletion on devnet
3678                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3679                        cfg.feature_flags.shared_object_deletion = true;
3680                    }
3681                }
3682                32 => {
3683                    // enable zklogin in multisig in devnet and testnet
3684                    if chain != Chain::Mainnet {
3685                        cfg.feature_flags.accept_zklogin_in_multisig = true;
3686                    }
3687                    // enable receiving objects in devnet and testnet
3688                    if chain != Chain::Mainnet {
3689                        cfg.transfer_receive_object_cost_base = Some(52);
3690                        cfg.feature_flags.receive_objects = true;
3691                    }
3692                    // Only enable random beacon on devnet
3693                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3694                        cfg.feature_flags.random_beacon = true;
3695                        cfg.random_beacon_reduction_lower_bound = Some(1600);
3696                        cfg.random_beacon_dkg_timeout_round = Some(3000);
3697                        cfg.random_beacon_min_round_interval_ms = Some(150);
3698                    }
3699                    // Only enable consensus digest in consensus commit prologue in devnet.
3700                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3701                        cfg.feature_flags.include_consensus_digest_in_prologue = true;
3702                    }
3703
3704                    // enable nw cert v2 on mainnet
3705                    cfg.feature_flags.narwhal_certificate_v2 = true;
3706                }
3707                33 => {
3708                    cfg.feature_flags.hardened_otw_check = true;
3709                    cfg.feature_flags.allow_receiving_object_id = true;
3710
3711                    // Enable transfer-to-object in mainnet
3712                    cfg.transfer_receive_object_cost_base = Some(52);
3713                    cfg.feature_flags.receive_objects = true;
3714
3715                    // Enable shared object deletion in testnet and devnet
3716                    if chain != Chain::Mainnet {
3717                        cfg.feature_flags.shared_object_deletion = true;
3718                    }
3719
3720                    cfg.feature_flags.enable_effects_v2 = true;
3721                }
3722                34 => {}
3723                35 => {
3724                    // Add costs for poseidon::poseidon_bn254
3725                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3726                        cfg.feature_flags.enable_poseidon = true;
3727                        cfg.poseidon_bn254_cost_base = Some(260);
3728                        cfg.poseidon_bn254_cost_per_block = Some(10);
3729                    }
3730
3731                    cfg.feature_flags.enable_coin_deny_list = true;
3732                }
3733                36 => {
3734                    // Only enable group ops on devnet
3735                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3736                        cfg.feature_flags.enable_group_ops_native_functions = true;
3737                        cfg.feature_flags.enable_group_ops_native_function_msm = true;
3738                        // Next values are arbitrary in a similar way as the other crypto native functions.
3739                        cfg.group_ops_bls12381_decode_scalar_cost = Some(52);
3740                        cfg.group_ops_bls12381_decode_g1_cost = Some(52);
3741                        cfg.group_ops_bls12381_decode_g2_cost = Some(52);
3742                        cfg.group_ops_bls12381_decode_gt_cost = Some(52);
3743                        cfg.group_ops_bls12381_scalar_add_cost = Some(52);
3744                        cfg.group_ops_bls12381_g1_add_cost = Some(52);
3745                        cfg.group_ops_bls12381_g2_add_cost = Some(52);
3746                        cfg.group_ops_bls12381_gt_add_cost = Some(52);
3747                        cfg.group_ops_bls12381_scalar_sub_cost = Some(52);
3748                        cfg.group_ops_bls12381_g1_sub_cost = Some(52);
3749                        cfg.group_ops_bls12381_g2_sub_cost = Some(52);
3750                        cfg.group_ops_bls12381_gt_sub_cost = Some(52);
3751                        cfg.group_ops_bls12381_scalar_mul_cost = Some(52);
3752                        cfg.group_ops_bls12381_g1_mul_cost = Some(52);
3753                        cfg.group_ops_bls12381_g2_mul_cost = Some(52);
3754                        cfg.group_ops_bls12381_gt_mul_cost = Some(52);
3755                        cfg.group_ops_bls12381_scalar_div_cost = Some(52);
3756                        cfg.group_ops_bls12381_g1_div_cost = Some(52);
3757                        cfg.group_ops_bls12381_g2_div_cost = Some(52);
3758                        cfg.group_ops_bls12381_gt_div_cost = Some(52);
3759                        cfg.group_ops_bls12381_g1_hash_to_base_cost = Some(52);
3760                        cfg.group_ops_bls12381_g2_hash_to_base_cost = Some(52);
3761                        cfg.group_ops_bls12381_g1_hash_to_cost_per_byte = Some(2);
3762                        cfg.group_ops_bls12381_g2_hash_to_cost_per_byte = Some(2);
3763                        cfg.group_ops_bls12381_g1_msm_base_cost = Some(52);
3764                        cfg.group_ops_bls12381_g2_msm_base_cost = Some(52);
3765                        cfg.group_ops_bls12381_g1_msm_base_cost_per_input = Some(52);
3766                        cfg.group_ops_bls12381_g2_msm_base_cost_per_input = Some(52);
3767                        cfg.group_ops_bls12381_msm_max_len = Some(32);
3768                        cfg.group_ops_bls12381_pairing_cost = Some(52);
3769                    }
3770                    // Enable shared object deletion on all networks.
3771                    cfg.feature_flags.shared_object_deletion = true;
3772
3773                    cfg.consensus_max_transaction_size_bytes = Some(256 * 1024); // 256KB
3774                    cfg.consensus_max_transactions_in_block_bytes = Some(6 * 1_024 * 1024);
3775                    // 6 MB
3776                }
3777                37 => {
3778                    cfg.feature_flags.reject_mutable_random_on_entry_functions = true;
3779
3780                    // Enable consensus digest in consensus commit prologue in testnet and devnet.
3781                    if chain != Chain::Mainnet {
3782                        cfg.feature_flags.include_consensus_digest_in_prologue = true;
3783                    }
3784                }
3785                38 => {
3786                    cfg.binary_module_handles = Some(100);
3787                    cfg.binary_struct_handles = Some(300);
3788                    cfg.binary_function_handles = Some(1500);
3789                    cfg.binary_function_instantiations = Some(750);
3790                    cfg.binary_signatures = Some(1000);
3791                    // constants and identifiers are proportional to the binary size,
3792                    // and they vastly depend on the code, so we are leaving them
3793                    // reasonably high
3794                    cfg.binary_constant_pool = Some(4000);
3795                    cfg.binary_identifiers = Some(10000);
3796                    cfg.binary_address_identifiers = Some(100);
3797                    cfg.binary_struct_defs = Some(200);
3798                    cfg.binary_struct_def_instantiations = Some(100);
3799                    cfg.binary_function_defs = Some(1000);
3800                    cfg.binary_field_handles = Some(500);
3801                    cfg.binary_field_instantiations = Some(250);
3802                    cfg.binary_friend_decls = Some(100);
3803                    // reduce dependencies maximum
3804                    cfg.max_package_dependencies = Some(32);
3805                    cfg.max_modules_in_publish = Some(64);
3806                    // bump execution version
3807                    cfg.execution_version = Some(3);
3808                }
3809                39 => {
3810                    // It is important that we keep this protocol version blank due to an issue with random.move.
3811                }
3812                40 => {}
3813                41 => {
3814                    // Enable group ops and all networks (but not msm)
3815                    cfg.feature_flags.enable_group_ops_native_functions = true;
3816                    // Next values are arbitrary in a similar way as the other crypto native functions.
3817                    cfg.group_ops_bls12381_decode_scalar_cost = Some(52);
3818                    cfg.group_ops_bls12381_decode_g1_cost = Some(52);
3819                    cfg.group_ops_bls12381_decode_g2_cost = Some(52);
3820                    cfg.group_ops_bls12381_decode_gt_cost = Some(52);
3821                    cfg.group_ops_bls12381_scalar_add_cost = Some(52);
3822                    cfg.group_ops_bls12381_g1_add_cost = Some(52);
3823                    cfg.group_ops_bls12381_g2_add_cost = Some(52);
3824                    cfg.group_ops_bls12381_gt_add_cost = Some(52);
3825                    cfg.group_ops_bls12381_scalar_sub_cost = Some(52);
3826                    cfg.group_ops_bls12381_g1_sub_cost = Some(52);
3827                    cfg.group_ops_bls12381_g2_sub_cost = Some(52);
3828                    cfg.group_ops_bls12381_gt_sub_cost = Some(52);
3829                    cfg.group_ops_bls12381_scalar_mul_cost = Some(52);
3830                    cfg.group_ops_bls12381_g1_mul_cost = Some(52);
3831                    cfg.group_ops_bls12381_g2_mul_cost = Some(52);
3832                    cfg.group_ops_bls12381_gt_mul_cost = Some(52);
3833                    cfg.group_ops_bls12381_scalar_div_cost = Some(52);
3834                    cfg.group_ops_bls12381_g1_div_cost = Some(52);
3835                    cfg.group_ops_bls12381_g2_div_cost = Some(52);
3836                    cfg.group_ops_bls12381_gt_div_cost = Some(52);
3837                    cfg.group_ops_bls12381_g1_hash_to_base_cost = Some(52);
3838                    cfg.group_ops_bls12381_g2_hash_to_base_cost = Some(52);
3839                    cfg.group_ops_bls12381_g1_hash_to_cost_per_byte = Some(2);
3840                    cfg.group_ops_bls12381_g2_hash_to_cost_per_byte = Some(2);
3841                    cfg.group_ops_bls12381_g1_msm_base_cost = Some(52);
3842                    cfg.group_ops_bls12381_g2_msm_base_cost = Some(52);
3843                    cfg.group_ops_bls12381_g1_msm_base_cost_per_input = Some(52);
3844                    cfg.group_ops_bls12381_g2_msm_base_cost_per_input = Some(52);
3845                    cfg.group_ops_bls12381_msm_max_len = Some(32);
3846                    cfg.group_ops_bls12381_pairing_cost = Some(52);
3847                }
3848                42 => {}
3849                43 => {
3850                    cfg.feature_flags.zklogin_max_epoch_upper_bound_delta = Some(30);
3851                    cfg.max_meter_ticks_per_package = Some(16_000_000);
3852                }
3853                44 => {
3854                    // Enable consensus digest in consensus commit prologue on all networks..
3855                    cfg.feature_flags.include_consensus_digest_in_prologue = true;
3856                    // Switch between Narwhal and Mysticeti per epoch in tests, devnet and testnet.
3857                    if chain != Chain::Mainnet {
3858                        cfg.feature_flags.consensus_choice = ConsensusChoice::SwapEachEpoch;
3859                    }
3860                }
3861                45 => {
3862                    // Use tonic networking for consensus, in tests and devnet.
3863                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3864                        cfg.feature_flags.consensus_network = ConsensusNetwork::Tonic;
3865                    }
3866
3867                    if chain != Chain::Mainnet {
3868                        // Enable leader scoring & schedule change on testnet for mysticeti.
3869                        cfg.feature_flags.mysticeti_leader_scoring_and_schedule = true;
3870                    }
3871                    cfg.min_move_binary_format_version = Some(6);
3872                    cfg.feature_flags.accept_zklogin_in_multisig = true;
3873
3874                    // Also bumps framework snapshot to fix binop issue.
3875
3876                    // enable bridge in devnet
3877                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3878                        cfg.feature_flags.bridge = true;
3879                    }
3880                }
3881                46 => {
3882                    // enable bridge in devnet and testnet
3883                    if chain != Chain::Mainnet {
3884                        cfg.feature_flags.bridge = true;
3885                    }
3886
3887                    // Enable resharing at same initial version
3888                    cfg.feature_flags.reshare_at_same_initial_version = true;
3889                }
3890                47 => {}
3891                48 => {
3892                    // Use tonic networking for Mysticeti.
3893                    cfg.feature_flags.consensus_network = ConsensusNetwork::Tonic;
3894
3895                    // Enable resolving abort code IDs to package ID instead of runtime module ID
3896                    cfg.feature_flags.resolve_abort_locations_to_package_id = true;
3897
3898                    // Enable random beacon on testnet.
3899                    if chain != Chain::Mainnet {
3900                        cfg.feature_flags.random_beacon = true;
3901                        cfg.random_beacon_reduction_lower_bound = Some(1600);
3902                        cfg.random_beacon_dkg_timeout_round = Some(3000);
3903                        cfg.random_beacon_min_round_interval_ms = Some(200);
3904                    }
3905
3906                    // Enable the committed sub dag digest inclusion on the commit output
3907                    cfg.feature_flags.mysticeti_use_committed_subdag_digest = true;
3908                }
3909                49 => {
3910                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3911                        cfg.move_binary_format_version = Some(7);
3912                    }
3913
3914                    // enable vdf in devnet
3915                    if chain != Chain::Mainnet && chain != Chain::Testnet {
3916                        cfg.feature_flags.enable_vdf = true;
3917                        // Set to 30x and 2x the cost of a signature verification for now. This
3918                        // should be updated along with other native crypto functions.
3919                        cfg.vdf_verify_vdf_cost = Some(1500);
3920                        cfg.vdf_hash_to_input_cost = Some(100);
3921                    }
3922
3923                    // Only enable consensus commit prologue V3 in devnet.
3924                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3925                        cfg.feature_flags
3926                            .record_consensus_determined_version_assignments_in_prologue = true;
3927                    }
3928
3929                    // Run Mysticeti consensus in testnet.
3930                    if chain != Chain::Mainnet {
3931                        cfg.feature_flags.consensus_choice = ConsensusChoice::Mysticeti;
3932                    }
3933
3934                    // Run Move verification on framework upgrades in its own VM
3935                    cfg.feature_flags.fresh_vm_on_framework_upgrade = true;
3936                }
3937                50 => {
3938                    // Enable checkpoint batching in testnet.
3939                    if chain != Chain::Mainnet {
3940                        cfg.checkpoint_summary_version_specific_data = Some(1);
3941                        cfg.min_checkpoint_interval_ms = Some(200);
3942                    }
3943
3944                    // Only enable prepose consensus commit prologue in checkpoints in devnet.
3945                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3946                        cfg.feature_flags
3947                            .prepend_prologue_tx_in_consensus_commit_in_checkpoints = true;
3948                    }
3949
3950                    cfg.feature_flags.mysticeti_num_leaders_per_round = Some(1);
3951
3952                    // Set max transaction deferral to 10 consensus rounds.
3953                    cfg.max_deferral_rounds_for_congestion_control = Some(10);
3954                }
3955                51 => {
3956                    cfg.random_beacon_dkg_version = Some(1);
3957
3958                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3959                        cfg.feature_flags.enable_coin_deny_list_v2 = true;
3960                    }
3961                }
3962                52 => {
3963                    if chain != Chain::Mainnet {
3964                        cfg.feature_flags.soft_bundle = true;
3965                        cfg.max_soft_bundle_size = Some(5);
3966                    }
3967
3968                    cfg.config_read_setting_impl_cost_base = Some(100);
3969                    cfg.config_read_setting_impl_cost_per_byte = Some(40);
3970
3971                    // Turn on shared object congestion control in devnet.
3972                    if chain != Chain::Testnet && chain != Chain::Mainnet {
3973                        cfg.max_accumulated_txn_cost_per_object_in_narwhal_commit = Some(100);
3974                        cfg.feature_flags.per_object_congestion_control_mode =
3975                            PerObjectCongestionControlMode::TotalTxCount;
3976                    }
3977
3978                    // Enable Mysticeti on mainnet.
3979                    cfg.feature_flags.consensus_choice = ConsensusChoice::Mysticeti;
3980
3981                    // Enable leader scoring & schedule change on mainnet for mysticeti.
3982                    cfg.feature_flags.mysticeti_leader_scoring_and_schedule = true;
3983
3984                    // Enable checkpoint batching on mainnet.
3985                    cfg.checkpoint_summary_version_specific_data = Some(1);
3986                    cfg.min_checkpoint_interval_ms = Some(200);
3987
3988                    // Enable consensus commit prologue V3 in testnet.
3989                    if chain != Chain::Mainnet {
3990                        cfg.feature_flags
3991                            .record_consensus_determined_version_assignments_in_prologue = true;
3992                        cfg.feature_flags
3993                            .prepend_prologue_tx_in_consensus_commit_in_checkpoints = true;
3994                    }
3995                    // Turn on enums in testnet and devnet
3996                    if chain != Chain::Mainnet {
3997                        cfg.move_binary_format_version = Some(7);
3998                    }
3999
4000                    if chain != Chain::Testnet && chain != Chain::Mainnet {
4001                        cfg.feature_flags.passkey_auth = true;
4002                    }
4003                    cfg.feature_flags.enable_coin_deny_list_v2 = true;
4004                }
4005                53 => {
4006                    // Do not allow bridge committee to finalize on mainnet.
4007                    cfg.bridge_should_try_to_finalize_committee = Some(chain != Chain::Mainnet);
4008
4009                    // Enable consensus commit prologue V3 on mainnet.
4010                    cfg.feature_flags
4011                        .record_consensus_determined_version_assignments_in_prologue = true;
4012                    cfg.feature_flags
4013                        .prepend_prologue_tx_in_consensus_commit_in_checkpoints = true;
4014
4015                    if chain == Chain::Unknown {
4016                        cfg.feature_flags.authority_capabilities_v2 = true;
4017                    }
4018
4019                    // Turns on shared object congestion control on testnet.
4020                    if chain != Chain::Mainnet {
4021                        cfg.max_accumulated_txn_cost_per_object_in_narwhal_commit = Some(100);
4022                        cfg.max_accumulated_txn_cost_per_object_in_mysticeti_commit = Some(10);
4023                        cfg.feature_flags.per_object_congestion_control_mode =
4024                            PerObjectCongestionControlMode::TotalTxCount;
4025                    }
4026
4027                    // Adjust stdlib gas costs
4028                    cfg.bcs_per_byte_serialized_cost = Some(2);
4029                    cfg.bcs_legacy_min_output_size_cost = Some(1);
4030                    cfg.bcs_failure_cost = Some(52);
4031                    cfg.debug_print_base_cost = Some(52);
4032                    cfg.debug_print_stack_trace_base_cost = Some(52);
4033                    cfg.hash_sha2_256_base_cost = Some(52);
4034                    cfg.hash_sha2_256_per_byte_cost = Some(2);
4035                    cfg.hash_sha2_256_legacy_min_input_len_cost = Some(1);
4036                    cfg.hash_sha3_256_base_cost = Some(52);
4037                    cfg.hash_sha3_256_per_byte_cost = Some(2);
4038                    cfg.hash_sha3_256_legacy_min_input_len_cost = Some(1);
4039                    cfg.type_name_get_base_cost = Some(52);
4040                    cfg.type_name_get_per_byte_cost = Some(2);
4041                    cfg.string_check_utf8_base_cost = Some(52);
4042                    cfg.string_check_utf8_per_byte_cost = Some(2);
4043                    cfg.string_is_char_boundary_base_cost = Some(52);
4044                    cfg.string_sub_string_base_cost = Some(52);
4045                    cfg.string_sub_string_per_byte_cost = Some(2);
4046                    cfg.string_index_of_base_cost = Some(52);
4047                    cfg.string_index_of_per_byte_pattern_cost = Some(2);
4048                    cfg.string_index_of_per_byte_searched_cost = Some(2);
4049                    cfg.vector_empty_base_cost = Some(52);
4050                    cfg.vector_length_base_cost = Some(52);
4051                    cfg.vector_push_back_base_cost = Some(52);
4052                    cfg.vector_push_back_legacy_per_abstract_memory_unit_cost = Some(2);
4053                    cfg.vector_borrow_base_cost = Some(52);
4054                    cfg.vector_pop_back_base_cost = Some(52);
4055                    cfg.vector_destroy_empty_base_cost = Some(52);
4056                    cfg.vector_swap_base_cost = Some(52);
4057                }
4058                54 => {
4059                    // Enable random beacon on mainnet.
4060                    cfg.feature_flags.random_beacon = true;
4061                    cfg.random_beacon_reduction_lower_bound = Some(1000);
4062                    cfg.random_beacon_dkg_timeout_round = Some(3000);
4063                    cfg.random_beacon_min_round_interval_ms = Some(500);
4064
4065                    // Turns on shared object congestion control on mainnet.
4066                    cfg.max_accumulated_txn_cost_per_object_in_narwhal_commit = Some(100);
4067                    cfg.max_accumulated_txn_cost_per_object_in_mysticeti_commit = Some(10);
4068                    cfg.feature_flags.per_object_congestion_control_mode =
4069                        PerObjectCongestionControlMode::TotalTxCount;
4070
4071                    // Enable soft bundle on mainnet.
4072                    cfg.feature_flags.soft_bundle = true;
4073                    cfg.max_soft_bundle_size = Some(5);
4074                }
4075                55 => {
4076                    // Turn on enums mainnet
4077                    cfg.move_binary_format_version = Some(7);
4078
4079                    // Assume 1KB per transaction and 500 transactions per block.
4080                    cfg.consensus_max_transactions_in_block_bytes = Some(512 * 1024);
4081                    // Assume 20_000 TPS * 5% max stake per validator / (minimum) 4 blocks per round = 250 transactions per block maximum
4082                    // Using a higher limit that is 512, to account for bursty traffic and system transactions.
4083                    cfg.consensus_max_num_transactions_in_block = Some(512);
4084
4085                    cfg.feature_flags.rethrow_serialization_type_layout_errors = true;
4086                }
4087                56 => {
4088                    if chain == Chain::Mainnet {
4089                        cfg.feature_flags.bridge = true;
4090                    }
4091                }
4092                57 => {
4093                    // Reduce minimum number of random beacon shares.
4094                    cfg.random_beacon_reduction_lower_bound = Some(800);
4095                }
4096                58 => {
4097                    if chain == Chain::Mainnet {
4098                        cfg.bridge_should_try_to_finalize_committee = Some(true);
4099                    }
4100
4101                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4102                        // Enable distributed vote scoring for devnet
4103                        cfg.feature_flags
4104                            .consensus_distributed_vote_scoring_strategy = true;
4105                    }
4106                }
4107                59 => {
4108                    // Enable round prober in consensus.
4109                    cfg.feature_flags.consensus_round_prober = true;
4110                }
4111                60 => {
4112                    cfg.max_type_to_layout_nodes = Some(512);
4113                    cfg.feature_flags.validate_identifier_inputs = true;
4114                }
4115                61 => {
4116                    if chain != Chain::Mainnet {
4117                        // Enable distributed vote scoring for testnet
4118                        cfg.feature_flags
4119                            .consensus_distributed_vote_scoring_strategy = true;
4120                    }
4121                    // Further reduce minimum number of random beacon shares.
4122                    cfg.random_beacon_reduction_lower_bound = Some(700);
4123
4124                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4125                        // Enable Mysticeti fastpath for devnet
4126                        cfg.feature_flags.mysticeti_fastpath = true;
4127                    }
4128                }
4129                62 => {
4130                    cfg.feature_flags.relocate_event_module = true;
4131                }
4132                63 => {
4133                    cfg.feature_flags.per_object_congestion_control_mode =
4134                        PerObjectCongestionControlMode::TotalGasBudgetWithCap;
4135                    cfg.gas_budget_based_txn_cost_cap_factor = Some(400_000);
4136                    cfg.max_accumulated_txn_cost_per_object_in_mysticeti_commit = Some(18_500_000);
4137                    cfg.max_accumulated_txn_cost_per_object_in_narwhal_commit = Some(240_000_000);
4138                }
4139                64 => {
4140                    cfg.feature_flags.per_object_congestion_control_mode =
4141                        PerObjectCongestionControlMode::TotalTxCount;
4142                    cfg.max_accumulated_txn_cost_per_object_in_narwhal_commit = Some(40);
4143                    cfg.max_accumulated_txn_cost_per_object_in_mysticeti_commit = Some(3);
4144                }
4145                65 => {
4146                    // Enable distributed vote scoring for mainnet
4147                    cfg.feature_flags
4148                        .consensus_distributed_vote_scoring_strategy = true;
4149                }
4150                66 => {
4151                    if chain == Chain::Mainnet {
4152                        // Revert the distributed vote scoring for mainnet (for one protocol upgrade)
4153                        cfg.feature_flags
4154                            .consensus_distributed_vote_scoring_strategy = false;
4155                    }
4156                }
4157                67 => {
4158                    // Enable it once again.
4159                    cfg.feature_flags
4160                        .consensus_distributed_vote_scoring_strategy = true;
4161                }
4162                68 => {
4163                    cfg.group_ops_bls12381_g1_to_uncompressed_g1_cost = Some(26);
4164                    cfg.group_ops_bls12381_uncompressed_g1_to_g1_cost = Some(52);
4165                    cfg.group_ops_bls12381_uncompressed_g1_sum_base_cost = Some(26);
4166                    cfg.group_ops_bls12381_uncompressed_g1_sum_cost_per_term = Some(13);
4167                    cfg.group_ops_bls12381_uncompressed_g1_sum_max_terms = Some(2000);
4168
4169                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4170                        cfg.feature_flags.uncompressed_g1_group_elements = true;
4171                    }
4172
4173                    cfg.feature_flags.per_object_congestion_control_mode =
4174                        PerObjectCongestionControlMode::TotalGasBudgetWithCap;
4175                    cfg.gas_budget_based_txn_cost_cap_factor = Some(400_000);
4176                    cfg.max_accumulated_txn_cost_per_object_in_mysticeti_commit = Some(18_500_000);
4177                    cfg.max_accumulated_randomness_txn_cost_per_object_in_mysticeti_commit =
4178                        Some(3_700_000); // 20% of above
4179                    cfg.max_txn_cost_overage_per_object_in_commit = Some(u64::MAX);
4180                    cfg.gas_budget_based_txn_cost_absolute_cap_commit_count = Some(50);
4181
4182                    // Further reduce minimum number of random beacon shares.
4183                    cfg.random_beacon_reduction_lower_bound = Some(500);
4184
4185                    cfg.feature_flags.disallow_new_modules_in_deps_only_packages = true;
4186                }
4187                69 => {
4188                    // Sets number of rounds allowed for fastpath voting in consensus.
4189                    cfg.consensus_voting_rounds = Some(40);
4190
4191                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4192                        // Enable smart ancestor selection for devnet
4193                        cfg.feature_flags.consensus_smart_ancestor_selection = true;
4194                    }
4195
4196                    if chain != Chain::Mainnet {
4197                        cfg.feature_flags.uncompressed_g1_group_elements = true;
4198                    }
4199                }
4200                70 => {
4201                    if chain != Chain::Mainnet {
4202                        // Enable smart ancestor selection for testnet
4203                        cfg.feature_flags.consensus_smart_ancestor_selection = true;
4204                        // Enable probing for accepted rounds in round prober for testnet
4205                        cfg.feature_flags
4206                            .consensus_round_prober_probe_accepted_rounds = true;
4207                    }
4208
4209                    cfg.poseidon_bn254_cost_per_block = Some(388);
4210
4211                    cfg.gas_model_version = Some(9);
4212                    cfg.feature_flags.native_charging_v2 = true;
4213                    cfg.bls12381_bls12381_min_sig_verify_cost_base = Some(44064);
4214                    cfg.bls12381_bls12381_min_pk_verify_cost_base = Some(49282);
4215                    cfg.ecdsa_k1_secp256k1_verify_keccak256_cost_base = Some(1470);
4216                    cfg.ecdsa_k1_secp256k1_verify_sha256_cost_base = Some(1470);
4217                    cfg.ecdsa_r1_secp256r1_verify_sha256_cost_base = Some(4225);
4218                    cfg.ecdsa_r1_secp256r1_verify_keccak256_cost_base = Some(4225);
4219                    cfg.ecvrf_ecvrf_verify_cost_base = Some(4848);
4220                    cfg.ed25519_ed25519_verify_cost_base = Some(1802);
4221
4222                    // Manually changed to be "under cost"
4223                    cfg.ecdsa_r1_ecrecover_keccak256_cost_base = Some(1173);
4224                    cfg.ecdsa_r1_ecrecover_sha256_cost_base = Some(1173);
4225                    cfg.ecdsa_k1_ecrecover_keccak256_cost_base = Some(500);
4226                    cfg.ecdsa_k1_ecrecover_sha256_cost_base = Some(500);
4227
4228                    cfg.groth16_prepare_verifying_key_bls12381_cost_base = Some(53838);
4229                    cfg.groth16_prepare_verifying_key_bn254_cost_base = Some(82010);
4230                    cfg.groth16_verify_groth16_proof_internal_bls12381_cost_base = Some(72090);
4231                    cfg.groth16_verify_groth16_proof_internal_bls12381_cost_per_public_input =
4232                        Some(8213);
4233                    cfg.groth16_verify_groth16_proof_internal_bn254_cost_base = Some(115502);
4234                    cfg.groth16_verify_groth16_proof_internal_bn254_cost_per_public_input =
4235                        Some(9484);
4236
4237                    cfg.hash_keccak256_cost_base = Some(10);
4238                    cfg.hash_blake2b256_cost_base = Some(10);
4239
4240                    // group ops
4241                    cfg.group_ops_bls12381_decode_scalar_cost = Some(7);
4242                    cfg.group_ops_bls12381_decode_g1_cost = Some(2848);
4243                    cfg.group_ops_bls12381_decode_g2_cost = Some(3770);
4244                    cfg.group_ops_bls12381_decode_gt_cost = Some(3068);
4245
4246                    cfg.group_ops_bls12381_scalar_add_cost = Some(10);
4247                    cfg.group_ops_bls12381_g1_add_cost = Some(1556);
4248                    cfg.group_ops_bls12381_g2_add_cost = Some(3048);
4249                    cfg.group_ops_bls12381_gt_add_cost = Some(188);
4250
4251                    cfg.group_ops_bls12381_scalar_sub_cost = Some(10);
4252                    cfg.group_ops_bls12381_g1_sub_cost = Some(1550);
4253                    cfg.group_ops_bls12381_g2_sub_cost = Some(3019);
4254                    cfg.group_ops_bls12381_gt_sub_cost = Some(497);
4255
4256                    cfg.group_ops_bls12381_scalar_mul_cost = Some(11);
4257                    cfg.group_ops_bls12381_g1_mul_cost = Some(4842);
4258                    cfg.group_ops_bls12381_g2_mul_cost = Some(9108);
4259                    cfg.group_ops_bls12381_gt_mul_cost = Some(27490);
4260
4261                    cfg.group_ops_bls12381_scalar_div_cost = Some(91);
4262                    cfg.group_ops_bls12381_g1_div_cost = Some(5091);
4263                    cfg.group_ops_bls12381_g2_div_cost = Some(9206);
4264                    cfg.group_ops_bls12381_gt_div_cost = Some(27804);
4265
4266                    cfg.group_ops_bls12381_g1_hash_to_base_cost = Some(2962);
4267                    cfg.group_ops_bls12381_g2_hash_to_base_cost = Some(8688);
4268
4269                    cfg.group_ops_bls12381_g1_msm_base_cost = Some(62648);
4270                    cfg.group_ops_bls12381_g2_msm_base_cost = Some(131192);
4271                    cfg.group_ops_bls12381_g1_msm_base_cost_per_input = Some(1333);
4272                    cfg.group_ops_bls12381_g2_msm_base_cost_per_input = Some(3216);
4273
4274                    cfg.group_ops_bls12381_uncompressed_g1_to_g1_cost = Some(677);
4275                    cfg.group_ops_bls12381_g1_to_uncompressed_g1_cost = Some(2099);
4276                    cfg.group_ops_bls12381_uncompressed_g1_sum_base_cost = Some(77);
4277                    cfg.group_ops_bls12381_uncompressed_g1_sum_cost_per_term = Some(26);
4278
4279                    cfg.group_ops_bls12381_pairing_cost = Some(26897);
4280                    cfg.group_ops_bls12381_uncompressed_g1_sum_max_terms = Some(1200);
4281
4282                    cfg.validator_validate_metadata_cost_base = Some(20000);
4283                }
4284                71 => {
4285                    cfg.sip_45_consensus_amplification_threshold = Some(5);
4286
4287                    // Enable bursts for congestion control. (10x the per-commit budget)
4288                    cfg.allowed_txn_cost_overage_burst_per_object_in_commit = Some(185_000_000);
4289                }
4290                72 => {
4291                    cfg.feature_flags.convert_type_argument_error = true;
4292
4293                    // Invariant: max_gas_price * base_tx_cost_fixed <= max_tx_gas
4294                    // max gas budget is in MIST and an absolute value 50_000 SUI
4295                    cfg.max_tx_gas = Some(50_000_000_000_000);
4296                    // max gas price is in MIST and an absolute value 50 SUI
4297                    cfg.max_gas_price = Some(50_000_000_000);
4298
4299                    cfg.feature_flags.variant_nodes = true;
4300                }
4301                73 => {
4302                    // Enable new marker table version.
4303                    cfg.use_object_per_epoch_marker_table_v2 = Some(true);
4304
4305                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4306                        // Assuming a round rate of max 15/sec, then using a gc depth of 60 allow blocks within a window of ~4 seconds
4307                        // to be included before be considered garbage collected.
4308                        cfg.consensus_gc_depth = Some(60);
4309                    }
4310
4311                    if chain != Chain::Mainnet {
4312                        // Enable zstd compression for consensus in testnet
4313                        cfg.feature_flags.consensus_zstd_compression = true;
4314                    }
4315
4316                    // Enable smart ancestor selection for mainnet
4317                    cfg.feature_flags.consensus_smart_ancestor_selection = true;
4318                    // Enable probing for accepted rounds in round prober for mainnet
4319                    cfg.feature_flags
4320                        .consensus_round_prober_probe_accepted_rounds = true;
4321
4322                    // Increase congestion control budget.
4323                    cfg.feature_flags.per_object_congestion_control_mode =
4324                        PerObjectCongestionControlMode::TotalGasBudgetWithCap;
4325                    cfg.gas_budget_based_txn_cost_cap_factor = Some(400_000);
4326                    cfg.max_accumulated_txn_cost_per_object_in_mysticeti_commit = Some(37_000_000);
4327                    cfg.max_accumulated_randomness_txn_cost_per_object_in_mysticeti_commit =
4328                        Some(7_400_000); // 20% of above
4329                    cfg.max_txn_cost_overage_per_object_in_commit = Some(u64::MAX);
4330                    cfg.gas_budget_based_txn_cost_absolute_cap_commit_count = Some(50);
4331                    cfg.allowed_txn_cost_overage_burst_per_object_in_commit = Some(370_000_000);
4332                }
4333                74 => {
4334                    // Enable nitro attestation verify native move function for devnet
4335                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4336                        cfg.feature_flags.enable_nitro_attestation = true;
4337                    }
4338                    cfg.nitro_attestation_parse_base_cost = Some(53 * 50);
4339                    cfg.nitro_attestation_parse_cost_per_byte = Some(50);
4340                    cfg.nitro_attestation_verify_base_cost = Some(49632 * 50);
4341                    cfg.nitro_attestation_verify_cost_per_cert = Some(52369 * 50);
4342
4343                    // Enable zstd compression for consensus in mainnet
4344                    cfg.feature_flags.consensus_zstd_compression = true;
4345
4346                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4347                        cfg.feature_flags.consensus_linearize_subdag_v2 = true;
4348                    }
4349                }
4350                75 => {
4351                    if chain != Chain::Mainnet {
4352                        cfg.feature_flags.passkey_auth = true;
4353                    }
4354                }
4355                76 => {
4356                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4357                        cfg.feature_flags.record_additional_state_digest_in_prologue = true;
4358                        cfg.consensus_commit_rate_estimation_window_size = Some(10);
4359                    }
4360                    cfg.feature_flags.minimize_child_object_mutations = true;
4361
4362                    if chain != Chain::Mainnet {
4363                        cfg.feature_flags.accept_passkey_in_multisig = true;
4364                    }
4365                }
4366                77 => {
4367                    cfg.feature_flags.uncompressed_g1_group_elements = true;
4368
4369                    if chain != Chain::Mainnet {
4370                        cfg.consensus_gc_depth = Some(60);
4371                        cfg.feature_flags.consensus_linearize_subdag_v2 = true;
4372                    }
4373                }
4374                78 => {
4375                    cfg.feature_flags.move_native_context = true;
4376                    cfg.tx_context_fresh_id_cost_base = Some(52);
4377                    cfg.tx_context_sender_cost_base = Some(30);
4378                    cfg.tx_context_epoch_cost_base = Some(30);
4379                    cfg.tx_context_epoch_timestamp_ms_cost_base = Some(30);
4380                    cfg.tx_context_sponsor_cost_base = Some(30);
4381                    cfg.tx_context_gas_price_cost_base = Some(30);
4382                    cfg.tx_context_gas_budget_cost_base = Some(30);
4383                    cfg.tx_context_ids_created_cost_base = Some(30);
4384                    cfg.tx_context_replace_cost_base = Some(30);
4385                    cfg.gas_model_version = Some(10);
4386
4387                    if chain != Chain::Mainnet {
4388                        cfg.feature_flags.record_additional_state_digest_in_prologue = true;
4389                        cfg.consensus_commit_rate_estimation_window_size = Some(10);
4390
4391                        // Enable execution time estimate mode for congestion control on testnet.
4392                        cfg.feature_flags.per_object_congestion_control_mode =
4393                            PerObjectCongestionControlMode::ExecutionTimeEstimate(
4394                                ExecutionTimeEstimateParams {
4395                                    target_utilization: 30,
4396                                    allowed_txn_cost_overage_burst_limit_us: 100_000, // 100 ms
4397                                    randomness_scalar: 20,
4398                                    max_estimate_us: 1_500_000, // 1.5s
4399                                    stored_observations_num_included_checkpoints: 10,
4400                                    stored_observations_limit: u64::MAX,
4401                                    stake_weighted_median_threshold: 0,
4402                                    default_none_duration_for_new_keys: false,
4403                                    observations_chunk_size: None,
4404                                },
4405                            );
4406                    }
4407                }
4408                79 => {
4409                    if chain != Chain::Mainnet {
4410                        cfg.feature_flags.consensus_median_based_commit_timestamp = true;
4411
4412                        // Increase threshold for bad nodes that won't be considered
4413                        // leaders in consensus in testnet
4414                        cfg.consensus_bad_nodes_stake_threshold = Some(30);
4415
4416                        cfg.feature_flags.consensus_batched_block_sync = true;
4417
4418                        // Enable verify nitro attestation in testnet.
4419                        cfg.feature_flags.enable_nitro_attestation = true
4420                    }
4421                    cfg.feature_flags.normalize_ptb_arguments = true;
4422
4423                    cfg.consensus_gc_depth = Some(60);
4424                    cfg.feature_flags.consensus_linearize_subdag_v2 = true;
4425                }
4426                80 => {
4427                    cfg.max_ptb_value_size = Some(1024 * 1024);
4428                }
4429                81 => {
4430                    cfg.feature_flags.consensus_median_based_commit_timestamp = true;
4431                    cfg.feature_flags.enforce_checkpoint_timestamp_monotonicity = true;
4432                    cfg.consensus_bad_nodes_stake_threshold = Some(30)
4433                }
4434                82 => {
4435                    cfg.feature_flags.max_ptb_value_size_v2 = true;
4436                }
4437                83 => {
4438                    if chain == Chain::Mainnet {
4439                        // The address that will sign the recovery transaction.
4440                        let aliased: [u8; 32] = Hex::decode(
4441                            "0x0b2da327ba6a4cacbe75dddd50e6e8bbf81d6496e92d66af9154c61c77f7332f",
4442                        )
4443                        .unwrap()
4444                        .try_into()
4445                        .unwrap();
4446
4447                        // Allow aliasing for the two addresses that contain stolen funds.
4448                        cfg.aliased_addresses.push(AliasedAddress {
4449                            original: Hex::decode("0xcd8962dad278d8b50fa0f9eb0186bfa4cbdecc6d59377214c88d0286a0ac9562").unwrap().try_into().unwrap(),
4450                            aliased,
4451                            allowed_tx_digests: vec![
4452                                Base58::decode("B2eGLFoMHgj93Ni8dAJBfqGzo8EWSTLBesZzhEpTPA4").unwrap().try_into().unwrap(),
4453                            ],
4454                        });
4455
4456                        cfg.aliased_addresses.push(AliasedAddress {
4457                            original: Hex::decode("0xe28b50cef1d633ea43d3296a3f6b67ff0312a5f1a99f0af753c85b8b5de8ff06").unwrap().try_into().unwrap(),
4458                            aliased,
4459                            allowed_tx_digests: vec![
4460                                Base58::decode("J4QqSAgp7VrQtQpMy5wDX4QGsCSEZu3U5KuDAkbESAge").unwrap().try_into().unwrap(),
4461                            ],
4462                        });
4463                    }
4464
4465                    // These features had to be deferred to v84 for mainnet in order to ship the recovery protocol
4466                    // upgrade as a patch to 1.48
4467                    if chain != Chain::Mainnet {
4468                        cfg.feature_flags.resolve_type_input_ids_to_defining_id = true;
4469                        cfg.transfer_party_transfer_internal_cost_base = Some(52);
4470
4471                        // Enable execution time estimate mode for congestion control on mainnet.
4472                        cfg.feature_flags.record_additional_state_digest_in_prologue = true;
4473                        cfg.consensus_commit_rate_estimation_window_size = Some(10);
4474                        cfg.feature_flags.per_object_congestion_control_mode =
4475                            PerObjectCongestionControlMode::ExecutionTimeEstimate(
4476                                ExecutionTimeEstimateParams {
4477                                    target_utilization: 30,
4478                                    allowed_txn_cost_overage_burst_limit_us: 100_000, // 100 ms
4479                                    randomness_scalar: 20,
4480                                    max_estimate_us: 1_500_000, // 1.5s
4481                                    stored_observations_num_included_checkpoints: 10,
4482                                    stored_observations_limit: u64::MAX,
4483                                    stake_weighted_median_threshold: 0,
4484                                    default_none_duration_for_new_keys: false,
4485                                    observations_chunk_size: None,
4486                                },
4487                            );
4488
4489                        // Enable the new depth-first block sync logic.
4490                        cfg.feature_flags.consensus_batched_block_sync = true;
4491
4492                        // Enable nitro attestation upgraded parsing logic and enable the
4493                        // native function on mainnet.
4494                        cfg.feature_flags.enable_nitro_attestation_upgraded_parsing = true;
4495                        cfg.feature_flags.enable_nitro_attestation = true;
4496                    }
4497                }
4498                84 => {
4499                    if chain == Chain::Mainnet {
4500                        cfg.feature_flags.resolve_type_input_ids_to_defining_id = true;
4501                        cfg.transfer_party_transfer_internal_cost_base = Some(52);
4502
4503                        // Enable execution time estimate mode for congestion control on mainnet.
4504                        cfg.feature_flags.record_additional_state_digest_in_prologue = true;
4505                        cfg.consensus_commit_rate_estimation_window_size = Some(10);
4506                        cfg.feature_flags.per_object_congestion_control_mode =
4507                            PerObjectCongestionControlMode::ExecutionTimeEstimate(
4508                                ExecutionTimeEstimateParams {
4509                                    target_utilization: 30,
4510                                    allowed_txn_cost_overage_burst_limit_us: 100_000, // 100 ms
4511                                    randomness_scalar: 20,
4512                                    max_estimate_us: 1_500_000, // 1.5s
4513                                    stored_observations_num_included_checkpoints: 10,
4514                                    stored_observations_limit: u64::MAX,
4515                                    stake_weighted_median_threshold: 0,
4516                                    default_none_duration_for_new_keys: false,
4517                                    observations_chunk_size: None,
4518                                },
4519                            );
4520
4521                        // Enable the new depth-first block sync logic.
4522                        cfg.feature_flags.consensus_batched_block_sync = true;
4523
4524                        // Enable nitro attestation upgraded parsing logic and enable the
4525                        // native function on mainnet.
4526                        cfg.feature_flags.enable_nitro_attestation_upgraded_parsing = true;
4527                        cfg.feature_flags.enable_nitro_attestation = true;
4528                    }
4529
4530                    // Limit the number of stored execution time observations at end of epoch.
4531                    cfg.feature_flags.per_object_congestion_control_mode =
4532                        PerObjectCongestionControlMode::ExecutionTimeEstimate(
4533                            ExecutionTimeEstimateParams {
4534                                target_utilization: 30,
4535                                allowed_txn_cost_overage_burst_limit_us: 100_000, // 100 ms
4536                                randomness_scalar: 20,
4537                                max_estimate_us: 1_500_000, // 1.5s
4538                                stored_observations_num_included_checkpoints: 10,
4539                                stored_observations_limit: 20,
4540                                stake_weighted_median_threshold: 0,
4541                                default_none_duration_for_new_keys: false,
4542                                observations_chunk_size: None,
4543                            },
4544                        );
4545                    cfg.feature_flags.allow_unbounded_system_objects = true;
4546                }
4547                85 => {
4548                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4549                        cfg.feature_flags.enable_party_transfer = true;
4550                    }
4551
4552                    cfg.feature_flags
4553                        .record_consensus_determined_version_assignments_in_prologue_v2 = true;
4554                    cfg.feature_flags.disallow_self_identifier = true;
4555                    cfg.feature_flags.per_object_congestion_control_mode =
4556                        PerObjectCongestionControlMode::ExecutionTimeEstimate(
4557                            ExecutionTimeEstimateParams {
4558                                target_utilization: 50,
4559                                allowed_txn_cost_overage_burst_limit_us: 500_000, // 500 ms
4560                                randomness_scalar: 20,
4561                                max_estimate_us: 1_500_000, // 1.5s
4562                                stored_observations_num_included_checkpoints: 10,
4563                                stored_observations_limit: 20,
4564                                stake_weighted_median_threshold: 0,
4565                                default_none_duration_for_new_keys: false,
4566                                observations_chunk_size: None,
4567                            },
4568                        );
4569                }
4570                86 => {
4571                    cfg.feature_flags.type_tags_in_object_runtime = true;
4572                    cfg.max_move_enum_variants = Some(move_core_types::VARIANT_COUNT_MAX);
4573
4574                    // Set a stake_weighted_median_threshold for congestion control.
4575                    cfg.feature_flags.per_object_congestion_control_mode =
4576                        PerObjectCongestionControlMode::ExecutionTimeEstimate(
4577                            ExecutionTimeEstimateParams {
4578                                target_utilization: 50,
4579                                allowed_txn_cost_overage_burst_limit_us: 500_000, // 500 ms
4580                                randomness_scalar: 20,
4581                                max_estimate_us: 1_500_000, // 1.5s
4582                                stored_observations_num_included_checkpoints: 10,
4583                                stored_observations_limit: 20,
4584                                stake_weighted_median_threshold: 3334,
4585                                default_none_duration_for_new_keys: false,
4586                                observations_chunk_size: None,
4587                            },
4588                        );
4589                    // Enable party transfer for testnet.
4590                    if chain != Chain::Mainnet {
4591                        cfg.feature_flags.enable_party_transfer = true;
4592                    }
4593                }
4594                87 => {
4595                    if chain == Chain::Mainnet {
4596                        cfg.feature_flags.record_time_estimate_processed = true;
4597                    }
4598                    cfg.feature_flags.better_adapter_type_resolution_errors = true;
4599                }
4600                88 => {
4601                    cfg.feature_flags.record_time_estimate_processed = true;
4602                    cfg.tx_context_rgp_cost_base = Some(30);
4603                    cfg.feature_flags
4604                        .ignore_execution_time_observations_after_certs_closed = true;
4605
4606                    // Disable backwards compatible behavior in execution time estimator for
4607                    // new protocol version.
4608                    cfg.feature_flags.per_object_congestion_control_mode =
4609                        PerObjectCongestionControlMode::ExecutionTimeEstimate(
4610                            ExecutionTimeEstimateParams {
4611                                target_utilization: 50,
4612                                allowed_txn_cost_overage_burst_limit_us: 500_000, // 500 ms
4613                                randomness_scalar: 20,
4614                                max_estimate_us: 1_500_000, // 1.5s
4615                                stored_observations_num_included_checkpoints: 10,
4616                                stored_observations_limit: 20,
4617                                stake_weighted_median_threshold: 3334,
4618                                default_none_duration_for_new_keys: true,
4619                                observations_chunk_size: None,
4620                            },
4621                        );
4622                }
4623                89 => {
4624                    cfg.feature_flags.dependency_linkage_error = true;
4625                    cfg.feature_flags.additional_multisig_checks = true;
4626                }
4627                90 => {
4628                    // 100x RGP
4629                    cfg.max_gas_price_rgp_factor_for_aborted_transactions = Some(100);
4630                    cfg.feature_flags.debug_fatal_on_move_invariant_violation = true;
4631                    cfg.feature_flags.additional_consensus_digest_indirect_state = true;
4632                    cfg.feature_flags.accept_passkey_in_multisig = true;
4633                    cfg.feature_flags.passkey_auth = true;
4634                    cfg.feature_flags.check_for_init_during_upgrade = true;
4635
4636                    // Enable Mysticeti fastpath handlers on testnet.
4637                    if chain != Chain::Mainnet {
4638                        cfg.feature_flags.mysticeti_fastpath = true;
4639                    }
4640                }
4641                91 => {
4642                    cfg.feature_flags.per_command_shared_object_transfer_rules = true;
4643                }
4644                92 => {
4645                    cfg.feature_flags.per_command_shared_object_transfer_rules = false;
4646                }
4647                93 => {
4648                    cfg.feature_flags
4649                        .consensus_checkpoint_signature_key_includes_digest = true;
4650                }
4651                94 => {
4652                    // Decrease stored observations limit 20->18 to stay within system object size limit.
4653                    cfg.feature_flags.per_object_congestion_control_mode =
4654                        PerObjectCongestionControlMode::ExecutionTimeEstimate(
4655                            ExecutionTimeEstimateParams {
4656                                target_utilization: 50,
4657                                allowed_txn_cost_overage_burst_limit_us: 500_000, // 500 ms
4658                                randomness_scalar: 20,
4659                                max_estimate_us: 1_500_000, // 1.5s
4660                                stored_observations_num_included_checkpoints: 10,
4661                                stored_observations_limit: 18,
4662                                stake_weighted_median_threshold: 3334,
4663                                default_none_duration_for_new_keys: true,
4664                                observations_chunk_size: None,
4665                            },
4666                        );
4667
4668                    // Enable party transfer on mainnet.
4669                    cfg.feature_flags.enable_party_transfer = true;
4670                }
4671                95 => {
4672                    cfg.type_name_id_base_cost = Some(52);
4673
4674                    // Reduce the frequency of checkpoint splitting under high TPS.
4675                    cfg.max_transactions_per_checkpoint = Some(20_000);
4676                }
4677                96 => {
4678                    // Enable artifacts digest in devnet.
4679                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4680                        cfg.feature_flags
4681                            .include_checkpoint_artifacts_digest_in_summary = true;
4682                    }
4683                    cfg.feature_flags.correct_gas_payment_limit_check = true;
4684                    cfg.feature_flags.authority_capabilities_v2 = true;
4685                    cfg.feature_flags.use_mfp_txns_in_load_initial_object_debts = true;
4686                    cfg.feature_flags.cancel_for_failed_dkg_early = true;
4687                    cfg.feature_flags.enable_coin_registry = true;
4688
4689                    // Enable Mysticeti fastpath handlers on mainnet.
4690                    cfg.feature_flags.mysticeti_fastpath = true;
4691                }
4692                97 => {
4693                    cfg.feature_flags.additional_borrow_checks = true;
4694                }
4695                98 => {
4696                    cfg.event_emit_auth_stream_cost = Some(52);
4697                    cfg.feature_flags.better_loader_errors = true;
4698                    cfg.feature_flags.generate_df_type_layouts = true;
4699                }
4700                99 => {
4701                    cfg.feature_flags.use_new_commit_handler = true;
4702                }
4703                100 => {
4704                    cfg.feature_flags.private_generics_verifier_v2 = true;
4705                }
4706                101 => {
4707                    cfg.feature_flags.create_root_accumulator_object = true;
4708                    cfg.max_updates_per_settlement_txn = Some(100);
4709                    if chain != Chain::Mainnet {
4710                        cfg.feature_flags.enable_poseidon = true;
4711                    }
4712                }
4713                102 => {
4714                    // Enable execution time observation chunking and increase limit to 180.
4715                    // max_move_object_size is 250 KB, we've experientially determined that fits ~ 18 estimates
4716                    // so if we have 10 chunks, that's 2.5MB, < 8MB max_serialized_tx_effects_size_bytes_system_tx
4717                    cfg.feature_flags.per_object_congestion_control_mode =
4718                        PerObjectCongestionControlMode::ExecutionTimeEstimate(
4719                            ExecutionTimeEstimateParams {
4720                                target_utilization: 50,
4721                                allowed_txn_cost_overage_burst_limit_us: 500_000, // 500 ms
4722                                randomness_scalar: 20,
4723                                max_estimate_us: 1_500_000, // 1.5s
4724                                stored_observations_num_included_checkpoints: 10,
4725                                stored_observations_limit: 180,
4726                                stake_weighted_median_threshold: 3334,
4727                                default_none_duration_for_new_keys: true,
4728                                observations_chunk_size: Some(18),
4729                            },
4730                        );
4731                    cfg.feature_flags.deprecate_global_storage_ops = true;
4732                }
4733                103 => {}
4734                104 => {
4735                    cfg.translation_per_command_base_charge = Some(1);
4736                    cfg.translation_per_input_base_charge = Some(1);
4737                    cfg.translation_pure_input_per_byte_charge = Some(1);
4738                    cfg.translation_per_type_node_charge = Some(1);
4739                    cfg.translation_per_reference_node_charge = Some(1);
4740                    cfg.translation_per_linkage_entry_charge = Some(10);
4741                    cfg.gas_model_version = Some(11);
4742                    cfg.feature_flags.abstract_size_in_object_runtime = true;
4743                    cfg.feature_flags.object_runtime_charge_cache_load_gas = true;
4744                    cfg.dynamic_field_hash_type_and_key_cost_base = Some(52);
4745                    cfg.dynamic_field_add_child_object_cost_base = Some(52);
4746                    cfg.dynamic_field_add_child_object_value_cost_per_byte = Some(1);
4747                    cfg.dynamic_field_borrow_child_object_cost_base = Some(52);
4748                    cfg.dynamic_field_borrow_child_object_child_ref_cost_per_byte = Some(1);
4749                    cfg.dynamic_field_remove_child_object_cost_base = Some(52);
4750                    cfg.dynamic_field_remove_child_object_child_cost_per_byte = Some(1);
4751                    cfg.dynamic_field_has_child_object_cost_base = Some(52);
4752                    cfg.dynamic_field_has_child_object_with_ty_cost_base = Some(52);
4753                    cfg.feature_flags.enable_ptb_execution_v2 = true;
4754
4755                    cfg.poseidon_bn254_cost_base = Some(260);
4756
4757                    cfg.feature_flags.consensus_skip_gced_accept_votes = true;
4758
4759                    if chain != Chain::Mainnet {
4760                        cfg.feature_flags
4761                            .enable_nitro_attestation_all_nonzero_pcrs_parsing = true;
4762                    }
4763
4764                    cfg.feature_flags
4765                        .include_cancelled_randomness_txns_in_prologue = true;
4766                }
4767                105 => {
4768                    cfg.feature_flags.enable_multi_epoch_transaction_expiration = true;
4769                    cfg.feature_flags.disable_preconsensus_locking = true;
4770
4771                    if chain != Chain::Mainnet {
4772                        cfg.feature_flags
4773                            .enable_nitro_attestation_always_include_required_pcrs_parsing = true;
4774                    }
4775                }
4776                106 => {
4777                    // est. 100 bytes per object * 76 (storage_gas_price)
4778                    cfg.accumulator_object_storage_cost = Some(7600);
4779
4780                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4781                        cfg.feature_flags.enable_accumulators = true;
4782                        cfg.feature_flags.enable_address_balance_gas_payments = true;
4783                        cfg.feature_flags.enable_authenticated_event_streams = true;
4784                        cfg.feature_flags.enable_object_funds_withdraw = true;
4785                    }
4786                }
4787                107 => {
4788                    cfg.feature_flags
4789                        .consensus_skip_gced_blocks_in_direct_finalization = true;
4790
4791                    // Trigger edge cases more often in integration tests.
4792                    if in_integration_test() {
4793                        cfg.consensus_gc_depth = Some(6);
4794                        cfg.consensus_max_num_transactions_in_block = Some(8);
4795                    }
4796                }
4797                108 => {
4798                    cfg.feature_flags.gas_rounding_halve_digits = true;
4799                    cfg.feature_flags.flexible_tx_context_positions = true;
4800                    cfg.feature_flags.disable_entry_point_signature_check = true;
4801
4802                    if chain != Chain::Mainnet {
4803                        cfg.feature_flags.address_aliases = true;
4804
4805                        cfg.feature_flags.enable_accumulators = true;
4806                        cfg.feature_flags.enable_address_balance_gas_payments = true;
4807                    }
4808
4809                    cfg.feature_flags.enable_poseidon = true;
4810                }
4811                109 => {
4812                    cfg.binary_variant_handles = Some(1024);
4813                    cfg.binary_variant_instantiation_handles = Some(1024);
4814                    cfg.feature_flags.restrict_hot_or_not_entry_functions = true;
4815                }
4816                110 => {
4817                    cfg.feature_flags
4818                        .enable_nitro_attestation_all_nonzero_pcrs_parsing = true;
4819                    cfg.feature_flags
4820                        .enable_nitro_attestation_always_include_required_pcrs_parsing = true;
4821                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4822                        cfg.feature_flags.split_checkpoints_in_consensus_handler = true;
4823                    }
4824                    cfg.feature_flags.validate_zklogin_public_identifier = true;
4825                    cfg.feature_flags.fix_checkpoint_signature_mapping = true;
4826                    cfg.feature_flags
4827                        .consensus_always_accept_system_transactions = true;
4828                    if chain != Chain::Mainnet {
4829                        cfg.feature_flags.enable_object_funds_withdraw = true;
4830                    }
4831                }
4832                111 => {
4833                    cfg.feature_flags.validator_metadata_verify_v2 = true;
4834                }
4835                112 => {
4836                    cfg.group_ops_ristretto_decode_scalar_cost = Some(7);
4837                    cfg.group_ops_ristretto_decode_point_cost = Some(200);
4838                    cfg.group_ops_ristretto_scalar_add_cost = Some(10);
4839                    cfg.group_ops_ristretto_point_add_cost = Some(500);
4840                    cfg.group_ops_ristretto_scalar_sub_cost = Some(10);
4841                    cfg.group_ops_ristretto_point_sub_cost = Some(500);
4842                    cfg.group_ops_ristretto_scalar_mul_cost = Some(11);
4843                    cfg.group_ops_ristretto_point_mul_cost = Some(1200);
4844                    cfg.group_ops_ristretto_scalar_div_cost = Some(151);
4845                    cfg.group_ops_ristretto_point_div_cost = Some(2500);
4846
4847                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4848                        cfg.feature_flags.enable_ristretto255_group_ops = true;
4849                    }
4850                }
4851                113 => {
4852                    cfg.feature_flags.address_balance_gas_check_rgp_at_signing = true;
4853                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4854                        cfg.feature_flags.defer_unpaid_amplification = true;
4855                    }
4856                }
4857                114 => {
4858                    cfg.feature_flags.randomize_checkpoint_tx_limit_in_tests = true;
4859                    cfg.feature_flags.address_balance_gas_reject_gas_coin_arg = true;
4860                    if chain != Chain::Mainnet {
4861                        cfg.feature_flags.split_checkpoints_in_consensus_handler = true;
4862                        cfg.feature_flags.enable_authenticated_event_streams = true;
4863                        cfg.feature_flags
4864                            .include_checkpoint_artifacts_digest_in_summary = true;
4865                    }
4866                }
4867                115 => {
4868                    cfg.feature_flags.normalize_depth_formula = true;
4869                }
4870                116 => {
4871                    cfg.feature_flags.gasless_transaction_drop_safety = true;
4872                    cfg.feature_flags.address_aliases = true;
4873                    cfg.feature_flags.relax_valid_during_for_owned_inputs = true;
4874                    // Disabled while debugging
4875                    cfg.feature_flags.defer_unpaid_amplification = false;
4876                    cfg.feature_flags.enable_display_registry = true;
4877                }
4878                117 => {}
4879                118 => {
4880                    cfg.feature_flags.use_coin_party_owner = true;
4881                }
4882                119 => {
4883                    // Enable new VM.
4884                    cfg.execution_version = Some(4);
4885                    cfg.feature_flags.address_balance_gas_reject_gas_coin_arg = false;
4886                    cfg.feature_flags.merge_randomness_into_checkpoint = true;
4887                    if chain != Chain::Mainnet {
4888                        cfg.feature_flags.enable_gasless = true;
4889                        cfg.gasless_max_computation_units = Some(50_000);
4890                        cfg.gasless_allowed_token_types = Some(vec![]);
4891                        cfg.feature_flags.enable_coin_reservation_obj_refs = true;
4892                        cfg.feature_flags
4893                            .convert_withdrawal_compatibility_ptb_arguments = true;
4894                    }
4895                    cfg.gasless_max_unused_inputs = Some(1);
4896                    cfg.gasless_max_pure_input_bytes = Some(32);
4897                    if chain == Chain::Testnet {
4898                        cfg.gasless_allowed_token_types = Some(vec![(TESTNET_USDC.to_string(), 0)]);
4899                    }
4900                    cfg.transfer_receive_object_cost_per_byte = Some(1);
4901                    cfg.transfer_receive_object_type_cost_per_byte = Some(2);
4902                }
4903                120 => {
4904                    cfg.feature_flags.disallow_jump_orphans = true;
4905                }
4906                121 => {
4907                    // Re-enable unpaid amplification deferral protection (testnet + devnet)
4908                    if chain != Chain::Mainnet {
4909                        cfg.feature_flags.defer_unpaid_amplification = true;
4910                        cfg.gasless_max_tps = Some(50);
4911                    }
4912                    cfg.feature_flags
4913                        .early_return_receive_object_mismatched_type = true;
4914                }
4915                122 => {
4916                    // Enable unpaid amplification deferral on mainnet
4917                    cfg.feature_flags.defer_unpaid_amplification = true;
4918                    // Enable bulletproofs range proofs on devnet
4919                    cfg.verify_bulletproofs_ristretto255_base_cost = Some(30000);
4920                    cfg.verify_bulletproofs_ristretto255_cost_per_bit_and_commitment = Some(6500);
4921                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4922                        cfg.feature_flags.enable_verify_bulletproofs_ristretto255 = true;
4923                    }
4924                    cfg.feature_flags.gasless_verify_remaining_balance = true;
4925                    cfg.include_special_package_amendments = match chain {
4926                        Chain::Mainnet => Some(MAINNET_LINKAGE_AMENDMENTS.clone()),
4927                        Chain::Testnet => Some(TESTNET_LINKAGE_AMENDMENTS.clone()),
4928                        Chain::Unknown => None,
4929                    };
4930                    cfg.gasless_max_tx_size_bytes = Some(16 * 1024);
4931                    cfg.gasless_max_tps = Some(300);
4932                    cfg.gasless_max_computation_units = Some(5_000);
4933                }
4934                123 => {
4935                    cfg.gas_model_version = Some(13);
4936                }
4937                124 => {
4938                    if chain != Chain::Mainnet && chain != Chain::Testnet {
4939                        cfg.feature_flags.timestamp_based_epoch_close = true;
4940                    }
4941                    cfg.gas_model_version = Some(14);
4942                    cfg.feature_flags.limit_groth16_pvk_inputs = true;
4943
4944                    // Bring mainnet in line with testnet: enable address balances, the
4945                    // gasless "free tier", coin reservations, and the rest of the
4946                    // accumulator/withdraw stack. These are all already enabled on
4947                    // testnet and devnet, so setting them unconditionally is a no-op
4948                    // there.
4949                    cfg.feature_flags.enable_accumulators = true;
4950                    cfg.feature_flags.enable_address_balance_gas_payments = true;
4951                    cfg.feature_flags.enable_authenticated_event_streams = true;
4952                    cfg.feature_flags.enable_coin_reservation_obj_refs = true;
4953                    cfg.feature_flags.enable_object_funds_withdraw = true;
4954                    cfg.feature_flags
4955                        .convert_withdrawal_compatibility_ptb_arguments = true;
4956                    cfg.feature_flags.split_checkpoints_in_consensus_handler = true;
4957                    cfg.feature_flags
4958                        .include_checkpoint_artifacts_digest_in_summary = true;
4959                    cfg.feature_flags.enable_gasless = true;
4960
4961                    // Set the mainnet allow-list. Testnet already has its USDC entry
4962                    // from v119, so only set this on mainnet to avoid clobbering the
4963                    // testnet value. $0.01 minimum transfer per stable; all listed
4964                    // tokens have 6 decimals.
4965                    if chain == Chain::Mainnet {
4966                        cfg.gasless_allowed_token_types = Some(vec![
4967                            (MAINNET_USDC.to_string(), 10_000),
4968                            (MAINNET_USDSUI.to_string(), 10_000),
4969                            (MAINNET_SUI_USDE.to_string(), 10_000),
4970                            (MAINNET_USDY.to_string(), 10_000),
4971                            (MAINNET_FDUSD.to_string(), 10_000),
4972                            (MAINNET_AUSD.to_string(), 10_000),
4973                            (MAINNET_USDB.to_string(), 10_000),
4974                        ]);
4975                    }
4976                }
4977                125 => {
4978                    cfg.feature_flags.granular_post_execution_checks = true;
4979                    if chain != Chain::Mainnet {
4980                        cfg.feature_flags.timestamp_based_epoch_close = true;
4981                    }
4982                }
4983                // Use this template when making changes:
4984                //
4985                //     // modify an existing constant.
4986                //     move_binary_format_version: Some(7),
4987                //
4988                //     // Add a new constant (which is set to None in prior versions).
4989                //     new_constant: Some(new_value),
4990                //
4991                //     // Remove a constant (ensure that it is never accessed during this version).
4992                //     max_move_object_size: None,
4993                _ => panic!("unsupported version {:?}", version),
4994            }
4995        }
4996
4997        cfg
4998    }
4999
5000    pub fn apply_seeded_test_overrides(&mut self, seed: &[u8; 32]) {
5001        if !self.feature_flags.randomize_checkpoint_tx_limit_in_tests
5002            || !self.feature_flags.split_checkpoints_in_consensus_handler
5003        {
5004            return;
5005        }
5006
5007        if !mysten_common::in_test_configuration() {
5008            return;
5009        }
5010
5011        use rand::{Rng, SeedableRng, rngs::StdRng};
5012        let mut rng = StdRng::from_seed(*seed);
5013        let max_txns = rng.gen_range(10..=100u64);
5014        info!("seeded test override: max_transactions_per_checkpoint = {max_txns}");
5015        self.max_transactions_per_checkpoint = Some(max_txns);
5016    }
5017
5018    // Extract the bytecode verifier config from this protocol config.
5019    // If used during signing, `signing_limits` should be set.
5020    // The third limit configures`sanity_check_with_regex_reference_safety`,
5021    // which runs the new regex-based reference safety check to check that it is strictly more
5022    // permissive than the current implementation.
5023    pub fn verifier_config(&self, signing_limits: Option<(usize, usize, usize)>) -> VerifierConfig {
5024        let (
5025            max_back_edges_per_function,
5026            max_back_edges_per_module,
5027            sanity_check_with_regex_reference_safety,
5028        ) = if let Some((
5029            max_back_edges_per_function,
5030            max_back_edges_per_module,
5031            sanity_check_with_regex_reference_safety,
5032        )) = signing_limits
5033        {
5034            (
5035                Some(max_back_edges_per_function),
5036                Some(max_back_edges_per_module),
5037                Some(sanity_check_with_regex_reference_safety),
5038            )
5039        } else {
5040            (None, None, None)
5041        };
5042
5043        let additional_borrow_checks = if signing_limits.is_some() {
5044            // always turn on additional borrow checks during signing
5045            true
5046        } else {
5047            self.additional_borrow_checks()
5048        };
5049        let deprecate_global_storage_ops = if signing_limits.is_some() {
5050            // always turn on additional vector borrow checks during signing
5051            true
5052        } else {
5053            self.deprecate_global_storage_ops()
5054        };
5055
5056        VerifierConfig {
5057            max_loop_depth: Some(self.max_loop_depth() as usize),
5058            max_generic_instantiation_length: Some(self.max_generic_instantiation_length() as usize),
5059            max_function_parameters: Some(self.max_function_parameters() as usize),
5060            max_basic_blocks: Some(self.max_basic_blocks() as usize),
5061            max_value_stack_size: self.max_value_stack_size() as usize,
5062            max_type_nodes: Some(self.max_type_nodes() as usize),
5063            max_push_size: Some(self.max_push_size() as usize),
5064            max_dependency_depth: Some(self.max_dependency_depth() as usize),
5065            max_fields_in_struct: Some(self.max_fields_in_struct() as usize),
5066            max_function_definitions: Some(self.max_function_definitions() as usize),
5067            max_data_definitions: Some(self.max_struct_definitions() as usize),
5068            max_constant_vector_len: Some(self.max_move_vector_len()),
5069            max_back_edges_per_function,
5070            max_back_edges_per_module,
5071            max_basic_blocks_in_script: None,
5072            max_identifier_len: self.max_move_identifier_len_as_option(), // Before protocol version 9, there was no limit
5073            disallow_self_identifier: self.feature_flags.disallow_self_identifier,
5074            allow_receiving_object_id: self.allow_receiving_object_id(),
5075            reject_mutable_random_on_entry_functions: self
5076                .reject_mutable_random_on_entry_functions(),
5077            bytecode_version: self.move_binary_format_version(),
5078            max_variants_in_enum: self.max_move_enum_variants_as_option(),
5079            additional_borrow_checks,
5080            better_loader_errors: self.better_loader_errors(),
5081            private_generics_verifier_v2: self.private_generics_verifier_v2(),
5082            sanity_check_with_regex_reference_safety: sanity_check_with_regex_reference_safety
5083                .map(|limit| limit as u128),
5084            deprecate_global_storage_ops,
5085            disable_entry_point_signature_check: self.disable_entry_point_signature_check(),
5086            switch_to_regex_reference_safety: false,
5087            disallow_jump_orphans: self.disallow_jump_orphans(),
5088        }
5089    }
5090
5091    pub fn binary_config(
5092        &self,
5093        override_deprecate_global_storage_ops_during_deserialization: Option<bool>,
5094    ) -> BinaryConfig {
5095        let deprecate_global_storage_ops =
5096            override_deprecate_global_storage_ops_during_deserialization
5097                .unwrap_or_else(|| self.deprecate_global_storage_ops());
5098        BinaryConfig::new(
5099            self.move_binary_format_version(),
5100            self.min_move_binary_format_version_as_option()
5101                .unwrap_or(VERSION_1),
5102            self.no_extraneous_module_bytes(),
5103            deprecate_global_storage_ops,
5104            TableConfig {
5105                module_handles: self.binary_module_handles_as_option().unwrap_or(u16::MAX),
5106                datatype_handles: self.binary_struct_handles_as_option().unwrap_or(u16::MAX),
5107                function_handles: self.binary_function_handles_as_option().unwrap_or(u16::MAX),
5108                function_instantiations: self
5109                    .binary_function_instantiations_as_option()
5110                    .unwrap_or(u16::MAX),
5111                signatures: self.binary_signatures_as_option().unwrap_or(u16::MAX),
5112                constant_pool: self.binary_constant_pool_as_option().unwrap_or(u16::MAX),
5113                identifiers: self.binary_identifiers_as_option().unwrap_or(u16::MAX),
5114                address_identifiers: self
5115                    .binary_address_identifiers_as_option()
5116                    .unwrap_or(u16::MAX),
5117                struct_defs: self.binary_struct_defs_as_option().unwrap_or(u16::MAX),
5118                struct_def_instantiations: self
5119                    .binary_struct_def_instantiations_as_option()
5120                    .unwrap_or(u16::MAX),
5121                function_defs: self.binary_function_defs_as_option().unwrap_or(u16::MAX),
5122                field_handles: self.binary_field_handles_as_option().unwrap_or(u16::MAX),
5123                field_instantiations: self
5124                    .binary_field_instantiations_as_option()
5125                    .unwrap_or(u16::MAX),
5126                friend_decls: self.binary_friend_decls_as_option().unwrap_or(u16::MAX),
5127                enum_defs: self.binary_enum_defs_as_option().unwrap_or(u16::MAX),
5128                enum_def_instantiations: self
5129                    .binary_enum_def_instantiations_as_option()
5130                    .unwrap_or(u16::MAX),
5131                variant_handles: self.binary_variant_handles_as_option().unwrap_or(u16::MAX),
5132                variant_instantiation_handles: self
5133                    .binary_variant_instantiation_handles_as_option()
5134                    .unwrap_or(u16::MAX),
5135            },
5136        )
5137    }
5138
5139    /// Override one or more settings in the config, for testing.
5140    /// This must be called at the beginning of the test, before get_for_(min|max)_version is
5141    /// called, since those functions cache their return value.
5142    #[cfg(not(msim))]
5143    pub fn apply_overrides_for_testing(
5144        override_fn: impl Fn(ProtocolVersion, Self) -> Self + Send + Sync + 'static,
5145    ) -> OverrideGuard {
5146        let mut cur = CONFIG_OVERRIDE.lock().unwrap();
5147        assert!(cur.is_none(), "config override already present");
5148        *cur = Some(Box::new(override_fn));
5149        OverrideGuard
5150    }
5151
5152    /// Override one or more settings in the config, for testing.
5153    /// This must be called at the beginning of the test, before get_for_(min|max)_version is
5154    /// called, since those functions cache their return value.
5155    #[cfg(msim)]
5156    pub fn apply_overrides_for_testing(
5157        override_fn: impl Fn(ProtocolVersion, Self) -> Self + Send + 'static,
5158    ) -> OverrideGuard {
5159        CONFIG_OVERRIDE.with(|ovr| {
5160            let mut cur = ovr.borrow_mut();
5161            assert!(cur.is_none(), "config override already present");
5162            *cur = Some(Box::new(override_fn));
5163            OverrideGuard
5164        })
5165    }
5166
5167    #[cfg(not(msim))]
5168    fn apply_config_override(version: ProtocolVersion, mut ret: Self) -> Self {
5169        if let Some(override_fn) = CONFIG_OVERRIDE.lock().unwrap().as_ref() {
5170            warn!(
5171                "overriding ProtocolConfig settings with custom settings (you should not see this log outside of tests)"
5172            );
5173            ret = override_fn(version, ret);
5174        }
5175        ret
5176    }
5177
5178    #[cfg(msim)]
5179    fn apply_config_override(version: ProtocolVersion, ret: Self) -> Self {
5180        CONFIG_OVERRIDE.with(|ovr| {
5181            if let Some(override_fn) = &*ovr.borrow() {
5182                warn!(
5183                    "overriding ProtocolConfig settings with custom settings (you should not see this log outside of tests)"
5184                );
5185                override_fn(version, ret)
5186            } else {
5187                ret
5188            }
5189        })
5190    }
5191}
5192
5193// Setters for tests.
5194// This is only needed for feature_flags. Please suffix each setter with `_for_testing`.
5195// Non-feature_flags should already have test setters defined through macros.
5196impl ProtocolConfig {
5197    pub fn set_advance_to_highest_supported_protocol_version_for_testing(&mut self, val: bool) {
5198        self.feature_flags
5199            .advance_to_highest_supported_protocol_version = val
5200    }
5201    pub fn set_commit_root_state_digest_supported_for_testing(&mut self, val: bool) {
5202        self.feature_flags.commit_root_state_digest = val
5203    }
5204    pub fn set_zklogin_auth_for_testing(&mut self, val: bool) {
5205        self.feature_flags.zklogin_auth = val
5206    }
5207    pub fn set_enable_jwk_consensus_updates_for_testing(&mut self, val: bool) {
5208        self.feature_flags.enable_jwk_consensus_updates = val
5209    }
5210    pub fn set_random_beacon_for_testing(&mut self, val: bool) {
5211        self.feature_flags.random_beacon = val
5212    }
5213
5214    pub fn set_upgraded_multisig_for_testing(&mut self, val: bool) {
5215        self.feature_flags.upgraded_multisig_supported = val
5216    }
5217    pub fn set_accept_zklogin_in_multisig_for_testing(&mut self, val: bool) {
5218        self.feature_flags.accept_zklogin_in_multisig = val
5219    }
5220
5221    pub fn set_shared_object_deletion_for_testing(&mut self, val: bool) {
5222        self.feature_flags.shared_object_deletion = val;
5223    }
5224
5225    pub fn set_narwhal_new_leader_election_schedule_for_testing(&mut self, val: bool) {
5226        self.feature_flags.narwhal_new_leader_election_schedule = val;
5227    }
5228
5229    pub fn set_receive_object_for_testing(&mut self, val: bool) {
5230        self.feature_flags.receive_objects = val
5231    }
5232    pub fn set_narwhal_certificate_v2_for_testing(&mut self, val: bool) {
5233        self.feature_flags.narwhal_certificate_v2 = val
5234    }
5235    pub fn set_verify_legacy_zklogin_address_for_testing(&mut self, val: bool) {
5236        self.feature_flags.verify_legacy_zklogin_address = val
5237    }
5238
5239    pub fn set_per_object_congestion_control_mode_for_testing(
5240        &mut self,
5241        val: PerObjectCongestionControlMode,
5242    ) {
5243        self.feature_flags.per_object_congestion_control_mode = val;
5244    }
5245
5246    pub fn set_consensus_choice_for_testing(&mut self, val: ConsensusChoice) {
5247        self.feature_flags.consensus_choice = val;
5248    }
5249
5250    pub fn set_consensus_network_for_testing(&mut self, val: ConsensusNetwork) {
5251        self.feature_flags.consensus_network = val;
5252    }
5253
5254    pub fn set_zklogin_max_epoch_upper_bound_delta_for_testing(&mut self, val: Option<u64>) {
5255        self.feature_flags.zklogin_max_epoch_upper_bound_delta = val
5256    }
5257
5258    pub fn set_disable_bridge_for_testing(&mut self) {
5259        self.feature_flags.bridge = false
5260    }
5261
5262    pub fn set_mysticeti_num_leaders_per_round_for_testing(&mut self, val: Option<usize>) {
5263        self.feature_flags.mysticeti_num_leaders_per_round = val;
5264    }
5265
5266    pub fn set_enable_soft_bundle_for_testing(&mut self, val: bool) {
5267        self.feature_flags.soft_bundle = val;
5268    }
5269
5270    pub fn set_passkey_auth_for_testing(&mut self, val: bool) {
5271        self.feature_flags.passkey_auth = val
5272    }
5273
5274    pub fn set_enable_party_transfer_for_testing(&mut self, val: bool) {
5275        self.feature_flags.enable_party_transfer = val
5276    }
5277
5278    pub fn set_consensus_distributed_vote_scoring_strategy_for_testing(&mut self, val: bool) {
5279        self.feature_flags
5280            .consensus_distributed_vote_scoring_strategy = val;
5281    }
5282
5283    pub fn set_consensus_round_prober_for_testing(&mut self, val: bool) {
5284        self.feature_flags.consensus_round_prober = val;
5285    }
5286
5287    pub fn set_disallow_new_modules_in_deps_only_packages_for_testing(&mut self, val: bool) {
5288        self.feature_flags
5289            .disallow_new_modules_in_deps_only_packages = val;
5290    }
5291
5292    pub fn set_correct_gas_payment_limit_check_for_testing(&mut self, val: bool) {
5293        self.feature_flags.correct_gas_payment_limit_check = val;
5294    }
5295
5296    pub fn set_address_aliases_for_testing(&mut self, val: bool) {
5297        self.feature_flags.address_aliases = val;
5298    }
5299
5300    pub fn set_consensus_round_prober_probe_accepted_rounds(&mut self, val: bool) {
5301        self.feature_flags
5302            .consensus_round_prober_probe_accepted_rounds = val;
5303    }
5304
5305    pub fn set_mysticeti_fastpath_for_testing(&mut self, val: bool) {
5306        self.feature_flags.mysticeti_fastpath = val;
5307    }
5308
5309    pub fn set_accept_passkey_in_multisig_for_testing(&mut self, val: bool) {
5310        self.feature_flags.accept_passkey_in_multisig = val;
5311    }
5312
5313    pub fn set_consensus_batched_block_sync_for_testing(&mut self, val: bool) {
5314        self.feature_flags.consensus_batched_block_sync = val;
5315    }
5316
5317    pub fn set_record_time_estimate_processed_for_testing(&mut self, val: bool) {
5318        self.feature_flags.record_time_estimate_processed = val;
5319    }
5320
5321    pub fn set_prepend_prologue_tx_in_consensus_commit_in_checkpoints_for_testing(
5322        &mut self,
5323        val: bool,
5324    ) {
5325        self.feature_flags
5326            .prepend_prologue_tx_in_consensus_commit_in_checkpoints = val;
5327    }
5328
5329    pub fn enable_accumulators_for_testing(&mut self) {
5330        self.feature_flags.enable_accumulators = true;
5331    }
5332
5333    pub fn disable_accumulators_for_testing(&mut self) {
5334        self.feature_flags.enable_accumulators = false;
5335        self.feature_flags.enable_address_balance_gas_payments = false;
5336    }
5337
5338    pub fn enable_coin_reservation_for_testing(&mut self) {
5339        self.feature_flags.enable_coin_reservation_obj_refs = true;
5340        self.feature_flags
5341            .convert_withdrawal_compatibility_ptb_arguments = true;
5342        // Ensure execution_version >= 4 so new_vm_enabled() returns true,
5343        // which is required for enable_coin_reservation_obj_refs() to return true.
5344        self.execution_version = Some(self.execution_version.map_or(4, |v| v.max(4)));
5345    }
5346
5347    pub fn disable_coin_reservation_for_testing(&mut self) {
5348        self.feature_flags.enable_coin_reservation_obj_refs = false;
5349        self.feature_flags
5350            .convert_withdrawal_compatibility_ptb_arguments = false;
5351    }
5352
5353    pub fn create_root_accumulator_object_for_testing(&mut self) {
5354        self.feature_flags.create_root_accumulator_object = true;
5355    }
5356
5357    pub fn disable_create_root_accumulator_object_for_testing(&mut self) {
5358        self.feature_flags.create_root_accumulator_object = false;
5359    }
5360
5361    pub fn enable_address_balance_gas_payments_for_testing(&mut self) {
5362        self.feature_flags.enable_accumulators = true;
5363        self.feature_flags.allow_private_accumulator_entrypoints = true;
5364        self.feature_flags.enable_address_balance_gas_payments = true;
5365        self.feature_flags.address_balance_gas_check_rgp_at_signing = true;
5366        self.feature_flags.address_balance_gas_reject_gas_coin_arg = false;
5367        self.execution_version = Some(self.execution_version.map_or(4, |v| v.max(4)))
5368    }
5369
5370    pub fn disable_address_balance_gas_payments_for_testing(&mut self) {
5371        self.feature_flags.enable_address_balance_gas_payments = false;
5372    }
5373
5374    pub fn enable_gasless_for_testing(&mut self) {
5375        self.enable_address_balance_gas_payments_for_testing();
5376        self.feature_flags.enable_gasless = true;
5377        self.feature_flags.gasless_verify_remaining_balance = true;
5378        self.gasless_max_computation_units = Some(5_000);
5379        self.gasless_allowed_token_types = Some(vec![]);
5380        self.gasless_max_tps = Some(1000);
5381        self.gasless_max_tx_size_bytes = Some(16 * 1024);
5382    }
5383
5384    pub fn disable_gasless_for_testing(&mut self) {
5385        self.feature_flags.enable_gasless = false;
5386        self.gasless_max_computation_units = None;
5387        self.gasless_allowed_token_types = None;
5388    }
5389
5390    pub fn enable_multi_epoch_transaction_expiration_for_testing(&mut self) {
5391        self.feature_flags.enable_multi_epoch_transaction_expiration = true;
5392    }
5393
5394    pub fn enable_authenticated_event_streams_for_testing(&mut self) {
5395        self.enable_accumulators_for_testing();
5396        self.feature_flags.enable_authenticated_event_streams = true;
5397        self.feature_flags
5398            .include_checkpoint_artifacts_digest_in_summary = true;
5399        self.feature_flags.split_checkpoints_in_consensus_handler = true;
5400    }
5401
5402    pub fn disable_authenticated_event_streams_for_testing(&mut self) {
5403        self.feature_flags.enable_authenticated_event_streams = false;
5404    }
5405
5406    pub fn disable_randomize_checkpoint_tx_limit_for_testing(&mut self) {
5407        self.feature_flags.randomize_checkpoint_tx_limit_in_tests = false;
5408    }
5409
5410    pub fn enable_non_exclusive_writes_for_testing(&mut self) {
5411        self.feature_flags.enable_non_exclusive_writes = true;
5412    }
5413
5414    pub fn set_relax_valid_during_for_owned_inputs_for_testing(&mut self, val: bool) {
5415        self.feature_flags.relax_valid_during_for_owned_inputs = val;
5416    }
5417
5418    pub fn set_ignore_execution_time_observations_after_certs_closed_for_testing(
5419        &mut self,
5420        val: bool,
5421    ) {
5422        self.feature_flags
5423            .ignore_execution_time_observations_after_certs_closed = val;
5424    }
5425
5426    pub fn set_consensus_checkpoint_signature_key_includes_digest_for_testing(
5427        &mut self,
5428        val: bool,
5429    ) {
5430        self.feature_flags
5431            .consensus_checkpoint_signature_key_includes_digest = val;
5432    }
5433
5434    pub fn set_cancel_for_failed_dkg_early_for_testing(&mut self, val: bool) {
5435        self.feature_flags.cancel_for_failed_dkg_early = val;
5436    }
5437
5438    pub fn set_use_mfp_txns_in_load_initial_object_debts_for_testing(&mut self, val: bool) {
5439        self.feature_flags.use_mfp_txns_in_load_initial_object_debts = val;
5440    }
5441
5442    pub fn set_authority_capabilities_v2_for_testing(&mut self, val: bool) {
5443        self.feature_flags.authority_capabilities_v2 = val;
5444    }
5445
5446    pub fn allow_references_in_ptbs_for_testing(&mut self) {
5447        self.feature_flags.allow_references_in_ptbs = true;
5448    }
5449
5450    pub fn set_consensus_skip_gced_accept_votes_for_testing(&mut self, val: bool) {
5451        self.feature_flags.consensus_skip_gced_accept_votes = val;
5452    }
5453
5454    pub fn set_enable_object_funds_withdraw_for_testing(&mut self, val: bool) {
5455        self.feature_flags.enable_object_funds_withdraw = val;
5456    }
5457
5458    pub fn set_split_checkpoints_in_consensus_handler_for_testing(&mut self, val: bool) {
5459        self.feature_flags.split_checkpoints_in_consensus_handler = val;
5460    }
5461
5462    pub fn set_merge_randomness_into_checkpoint_for_testing(&mut self, val: bool) {
5463        self.feature_flags.merge_randomness_into_checkpoint = val;
5464    }
5465}
5466
5467#[cfg(not(msim))]
5468type OverrideFn = dyn Fn(ProtocolVersion, ProtocolConfig) -> ProtocolConfig + Send + Sync;
5469
5470#[cfg(not(msim))]
5471static CONFIG_OVERRIDE: Mutex<Option<Box<OverrideFn>>> = Mutex::new(None);
5472
5473#[cfg(msim)]
5474type OverrideFn = dyn Fn(ProtocolVersion, ProtocolConfig) -> ProtocolConfig + Send;
5475
5476#[cfg(msim)]
5477thread_local! {
5478    static CONFIG_OVERRIDE: RefCell<Option<Box<OverrideFn>>> = RefCell::new(None);
5479}
5480
5481#[must_use]
5482pub struct OverrideGuard;
5483
5484#[cfg(not(msim))]
5485impl Drop for OverrideGuard {
5486    fn drop(&mut self) {
5487        info!("restoring override fn");
5488        *CONFIG_OVERRIDE.lock().unwrap() = None;
5489    }
5490}
5491
5492#[cfg(msim)]
5493impl Drop for OverrideGuard {
5494    fn drop(&mut self) {
5495        info!("restoring override fn");
5496        CONFIG_OVERRIDE.with(|ovr| {
5497            *ovr.borrow_mut() = None;
5498        });
5499    }
5500}
5501
5502/// Defines which limit got crossed.
5503/// The value which crossed the limit and value of the limit crossed are embedded
5504#[derive(PartialEq, Eq)]
5505pub enum LimitThresholdCrossed {
5506    None,
5507    Soft(u128, u128),
5508    Hard(u128, u128),
5509}
5510
5511/// Convenience function for comparing limit ranges
5512/// V::MAX must be at >= U::MAX and T::MAX
5513pub fn check_limit_in_range<T: Into<V>, U: Into<V>, V: PartialOrd + Into<u128>>(
5514    x: T,
5515    soft_limit: U,
5516    hard_limit: V,
5517) -> LimitThresholdCrossed {
5518    let x: V = x.into();
5519    let soft_limit: V = soft_limit.into();
5520
5521    debug_assert!(soft_limit <= hard_limit);
5522
5523    // It is important to preserve this comparison order because if soft_limit == hard_limit
5524    // we want LimitThresholdCrossed::Hard
5525    if x >= hard_limit {
5526        LimitThresholdCrossed::Hard(x.into(), hard_limit.into())
5527    } else if x < soft_limit {
5528        LimitThresholdCrossed::None
5529    } else {
5530        LimitThresholdCrossed::Soft(x.into(), soft_limit.into())
5531    }
5532}
5533
5534#[macro_export]
5535macro_rules! check_limit {
5536    ($x:expr, $hard:expr) => {
5537        check_limit!($x, $hard, $hard)
5538    };
5539    ($x:expr, $soft:expr, $hard:expr) => {
5540        check_limit_in_range($x as u64, $soft, $hard)
5541    };
5542}
5543
5544/// Used to check which limits were crossed if the TX is metered (not system tx)
5545/// Args are: is_metered, value_to_check, metered_limit, unmetered_limit
5546/// metered_limit is always less than or equal to unmetered_hard_limit
5547#[macro_export]
5548macro_rules! check_limit_by_meter {
5549    ($is_metered:expr, $x:expr, $metered_limit:expr, $unmetered_hard_limit:expr, $metric:expr) => {{
5550        // If this is metered, we use the metered_limit limit as the upper bound
5551        let (h, metered_str) = if $is_metered {
5552            ($metered_limit, "metered")
5553        } else {
5554            // Unmetered gets more headroom
5555            ($unmetered_hard_limit, "unmetered")
5556        };
5557        use sui_protocol_config::check_limit_in_range;
5558        let result = check_limit_in_range($x as u64, $metered_limit, h);
5559        match result {
5560            LimitThresholdCrossed::None => {}
5561            LimitThresholdCrossed::Soft(_, _) => {
5562                $metric.with_label_values(&[metered_str, "soft"]).inc();
5563            }
5564            LimitThresholdCrossed::Hard(_, _) => {
5565                $metric.with_label_values(&[metered_str, "hard"]).inc();
5566            }
5567        };
5568        result
5569    }};
5570}
5571
5572// Amendments tables
5573
5574pub type Amendments = BTreeMap<AccountAddress, BTreeMap<AccountAddress, AccountAddress>>;
5575
5576static MAINNET_LINKAGE_AMENDMENTS: LazyLock<Arc<Amendments>> =
5577    LazyLock::new(|| parse_amendments(include_str!("mainnet_amendments.json")));
5578
5579static TESTNET_LINKAGE_AMENDMENTS: LazyLock<Arc<Amendments>> =
5580    LazyLock::new(|| parse_amendments(include_str!("testnet_amendments.json")));
5581
5582fn parse_amendments(json: &str) -> Arc<Amendments> {
5583    #[derive(serde::Deserialize)]
5584    struct AmendmentEntry {
5585        root: String,
5586        deps: Vec<DepEntry>,
5587    }
5588
5589    #[derive(serde::Deserialize)]
5590    struct DepEntry {
5591        original_id: String,
5592        version_id: String,
5593    }
5594
5595    let entries: Vec<AmendmentEntry> =
5596        serde_json::from_str(json).expect("Failed to parse amendments JSON");
5597    let mut amendments = BTreeMap::new();
5598    for entry in entries {
5599        let root_id = AccountAddress::from_hex_literal(&entry.root).unwrap();
5600        let mut dep_ids = BTreeMap::new();
5601        for dep in entry.deps {
5602            let orig_id = AccountAddress::from_hex_literal(&dep.original_id).unwrap();
5603            let upgraded_id = AccountAddress::from_hex_literal(&dep.version_id).unwrap();
5604            assert!(
5605                dep_ids.insert(orig_id, upgraded_id).is_none(),
5606                "Duplicate original ID in amendments table"
5607            );
5608        }
5609        assert!(
5610            amendments.insert(root_id, dep_ids).is_none(),
5611            "Duplicate root ID in amendments table"
5612        );
5613    }
5614    Arc::new(amendments)
5615}
5616
5617#[cfg(all(test, not(msim)))]
5618mod test {
5619    use insta::assert_yaml_snapshot;
5620
5621    use super::*;
5622
5623    #[test]
5624    fn snapshot_tests() {
5625        println!("\n============================================================================");
5626        println!("!                                                                          !");
5627        println!("! IMPORTANT: never update snapshots from this test. only add new versions! !");
5628        println!("!                                                                          !");
5629        println!("============================================================================\n");
5630        for chain_id in &[Chain::Unknown, Chain::Mainnet, Chain::Testnet] {
5631            // make Chain::Unknown snapshots compatible with pre-chain-id snapshots so that we
5632            // don't break the release-time compatibility tests. Once Chain Id configs have been
5633            // released everywhere, we can remove this and only test Mainnet and Testnet
5634            let chain_str = match chain_id {
5635                Chain::Unknown => "".to_string(),
5636                _ => format!("{:?}_", chain_id),
5637            };
5638            for i in MIN_PROTOCOL_VERSION..=MAX_PROTOCOL_VERSION {
5639                let cur = ProtocolVersion::new(i);
5640                assert_yaml_snapshot!(
5641                    format!("{}version_{}", chain_str, cur.as_u64()),
5642                    ProtocolConfig::get_for_version(cur, *chain_id)
5643                );
5644            }
5645        }
5646    }
5647
5648    #[test]
5649    fn test_getters() {
5650        let prot: ProtocolConfig =
5651            ProtocolConfig::get_for_version(ProtocolVersion::new(1), Chain::Unknown);
5652        assert_eq!(
5653            prot.max_arguments(),
5654            prot.max_arguments_as_option().unwrap()
5655        );
5656    }
5657
5658    #[test]
5659    fn test_setters() {
5660        let mut prot: ProtocolConfig =
5661            ProtocolConfig::get_for_version(ProtocolVersion::new(1), Chain::Unknown);
5662        prot.set_max_arguments_for_testing(123);
5663        assert_eq!(prot.max_arguments(), 123);
5664
5665        prot.set_max_arguments_from_str_for_testing("321".to_string());
5666        assert_eq!(prot.max_arguments(), 321);
5667
5668        prot.disable_max_arguments_for_testing();
5669        assert_eq!(prot.max_arguments_as_option(), None);
5670
5671        prot.set_attr_for_testing("max_arguments".to_string(), "456".to_string());
5672        assert_eq!(prot.max_arguments(), 456);
5673    }
5674
5675    #[test]
5676    fn test_get_for_version_if_supported_applies_test_overrides() {
5677        let before =
5678            ProtocolConfig::get_for_version_if_supported(ProtocolVersion::new(1), Chain::Unknown)
5679                .unwrap();
5680
5681        assert!(!before.enable_coin_reservation_obj_refs());
5682
5683        let _guard = ProtocolConfig::apply_overrides_for_testing(|_, mut cfg| {
5684            cfg.enable_coin_reservation_for_testing();
5685            cfg
5686        });
5687
5688        let after =
5689            ProtocolConfig::get_for_version_if_supported(ProtocolVersion::new(1), Chain::Unknown)
5690                .unwrap();
5691
5692        assert!(after.enable_coin_reservation_obj_refs());
5693    }
5694
5695    #[test]
5696    #[should_panic(expected = "unsupported version")]
5697    fn max_version_test() {
5698        // When this does not panic, version higher than MAX_PROTOCOL_VERSION exists.
5699        // To fix, bump MAX_PROTOCOL_VERSION or disable this check for the version.
5700        let _ = ProtocolConfig::get_for_version_impl(
5701            ProtocolVersion::new(MAX_PROTOCOL_VERSION + 1),
5702            Chain::Unknown,
5703        );
5704    }
5705
5706    #[test]
5707    fn lookup_by_string_test() {
5708        let prot: ProtocolConfig =
5709            ProtocolConfig::get_for_version(ProtocolVersion::new(1), Chain::Unknown);
5710        // Does not exist
5711        assert!(prot.lookup_attr("some random string".to_string()).is_none());
5712
5713        assert!(
5714            prot.lookup_attr("max_arguments".to_string())
5715                == Some(ProtocolConfigValue::u32(prot.max_arguments())),
5716        );
5717
5718        // We didnt have this in version 1
5719        assert!(
5720            prot.lookup_attr("max_move_identifier_len".to_string())
5721                .is_none()
5722        );
5723
5724        // But we did in version 9
5725        let prot: ProtocolConfig =
5726            ProtocolConfig::get_for_version(ProtocolVersion::new(9), Chain::Unknown);
5727        assert!(
5728            prot.lookup_attr("max_move_identifier_len".to_string())
5729                == Some(ProtocolConfigValue::u64(prot.max_move_identifier_len()))
5730        );
5731
5732        let prot: ProtocolConfig =
5733            ProtocolConfig::get_for_version(ProtocolVersion::new(1), Chain::Unknown);
5734        // We didnt have this in version 1
5735        assert!(
5736            prot.attr_map()
5737                .get("max_move_identifier_len")
5738                .unwrap()
5739                .is_none()
5740        );
5741        // We had this in version 1
5742        assert!(
5743            prot.attr_map().get("max_arguments").unwrap()
5744                == &Some(ProtocolConfigValue::u32(prot.max_arguments()))
5745        );
5746
5747        // Check feature flags
5748        let prot: ProtocolConfig =
5749            ProtocolConfig::get_for_version(ProtocolVersion::new(1), Chain::Unknown);
5750        // Does not exist
5751        assert!(
5752            prot.feature_flags
5753                .lookup_attr("some random string".to_owned())
5754                .is_none()
5755        );
5756        assert!(
5757            !prot
5758                .feature_flags
5759                .attr_map()
5760                .contains_key("some random string")
5761        );
5762
5763        // Was false in v1
5764        assert!(
5765            prot.feature_flags
5766                .lookup_attr("package_upgrades".to_owned())
5767                == Some(false)
5768        );
5769        assert!(
5770            prot.feature_flags
5771                .attr_map()
5772                .get("package_upgrades")
5773                .unwrap()
5774                == &false
5775        );
5776        let prot: ProtocolConfig =
5777            ProtocolConfig::get_for_version(ProtocolVersion::new(4), Chain::Unknown);
5778        // Was true from v3 and up
5779        assert!(
5780            prot.feature_flags
5781                .lookup_attr("package_upgrades".to_owned())
5782                == Some(true)
5783        );
5784        assert!(
5785            prot.feature_flags
5786                .attr_map()
5787                .get("package_upgrades")
5788                .unwrap()
5789                == &true
5790        );
5791    }
5792
5793    #[test]
5794    fn limit_range_fn_test() {
5795        let low = 100u32;
5796        let high = 10000u64;
5797
5798        assert!(check_limit!(1u8, low, high) == LimitThresholdCrossed::None);
5799        assert!(matches!(
5800            check_limit!(255u16, low, high),
5801            LimitThresholdCrossed::Soft(255u128, 100)
5802        ));
5803        // This wont compile because lossy
5804        //assert!(check_limit!(100000000u128, low, high) == LimitThresholdCrossed::None);
5805        // This wont compile because lossy
5806        //assert!(check_limit!(100000000usize, low, high) == LimitThresholdCrossed::None);
5807
5808        assert!(matches!(
5809            check_limit!(2550000u64, low, high),
5810            LimitThresholdCrossed::Hard(2550000, 10000)
5811        ));
5812
5813        assert!(matches!(
5814            check_limit!(2550000u64, high, high),
5815            LimitThresholdCrossed::Hard(2550000, 10000)
5816        ));
5817
5818        assert!(matches!(
5819            check_limit!(1u8, high),
5820            LimitThresholdCrossed::None
5821        ));
5822
5823        assert!(check_limit!(255u16, high) == LimitThresholdCrossed::None);
5824
5825        assert!(matches!(
5826            check_limit!(2550000u64, high),
5827            LimitThresholdCrossed::Hard(2550000, 10000)
5828        ));
5829    }
5830
5831    #[test]
5832    fn linkage_amendments_load() {
5833        let mainnet = LazyLock::force(&MAINNET_LINKAGE_AMENDMENTS);
5834        let testnet = LazyLock::force(&TESTNET_LINKAGE_AMENDMENTS);
5835        assert!(!mainnet.is_empty(), "mainnet amendments must not be empty");
5836        assert!(!testnet.is_empty(), "testnet amendments must not be empty");
5837    }
5838
5839    #[test]
5840    fn render_scalar_fields_use_precision_safe_encoding() {
5841        use mysten_common::rpc_format::Unmetered;
5842
5843        let config = ProtocolConfig::get_for_max_version_UNSAFE();
5844        let rendered = config
5845            .render::<serde_json::Value>(&mut Unmetered)
5846            .expect("render should succeed");
5847
5848        let max_args = rendered
5849            .get("max_arguments")
5850            .expect("max_arguments set at max version");
5851        assert!(
5852            max_args.is_number(),
5853            "u32 should render as number, got {max_args:?}",
5854        );
5855
5856        let max_tx_size = rendered
5857            .get("max_tx_size_bytes")
5858            .expect("max_tx_size_bytes set at max version");
5859        assert!(
5860            max_tx_size.is_string(),
5861            "u64 should render as string, got {max_tx_size:?}",
5862        );
5863    }
5864
5865    #[test]
5866    fn render_includes_non_scalar_gasless_allowlist_as_json() {
5867        use mysten_common::rpc_format::Unmetered;
5868        use serde_json::json;
5869
5870        let mut config = ProtocolConfig::get_for_max_version_UNSAFE();
5871        config.set_gasless_allowed_token_types_for_testing(vec![
5872            ("0xa::usdc::USDC".to_string(), 10_000),
5873            ("0xb::usdt::USDT".to_string(), 0),
5874        ]);
5875
5876        let rendered = config
5877            .render::<serde_json::Value>(&mut Unmetered)
5878            .expect("render should succeed under Unmetered budget");
5879        let allowlist = rendered
5880            .get("gasless_allowed_token_types")
5881            .expect("entry should be present after the testing setter");
5882
5883        // u64 values render as strings to preserve JS precision; the tuple becomes a 2-element
5884        // JSON array.
5885        assert_eq!(
5886            allowlist,
5887            &json!([["0xa::usdc::USDC", "10000"], ["0xb::usdt::USDT", "0"],]),
5888        );
5889    }
5890
5891    #[test]
5892    fn render_targets_prost_value_for_grpc() {
5893        use mysten_common::rpc_format::Unmetered;
5894        use prost_types::value::Kind;
5895
5896        let mut config = ProtocolConfig::get_for_max_version_UNSAFE();
5897        config.set_gasless_allowed_token_types_for_testing(vec![(
5898            "0xa::usdc::USDC".to_string(),
5899            10_000,
5900        )]);
5901
5902        let rendered = config
5903            .render::<prost_types::Value>(&mut Unmetered)
5904            .expect("render to prost Value should succeed");
5905        let allowlist = rendered
5906            .get("gasless_allowed_token_types")
5907            .expect("entry should be present after the testing setter");
5908
5909        // Outer ListValue with one inner ListValue carrying [coin_type_string, amount_string].
5910        let Some(Kind::ListValue(outer)) = &allowlist.kind else {
5911            panic!(
5912                "expected ListValue at the top level, got {:?}",
5913                allowlist.kind
5914            );
5915        };
5916        assert_eq!(outer.values.len(), 1, "one allowlisted entry");
5917        let Some(Kind::ListValue(entry)) = &outer.values[0].kind else {
5918            panic!("expected each entry to be a ListValue");
5919        };
5920        assert_eq!(entry.values.len(), 2, "entry has (coin_type, amount)");
5921
5922        let Some(Kind::StringValue(coin_type)) = &entry.values[0].kind else {
5923            panic!("expected coin_type as StringValue");
5924        };
5925        assert_eq!(coin_type, "0xa::usdc::USDC");
5926
5927        // u64 amount renders as a string, not a NumberValue — this is the precision-safe path.
5928        let Some(Kind::StringValue(amount)) = &entry.values[1].kind else {
5929            panic!(
5930                "expected minimum_transfer_amount as StringValue (precision-safe u64); got {:?}",
5931                entry.values[1].kind,
5932            );
5933        };
5934        assert_eq!(amount, "10000");
5935    }
5936
5937    #[test]
5938    fn render_emits_null_for_unset_protocol_versions() {
5939        use mysten_common::rpc_format::Unmetered;
5940
5941        let config = ProtocolConfig::get_for_version(1.into(), Chain::Unknown);
5942        let rendered = config
5943            .render::<serde_json::Value>(&mut Unmetered)
5944            .expect("render should succeed");
5945        // The gasless allowlist key is present in every version's keyset, but renders as JSON
5946        // `null` for versions that predate the feature. This keeps the keyset stable across
5947        // protocol versions so clients can distinguish "unknown key" from "present but unset".
5948        let entry = rendered
5949            .get("gasless_allowed_token_types")
5950            .expect("key should be present for every protocol version");
5951        assert!(
5952            entry.is_null(),
5953            "value should be null for pre-feature protocol version, got {entry:?}",
5954        );
5955    }
5956}