near-primitives 0.35.1

This crate provides the base set of primitives used by other nearcore crates
Documentation
use crate::merkle::MerklePath;
use crate::types::StaticSignatureDifferentiator;
use crate::validator_signer::ValidatorSigner;
use near_crypto::{PublicKey, Signature};
use near_primitives_core::hash::CryptoHash;
use near_primitives_core::types::{AccountId, MerkleHash, ShardId};

#[derive(borsh::BorshSerialize, borsh::BorshDeserialize, Debug, Clone, PartialEq, Eq, Hash)]
pub enum SpiceDataIdentifier {
    ReceiptProof { block_hash: CryptoHash, from_shard_id: ShardId, to_shard_id: ShardId },
    Witness { block_hash: CryptoHash, shard_id: ShardId },
}

impl SpiceDataIdentifier {
    pub fn block_hash(&self) -> &CryptoHash {
        match self {
            SpiceDataIdentifier::ReceiptProof { block_hash, .. } => block_hash,
            SpiceDataIdentifier::Witness { block_hash, .. } => block_hash,
        }
    }
}

#[derive(borsh::BorshSerialize, borsh::BorshDeserialize, Debug, Clone, PartialEq, Eq, Hash)]
pub struct SpiceDataCommitment {
    pub hash: CryptoHash,
    pub root: MerkleHash,
    pub encoded_length: u64,
}

#[derive(borsh::BorshSerialize, borsh::BorshDeserialize, Debug, Clone, PartialEq, Eq)]
pub struct SpiceDataPart {
    pub part_ord: u64,
    pub part: Box<[u8]>,
    pub merkle_proof: MerklePath,
}

// TODO(spice): Version this struct since it is sent over the network.
/// Partial data for spice with unverified signature.
#[derive(borsh::BorshSerialize, borsh::BorshDeserialize, Debug, Clone, PartialEq, Eq)]
pub struct SpicePartialData {
    inner: SpicePartialDataInner,
    sender: AccountId,
    signature: Signature,
}

impl SpicePartialData {
    pub fn new(
        id: SpiceDataIdentifier,
        commitment: SpiceDataCommitment,
        parts: Vec<SpiceDataPart>,
        signer: &ValidatorSigner,
    ) -> Self {
        let inner = SpicePartialDataInner { id, commitment, parts };
        let signature = signer.sign_bytes(&inner.serialize_for_signing());
        Self { inner, signature, sender: signer.validator_id().clone() }
    }

    pub fn block_hash(&self) -> &CryptoHash {
        self.inner.id.block_hash()
    }

    pub fn sender(&self) -> &AccountId {
        &self.sender
    }

    pub fn into_verified(self, public_key: &PublicKey) -> Option<SpiceVerifiedPartialData> {
        let data = self.inner.serialize_for_signing();
        if !self.signature.verify(&data, public_key) {
            return None;
        }
        Some(SpiceVerifiedPartialData {
            id: self.inner.id,
            commitment: self.inner.commitment,
            parts: self.inner.parts,
            sender: self.sender,
        })
    }
}

#[derive(borsh::BorshSerialize, borsh::BorshDeserialize, Debug, Clone, PartialEq, Eq)]
struct SpicePartialDataInner {
    // We include id to allow finding recipients and producers when receiving the data.
    id: SpiceDataIdentifier,
    commitment: SpiceDataCommitment,
    parts: Vec<SpiceDataPart>,
}

impl SpicePartialDataInner {
    fn serialize_for_signing(&self) -> Vec<u8> {
        static SIGNATURE_DIFFERENTIATOR: StaticSignatureDifferentiator = "SpicePartialData";
        let data = (self, SIGNATURE_DIFFERENTIATOR);
        borsh::to_vec(&data).unwrap()
    }
}

/// Spice partial data with verified signature.
#[derive(borsh::BorshSerialize, borsh::BorshDeserialize, Debug, Clone, PartialEq, Eq)]
pub struct SpiceVerifiedPartialData {
    pub id: SpiceDataIdentifier,
    pub commitment: SpiceDataCommitment,
    pub parts: Vec<SpiceDataPart>,
    pub sender: AccountId,
}

// Outside of tests it should be impossible to create spice partial data with invalid signature.
pub fn testonly_create_spice_partial_data(
    id: SpiceDataIdentifier,
    commitment: SpiceDataCommitment,
    parts: Vec<SpiceDataPart>,
    signature: Signature,
    sender: AccountId,
) -> SpicePartialData {
    SpicePartialData { inner: SpicePartialDataInner { id, commitment, parts }, signature, sender }
}