|
| 1 | +mod helpers; |
| 2 | + |
| 3 | +use { |
| 4 | + bincode::Options, |
| 5 | + helpers::*, |
| 6 | + p_stake_interface::state::{StakeStateV2, StakeStateV2Layout, StakeStateV2View}, |
| 7 | + proptest::prelude::*, |
| 8 | + solana_stake_interface::state::StakeStateV2 as LegacyStakeStateV2, |
| 9 | + wincode::ZeroCopy, |
| 10 | +}; |
| 11 | + |
| 12 | +fn assert_legacy_and_view_agree(bytes: &[u8]) { |
| 13 | + let legacy: LegacyStakeStateV2 = bincode_opts().deserialize(bytes).unwrap(); |
| 14 | + let view = StakeStateV2::from_bytes(bytes).unwrap(); |
| 15 | + |
| 16 | + match (legacy, view) { |
| 17 | + (LegacyStakeStateV2::Uninitialized, StakeStateV2View::Uninitialized) => {} |
| 18 | + (LegacyStakeStateV2::RewardsPool, StakeStateV2View::RewardsPool) => {} |
| 19 | + (LegacyStakeStateV2::Initialized(legacy_meta), StakeStateV2View::Initialized(meta)) => { |
| 20 | + assert_meta_compat(meta, &legacy_meta); |
| 21 | + } |
| 22 | + ( |
| 23 | + LegacyStakeStateV2::Stake(legacy_meta, legacy_stake, legacy_flags), |
| 24 | + StakeStateV2View::Stake { meta, stake }, |
| 25 | + ) => { |
| 26 | + assert_meta_compat(meta, &legacy_meta); |
| 27 | + assert_stake_compat(stake, &legacy_stake); |
| 28 | + |
| 29 | + // ABI: stake_flags byte must match legacy exactly. |
| 30 | + let layout = StakeStateV2Layout::from_bytes(bytes).unwrap(); |
| 31 | + assert_eq!(layout.stake_flags, stake_flags_byte(&legacy_flags)); |
| 32 | + } |
| 33 | + |
| 34 | + (o, v) => panic!("variant mismatch legacy={o:?} new={v:?}"), |
| 35 | + } |
| 36 | +} |
| 37 | + |
| 38 | +proptest! { |
| 39 | + #![proptest_config(ProptestConfig::with_cases(10000))] |
| 40 | + |
| 41 | + // legacy bincode == new layout wincode bytes |
| 42 | + #[test] |
| 43 | + fn prop_wincode_roundtrips_legacy_bytes(legacy in arb_legacy_state()) { |
| 44 | + let expected = serialize_legacy(&legacy); |
| 45 | + prop_assert_eq!(expected.len(), LAYOUT_LEN); |
| 46 | + |
| 47 | + let new_layout = StakeStateV2Layout::from_bytes(&expected[..]).unwrap(); |
| 48 | + let mut actual = [0u8; 200]; |
| 49 | + wincode::serialize_into(&mut actual.as_mut_slice(), new_layout).unwrap(); |
| 50 | + |
| 51 | + prop_assert_eq!(expected.as_slice(), &actual); |
| 52 | + } |
| 53 | + |
| 54 | + // both the legacy decoder and zero-copy view interpret trailing bytes the same |
| 55 | + #[test] |
| 56 | + fn prop_unpadded_legacy_prefix_is_compatible(legacy in arb_legacy_state(), mut tail in any::<[u8; 200]>()) { |
| 57 | + let prefix = serialize_legacy_unpadded(&legacy); |
| 58 | + tail[..prefix.len()].copy_from_slice(&prefix); |
| 59 | + assert_legacy_and_view_agree(&tail[..]); |
| 60 | + } |
| 61 | + |
| 62 | + // arbitrary 200-byte blobs with a valid tag must parse identically in legacy bincode and the zero-copy view |
| 63 | + #[test] |
| 64 | + fn prop_any_200_bytes_with_valid_tag_legacy_and_new_agree(mut bytes in any::<[u8; 200]>(), tag in 0u32..=3u32) { |
| 65 | + bytes[..4].copy_from_slice(&tag.to_le_bytes()); |
| 66 | + assert_legacy_and_view_agree(&bytes[..]); |
| 67 | + } |
| 68 | +} |
0 commit comments