Skip to content

Commit 5b63cf6

Browse files
authored
Merge pull request #879 from CosmWasm/query_validator
Add single validator query
2 parents afe6466 + d653b37 commit 5b63cf6

File tree

8 files changed

+138
-25
lines changed

8 files changed

+138
-25
lines changed

CHANGELOG.md

+7
Original file line numberDiff line numberDiff line change
@@ -173,6 +173,12 @@ and this project adheres to
173173
- cosmwasm-std: Remove `StakingMsg::Withdraw` in favour of
174174
`DistributionMsg::SetWithdrawAddress` and
175175
`DistributionMsg::WithdrawDelegatorReward` ([#848]).
176+
- cosmwasm-std: Rename `StakingQuery::Validators`, `ValidatorsResponse` and
177+
`QuerierWrapper::query_validators` to `StakingQuery::AllValidators`,
178+
`AllValidatorsResponse` and `QuerierWrapper.query_all_validators`. Add
179+
`StakingQuery::Validator`, `ValidatorResponse` and
180+
`QuerierWrapper::query_validator` to allow querying a single validator.
181+
([#879])
176182

177183
[#696]: https://github.com/CosmWasm/cosmwasm/issues/696
178184
[#697]: https://github.com/CosmWasm/cosmwasm/issues/697
@@ -186,6 +192,7 @@ and this project adheres to
186192
[#871]: https://github.com/CosmWasm/cosmwasm/issues/871
187193
[#861]: https://github.com/CosmWasm/cosmwasm/issues/861
188194
[#848]: https://github.com/CosmWasm/cosmwasm/issues/848
195+
[#879]: https://github.com/CosmWasm/cosmwasm/pull/879
189196

190197
### Deprecated
191198

contracts/reflect/schema/query_msg.json

+25-3
Original file line numberDiff line numberDiff line change
@@ -420,17 +420,39 @@
420420
"additionalProperties": false
421421
},
422422
{
423-
"description": "Returns all registered Validators on the system",
423+
"description": "Returns all validators in the currently active validator set.\n\nThe query response type is `AllValidatorsResponse`.",
424424
"type": "object",
425425
"required": [
426-
"validators"
426+
"all_validators"
427427
],
428428
"properties": {
429-
"validators": {
429+
"all_validators": {
430430
"type": "object"
431431
}
432432
},
433433
"additionalProperties": false
434+
},
435+
{
436+
"description": "Returns the validator at the given address. Returns None if the validator is not part of the currently active validator set.\n\nThe query response type is `ValidatorResponse`.",
437+
"type": "object",
438+
"required": [
439+
"validator"
440+
],
441+
"properties": {
442+
"validator": {
443+
"type": "object",
444+
"required": [
445+
"address"
446+
],
447+
"properties": {
448+
"address": {
449+
"description": "The validator's address (e.g. (e.g. cosmosvaloper1...))",
450+
"type": "string"
451+
}
452+
}
453+
}
454+
},
455+
"additionalProperties": false
434456
}
435457
]
436458
},

contracts/staking/src/contract.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -24,8 +24,8 @@ pub fn instantiate(
2424
msg: InstantiateMsg,
2525
) -> StdResult<Response> {
2626
// ensure the validator is registered
27-
let vals = deps.querier.query_validators()?;
28-
if !vals.iter().any(|v| v.address == msg.validator) {
27+
let validator = deps.querier.query_validator(msg.validator.clone())?;
28+
if validator.is_none() {
2929
return Err(StdError::generic_err(format!(
3030
"{} is not in the current validator set",
3131
msg.validator

packages/std/src/lib.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -46,8 +46,8 @@ pub use crate::query::{
4646
};
4747
#[cfg(feature = "staking")]
4848
pub use crate::query::{
49-
AllDelegationsResponse, BondedDenomResponse, Delegation, FullDelegation, StakingQuery,
50-
Validator, ValidatorsResponse,
49+
AllDelegationsResponse, AllValidatorsResponse, BondedDenomResponse, Delegation, FullDelegation,
50+
StakingQuery, Validator, ValidatorResponse,
5151
};
5252
pub use crate::results::{
5353
attr, wasm_execute, wasm_instantiate, Attribute, BankMsg, ContractResult, CosmosMsg, Empty,

packages/std/src/mock.rs

+65-7
Original file line numberDiff line numberDiff line change
@@ -15,8 +15,8 @@ use crate::query::{
1515
};
1616
#[cfg(feature = "staking")]
1717
use crate::query::{
18-
AllDelegationsResponse, BondedDenomResponse, DelegationResponse, FullDelegation, StakingQuery,
19-
Validator, ValidatorsResponse,
18+
AllDelegationsResponse, AllValidatorsResponse, BondedDenomResponse, DelegationResponse,
19+
FullDelegation, StakingQuery, Validator, ValidatorResponse,
2020
};
2121
use crate::results::{ContractResult, Empty, SystemResult};
2222
use crate::serde::{from_slice, to_binary};
@@ -459,12 +459,21 @@ impl StakingQuerier {
459459
};
460460
to_binary(&res).into()
461461
}
462-
StakingQuery::Validators {} => {
463-
let res = ValidatorsResponse {
462+
StakingQuery::AllValidators {} => {
463+
let res = AllValidatorsResponse {
464464
validators: self.validators.clone(),
465465
};
466466
to_binary(&res).into()
467467
}
468+
StakingQuery::Validator { address } => {
469+
let validator: Option<Validator> = self
470+
.validators
471+
.iter()
472+
.find(|validator| validator.address == *address)
473+
.cloned();
474+
let res = ValidatorResponse { validator };
475+
to_binary(&res).into()
476+
}
468477
StakingQuery::AllDelegations { delegator } => {
469478
let delegations: Vec<_> = self
470479
.delegations
@@ -874,7 +883,7 @@ mod tests {
874883

875884
#[cfg(feature = "staking")]
876885
#[test]
877-
fn staking_querier_validators() {
886+
fn staking_querier_all_validators() {
878887
let val1 = Validator {
879888
address: String::from("validator-one"),
880889
commission: Decimal::percent(1),
@@ -892,13 +901,62 @@ mod tests {
892901

893902
// one match
894903
let raw = staking
895-
.query(&StakingQuery::Validators {})
904+
.query(&StakingQuery::AllValidators {})
896905
.unwrap()
897906
.unwrap();
898-
let vals: ValidatorsResponse = from_binary(&raw).unwrap();
907+
let vals: AllValidatorsResponse = from_binary(&raw).unwrap();
899908
assert_eq!(vals.validators, vec![val1, val2]);
900909
}
901910

911+
#[cfg(feature = "staking")]
912+
#[test]
913+
fn staking_querier_validator() {
914+
let address1 = String::from("validator-one");
915+
let address2 = String::from("validator-two");
916+
let address_non_existent = String::from("wannabe-validator");
917+
918+
let val1 = Validator {
919+
address: address1.clone(),
920+
commission: Decimal::percent(1),
921+
max_commission: Decimal::percent(3),
922+
max_change_rate: Decimal::percent(1),
923+
};
924+
let val2 = Validator {
925+
address: address2.clone(),
926+
commission: Decimal::permille(15),
927+
max_commission: Decimal::permille(40),
928+
max_change_rate: Decimal::permille(5),
929+
};
930+
931+
let staking = StakingQuerier::new("ustake", &[val1.clone(), val2.clone()], &[]);
932+
933+
// query 1
934+
let raw = staking
935+
.query(&StakingQuery::Validator { address: address1 })
936+
.unwrap()
937+
.unwrap();
938+
let res: ValidatorResponse = from_binary(&raw).unwrap();
939+
assert_eq!(res.validator, Some(val1));
940+
941+
// query 2
942+
let raw = staking
943+
.query(&StakingQuery::Validator { address: address2 })
944+
.unwrap()
945+
.unwrap();
946+
let res: ValidatorResponse = from_binary(&raw).unwrap();
947+
assert_eq!(res.validator, Some(val2));
948+
949+
// query non-existent
950+
let raw = staking
951+
.query(&StakingQuery::Validator {
952+
address: address_non_existent,
953+
})
954+
.unwrap()
955+
.unwrap();
956+
let res: ValidatorResponse = from_binary(&raw).unwrap();
957+
assert_eq!(res.validator, None);
958+
}
959+
902960
#[cfg(feature = "staking")]
903961
// gets delegators from query or panic
904962
fn get_all_delegators<U: Into<String>>(

packages/std/src/query/mod.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -15,8 +15,8 @@ mod wasm;
1515
pub use bank::{AllBalanceResponse, BalanceResponse, BankQuery};
1616
#[cfg(feature = "staking")]
1717
pub use staking::{
18-
AllDelegationsResponse, BondedDenomResponse, Delegation, DelegationResponse, FullDelegation,
19-
StakingQuery, Validator, ValidatorsResponse,
18+
AllDelegationsResponse, AllValidatorsResponse, BondedDenomResponse, Delegation,
19+
DelegationResponse, FullDelegation, StakingQuery, Validator, ValidatorResponse,
2020
};
2121
#[cfg(feature = "stargate")]
2222
pub use stargate::StargateResponse;

packages/std/src/query/staking.rs

+20-4
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,18 @@ pub enum StakingQuery {
1919
delegator: String,
2020
validator: String,
2121
},
22-
/// Returns all registered Validators on the system
23-
Validators {},
22+
/// Returns all validators in the currently active validator set.
23+
///
24+
/// The query response type is `AllValidatorsResponse`.
25+
AllValidators {},
26+
/// Returns the validator at the given address. Returns None if the validator is
27+
/// not part of the currently active validator set.
28+
///
29+
/// The query response type is `ValidatorResponse`.
30+
Validator {
31+
/// The validator's address (e.g. (e.g. cosmosvaloper1...))
32+
address: String,
33+
},
2434
}
2535

2636
/// BondedDenomResponse is data format returned from StakingRequest::BondedDenom query
@@ -85,12 +95,18 @@ pub struct FullDelegation {
8595
pub accumulated_rewards: Vec<Coin>,
8696
}
8797

88-
/// ValidatorsResponse is data format returned from StakingRequest::Validators query
98+
/// The data format returned from StakingRequest::AllValidators query
8999
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
90-
pub struct ValidatorsResponse {
100+
pub struct AllValidatorsResponse {
91101
pub validators: Vec<Validator>,
92102
}
93103

104+
/// The data format returned from StakingRequest::Validator query
105+
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
106+
pub struct ValidatorResponse {
107+
pub validator: Option<Validator>,
108+
}
109+
94110
/// Instances are created in the querier.
95111
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
96112
pub struct Validator {

packages/std/src/traits.rs

+15-5
Original file line numberDiff line numberDiff line change
@@ -12,8 +12,8 @@ use crate::query::{
1212
};
1313
#[cfg(feature = "staking")]
1414
use crate::query::{
15-
AllDelegationsResponse, BondedDenomResponse, Delegation, DelegationResponse, FullDelegation,
16-
StakingQuery, Validator, ValidatorsResponse,
15+
AllDelegationsResponse, AllValidatorsResponse, BondedDenomResponse, Delegation,
16+
DelegationResponse, FullDelegation, StakingQuery, Validator, ValidatorResponse,
1717
};
1818
use crate::results::{ContractResult, Empty, SystemResult};
1919
use crate::serde::{from_binary, to_binary, to_vec};
@@ -261,12 +261,22 @@ impl<'a> QuerierWrapper<'a> {
261261
}
262262

263263
#[cfg(feature = "staking")]
264-
pub fn query_validators(&self) -> StdResult<Vec<Validator>> {
265-
let request = StakingQuery::Validators {}.into();
266-
let res: ValidatorsResponse = self.query(&request)?;
264+
pub fn query_all_validators(&self) -> StdResult<Vec<Validator>> {
265+
let request = StakingQuery::AllValidators {}.into();
266+
let res: AllValidatorsResponse = self.query(&request)?;
267267
Ok(res.validators)
268268
}
269269

270+
#[cfg(feature = "staking")]
271+
pub fn query_validator<U: Into<String>>(&self, address: U) -> StdResult<Option<Validator>> {
272+
let request = StakingQuery::Validator {
273+
address: address.into(),
274+
}
275+
.into();
276+
let res: ValidatorResponse = self.query(&request)?;
277+
Ok(res.validator)
278+
}
279+
270280
#[cfg(feature = "staking")]
271281
pub fn query_bonded_denom(&self) -> StdResult<String> {
272282
let request = StakingQuery::BondedDenom {}.into();

0 commit comments

Comments
 (0)