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

new cosmwasm doc #1968

Open
wants to merge 11 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 3 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
40 changes: 16 additions & 24 deletions docs/CAPABILITIES.md
Original file line number Diff line number Diff line change
@@ -1,27 +1,19 @@
# Capabilities

Capabilities are a mechanism to negotiate functionality between a contract and
an environment (i.e. the chain that embeds cosmwasm-vm/[wasmvm]) in a very
primitive way. The contract defines required capabilities. The environment
defines it's capabilities. If the required capabilities are all available, the
contract can be used. Doing this check when the contract is first stored ensures
missing capabilities are detected early and not when a user tries to execute a
certain code path.

## Origin and Disambiguation

Before August 2022, we had two types of "features": app level features in the
CosmWasm VM and Cargo's build system features. In order to avoid the confusion,
the former have been renamed to capabilities.

Capabilities can be implemented in any language that compiles to Wasm whereas
features are Rust build system specific.

## Required capabilities

The contract defines required capabilities using marker export functions that
take no arguments and return no value. The name of the export needs to start
with "requires\_" followed by the name of the capability.
# Overview
**Capabilities** in CosmosWasm are a fundamental concept, designed to facilitate the interaction between a smart contract and its operational environment (i.e., the blockchain implementing the cosmwasm-vm or [wasmvm]). This mechanism operates by matching the capabilities required by a contract with those offered by the environment.

# The Essence of Capabilities
**Purpose:** They serve as a negotiation tool, ensuring that a contract only operates in an environment where all its necessary functionalities are present.
**Early Detection:** By verifying capabilities when a contract is first stored, any discrepancies or missing capabilities are identified early. This preemptive approach prevents potential runtime failures during execution.

# Historical Context: Origin and Disambiguation
**Prior to August 2022:** The term "features" was used ambiguously to describe both app-level features in the CosmWasm VM and features in Cargo's build system.
+**Redefinition:** To mitigate this confusion, app-level features in the CosmWasm VM were renamed to "capabilities," distinguishing them from Cargo's build system features.
+**Language Independence:** Unlike Rust-specific features, capabilities can be implemented in any language that compiles to Wasm, broadening their applicability.
costa2400 marked this conversation as resolved.
Show resolved Hide resolved

# Defining Required Capabilities
## Implementation
**Marker Export Functions:** Contracts specify required capabilities using these functions, which have no arguments and return no value.
**Naming Convention:** The export name begins with "requires\_" followed by the capability name, establishing a clear and standardized method for declaration.

An example of such markers in cosmwasm-std are those:

Expand Down
49 changes: 24 additions & 25 deletions docs/STORAGE_KEYS.md
Original file line number Diff line number Diff line change
@@ -1,17 +1,21 @@
# Storage keys

CosmWasm provides a generic key value store to contract developers via the
`Storage` trait. This is powerful but the nature of low level byte operations
makes it hard to use for high level storage types. In this document we discuss
the foundations of storage key composition all the way up to cw-storage-plus.

In a simple world, all you need is a `&[u8]` key which you can get e.g. using
`&17u64.to_be_bytes()`. This is an 8 bytes key with an encoded integer. But if
you have multiple data types in your contract, you want to prefix those keys in
order to avoid collisions. A simple concatenation is not sufficient because you
want to avoid collisions when part of the prefixes and part of the key overlap.
E.g. `b"keya" | b"x"` and `b"key" | b"ax"` (`|` denotes concatenation) must not
have the same binary representation.
CosmWasm introduces a versatile key-value store accessible to contract developers
through the `Storage` trait. This low-level byte operation-based system, while powerful,
can be challenging for managing high-level storage types. This documentation explores
the evolution of storage key composition in CosmWasm, leading up to the current
implementation in cw-storage-plus.

# The Challenge of Key Composition

The fundamental requirement for storage keys in CosmWasm is a `&[u8]` key, which
can be derived from basic types like integers (e.g., `&17u64.to_be_bytes()`).
costa2400 marked this conversation as resolved.
Show resolved Hide resolved
However, when handling various data types within a contract, it's crucial to
use prefixed keys to prevent data collisions.
Simple concatenation of keys is insufficient due to potential overlap issues.
For instance, `b"keya" | b"x"` and `b"key" | b"ax"` should not yield the same binary representation, where | denotes concatenation.

# Evolution of Key Namespacing

In the early days, multiple approaches of key namespacing were discussed and
were documented here: https://github.com/webmaster128/key-namespacing. The "0x00
Expand Down Expand Up @@ -39,12 +43,11 @@ pub fn to_length_prefixed_nested(namespaces: &[&[u8]]) -> Vec<u8>
fn concat(namespace: &[u8], key: &[u8]) -> Vec<u8>
```

With the emerging cw-storage-plus we see two additions to that approach:
# Transition to `cw-storage-plus`
With the introduction of cw-storage-plus, there were significant enhancements:

1. Manually creating the namespace and concatenating it with `concat` makes no
sense anymore. Instead `namespace` and `key` are always provided and a
composed database key is created.
2. Using a multi component namespace becomes the norm.
1. **Simplified Key Composition:** The manual creation of namespaces followed by concatenation using concat was replaced by a more integrated approach, where namespace and key are provided together to create a composed database key.
2. **Multi-component Namespaces:** Using multiple components in a namespace became commonplace.

This led to the following addition in cw-storage-plus:

Expand All @@ -68,14 +71,10 @@ With the deprecation if cosmwasm-storage and the adoption of the system in
cw-storage-plus, it is time to do a few changes to the Length-prefixed keys
standard, without breaking existing users.

1. Remove the single component `to_length_prefixed` implementation and fully
commit to the multi-component version. This shifts focus from the recursive
implementation to the compatible iterative implementation.
2. Rename "namespaces" to just "namespace" and let one namespace have multiple
components.
3. Adopt the combined namespace + key encoder `namespaces_with_key` from
cw-storage-plus.
4. Add a decomposition implementation
1. **Removal of Single Component Implementation:** The `to_length_prefixed` single-component version will be deprecated in favor of the multi-component `to_length_prefixed_nested` approach.
2. **Terminology Adjustment:** The term "namespaces" will be simplified to "namespace", encompassing multiple components.
3. **Adoption of Combined Encoder:** The `namespaces_with_key` function from `cw-storage-plus` will be the standard for key encoding.
4. **Decomposition Feature:** Introduction of a feature to decompose keys for enhanced flexibility.

Given the importance of Length-prefixed keys for the entire CosmWasm ecosystem,
those implementations should be maintained in cosmwasm-std. The generic approach
Expand Down
4 changes: 2 additions & 2 deletions packages/std/src/deps.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,13 +15,13 @@ pub struct OwnedDeps<S: Storage, A: Api, Q: Querier, C: CustomQuery = Empty> {
pub querier: Q,
pub custom_query_type: PhantomData<C>,
}

/// A mutable version of `Deps` that allows changes to the dependencies.
Copy link
Collaborator

Choose a reason for hiding this comment

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

Nice to have a doc comment here. I think we can be a bit more specific:

Suggested change
/// A mutable version of `Deps` that allows changes to the dependencies.
/// A mutable version of `Deps` that allows writing to storage.

pub struct DepsMut<'a, C: CustomQuery = Empty> {
pub storage: &'a mut dyn Storage,
pub api: &'a dyn Api,
pub querier: QuerierWrapper<'a, C>,
}

/// Represents dependencies required by a contract for execution, like storage and API access.
#[derive(Clone)]
pub struct Deps<'a, C: CustomQuery = Empty> {
pub storage: &'a dyn Storage,
Expand Down
4 changes: 2 additions & 2 deletions packages/std/src/errors/std_error.rs
Original file line number Diff line number Diff line change
Expand Up @@ -415,7 +415,7 @@ impl fmt::Display for OverflowOperation {
write!(f, "{self:?}")
}
}

///An error struct used when a numeric operation results in an overflow.
Copy link
Collaborator

Choose a reason for hiding this comment

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

nit: In general, we use a leading space for doc comments

#[derive(Error, Debug, PartialEq, Eq)]
#[error("Cannot {operation} with given operands")]
pub struct OverflowError {
Expand Down Expand Up @@ -449,7 +449,7 @@ impl ConversionOverflowError {
}
}
}

///An error struct used when a division operation in a contract attempts to divide by zero.
#[derive(Error, Debug, Default, PartialEq, Eq)]
#[error("Cannot divide by zero")]
pub struct DivideByZeroError;
Expand Down
1 change: 1 addition & 0 deletions packages/std/src/hex_binary.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ use crate::{Binary, StdError, StdResult};
/// This is similar to `cosmwasm_std::Binary` but uses hex.
/// See also <https://github.com/CosmWasm/cosmwasm/blob/main/docs/MESSAGE_TYPES.md>.
#[derive(Clone, Default, PartialEq, Eq, Hash, PartialOrd, Ord, JsonSchema)]
///A wrapper around a vector (Vec) for hex-encoded binary data, supporting encoding and decoding operations.
Copy link
Collaborator

Choose a reason for hiding this comment

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

There already is a doc string above the #[derive(...)]. I'd say we use yours as the first paragraph, but keep the reference to the message types and Binary type.

pub struct HexBinary(#[schemars(with = "String")] Vec<u8>);

impl HexBinary {
Expand Down
10 changes: 6 additions & 4 deletions packages/std/src/ibc.rs
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,7 @@ pub enum IbcMsg {
/// Port is auto-assigned to the contract's IBC port
CloseChannel { channel_id: String },
}

///Details about an IBC endpoint, including its port and channel information.
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
pub struct IbcEndpoint {
pub port_id: String,
Expand Down Expand Up @@ -192,7 +192,7 @@ impl Ord for IbcTimeoutBlock {
}
}
}

///Struct representing an IBC packet, which contains data transferred across different blockchains.
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
#[non_exhaustive]
pub struct IbcPacket {
Expand Down Expand Up @@ -225,7 +225,7 @@ impl IbcPacket {
}
}
}

///Represents an acknowledgement message in the IBC protocol.
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
#[non_exhaustive]
pub struct IbcAcknowledgement {
Expand Down Expand Up @@ -304,6 +304,7 @@ impl From<IbcChannelOpenMsg> for IbcChannel {
pub type IbcChannelOpenResponse = Option<Ibc3ChannelOpenResponse>;

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
///A response format for IBC channel open queries, providing details about the channel's status.
pub struct Ibc3ChannelOpenResponse {
/// We can set the channel version to a different one than we were called with
pub version: String,
Expand Down Expand Up @@ -411,7 +412,7 @@ impl IbcPacketReceiveMsg {
}
}

/// The message that is passed into `ibc_packet_ack`
/// Message format used when acknowledging the receipt of an IBC packet.
costa2400 marked this conversation as resolved.
Show resolved Hide resolved
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
#[non_exhaustive]
pub struct IbcPacketAckMsg {
Expand Down Expand Up @@ -456,6 +457,7 @@ impl IbcPacketTimeoutMsg {
/// or that cannot redispatch messages (like the handshake callbacks)
/// will use other Response types
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
///A general response structure for IBC handler operations, used when a specific response format isn't required.
Copy link
Collaborator

Choose a reason for hiding this comment

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

This also already has a doc comment above the #[derive(...)]

#[non_exhaustive]
pub struct IbcBasicResponse<T = Empty> {
/// Optional list of messages to pass. These will be executed in order.
Expand Down
3 changes: 2 additions & 1 deletion packages/std/src/math/fraction.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,8 @@ impl<T: Copy + From<u8> + PartialEq> Fraction<T> for (T, T) {
}
}
}

//Adds a function to a type that allows you to multiply instances of that type by a fraction.
//It's useful for calculations involving fractional multiplication.
#[macro_export]
macro_rules! impl_mul_fraction {
($Uint:ident) => {
Expand Down
2 changes: 1 addition & 1 deletion packages/std/src/math/signed_decimal.rs
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ use super::Int128;
pub struct SignedDecimal(#[schemars(with = "String")] Int128);

forward_ref_partial_eq!(SignedDecimal, SignedDecimal);

///Similar to `SignedDecimal256RangeExceeded`, but for standard `SignedDecimal` operations.
Copy link
Collaborator

Choose a reason for hiding this comment

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

If we use brackets here, we can link to the type, so it's easier to navigate:

Suggested change
///Similar to `SignedDecimal256RangeExceeded`, but for standard `SignedDecimal` operations.
/// Similar to [`crate::SignedDecimal256RangeExceeded`], but for standard `SignedDecimal` operations.

#[derive(Error, Debug, PartialEq, Eq)]
#[error("SignedDecimal range exceeded")]
pub struct SignedDecimalRangeExceeded;
Expand Down
2 changes: 1 addition & 1 deletion packages/std/src/math/signed_decimal_256.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@ use super::Int256;
pub struct SignedDecimal256(#[schemars(with = "String")] Int256);

forward_ref_partial_eq!(SignedDecimal256, SignedDecimal256);

/// Indicates an error when a SignedDecimal256 operation exceeds its range.
#[derive(Error, Debug, PartialEq, Eq)]
#[error("SignedDecimal256 range exceeded")]
pub struct SignedDecimal256RangeExceeded;
Expand Down
6 changes: 5 additions & 1 deletion packages/std/src/query/bank.rs
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ pub enum BankQuery {
#[cfg(feature = "cosmwasm_1_3")]
AllDenomMetadata { pagination: Option<PageRequest> },
}

///A response format that provides information about the supply of a specific asset or token.
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
#[serde(rename_all = "snake_case")]
#[non_exhaustive]
Expand All @@ -50,6 +50,7 @@ impl QueryResponseType for SupplyResponse {}

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
#[serde(rename_all = "snake_case")]
///This struct holds the response information for a balance query, detailing the amounts held.
#[non_exhaustive]
pub struct BalanceResponse {
/// Always returns a Coin with the requested denom.
Expand All @@ -63,6 +64,8 @@ impl QueryResponseType for BalanceResponse {}

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
#[serde(rename_all = "snake_case")]
///A response structure that contains all balances for a query,
///typically used in balance-related requests.
Copy link
Collaborator

Choose a reason for hiding this comment

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

This is not "typically used in balance-related requests." It is the response to the BankQuery::AllBalances query (and only to that one)

#[non_exhaustive]
pub struct AllBalanceResponse {
/// Returns all non-zero coins held by this account.
Expand All @@ -76,6 +79,7 @@ impl QueryResponseType for AllBalanceResponse {}
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
#[serde(rename_all = "snake_case")]
#[non_exhaustive]
/// A response format that contains metadata for a specific denomination in a query.
Copy link
Collaborator

Choose a reason for hiding this comment

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

Please move the doc-comment above the # attributes.

pub struct DenomMetadataResponse {
/// The metadata for the queried denom.
pub metadata: DenomMetadata,
Expand Down
5 changes: 3 additions & 2 deletions packages/std/src/query/ibc.rs
Original file line number Diff line number Diff line change
Expand Up @@ -29,15 +29,15 @@ pub enum IbcQuery {
},
// TODO: Add more
}

/// Contains the response information for a query about a port ID in IBC.
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
#[non_exhaustive]
pub struct PortIdResponse {
pub port_id: String,
}

impl_response_constructor!(PortIdResponse, port_id: String);

///: A response format containing a list of IBC channels and their details.
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
#[non_exhaustive]
pub struct ListChannelsResponse {
Expand All @@ -47,6 +47,7 @@ pub struct ListChannelsResponse {
impl_response_constructor!(ListChannelsResponse, channels: Vec<IbcChannel>);

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
///Contains details about a channel, typically used in IBC (Inter-Blockchain Communication) queries.
#[non_exhaustive]
pub struct ChannelResponse {
pub channel: Option<IbcChannel>,
Expand Down
1 change: 1 addition & 0 deletions packages/std/src/query/staking.rs
Original file line number Diff line number Diff line change
Expand Up @@ -99,6 +99,7 @@ impl_response_constructor!(DelegationResponse, delegation: Option<FullDelegation
///
/// Instances are created in the querier.
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
/// Detailed information about a delegation, including rewards and redelegation capabilities.
Copy link
Collaborator

Choose a reason for hiding this comment

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

Also has a doc-comment above already

#[non_exhaustive]
pub struct FullDelegation {
pub delegator: Addr,
Expand Down
2 changes: 1 addition & 1 deletion packages/std/src/query/wasm.rs
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ pub enum WasmQuery {
#[cfg(feature = "cosmwasm_1_2")]
CodeInfo { code_id: u64 },
}

/// A response format that provides detailed information about a contract.
#[non_exhaustive]
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
pub struct ContractInfoResponse {
Expand Down
1 change: 1 addition & 0 deletions packages/std/src/results/empty.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ use serde::{Deserialize, Serialize};
/// contains no meaningful data. Previously we used enums without cases,
/// but those cannot represented as valid JSON Schema (https://github.com/CosmWasm/cosmwasm/issues/451)
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema, Default)]
///An error struct used when a division operation in a contract attempts to divide by zero.
Copy link
Collaborator

Choose a reason for hiding this comment

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

Same, already above

pub struct Empty {}

#[cfg(test)]
Expand Down
1 change: 1 addition & 0 deletions packages/std/src/results/events.rs
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ use crate::forward_ref_partial_eq;
/// [*Cosmos SDK* StringEvent]: https://github.com/cosmos/cosmos-sdk/blob/v0.42.5/proto/cosmos/base/abci/v1beta1/abci.proto#L56-L70
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
#[non_exhaustive]
///Represents a full event in the Cosmos SDK, used for logging and tracking contract activities.
costa2400 marked this conversation as resolved.
Show resolved Hide resolved
pub struct Event {
/// The event type. This is renamed to "ty" because "type" is reserved in Rust. This sucks, we know.
#[serde(rename = "type")]
Expand Down
2 changes: 1 addition & 1 deletion packages/std/src/storage.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ use core::ops::{Bound, RangeBounds};
#[cfg(feature = "iterator")]
use crate::iterator::{Order, Record};
use crate::traits::Storage;

///Represents a storage mechanism that exists only in memory (not persisted).
#[derive(Default)]
pub struct MemoryStorage {
data: BTreeMap<Vec<u8>, Vec<u8>>,
Expand Down
2 changes: 1 addition & 1 deletion packages/std/src/traits.rs
Original file line number Diff line number Diff line change
Expand Up @@ -194,7 +194,7 @@ pub trait Querier {
/// helper methods
fn raw_query(&self, bin_request: &[u8]) -> QuerierResult;
}

/// A wrapper struct that enables querying blockchain state from within a contract.
#[derive(Clone)]
pub struct QuerierWrapper<'a, C: CustomQuery = Empty> {
querier: &'a dyn Querier,
Expand Down
8 changes: 4 additions & 4 deletions packages/std/src/types.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ use serde::{Deserialize, Serialize};
use crate::addresses::Addr;
use crate::coin::Coin;
use crate::timestamp::Timestamp;

///Holds the environmental information of the contract's execution context, like block info and transaction details.
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
pub struct Env {
pub block: BlockInfo,
Expand All @@ -14,7 +14,7 @@ pub struct Env {
pub transaction: Option<TransactionInfo>,
pub contract: ContractInfo,
}

/// Contains information about a transaction, like its ID and other relevant data.
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
pub struct TransactionInfo {
/// The position of this transaction in the block. The first
Expand All @@ -25,7 +25,7 @@ pub struct TransactionInfo {
///
pub index: u32,
}

/// Represents information about a specific blockchain block, such as its height, timestamp, and other relevant data.
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
pub struct BlockInfo {
/// The height of a block is the number of blocks preceding it in the blockchain.
Expand Down Expand Up @@ -104,7 +104,7 @@ pub struct MessageInfo {
/// is executed such that the new balance is visible during contract execution.
pub funds: Vec<Coin>,
}

///Stores information about a smart contract, like its creator, admin, and other relevant details.
costa2400 marked this conversation as resolved.
Show resolved Hide resolved
#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, Eq, JsonSchema)]
pub struct ContractInfo {
pub address: Addr,
Expand Down
Loading