Skip to content

Commit

Permalink
added query to check if delegate is registered
Browse files Browse the repository at this point in the history
  • Loading branch information
NoahSaso committed Nov 27, 2024
1 parent 590eaf0 commit ed4a6f8
Show file tree
Hide file tree
Showing 5 changed files with 163 additions and 8 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -623,6 +623,36 @@
},
"additionalProperties": false
},
{
"description": "Returns registration info for a delegate, optionally at a given height.",
"type": "object",
"required": [
"registration"
],
"properties": {
"registration": {
"type": "object",
"required": [
"delegate"
],
"properties": {
"delegate": {
"type": "string"
},
"height": {
"type": [
"integer",
"null"
],
"format": "uint64",
"minimum": 0.0
}
},
"additionalProperties": false
}
},
"additionalProperties": false
},
{
"description": "Returns the paginated list of active delegates.",
"type": "object",
Expand Down Expand Up @@ -975,6 +1005,43 @@
}
}
},
"registration": {
"$schema": "http://json-schema.org/draft-07/schema#",
"title": "RegistrationResponse",
"type": "object",
"required": [
"height",
"power",
"registered"
],
"properties": {
"height": {
"description": "The height at which registration was checked.",
"type": "integer",
"format": "uint64",
"minimum": 0.0
},
"power": {
"description": "The total voting power delegated to the delegate. If not registered, this may still be nonzero if the delegate was registered in the past.",
"allOf": [
{
"$ref": "#/definitions/Uint128"
}
]
},
"registered": {
"description": "Whether or not the delegate is registered.",
"type": "boolean"
}
},
"additionalProperties": false,
"definitions": {
"Uint128": {
"description": "A thin wrapper around u128 that is using strings for JSON encoding/decoding, such that the full u128 range can be used for clients that convert JSON numbers to floats, like JavaScript and jq.\n\n# Examples\n\nUse `from` to create instances of this and `u128` to get the value out:\n\n``` # use cosmwasm_std::Uint128; let a = Uint128::from(123u128); assert_eq!(a.u128(), 123);\n\nlet b = Uint128::from(42u64); assert_eq!(b.u128(), 42);\n\nlet c = Uint128::from(70u32); assert_eq!(c.u128(), 70); ```",
"type": "string"
}
}
},
"unvoted_delegated_voting_power": {
"$schema": "http://json-schema.org/draft-07/schema#",
"title": "UnvotedDelegatedVotingPowerResponse",
Expand Down
27 changes: 26 additions & 1 deletion contracts/delegation/dao-vote-delegation/src/contract.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,8 @@ use dao_interface::helpers::OptionalUpdate;
use dao_interface::state::{ProposalModule, ProposalModuleStatus};
use dao_interface::voting::InfoResponse;
use dao_voting::delegation::{
calculate_delegated_vp, DelegationResponse, UnvotedDelegatedVotingPowerResponse,
calculate_delegated_vp, DelegationResponse, RegistrationResponse,
UnvotedDelegatedVotingPowerResponse,
};
use dao_voting::voting;
use semver::Version;
Expand Down Expand Up @@ -462,6 +463,9 @@ fn execute_update_config(
pub fn query(deps: Deps, env: Env, msg: QueryMsg) -> StdResult<Binary> {
match msg {
QueryMsg::Info {} => Ok(to_json_binary(&query_info(deps)?)?),

Check warning on line 465 in contracts/delegation/dao-vote-delegation/src/contract.rs

View check run for this annotation

Codecov / codecov/patch

contracts/delegation/dao-vote-delegation/src/contract.rs#L465

Added line #L465 was not covered by tests
QueryMsg::Registration { delegate, height } => Ok(to_json_binary(&query_registration(
deps, env, delegate, height,
)?)?),
QueryMsg::Delegates { start_after, limit } => Ok(to_json_binary(&query_delegates(
deps,
env,
Expand Down Expand Up @@ -502,6 +506,27 @@ fn query_info(deps: Deps) -> StdResult<InfoResponse> {
Ok(InfoResponse { info })
}

Check warning on line 507 in contracts/delegation/dao-vote-delegation/src/contract.rs

View check run for this annotation

Codecov / codecov/patch

contracts/delegation/dao-vote-delegation/src/contract.rs#L504-L507

Added lines #L504 - L507 were not covered by tests

fn query_registration(
deps: Deps,
env: Env,
delegate: String,
height: Option<u64>,
) -> StdResult<RegistrationResponse> {
let height = height.unwrap_or(env.block.height);
let delegate = deps.api.addr_validate(&delegate)?;

let registered = is_delegate_registered(deps, &delegate, Some(height))?;
let power = DELEGATED_VP
.load(deps.storage, delegate, height)?
.unwrap_or_default();

Ok(RegistrationResponse {
registered,
power,
height,
})
}

fn query_delegates(
deps: Deps,
env: Env,
Expand Down
40 changes: 34 additions & 6 deletions contracts/delegation/dao-vote-delegation/src/testing/suite.rs
Original file line number Diff line number Diff line change
Expand Up @@ -248,6 +248,20 @@ impl DaoVoteDelegationTestingSuite {

/// QUERIES
impl DaoVoteDelegationTestingSuite {
/// get whether a delegate is registered
pub fn registered(&self, delegate: impl Into<String>, height: Option<u64>) -> bool {
self.querier()
.query_wasm_smart::<dao_voting::delegation::RegistrationResponse>(
&self.delegation_addr,
&crate::msg::QueryMsg::Registration {
delegate: delegate.into(),
height,
},
)
.unwrap()
.registered
}

/// get the delegates
pub fn delegates(
&self,
Expand Down Expand Up @@ -361,18 +375,32 @@ impl DaoVoteDelegationTestingSuite {
.any(|d| d.delegate == delegate.into() && d.percent == percent && d.active));
}

/// assert that a delegate is registered
pub fn assert_delegate_registered(
&self,
delegate: impl Into<String> + Copy,
height: Option<u64>,
) {
let registered = self.registered(delegate, height);
assert!(registered);
}

/// assert that a delegate is not registered
pub fn assert_delegate_not_registered(
&self,
delegate: impl Into<String> + Copy,
height: Option<u64>,
) {
let registered = self.registered(delegate, height);
assert!(!registered);
}

/// assert that there are N delegates
pub fn assert_delegates_count(&self, count: u32) {
let delegates = self.delegates(None, None);
assert_eq!(delegates.len() as u32, count);
}

/// assert a delegate is registered
pub fn assert_registered(&self, delegate: impl Into<String> + Copy) {
let delegates = self.delegates(None, None);
assert!(delegates.iter().any(|d| d.delegate == delegate.into()));
}

/// assert a delegate's total delegated voting power
pub fn assert_delegate_total_delegated_vp(
&self,
Expand Down
20 changes: 19 additions & 1 deletion contracts/delegation/dao-vote-delegation/src/testing/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -41,17 +41,24 @@ fn test_simple() {
.collect::<Vec<_>>()
);

suite.assert_delegate_not_registered(ADDR0, None);

// register ADDR0 as a delegate
suite.register(ADDR0);
suite.assert_delegates_count(1);
suite.assert_registered(ADDR0);

// delegate 100% of addr1's voting power to ADDR0
suite.delegate(ADDR1, ADDR0, Decimal::percent(100));

// delegations take effect on the next block
suite.advance_block();

// ensure registered
suite.assert_delegate_registered(ADDR0, None);
suite.assert_delegate_registered(ADDR0, Some(suite.app.block_info().height));
// historical check works
suite.assert_delegate_not_registered(ADDR0, Some(suite.app.block_info().height - 1));

suite.assert_delegations_count(ADDR1, 1);
suite.assert_delegation(ADDR1, ADDR0, Decimal::percent(100));
suite.assert_delegate_total_delegated_vp(ADDR0, suite.members[1].weight);
Expand Down Expand Up @@ -185,12 +192,15 @@ fn test_simple() {

suite.assert_delegate_total_delegated_vp(ADDR0, total_vp_except_addr0);

suite.assert_delegate_registered(ADDR0, None);

// unregister ADDR0 as a delegate
suite.unregister(ADDR0);

// delegations take effect on the next block
suite.advance_block();

suite.assert_delegate_not_registered(ADDR0, None);
suite.assert_delegates_count(0);

// propose another proposal
Expand Down Expand Up @@ -621,6 +631,10 @@ fn test_allow_register_after_unregister_same_block() {
suite.register(ADDR0);
suite.unregister(ADDR0);
suite.register(ADDR0);

// ensure registered
suite.advance_block();
suite.assert_delegate_registered(ADDR0, None);
}

#[test]
Expand All @@ -632,6 +646,10 @@ fn test_allow_register_after_unregister_next_block() {
suite.unregister(ADDR0);
suite.advance_block();
suite.register(ADDR0);

// ensure registered
suite.advance_block();
suite.assert_delegate_registered(ADDR0, None);
}

#[test]
Expand Down
17 changes: 17 additions & 0 deletions packages/dao-voting/src/delegation.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,12 @@ pub enum QueryMsg {
/// Returns contract version info
#[returns(InfoResponse)]
Info {},
/// Returns registration info for a delegate, optionally at a given height.
#[returns(RegistrationResponse)]
Registration {
delegate: String,
height: Option<u64>,
},
/// Returns the paginated list of active delegates.
#[returns(DelegatesResponse)]
Delegates {
Expand Down Expand Up @@ -50,6 +56,17 @@ pub enum QueryMsg {
},
}

#[cw_serde]
pub struct RegistrationResponse {
/// Whether or not the delegate is registered.
pub registered: bool,
/// The total voting power delegated to the delegate. If not registered,
/// this may still be nonzero if the delegate was registered in the past.
pub power: Uint128,
/// The height at which registration was checked.
pub height: u64,
}

#[cw_serde]
pub struct DelegatesResponse {
/// The delegates.
Expand Down

0 comments on commit ed4a6f8

Please sign in to comment.