# beacon_chain # Copyright (c) 2018-2019 Status Research & Development GmbH # Licensed and distributed under either of # * MIT license (license terms in the root directory or at https://opensource.org/licenses/MIT). # * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0). # at your option. This file may not be copied, modified, or distributed except according to those terms. import options, sequtils, stew/endians2, chronicles, eth/trie/[db], ./mocking/merkle_minimal, ../beacon_chain/[beacon_chain_db, block_pool, extras, ssz, state_transition, validator_pool], ../beacon_chain/spec/[beaconstate, crypto, datatypes, digest, helpers, validator, state_transition_block] func makeFakeValidatorPrivKey(i: int): ValidatorPrivKey = # 0 is not a valid BLS private key - 1000 helps interop with rust BLS library, # lighthouse. # TODO: switch to https://github.com/ethereum/eth2.0-pm/issues/60 var bytes = uint64(i + 1000).toBytesLE() copyMem(addr result, addr bytes[0], sizeof(bytes)) func makeFakeHash(i: int): Eth2Digest = var bytes = uint64(i).toBytesLE() static: doAssert sizeof(bytes) <= sizeof(result.data) copyMem(addr result.data[0], addr bytes[0], sizeof(bytes)) func hackPrivKey(v: Validator): ValidatorPrivKey = ## Extract private key, per above hack var bytes: array[8, byte] static: doAssert sizeof(bytes) <= sizeof(v.withdrawal_credentials.data) copyMem( addr bytes, unsafeAddr v.withdrawal_credentials.data[0], sizeof(bytes)) let i = int(uint64.fromBytesLE(bytes)) makeFakeValidatorPrivKey(i) func makeDeposit(i: int, flags: UpdateFlags): Deposit = ## Ugly hack for now: we stick the private key in withdrawal_credentials ## which means we can repro private key and randao reveal from this data, ## for testing :) let privkey = makeFakeValidatorPrivKey(i) pubkey = privkey.toPubKey() withdrawal_credentials = makeFakeHash(i) domain = compute_domain(DOMAIN_DEPOSIT, GENESIS_FORK_VERSION) result = Deposit( data: DepositData( pubkey: pubkey, withdrawal_credentials: withdrawal_credentials, amount: MAX_EFFECTIVE_BALANCE, ) ) if skipBLSValidation notin flags: let signing_root = compute_signing_root(result.getDepositMessage, domain) result.data.signature = bls_sign(privkey, signing_root.data) proc attachMerkleProofs*(deposits: var seq[Deposit]) = let deposit_data_roots = mapIt(deposits, it.data.hash_tree_root) var deposit_data_sums: seq[Eth2Digest] for prefix_root in hash_tree_roots_prefix( deposit_data_roots, 1'i64 shl DEPOSIT_CONTRACT_TREE_DEPTH): deposit_data_sums.add prefix_root for val_idx in 0 ..< deposits.len: let merkle_tree = merkleTreeFromLeaves(deposit_data_roots[0..val_idx]) deposits[val_idx].proof[0..31] = merkle_tree.getMerkleProof(val_idx) deposits[val_idx].proof[32].data[0..7] = int_to_bytes8((val_idx + 1).uint64) doAssert is_valid_merkle_branch( deposit_data_roots[val_idx], deposits[val_idx].proof, DEPOSIT_CONTRACT_TREE_DEPTH + 1, val_idx.uint64, deposit_data_sums[val_idx]) proc makeInitialDeposits*( n = SLOTS_PER_EPOCH, flags: UpdateFlags = {}): seq[Deposit] = for i in 0.. committee index -> # montonoic enumerable index, is wasteful and slow. Most test callers # want ValidatorIndex, so that's supported too. let validator = state.validators[validator_index] sac_index = committee.find(validator_index) data = makeAttestationData(state, slot, index, beacon_block_root) doAssert sac_index != -1, "find_beacon_committee should guarantee this" var aggregation_bits = CommitteeValidatorsBits.init(committee.len) aggregation_bits.setBit sac_index let sig = if skipBLSValidation notin flags: get_attestation_signature(state.fork, state.genesis_validators_root, data, hackPrivKey(validator)) else: ValidatorSig() Attestation( data: data, aggregation_bits: aggregation_bits, signature: sig ) proc find_beacon_committee( state: BeaconState, validator_index: ValidatorIndex, cache: var StateCache): auto = let epoch = compute_epoch_at_slot(state.slot) for epoch_committee_index in 0'u64 ..< get_committee_count_at_slot( state, epoch.compute_start_slot_at_epoch) * SLOTS_PER_EPOCH: let slot = ((epoch_committee_index mod SLOTS_PER_EPOCH) + epoch.compute_start_slot_at_epoch.uint64).Slot index = epoch_committee_index div SLOTS_PER_EPOCH committee = get_beacon_committee(state, slot, index, cache) if validator_index in committee: return (committee, slot, index) doAssert false proc makeAttestation*( state: BeaconState, beacon_block_root: Eth2Digest, validator_index: ValidatorIndex, cache: var StateCache, flags: UpdateFlags = {}): Attestation = let (committee, slot, index) = find_beacon_committee(state, validator_index, cache) makeAttestation(state, beacon_block_root, committee, slot, index, validator_index, cache, flags) proc makeFullAttestations*( state: BeaconState, beacon_block_root: Eth2Digest, slot: Slot, cache: var StateCache, flags: UpdateFlags = {}): seq[Attestation] = # Create attestations in which the full committee participates for each shard # that should be attested to during a particular slot let count = get_committee_count_at_slot(state, slot) for index in 0..= 1 # Initial attestation var attestation = Attestation( aggregation_bits: CommitteeValidatorsBits.init(committee.len), data: data, signature: get_attestation_signature( state.fork, state.genesis_validators_root, data, hackPrivKey(state.validators[committee[0]])) ) # Aggregate the remainder attestation.aggregation_bits.setBit 0 for j in 1 ..< committee.len(): attestation.aggregation_bits.setBit j if skipBLSValidation notin flags: attestation.signature.aggregate(get_attestation_signature( state.fork, state.genesis_validators_root, data, hackPrivKey(state.validators[committee[j]]) )) result.add attestation