# CosmWasm with Testnet

The following is a quick guide that shows the basics of deploying a contract to a Osmosis Testnet (osmo-test-4). It covers:

  • Initial Setup
    • Setup Rust
    • Setup Osmosis Testnet via Osmosis Installer
    • Setup Client
  • Deploy a Smart Contract
    • Clone cw-template
    • Compile the wasm contract with stable toolchain
    • Optimized Compilation
    • Store to Osmosis Testnet chain
    • Instantiate the contract
  • Execute the Contract
    • Get contract’s count
    • Increment contract’s count
    • Reset contract’s count
  • Osmo Contract Explorer
    • Upload the code
    • Execute the contract

TIP

Please note this a detailed guide on how to deploy via osmosisd, it also covers additional tooling and useful tips. You can also deploy to testnet with Beaker with a couple of commands.

# Initial Setup

This tutorial uses a Osmosis specific development tools to deploy contracts to Osmosis Testnet(osmo-test-4).

# Setup Rust

Rust is the main programming language used for CosmWasm smart contracts. While WASM smart contracts can theoretically be written in any programming language, CosmWasm libraries and tooling work best with Rust.

First, install rustup (opens new window).

Then run the following commands:

# 1. Set 'stable' as the default release channel:
rustup default stable
cargo version
# If this is lower than 1.50.0+, update
rustup update stable

# 2. Add WASM as the compilation target:
rustup target list --installed
rustup target add wasm32-unknown-unknown

# 3. Install the following packages to generate the contract:
cargo install cargo-generate --features vendored-openssl
cargo install cargo-run-script
1
2
3
4
5
6
7
8
9
10
11
12
13

# Setup Osmosis Testnet

You can easily set up an Osmosis Testnet environment using the Osmosis Installer (opens new window).

Run the following and choose option #2 (Client Node) and #2 (Testnet) in order.

curl -sL https://get.osmosis.zone/install > i.py && python3 i.py
1

Now you have successfully completed setting up an Osmosis client node in Testnet. In order to use osmosisd from the cli, either reload your terminal or refresh your profile with : ‘source ~/.profile’

# Setup the Client

First, create a wallet with the command following:

# add wallets for testing
osmosisd keys add wallet
1
2

When you run the commands above, osmosisd will prompt you all the information related to that wallet in YAML (.yml) format.

- name: wallet
  type: local
  address: osmo1v9yrqx8aaddlna29zxngr4ye3jnxtpprrej532
  pubkey: '{"@type":"/cosmos.crypto.secp256k1.PubKey","key":"AmKFqbczx7j/sYlqO2irXUSsQMdEN9Ugg1W2AOm7knh3"}'
  mnemonic: ""

**Important** write this mnemonic phrase in a safe place.
It is the only way to recover your account if you ever forget your password.

divert cliff issue spirit penalty chief improve neck enjoy pipe sing loop inherit behind space next tourist acid axis easy never ball enemy moment
1
2
3
4
5
6
7
8
9
10

You need some tokens named OSMO(uosmo) in your address to interact with the network.

# faucet

# Official Faucet

You can request tokens from the official faucet at faucet.osmosis.zone (opens new window)

# Discord Faucet

Youcan also participate in the Osmosis discord (opens new window) to request a faucet of the Osmosis Testnet. After gaining access to the testnet channel on the #roles channel of the discord, you can request a testnet token by sending the following message on the #faucet channel:

$request <address>
1

Then, you can check that your faucet request has been successful by checking the balance of your wallet bank account by trying the command:

osmosisd query bank balances $(osmosisd keys show -a wallet)
1
  • osmosisd query bank balances [address] commands query the total balance of an account.
  • osmosisd keys show -a wallet commands returns the address of the wallet that you created.

# Deploy a Smart Contract

# Clone cw-template

For this example, we will use the cw-template (opens new window) repo with counter example.

cargo generate --git https://github.com/osmosis-labs/cw-tpl-osmosis.git --name my-first-contract
cd my-first-contract
1
2

# Compile the wasm contract with stable toolchain

To deploy smart contracts, you must compile the code and make it an executable wasm binary file. We will compile the wasm contract with stable toolchain.

Compile using the command below:

# Set 'stable' as the default release channel:
rustup default stable
cargo wasm
1
2
3
  • toolchain means the compiler of Rust, and rust has three release channels: stable, beta, and nightly, of which the stable channel is the most recently released version. rustup helps you manage these different versions easily.

  • cargo is the Rust package manager. In the .cargo/config (opens new window), you can view the wasm compilation options as follows:

    wasm = "build --release --target wasm32-unknown-unknown"
    
    1

    So when we run the cargo wasm command, the cargo build --release —-target wasm32-unknown-unknown command is executed according to the option in the config file above.

After this compiles, it should produce a file in target/wasm32-unknown-unknown/release/my_first_contract.wasm. If you check the size of the file by using the ls -lh command, it shows around 1.8M. This is a release build, but not stripped of all unneeded code. To produce a much smaller version, you can run this which tells the compiler to strip all unused code out:

RUSTFLAGS='-C link-arg=-s' cargo wasm
1

This produces a file about 155K. To reduce gas costs, the binary size should be as small as possible. This will result in a less costly deployment, and lower fees on every interaction.

Also, if you don’t use compilation optimization, CosmWasm smart contract will not be deployed well due to exceeds limit error.

# Optimized Compilation

You can do further optimization using rust-optimizer (opens new window). rust-optimizer produces reproducible builds of CosmWasm smart contracts and does heavy optimization on the build size, using binary stripping and wasm-opt.

sudo docker run --rm -v "$(pwd)":/code \
    --mount type=volume,source="$(basename "$(pwd)")_cache",target=/code/target \
    --mount type=volume,source=registry_cache,target=/usr/local/cargo/registry \
    cosmwasm/rust-optimizer:0.12.6
1
2
3
4

Binary file will be at artifacts/my_first_contract.wasm folder and its size will be about 130K, which is more smaller than when only RUTFLAGS was used.

# Store to Osmosis Testnet chain

We have the wasm binary executable ready. Now it is time to store the code to the Osmosis Testnet blockchain.

# store the code on chain
RES=$(osmosisd tx wasm store artifacts/my_first_contract.wasm --from wallet --gas-prices 0.1uosmo --gas auto --gas-adjustment 1.3 -y --output json -b block)
1
2
  • osmosisd tx wasm store : upload a wasm binary
  • --from : name or address of private key with which to sign.
  • --gas-prices : gas prices in decimal format to determine the transaction fee.
  • --gas : gas limit to set per-transaction. set to "auto" to calculate sufficient gas automatically
  • --gas-adjustment : adjustment factor to be multiplied against the estimate returned by the tx simulation.
  • -y : to skip tx broadcasting prompt confirmation.
  • --output : output format.
  • -b : transaction broadcasting mode

Once that is complete, you can get the CODE_ID easily using jq.

jq is an open source that helps extract data from JSON. Install it according to your OS using the following command:

# Linux
sudo apt-get install jq

# Mac
brew install jq
1
2
3
4
5

If you are using an operating system other than the one listed above, please refer to the official document (opens new window).

Run the following command to set the CODE_ID as a variable:

# get CODE_ID
CODE_ID=$(echo $RES | jq -r '.logs[0].events[-1].attributes[0].value')
echo $CODE_ID
1
2
3

# Instantiate the contract

We can now create an instance of this wasm contract. First, set the initial state of the instance in the INIT variable and run the instantiate command.

# set the initial state of the instance
INIT='{"count":100}'

# instantiate the contract
osmosisd tx wasm instantiate $CODE_ID "$INIT" \
    --from wallet --label "my first contract" --gas-prices 0.025uosmo --gas auto --gas-adjustment 1.3 -b block -y --no-admin
1
2
3
4
5
6
  • osmosisd tx wasm instantiate : instantiate a wasm contract using CODE_ID of the uploaded binary.
  • --label : human-readable name for this contract in lists.
  • --no-admin : you must set this explicitly if you don’t want an admin.

If you have succeeded in instantiating the contract, you can search for output txhash in Osmosis Explorer (opens new window) to verify your deployment.

Get the contract address using the command following:

CONTRACT_ADDR=$(osmosisd query wasm list-contract-by-code $CODE_ID --output json | jq -r '.contracts[0]')
1
  • osmosisd query wasm list-contract-by-code : list wasm all bytecode on the chain for given code id

# Execute the Contract

Now, let's see if the contract we deployed works well.

# Get contract’s count

Send a get_count query to check the count value. The previously set INIT state is output as it is.: {"data":{"count":100}}

QUERY='{"get_count":{}}'
osmosisd query wasm contract-state smart $CONTRACT_ADDR "$QUERY" --output json
1
2
  • osmosisd query wasm contract-state smart : calls contract with given address with query data and prints the returned result

# Increment contract’s count

This time, let's send an increment transaction that increases the count value by +1. Because the transaction changes the internal state of the contract, you must pay gas fees.

If you run the get_count query again after sending the increment transaction, you can see that +1 has increased from the previous count value.

TRY_INCREMENT='{"increment": {}}'
osmosisd tx wasm execute $CONTRACT_ADDR "$TRY_INCREMENT" --from wallet --gas-prices 0.025uosmo --gas auto --gas-adjustment 1.3 -y
1
2
  • osmosisd tx wasm execute : execute a command on a wasm contract

# Reset contract’s count

Lastly, let’s send a reset transaction. Like increment, reset transaction also changes the internal state of contract, so you must pay gas fees.

RESET='{"reset": {"count": 0}}'
osmosisd tx wasm execute $CONTRACT_ADDR "$RESET" --from wallet --gas-prices 0.025uosmo --gas auto --gas-adjustment 1.3 -y
1
2

# Osmo Contract Explorer

You can also instantiate the contract using intuitive GUI in Osmo Contract Explorer (opens new window).

Let’s do it together

# Upload the code

When you access the site, you will see a list of the code we uploaded to the chain and the contract instances created using the corresponding CODE_ID.

First, Login by connecting your wallet. Then click the Create a Contract button and enter the initial state of the instance in Instantiate Message. After also entering label and admin, you can easily create a contract instance by clicking the Instantiate Contract button.

# Execute the contract

Now, let's send a transaction to the contract to see if the contract works well.

You can send a query in the Read Contract section and you can send transactions that change the internal state of the contract in the Write Contract section.

# get_count

In the Read Contract section, type get_count messages and press the Run query button to get the count value.

# increment

In the Write Contract section, type increment messages and the OSMO to pay and click the Execute Contract button to execute the transaction.

# reset

Congratulations! Now you deployed your wasm smart contract on Osmosis Testnet successfully.