Skip to content

Commit

Permalink
Basic unit tests for ProveReplicaUpdate3
Browse files Browse the repository at this point in the history
  • Loading branch information
anorth committed Sep 3, 2023
1 parent 18940d8 commit 1e797aa
Show file tree
Hide file tree
Showing 7 changed files with 475 additions and 215 deletions.
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions actors/miner/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ fil_actor_account = { workspace = true }
fil_actor_reward = { workspace = true }
fil_actor_power = { workspace = true }
fil_actor_market = { workspace = true }
fil_actor_verifreg = { workspace = true }
rand = { workspace = true }
test-case = { workspace = true }

Expand Down
29 changes: 14 additions & 15 deletions actors/miner/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1045,7 +1045,7 @@ impl Actor {

// Validate parameters.
rt.validate_immediate_caller_is(
info.control_addresses.iter().chain(&[info.owner, info.worker]),
info.control_addresses.iter().chain(&[info.worker, info.owner]),
)?;
if params.sector_proofs.is_empty() == params.aggregate_proof.is_empty() {
return Err(actor_error!(
Expand Down Expand Up @@ -1154,7 +1154,7 @@ impl Actor {
}
let proven_batch = proven_batch_gen.gen();

// Activate data and compute CommD.
// Activate data.
let data_activation_inputs: Vec<SectorPiecesActivationInput> = proven_manifests
.iter()
.map(|(update, info)| SectorPiecesActivationInput {
Expand Down Expand Up @@ -1855,7 +1855,7 @@ impl Actor {
sector_expiry: precommit.info.expiration,
sector_number: precommit.info.sector_number,
sector_type: precommit.info.seal_proof,
expected_commd: Some(precommit.info.unsealed_cid.clone()),
expected_commd: Some(precommit.info.unsealed_cid.clone()), // Check CommD
}
})
.collect();
Expand Down Expand Up @@ -5365,25 +5365,26 @@ struct ReplicaUpdateActivatedData {

// Activates data pieces by claiming allocations with the verified registry.
// Pieces are grouped by sector and succeed or fail in sector groups.
// Activation inputs specify an expected CommD for the sector,
// against which the CommD computed from the pieces is checked.
// If an activation input specifies an expected CommD for the sector, a CommD
// is calculated from the pieces and must match.
// This method never returns CommDs in the output type; either the caller provided
// them and they are correct, or the caller did not provide anything that needs checking.
fn activate_sectors_pieces(
rt: &impl Runtime,
activation_inputs: Vec<SectorPiecesActivationInput>,
all_or_nothing: bool,
) -> Result<(BatchReturn, Vec<DataActivationOutput>), ActorError> {
// Get a flattened list of verified claims for all activated sectors
let mut verified_claims = Vec::new();
let mut computed_commds = Vec::new();
let mut unverified_spaces = Vec::new();
for activation_info in activation_inputs {
let computed_commd = unsealed_cid_from_pieces(
rt,
&activation_info.piece_manifests,
activation_info.sector_type,
)?;
// Check a declared CommD matches that computed from the data.
if let Some(declared_commd) = activation_info.expected_commd {
let computed_commd = unsealed_cid_from_pieces(
rt,
&activation_info.piece_manifests,
activation_info.sector_type,
)?;
if !declared_commd.eq(&computed_commd) {
return Err(actor_error!(
illegal_argument,
Expand All @@ -5394,7 +5395,6 @@ fn activate_sectors_pieces(
));
}
}
computed_commds.push(computed_commd);

let mut sector_claims = vec![];
let mut unverified_space = BigInt::zero();
Expand Down Expand Up @@ -5429,12 +5429,11 @@ fn activate_sectors_pieces(
let activation_outputs = claim_res
.sector_claims
.iter()
.zip(claim_res.sector_results.successes(&computed_commds))
.zip(claim_res.sector_results.successes(&unverified_spaces))
.map(|((sector_claim, computed_commd), unverified_space)| DataActivationOutput {
.map(|(sector_claim, unverified_space)| DataActivationOutput {
unverified_space: unverified_space.clone(),
verified_space: sector_claim.claimed_space.clone(),
unsealed_cid: computed_commd.0,
unsealed_cid: None,
})
.collect();

Expand Down
5 changes: 3 additions & 2 deletions actors/miner/tests/miner_actor_test_precommit_batch.rs
Original file line number Diff line number Diff line change
Expand Up @@ -148,7 +148,8 @@ mod miner_actor_precommit_batch {
SectorDeals, VerifyDealsForActivationParams, VerifyDealsForActivationReturn,
};
use fil_actor_miner::{
new_deadline_info_from_offset_and_epoch, Actor, Method, PreCommitSectorBatchParams2,
new_deadline_info_from_offset_and_epoch, Actor, CompactCommD, Method,
PreCommitSectorBatchParams2,
};
use fil_actors_runtime::{STORAGE_MARKET_ACTOR_ADDR, STORAGE_POWER_ACTOR_ADDR};
use fvm_ipld_encoding::ipld_block::IpldBlock;
Expand Down Expand Up @@ -332,7 +333,7 @@ mod miner_actor_precommit_batch {
precommit_epoch - 1,
sector_expiration,
vec![1],
Some(make_piece_cid(&[1])),
CompactCommD::new(Some(make_piece_cid(&[1]))),
);
let sectors = vec![sector];
{
Expand Down
161 changes: 161 additions & 0 deletions actors/miner/tests/prove_replica_test.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,161 @@
use fil_actor_market::NO_ALLOCATION_ID;
use fvm_ipld_blockstore::Blockstore;
use fvm_shared::deal::DealID;
use fvm_shared::piece::PaddedPieceSize;
use fvm_shared::sector::SectorNumber;
use fvm_shared::{bigint::Zero, clock::ChainEpoch, econ::TokenAmount, ActorID};

use fil_actor_miner::ext::verifreg::AllocationID;
use fil_actor_miner::{
CompactCommD, PieceActivationManifest, SectorOnChainInfo, SectorPreCommitInfo,
SectorPreCommitOnChainInfo, SectorUpdateManifest, State, VerifiedAllocationKey,
};
use fil_actor_miner::{DataActivationNotification, ProveReplicaUpdates3Return};
use fil_actors_runtime::cbor::serialize;
use fil_actors_runtime::test_utils::{make_piece_cid, make_sealed_cid, MockRuntime};
use fil_actors_runtime::{runtime::Runtime, BatchReturn, EPOCHS_IN_DAY, STORAGE_MARKET_ACTOR_ADDR};
use util::*;

mod util;

const DEFAULT_SECTOR_EXPIRATION_DAYS: ChainEpoch = 220;

#[test]
fn prove_single_sectorxxx() {
let h = ActorHarness::new_with_options(HarnessOptions::default());
let rt = h.new_runtime();
rt.set_balance(BIG_BALANCE.clone());
let client_id: ActorID = 1000;

h.construct_and_verify(&rt);

// Onboard a batch of empty sectors.
rt.set_epoch(1);
let sector_expiry = *rt.epoch.borrow() + DEFAULT_SECTOR_EXPIRATION_DAYS * EPOCHS_IN_DAY;
let first_sector_number = 100;
let sector_count = 4;
let sectors = onboard_empty_sectors(&rt, &h, sector_expiry, first_sector_number, sector_count);

// Update them in batch, each with a single piece.
let st: State = h.get_state(&rt);
let store = rt.store();
let piece_size = h.sector_size as u64;
let sector_updates = vec![
make_update_manifest(&st, store, &sectors[0], &[(piece_size, 0, 0, 0)]), // No alloc or deal
make_update_manifest(&st, store, &sectors[1], &[(piece_size, client_id, 1000, 0)]), // Just an alloc
make_update_manifest(&st, store, &sectors[2], &[(piece_size, 0, 0, 2000)]), // Just a deal
make_update_manifest(&st, store, &sectors[3], &[(piece_size, client_id, 1001, 2001)]), // Alloc and deal
];

let result = h.prove_replica_updates3_batch(&rt, &sector_updates, true, true).unwrap();
assert_eq!(
ProveReplicaUpdates3Return { activation_results: BatchReturn::ok(sectors.len() as u32) },
result
);
}

fn onboard_empty_sectors(
rt: &MockRuntime,
h: &ActorHarness,
expiration: ChainEpoch,
first_sector_number: SectorNumber,
count: usize,
) -> Vec<SectorOnChainInfo> {
let precommit_epoch = *rt.epoch.borrow();

// Precommit the sectors.
let precommits =
make_empty_precommits(h, first_sector_number, precommit_epoch - 1, expiration, count);
h.pre_commit_sector_batch_v2(rt, &precommits, true, &TokenAmount::zero()).unwrap();
let precommits: Vec<SectorPreCommitOnChainInfo> =
precommits.iter().map(|sector| h.get_precommit(rt, sector.sector_number)).collect();

// Prove the sectors.
// Note: migrate this to ProveCommitSectors2 (batch) when the harness supports it.
rt.set_epoch(precommit_epoch + rt.policy.pre_commit_challenge_delay + 1);
let sectors: Vec<SectorOnChainInfo> = precommits
.iter()
.map(|pc| {
h.prove_commit_sector_and_confirm(
rt,
pc,
h.make_prove_commit_params(pc.info.sector_number),
ProveCommitConfig::default(),
)
.unwrap()
})
.collect();

// Window PoST to activate the sectors, a pre-requisite for upgrading.
h.advance_and_submit_posts(rt, &sectors);
sectors
}

fn make_empty_precommits(
h: &ActorHarness,
first_sector_number: SectorNumber,
challenge: ChainEpoch,
expiration: ChainEpoch,
count: usize,
) -> Vec<SectorPreCommitInfo> {
(0..count)
.map(|i| {
let sector_number = first_sector_number + i as u64;
h.make_pre_commit_params_v2(
sector_number,
challenge,
expiration,
vec![],
CompactCommD::empty(),
)
})
.collect()
}

fn make_update_manifest(
st: &State,
store: &impl Blockstore,
sector: &SectorOnChainInfo,
piece_specs: &[(u64, ActorID, AllocationID, DealID)],
) -> SectorUpdateManifest {
let (deadline, partition) = st.find_sector(store, sector.sector_number).unwrap();
let new_sealed_cid = make_sealed_cid(format!("sealed{}", sector.sector_number).as_bytes());
let pieces: Vec<PieceActivationManifest> = piece_specs
.iter()
.enumerate()
.map(|(i, (sz, client, alloc, deal))| make_piece_manifest(i, *sz, *client, *alloc, *deal))
.collect();
SectorUpdateManifest {
sector: sector.sector_number,
deadline,
partition,
new_sealed_cid,
pieces,
}
}

fn make_piece_manifest(
seq: usize,
size: u64,
alloc_client: ActorID,
alloc_id: AllocationID,
deal: DealID,
) -> PieceActivationManifest {
PieceActivationManifest {
cid: make_piece_cid(format!("piece-{}", seq).as_bytes()),
size: PaddedPieceSize(size),
verified_allocation_key: if alloc_id != NO_ALLOCATION_ID {
Some(VerifiedAllocationKey { client: alloc_client, id: alloc_id })
} else {
None
},
notify: if deal != 0 {
vec![DataActivationNotification {
address: STORAGE_MARKET_ACTOR_ADDR,
payload: serialize(&deal, "deal id").unwrap(),
}]
} else {
vec![]
},
}
}
Loading

0 comments on commit 1e797aa

Please sign in to comment.