From b3f5f17d7c7a85ecf075b13a220ad084dda3648e Mon Sep 17 00:00:00 2001 From: Lukasz Klimek <842586+lklimek@users.noreply.github.com> Date: Thu, 25 Jul 2024 13:13:11 +0200 Subject: [PATCH] feat(sdk): asset lock quorum verify against platform --- packages/rs-sdk/src/error.rs | 8 ++ .../src/platform/transition/put_identity.rs | 78 +++++++++++++++++-- 2 files changed, 80 insertions(+), 6 deletions(-) diff --git a/packages/rs-sdk/src/error.rs b/packages/rs-sdk/src/error.rs index 0a5024f541d..892824e533c 100644 --- a/packages/rs-sdk/src/error.rs +++ b/packages/rs-sdk/src/error.rs @@ -49,6 +49,14 @@ pub enum Error { /// Epoch not found; we must have at least one epoch #[error("No epoch found on the Platform; it should never happen")] EpochNotFound, + /// Quorum not found; try again later + #[error("Quorum {quorum_hash_hex} of type {quorum_type} not yet available on the platform at height {core_chain_locked_height}: {e}; try again later")] + QuorumNotFound { + quorum_hash_hex: String, + quorum_type: u32, + core_chain_locked_height: u32, + e: ContextProviderError, + }, /// SDK operation timeout reached error #[error("SDK operation timeout {} secs reached: {1}", .0.as_secs())] TimeoutReached(Duration, String), diff --git a/packages/rs-sdk/src/platform/transition/put_identity.rs b/packages/rs-sdk/src/platform/transition/put_identity.rs index 59422aa6b2b..d4bb8b946cd 100644 --- a/packages/rs-sdk/src/platform/transition/put_identity.rs +++ b/packages/rs-sdk/src/platform/transition/put_identity.rs @@ -1,20 +1,21 @@ +use crate::platform::block_info_from_metadata::block_info_from_metadata; use crate::platform::transition::broadcast_identity::BroadcastRequestForNewIdentity; use crate::platform::transition::broadcast_request::BroadcastRequestForStateTransition; use crate::platform::Fetch; use crate::{Error, Sdk}; - +use dapi_grpc::platform::v0::get_epochs_info_request::{self, GetEpochsInfoRequestV0}; +use dapi_grpc::platform::v0::GetEpochsInfoRequest; use dapi_grpc::platform::VersionedGrpcResponse; use dapi_grpc::tonic::Code; +use dpp::dashcore::hashes::Hash; use dpp::dashcore::PrivateKey; use dpp::identity::signer::Signer; use dpp::prelude::{AssetLockProof, Identity}; -use drive_proof_verifier::error::ContextProviderError; -use drive_proof_verifier::DataContractProvider; - -use crate::platform::block_info_from_metadata::block_info_from_metadata; use dpp::state_transition::proof_result::StateTransitionProofResult; use drive::drive::Drive; -use rs_dapi_client::{DapiClientError, DapiRequest, RequestSettings}; +use drive_proof_verifier::error::ContextProviderError; +use drive_proof_verifier::{ContextProvider, DataContractProvider}; +use rs_dapi_client::{DapiClientError, DapiRequest, DapiRequestExecutor, RequestSettings}; #[async_trait::async_trait] /// A trait for putting an identity to platform @@ -37,6 +38,71 @@ pub trait PutIdentity { ) -> Result; } +#[async_trait::async_trait] +pub trait AssetLockProofVerifier { + /// Verifies the asset lock proof against the platform + async fn verify(&self, sdk: &Sdk) -> Result<(), Error>; +} + +#[async_trait::async_trait] +impl AssetLockProofVerifier for AssetLockProof { + async fn verify(&self, sdk: &Sdk) -> Result<(), Error> { + let context_provider = sdk + .context_provider() + .ok_or(Error::Config("Context Provider not configured".to_string()))?; + + // Check status of Platform first + // TODO: implement some caching mechanism to avoid fetching the same data multiple times + let request = GetEpochsInfoRequest { + version: Some(get_epochs_info_request::Version::V0( + GetEpochsInfoRequestV0 { + ascending: false, + count: 1, + prove: true, + start_epoch: None, + }, + )), + }; + let response = sdk.execute(request, RequestSettings::default()).await?; + + let platform_core_chain_locked_height = response.metadata()?.core_chain_locked_height; + let proof = response.proof_owned()?; + let platform_quorum_hash = proof.quorum_hash.try_into().map_err(|e: Vec| { + Error::Protocol(dpp::ProtocolError::DecodingError(format!( + "Invalid quorum hash size {}, expected 32 bytes", + e.len() + ))) + })?; + + let platform_quorum_type = proof.quorum_type; + + let (quorum_hash, core_chain_locked_height) = match self { + AssetLockProof::Chain(v) => (platform_quorum_hash, v.core_chain_locked_height), + AssetLockProof::Instant(v) => ( + v.instant_lock().cyclehash.to_raw_hash().to_byte_array(), + platform_core_chain_locked_height, + ), + }; + + // Try to fetch the quorum public key; if it fails, the + let result = context_provider.get_quorum_public_key( + platform_quorum_type, + quorum_hash, + core_chain_locked_height, + ); + + match result { + Err(ContextProviderError::InvalidQuorum(s)) => Err(Error::QuorumNotFound { + e: ContextProviderError::InvalidQuorum(s), + quorum_hash_hex: hex::encode(quorum_hash), + quorum_type: platform_quorum_type, + core_chain_locked_height, + }), + Err(e) => Err(e.into()), + Ok(_) => Ok(()), + } + } +} #[async_trait::async_trait] impl PutIdentity for Identity { async fn put_to_platform(