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

Use indexed map for proposal modules #748

Open
wants to merge 4 commits into
base: development
Choose a base branch
from
Open
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
209 changes: 145 additions & 64 deletions Cargo.lock

Large diffs are not rendered by default.

4 changes: 4 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -129,6 +129,10 @@ cw4-voting-v1 = { package = "cw4-voting", version = "0.1.0" }
voting-v1 = { package = "dao-voting", version = "0.1.0" }
stake-cw20-v03 = { package = "stake-cw20", version = "0.2.6" }

# v2 dependencies. used for state migrations
cw-core-v2 = { git = "https://github.com/DA0-DA0/dao-contracts.git", rev = "7f89ad1604e8022f202aef729853b0c8c7196988", package = "dao-dao-core" }
dao-interface-v2 = { git = "https://github.com/DA0-DA0/dao-contracts.git", rev = "7f89ad1604e8022f202aef729853b0c8c7196988", package = "dao-interface" }

# TODO remove when upstream PR merged and new release tagged: https://github.com/CosmWasm/cw-multi-test/pull/51
[patch.crates-io]
cw-multi-test = { git = "https://github.com/JakeHartnell/cw-multi-test.git", branch = "bank-supply-support" }
2 changes: 2 additions & 0 deletions contracts/dao-dao-core/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,8 @@ dao-interface = { workspace = true }
dao-dao-macros = { workspace = true }
cw-paginate-storage = { workspace = true }
cw-core-v1 = { workspace = true, features = ["library"] }
cw-core-v2 = { workspace = true, features = ["library"] }
dao-interface-v2 = { workspace = true }

[dev-dependencies]
cw-multi-test = { workspace = true, features = ["stargate"] }
Expand Down
168 changes: 98 additions & 70 deletions contracts/dao-dao-core/src/contract.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,9 @@ use cosmwasm_std::{
Reply, Response, StdError, StdResult, SubMsg, WasmMsg,
};
use cw2::{get_contract_version, set_contract_version, ContractVersion};
use cw_paginate_storage::{paginate_map, paginate_map_keys, paginate_map_values};
use cw_storage_plus::Map;
use cw_utils::{parse_reply_instantiate_data, Duration};
use cw_paginate_storage::{paginate_map, paginate_map_keys};
use cw_storage_plus::{Bound, Map};
use cw_utils::{maybe_addr, parse_reply_instantiate_data, Duration};
use dao_interface::{
msg::{ExecuteMsg, InitialItem, InstantiateMsg, MigrateMsg, QueryMsg},
query::{
Expand All @@ -23,8 +23,8 @@ use dao_interface::{

use crate::error::ContractError;
use crate::state::{
ACTIVE_PROPOSAL_MODULE_COUNT, ADMIN, CONFIG, CW20_LIST, CW721_LIST, ITEMS, NOMINATED_ADMIN,
PAUSED, PROPOSAL_MODULES, SUBDAO_LIST, TOTAL_PROPOSAL_MODULE_COUNT, VOTING_MODULE,
proposal_modules, ACTIVE_PROPOSAL_MODULE_COUNT, ADMIN, CONFIG, CW20_LIST, CW721_LIST, ITEMS,
NOMINATED_ADMIN, PAUSED, SUBDAO_LIST, TOTAL_PROPOSAL_MODULE_COUNT, VOTING_MODULE,
};

pub(crate) const CONTRACT_NAME: &str = "crates.io:dao-dao-core";
Expand Down Expand Up @@ -196,7 +196,7 @@ pub fn execute_proposal_hook(
sender: Addr,
msgs: Vec<CosmosMsg<Empty>>,
) -> Result<Response, ContractError> {
let module = PROPOSAL_MODULES
let module = proposal_modules()
.may_load(deps.storage, sender.clone())?
.ok_or(ContractError::Unauthorized {})?;

Expand Down Expand Up @@ -342,7 +342,7 @@ pub fn execute_update_proposal_modules(
let disable_count = to_disable.len() as u32;
for addr in to_disable {
let addr = deps.api.addr_validate(&addr)?;
let mut module = PROPOSAL_MODULES
let mut module = proposal_modules()
.load(deps.storage, addr.clone())
.map_err(|_| ContractError::ProposalModuleDoesNotExist {
address: addr.clone(),
Expand All @@ -355,7 +355,7 @@ pub fn execute_update_proposal_modules(
}

module.status = ProposalModuleStatus::Disabled {};
PROPOSAL_MODULES.save(deps.storage, addr, &module)?;
proposal_modules().save(deps.storage, addr, &module)?;
}

// If disabling this module will cause there to be no active modules, return error.
Expand Down Expand Up @@ -558,18 +558,22 @@ pub fn query(deps: Deps, env: Env, msg: QueryMsg) -> StdResult<Binary> {
QueryMsg::Info {} => query_info(deps),
QueryMsg::ListItems { start_after, limit } => query_list_items(deps, start_after, limit),
QueryMsg::PauseInfo {} => query_paused(deps, env),
QueryMsg::ProposalModules { start_after, limit } => {
query_proposal_modules(deps, start_after, limit)
}
QueryMsg::ProposalModules {
start_after,
limit,
include_disabled,
} => to_binary(&query_proposal_modules(
deps,
start_after,
limit,
include_disabled,
)?),
QueryMsg::ProposalModuleCount {} => query_proposal_module_count(deps),
QueryMsg::TotalPowerAtHeight { height } => query_total_power_at_height(deps, height),
QueryMsg::VotingModule {} => query_voting_module(deps),
QueryMsg::VotingPowerAtHeight { address, height } => {
query_voting_power_at_height(deps, address, height)
}
QueryMsg::ActiveProposalModules { start_after, limit } => {
query_active_proposal_modules(deps, start_after, limit)
}
QueryMsg::ListSubDaos { start_after, limit } => {
query_list_sub_daos(deps, start_after, limit)
}
Expand Down Expand Up @@ -601,57 +605,43 @@ pub fn query_proposal_modules(
deps: Deps,
start_after: Option<String>,
limit: Option<u32>,
) -> StdResult<Binary> {
// This query is will run out of gas due to the size of the
// returned message before it runs out of compute so taking a
// limit here is still nice. As removes happen in constant time
// the contract is still recoverable if too many items end up in
// here.
//
// Further, as the `range` method on a map returns an iterator it
// is possible (though implementation dependent) that new keys are
// loaded on demand as the iterator is moved. Should this be the
// case we are only paying for what we need here.
//
// Even if this does lock up one can determine the existing
// proposal modules by looking at past transactions on chain.
to_binary(&paginate_map_values(
deps,
&PROPOSAL_MODULES,
start_after
.map(|s| deps.api.addr_validate(&s))
.transpose()?,
limit,
cosmwasm_std::Order::Ascending,
)?)
}

pub fn query_active_proposal_modules(
deps: Deps,
start_after: Option<String>,
limit: Option<u32>,
) -> StdResult<Binary> {
// Note: this is not gas efficient as we need to potentially visit all modules in order to
// filter out the modules with active status.
let values = paginate_map_values(
deps,
&PROPOSAL_MODULES,
start_after
.map(|s| deps.api.addr_validate(&s))
.transpose()?,
None,
cosmwasm_std::Order::Ascending,
)?;

let limit = limit.unwrap_or(values.len() as u32);
include_disabled: Option<bool>,
) -> StdResult<Vec<ProposalModule>> {
let start_after_bound = maybe_addr(deps.api, start_after)?.map(Bound::exclusive);
let include_disabled = include_disabled.unwrap_or(false);

let items_iter: Box<dyn Iterator<Item = StdResult<ProposalModule>>> = if include_disabled {
Box::new(
proposal_modules()
.range(
deps.storage,
start_after_bound,
None,
cosmwasm_std::Order::Ascending,
)
.map(|x| x.map(|y| y.1)),
)
} else {
Box::new(
proposal_modules()
.idx
.status
.prefix(ProposalModuleStatus::Enabled.to_string())
.range(
deps.storage,
start_after_bound,
None,
cosmwasm_std::Order::Ascending,
)
.map(|x| x.map(|y| y.1)),
)
};

to_binary::<Vec<ProposalModule>>(
&values
.into_iter()
.filter(|module: &ProposalModule| module.status == ProposalModuleStatus::Enabled)
.take(limit as usize)
.collect(),
)
if let Some(l) = limit {
items_iter.take(l as usize).collect()
} else {
items_iter.collect()
}
}

fn get_pause_info(deps: Deps, env: Env) -> StdResult<PauseInfoResponse> {
Expand All @@ -675,10 +665,7 @@ pub fn query_dump_state(deps: Deps, env: Env) -> StdResult<Binary> {
let admin = ADMIN.load(deps.storage)?;
let config = CONFIG.load(deps.storage)?;
let voting_module = VOTING_MODULE.load(deps.storage)?;
let proposal_modules = PROPOSAL_MODULES
.range(deps.storage, None, None, cosmwasm_std::Order::Ascending)
.map(|kv| Ok(kv?.1))
.collect::<StdResult<Vec<ProposalModule>>>()?;
let proposal_modules = query_proposal_modules(deps, None, None, None)?;
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Only active proposal modules are listed here now

let pause_info = get_pause_info(deps, env)?;
let version = get_contract_version(deps.storage)?;
let active_proposal_module_count = ACTIVE_PROPOSAL_MODULE_COUNT.load(deps.storage)?;
Expand Down Expand Up @@ -883,7 +870,7 @@ pub fn migrate(deps: DepsMut, env: Env, msg: MigrateMsg) -> Result<Response, Con
status: ProposalModuleStatus::Enabled {},
prefix,
};
PROPOSAL_MODULES.save(deps.storage, address, proposal_module)?;
proposal_modules().save(deps.storage, address, proposal_module)?;
Ok(())
})?;

Expand Down Expand Up @@ -923,6 +910,47 @@ pub fn migrate(deps: DepsMut, env: Env, msg: MigrateMsg) -> Result<Response, Con

Ok(response)
}
MigrateMsg::FromV2 {} => {
// `CONTRACT_VERSION` here is from the data section of the
// blob we are migrating to. `version` is from storage. If
// the version in storage matches the version in the blob
// we are not upgrading.
if version == CONTRACT_VERSION {
return Err(ContractError::AlreadyMigrated {});
}

use cw_core_v2 as v2;
use dao_interface_v2 as v2_interface;

let current_keys = v2::state::PROPOSAL_MODULES
.keys(deps.storage, None, None, Order::Ascending)
.collect::<StdResult<Vec<Addr>>>()?;

// Update proposal modules to v3.
current_keys
.into_iter()
.enumerate()
.try_for_each::<_, StdResult<()>>(|(_idx, address)| {
let proposal_module =
v2::state::PROPOSAL_MODULES.load(deps.storage, address.clone())?;
let proposal_module = &ProposalModule {
address: address.clone(),
status: match proposal_module.status {
v2_interface::state::ProposalModuleStatus::Disabled => {
ProposalModuleStatus::Disabled
}
v2_interface::state::ProposalModuleStatus::Enabled => {
ProposalModuleStatus::Enabled
}
},
prefix: proposal_module.prefix.clone(),
};
proposal_modules().save(deps.storage, address, proposal_module)?;
Ok(())
})?;

Ok(Response::default())
}
MigrateMsg::FromCompatible {} => Ok(Response::default()),
}
}
Expand All @@ -942,7 +970,7 @@ pub fn reply(deps: DepsMut, _env: Env, msg: Reply) -> Result<Response, ContractE
prefix,
};

PROPOSAL_MODULES.save(deps.storage, prop_module_addr, &prop_module)?;
proposal_modules().save(deps.storage, prop_module_addr, &prop_module)?;

// Save active and total proposal module counts.
ACTIVE_PROPOSAL_MODULE_COUNT
Expand Down
26 changes: 23 additions & 3 deletions contracts/dao-dao-core/src/state.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
use cosmwasm_std::{Addr, Empty};
use cw_storage_plus::{Item, Map};
use cw_storage_plus::{Index, IndexList, IndexedMap, Item, Map, MultiIndex};
use cw_utils::Expiration;
use dao_interface::state::{Config, ProposalModule};

Expand Down Expand Up @@ -30,10 +30,30 @@ pub const PAUSED: Item<Expiration> = Item::new("paused");
/// The voting module associated with this contract.
pub const VOTING_MODULE: Item<Addr> = Item::new("voting_module");

pub struct ProposalModulesIndexes<'a> {
pub status: MultiIndex<'a, String, ProposalModule, Addr>,
}

impl<'a> IndexList<ProposalModule> for ProposalModulesIndexes<'a> {
fn get_indexes(&'_ self) -> Box<dyn Iterator<Item = &'_ dyn Index<ProposalModule>> + '_> {
let v: Vec<&dyn Index<ProposalModule>> = vec![&self.status];
Box::new(v.into_iter())
}
}

/// The proposal modules associated with this contract.
/// When we change the data format of this map, we update the key (previously "proposal_modules")
/// When we change the data format of this map, we update the key (previously "proposal_modules" or "proposal_modules_v2")
/// to create a new namespace for the changed state.
pub const PROPOSAL_MODULES: Map<Addr, ProposalModule> = Map::new("proposal_modules_v2");
pub fn proposal_modules<'a>() -> IndexedMap<'a, Addr, ProposalModule, ProposalModulesIndexes<'a>> {
let indexes = ProposalModulesIndexes {
status: MultiIndex::new(
|_x, d: &ProposalModule| d.status.to_string(),
"proposal_modules_v3",
"proposal_modules_v3__status",
),
};
IndexedMap::new("proposal_modules_v3", indexes)
}

/// The count of active proposal modules associated with this contract.
pub const ACTIVE_PROPOSAL_MODULE_COUNT: Item<u32> = Item::new("active_proposal_module_count");
Expand Down
Loading