Fix post_beacon_pool_payload_attestations_valid for Gloas

PR #9178 added this Gloas-only test using ApiTester::new() which
produces a phase0 chain even under FORK_NAME=gloas, so
head.beacon_state.get_ptc(...) errored with IncorrectStateVariant.
After switching to new_with_hard_forks() three further issues
surfaced:

1. The slot clock is left at head_slot + 1 by the harness setup, so
   a payload attestation for head_slot fails gossip propagation as a
   PastSlot. Rewind the clock to head_slot in
   make_valid_payload_attestation_message.

2. With VALIDATOR_COUNT = 32 and 32 slots/epoch, a slot's committees
   often hold only a single validator. The PTC for that slot then
   has one distinct validator regardless of PTCSize. The original
   test chained JSON and SSZ sub-tests on the same harness with
   ptc_offset 0 and 1 and asserted both gossip-published, but the
   second message is a duplicate (same slot/validator) and is
   silently dropped as PriorPayloadAttestationMessageKnown — so the
   second recv() hangs forever. Split the SSZ variant into its own
   test with its own harness so the two don't collide in the
   ObservedPayloadAttesters cache.

3. Switching the JSON test to new_with_hard_forks() so the chain
   actually reaches Gloas under FORK_NAME=gloas (same fix as the
   sibling tests added in #8415 and #9100).

Verified locally: full Gloas suite 197/197 passed (350s).
This commit is contained in:
dapplion
2026-04-28 14:03:00 +02:00
parent f583e1886c
commit 85b6c4d93f

View File

@@ -2803,6 +2803,12 @@ impl ApiTester {
let fork = head.beacon_state.fork();
let genesis_validators_root = self.chain.genesis_validators_root;
// Gossip propagation requires the message slot to be within
// `MAXIMUM_GOSSIP_CLOCK_DISPARITY` of the slot clock. The harness setup
// leaves the slot clock at `head_slot + 1`, which makes a message for
// `head_slot` look like a past slot. Rewind the clock to the head slot.
self.chain.slot_clock.set_slot(head_slot.as_u64());
let ptc = head
.beacon_state
.get_ptc(head_slot, &self.chain.spec)
@@ -8339,9 +8345,22 @@ async fn post_beacon_pool_payload_attestations_valid() {
if !fork_name_from_env().is_some_and(|f| f.gloas_enabled()) {
return;
}
ApiTester::new()
ApiTester::new_with_hard_forks()
.await
.test_post_beacon_pool_payload_attestations_valid()
.await;
}
#[tokio::test(flavor = "multi_thread", worker_threads = 2)]
async fn post_beacon_pool_payload_attestations_valid_ssz() {
if !fork_name_from_env().is_some_and(|f| f.gloas_enabled()) {
return;
}
// Use a separate harness from the JSON variant so that the SSZ sub-test does
// not collide with the JSON sub-test in the gossip dedup cache (with the
// small `VALIDATOR_COUNT` used by these tests, the slot's PTC may hold only
// one distinct validator, making the second message a duplicate).
ApiTester::new_with_hard_forks()
.await
.test_post_beacon_pool_payload_attestations_valid_ssz()
.await;