use crate::platform::transition::broadcast_request::BroadcastRequestForStateTransition;
use std::sync::Arc;
use crate::{Error, Sdk};
use crate::platform::block_info_from_metadata::block_info_from_metadata;
use crate::platform::transition::put_settings::PutSettings;
use crate::platform::Identifier;
use dapi_grpc::platform::VersionedGrpcResponse;
use dpp::data_contract::document_type::accessors::DocumentTypeV0Getters;
use dpp::data_contract::document_type::DocumentType;
use dpp::data_contract::DataContract;
use dpp::document::{Document, DocumentV0Getters};
use dpp::identity::signer::Signer;
use dpp::identity::IdentityPublicKey;
use dpp::state_transition::documents_batch_transition::methods::v0::DocumentsBatchTransitionMethodsV0;
use dpp::state_transition::documents_batch_transition::DocumentsBatchTransition;
use dpp::state_transition::proof_result::StateTransitionProofResult;
use dpp::state_transition::StateTransition;
use drive::drive::Drive;
use rs_dapi_client::{DapiRequest, RequestSettings};
#[async_trait::async_trait]
pub trait TransferDocument<S: Signer> {
async fn transfer_document_to_identity(
&self,
recipient_id: Identifier,
sdk: &Sdk,
document_type: DocumentType,
identity_public_key: IdentityPublicKey,
signer: &S,
settings: Option<PutSettings>,
) -> Result<StateTransition, Error>;
async fn wait_for_response(
&self,
sdk: &Sdk,
state_transition: StateTransition,
data_contract: Arc<DataContract>,
) -> Result<Document, Error>;
async fn transfer_document_to_identity_and_wait_for_response(
&self,
recipient_id: Identifier,
sdk: &Sdk,
document_type: DocumentType,
identity_public_key: IdentityPublicKey,
data_contract: Arc<DataContract>,
signer: &S,
) -> Result<Document, Error>;
}
#[async_trait::async_trait]
impl<S: Signer> TransferDocument<S> for Document {
async fn transfer_document_to_identity(
&self,
recipient_id: Identifier,
sdk: &Sdk,
document_type: DocumentType,
identity_public_key: IdentityPublicKey,
signer: &S,
settings: Option<PutSettings>,
) -> Result<StateTransition, Error> {
let new_identity_contract_nonce = sdk
.get_identity_contract_nonce(
self.owner_id(),
document_type.data_contract_id(),
true,
settings,
)
.await?;
let settings = settings.unwrap_or_default();
let transition = DocumentsBatchTransition::new_document_transfer_transition_from_document(
self.clone(),
document_type.as_ref(),
recipient_id,
&identity_public_key,
new_identity_contract_nonce,
settings.user_fee_increase.unwrap_or_default(),
signer,
sdk.version(),
None,
None,
None,
)?;
let request = transition.broadcast_request_for_state_transition()?;
request
.clone()
.execute(sdk, settings.request_settings)
.await?;
Ok(transition)
}
async fn wait_for_response(
&self,
sdk: &Sdk,
state_transition: StateTransition,
data_contract: Arc<DataContract>,
) -> Result<Document, Error> {
let request = state_transition.wait_for_state_transition_result_request()?;
let response = request.execute(sdk, RequestSettings::default()).await?;
let block_info = block_info_from_metadata(response.metadata()?)?;
let proof = response.proof_owned()?;
let (_, result) = Drive::verify_state_transition_was_executed_with_proof(
&state_transition,
&block_info,
proof.grovedb_proof.as_slice(),
&|_| Ok(Some(data_contract.clone())),
sdk.version(),
)?;
match result {
StateTransitionProofResult::VerifiedDocuments(mut documents) => {
let document = documents
.remove(self.id_ref())
.ok_or(Error::InvalidProvedResponse(
"did not prove the sent document".to_string(),
))?
.ok_or(Error::InvalidProvedResponse(
"expected there to actually be a document".to_string(),
))?;
Ok(document)
}
_ => Err(Error::DapiClientError("proved a non document".to_string())),
}
}
async fn transfer_document_to_identity_and_wait_for_response(
&self,
recipient_id: Identifier,
sdk: &Sdk,
document_type: DocumentType,
identity_public_key: IdentityPublicKey,
data_contract: Arc<DataContract>,
signer: &S,
) -> Result<Document, Error> {
let state_transition = self
.transfer_document_to_identity(
recipient_id,
sdk,
document_type,
identity_public_key,
signer,
None,
)
.await?;
let document = <Self as TransferDocument<S>>::wait_for_response(
self,
sdk,
state_transition,
data_contract,
)
.await?;
Ok(document)
}
}