Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(platform): get identities for non unique public key hash #2324

Draft
wants to merge 3 commits into
base: v1.8-dev
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 4 additions & 2 deletions packages/dapi-grpc/build.rs
Original file line number Diff line number Diff line change
Expand Up @@ -47,14 +47,15 @@ fn configure_platform(mut platform: MappingConfig) -> MappingConfig {
// Derive features for versioned messages
//
// "GetConsensusParamsRequest" is excluded as this message does not support proofs
const VERSIONED_REQUESTS: [&str; 30] = [
const VERSIONED_REQUESTS: [&str; 31] = [
"GetDataContractHistoryRequest",
"GetDataContractRequest",
"GetDataContractsRequest",
"GetDocumentsRequest",
"GetIdentitiesByPublicKeyHashesRequest",
"GetIdentitiesRequest",
"GetIdentitiesBalancesRequest",
"GetIdentitiesForNonUniquePublicKeyHashRequest",
"GetIdentityNonceRequest",
"GetIdentityContractNonceRequest",
"GetIdentityBalanceAndRevisionRequest",
Expand Down Expand Up @@ -85,14 +86,15 @@ fn configure_platform(mut platform: MappingConfig) -> MappingConfig {
// - "GetStatusResponse"
//
// "GetEvonodesProposedEpochBlocksResponse" is used for 2 Requests
const VERSIONED_RESPONSES: [&str; 29] = [
const VERSIONED_RESPONSES: [&str; 30] = [
"GetDataContractHistoryResponse",
"GetDataContractResponse",
"GetDataContractsResponse",
"GetDocumentsResponse",
"GetIdentitiesByPublicKeyHashesResponse",
"GetIdentitiesResponse",
"GetIdentitiesBalancesResponse",
"GetIdentitiesForNonUniquePublicKeyHashResponse",
"GetIdentityBalanceAndRevisionResponse",
"GetIdentityBalanceResponse",
"GetIdentityNonceResponse",
Expand Down
27 changes: 27 additions & 0 deletions packages/dapi-grpc/protos/platform/v0/platform.proto
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,8 @@ service Platform {
rpc getDocuments(GetDocumentsRequest) returns (GetDocumentsResponse);
rpc getIdentityByPublicKeyHash(GetIdentityByPublicKeyHashRequest)
returns (GetIdentityByPublicKeyHashResponse);
rpc getIdentitiesForNonUniquePublicKeyHash(GetIdentitiesForNonUniquePublicKeyHashRequest)
returns (GetIdentitiesForNonUniquePublicKeyHashResponse);
rpc waitForStateTransitionResult(WaitForStateTransitionResultRequest)
returns (WaitForStateTransitionResultResponse);
rpc getConsensusParams(GetConsensusParamsRequest)
Expand Down Expand Up @@ -592,6 +594,31 @@ message GetIdentityByPublicKeyHashResponse {
oneof version { GetIdentityByPublicKeyHashResponseV0 v0 = 1; }
}

message GetIdentitiesForNonUniquePublicKeyHashRequest {
message GetIdentitiesForNonUniquePublicKeyHashRequestV0 {
bytes public_key_hash = 1; // The non-unique public key hash for which identities are requested
bool prove = 2; // Flag to request proof for the response
}
oneof version { GetIdentitiesForNonUniquePublicKeyHashRequestV0 v0 = 1; }
}

message GetIdentitiesForNonUniquePublicKeyHashResponse {
message GetIdentitiesForNonUniquePublicKeyHashResponseV0 {
message Identities {
repeated bytes identities = 1;
}

oneof result {
Identities identities = 1; // The list of identity data corresponding to the requested public key hash
Proof proof = 2; // Cryptographic proof for the identities data, if requested
}

ResponseMetadata metadata = 3; // Metadata about the blockchain state
}
oneof version { GetIdentitiesForNonUniquePublicKeyHashResponseV0 v0 = 1; }
}


message WaitForStateTransitionResultRequest {
message WaitForStateTransitionResultRequestV0 {
bytes state_transition_hash = 1; // The hash of the state transition to wait for
Expand Down
8 changes: 8 additions & 0 deletions packages/rs-dapi-client/src/transport/grpc.rs
Original file line number Diff line number Diff line change
Expand Up @@ -500,3 +500,11 @@ impl_transport_request_grpc!(
RequestSettings::default(),
get_status
);

impl_transport_request_grpc!(
platform_proto::GetIdentitiesForNonUniquePublicKeyHashRequest,
platform_proto::GetIdentitiesForNonUniquePublicKeyHashResponse,
PlatformGrpcClient,
RequestSettings::default(),
get_identities_for_non_unique_public_key_hash
);
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
use crate::error::query::QueryError;
use crate::error::Error;
use crate::platform_types::platform::Platform;
use crate::platform_types::platform_state::PlatformState;
use crate::query::QueryValidationResult;
use dapi_grpc::platform::v0::get_identities_for_non_unique_public_key_hash_request::Version as RequestVersion;
use dapi_grpc::platform::v0::get_identities_for_non_unique_public_key_hash_response::Version as ResponseVersion;
use dapi_grpc::platform::v0::{
GetIdentitiesForNonUniquePublicKeyHashRequest, GetIdentitiesForNonUniquePublicKeyHashResponse,
GetIdentityByPublicKeyHashResponse,
};
use dpp::version::PlatformVersion;

mod v0;

impl<C> Platform<C> {
/// Querying of identities for a non-unique public key hash
pub fn query_identities_for_non_unique_public_key_hash(
&self,
GetIdentitiesForNonUniquePublicKeyHashRequest { version }: GetIdentitiesForNonUniquePublicKeyHashRequest,
platform_state: &PlatformState,
platform_version: &PlatformVersion,
) -> Result<QueryValidationResult<GetIdentitiesForNonUniquePublicKeyHashResponse>, Error> {
let Some(version) = version else {
return Ok(QueryValidationResult::new_with_error(
QueryError::DecodingError(
"could not decode identities for non unique public key hash query".to_string(),
),
));
};

let feature_version_bounds = &platform_version
.drive_abci
.query
.identity_based_queries
.identities_for_non_unique_public_key_hash;

let feature_version = match &version {
RequestVersion::V0(_) => 0,
};

if !feature_version_bounds.check_version(feature_version) {
return Ok(QueryValidationResult::new_with_error(
QueryError::UnsupportedQueryVersion(
"identities_for_non_unique_public_key_hash".to_string(),
feature_version_bounds.min_version,
feature_version_bounds.max_version,
platform_version.protocol_version,
feature_version,
),
));
}

match version {
RequestVersion::V0(request_v0) => {
let request = self.query_identities_for_non_unique_public_key_hash_v0(
request_v0,
platform_state,
platform_version,
)?;

Ok(request.map(
|response_v0| GetIdentitiesForNonUniquePublicKeyHashResponse {
version: Some(ResponseVersion::V0(response_v0)),
},
))
}
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,152 @@
use crate::error::query::QueryError;
use crate::error::Error;
use crate::platform_types::platform::Platform;
use crate::platform_types::platform_state::PlatformState;
use crate::query::QueryValidationResult;
use dapi_grpc::platform::v0::get_identities_for_non_unique_public_key_hash_request::GetIdentitiesForNonUniquePublicKeyHashRequestV0;
use dapi_grpc::platform::v0::get_identities_for_non_unique_public_key_hash_response::{
get_identities_for_non_unique_public_key_hash_response_v0,
GetIdentitiesForNonUniquePublicKeyHashResponseV0,
};
use dpp::check_validation_result_with_data;
use dpp::platform_value::Bytes20;
use dpp::serialization::PlatformSerializable;
use dpp::validation::ValidationResult;
use dpp::version::PlatformVersion;

impl<C> Platform<C> {
pub(super) fn query_identities_for_non_unique_public_key_hash_v0(
&self,
GetIdentitiesForNonUniquePublicKeyHashRequestV0 {
public_key_hash,
prove,
}: GetIdentitiesForNonUniquePublicKeyHashRequestV0,
platform_state: &PlatformState,
platform_version: &PlatformVersion,
) -> Result<QueryValidationResult<GetIdentitiesForNonUniquePublicKeyHashResponseV0>, Error>
{
let public_key_hash =
check_validation_result_with_data!(Bytes20::from_vec(public_key_hash)
.map(|bytes| bytes.0)
.map_err(|_| QueryError::InvalidArgument(
"public key hash must be 20 bytes long".to_string()
)));

let response = if prove {
let proof = self
.drive
.prove_full_identities_for_non_unique_public_key_hash(
public_key_hash,
None,
None,
platform_version,
)?;

GetIdentitiesForNonUniquePublicKeyHashResponseV0 {
result: Some(
get_identities_for_non_unique_public_key_hash_response_v0::Result::Proof(
self.response_proof_v0(platform_state, proof),
),
),
metadata: Some(self.response_metadata_v0(platform_state)),
}
} else {
let identities = self
.drive
.fetch_full_identities_for_non_unique_public_key_hash(
public_key_hash,
None,
None,
platform_version,
)?
.into_iter()
.map(|identity| {
identity
.serialize_consume_to_bytes()
.map_err(Error::Protocol)
})
.collect::<Result<Vec<_>, Error>>()?;

GetIdentitiesForNonUniquePublicKeyHashResponseV0 {
metadata: Some(self.response_metadata_v0(platform_state)),
result: Some(
get_identities_for_non_unique_public_key_hash_response_v0::Result::Identities(
get_identities_for_non_unique_public_key_hash_response_v0::Identities {
identities,
},
),
),
}
};

Ok(QueryValidationResult::new_with_data(response))
}
}

#[cfg(test)]
mod tests {
use super::*;
use crate::query::tests::setup_platform;
use dpp::dashcore::Network;

#[test]
fn test_invalid_public_key_hash() {
let (platform, state, version) = setup_platform(None, Network::Testnet, None);

let request = GetIdentitiesForNonUniquePublicKeyHashRequestV0 {
public_key_hash: vec![0; 8],
prove: false,
};

let result = platform
.query_identities_for_non_unique_public_key_hash_v0(request, &state, version)
.expect("expected query to succeed");

assert!(matches!(
result.errors.as_slice(),
[QueryError::InvalidArgument(msg)] if msg == &"public key hash must be 20 bytes long".to_string()
));
}

#[test]
fn test_identity_not_found() {
let (platform, state, version) = setup_platform(None, Network::Testnet, None);

let public_key_hash = vec![0; 20];
let request = GetIdentitiesForNonUniquePublicKeyHashRequestV0 {
public_key_hash: public_key_hash.clone(),
prove: false,
};

let result = platform
.query_identities_for_non_unique_public_key_hash_v0(request, &state, version)
.expect("expected query to succeed");

assert!(result.errors.is_empty());
}

#[test]
fn test_identity_absence_proof() {
let (platform, state, version) = setup_platform(None, Network::Testnet, None);

let public_key_hash = vec![0; 20];
let request = GetIdentitiesForNonUniquePublicKeyHashRequestV0 {
public_key_hash: public_key_hash.clone(),
prove: true,
};

let result = platform
.query_identities_for_non_unique_public_key_hash_v0(request, &state, version)
.expect("expected query to succeed");

assert!(matches!(
result.data,
Some(GetIdentitiesForNonUniquePublicKeyHashResponseV0 {
result: Some(
get_identities_for_non_unique_public_key_hash_response_v0::Result::Proof(_)
),
metadata: Some(_),
})
));
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ mod balance;
mod balance_and_revision;
mod balances;
mod identities_contract_keys;
mod identities_for_non_unique_public_key_hash;
mod identity;
mod identity_by_public_key_hash;
mod identity_contract_nonce;
Expand Down
15 changes: 14 additions & 1 deletion packages/rs-drive-abci/src/query/service.rs
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,8 @@ use dapi_grpc::platform::v0::{
GetEvonodesProposedEpochBlocksByIdsRequest, GetEvonodesProposedEpochBlocksByRangeRequest,
GetEvonodesProposedEpochBlocksResponse, GetIdentitiesBalancesRequest,
GetIdentitiesBalancesResponse, GetIdentitiesContractKeysRequest,
GetIdentitiesContractKeysResponse, GetIdentityBalanceAndRevisionRequest,
GetIdentitiesContractKeysResponse, GetIdentitiesForNonUniquePublicKeyHashRequest,
GetIdentitiesForNonUniquePublicKeyHashResponse, GetIdentityBalanceAndRevisionRequest,
GetIdentityBalanceAndRevisionResponse, GetIdentityBalanceRequest, GetIdentityBalanceResponse,
GetIdentityByPublicKeyHashRequest, GetIdentityByPublicKeyHashResponse,
GetIdentityContractNonceRequest, GetIdentityContractNonceResponse, GetIdentityKeysRequest,
Expand Down Expand Up @@ -404,6 +405,18 @@ impl PlatformService for QueryService {
.await
}

async fn get_identities_for_non_unique_public_key_hash(
&self,
request: Request<GetIdentitiesForNonUniquePublicKeyHashRequest>,
) -> Result<Response<GetIdentitiesForNonUniquePublicKeyHashResponse>, Status> {
self.handle_blocking_query(
request,
Platform::<DefaultCoreRPC>::query_identities_for_non_unique_public_key_hash,
"get_identities_for_non_unique_public_key_hash",
)
.await
}

async fn wait_for_state_transition_result(
&self,
_request: Request<WaitForStateTransitionResultRequest>,
Expand Down
Loading