Getting started
Gno.land is a Layer 1 blockchain where smart contracts are written in
Gno, a deterministic variant of Go. If you
know Go, you can write Gno minus what doesn't fit on-chain: no
goroutines, no os, no networking, and only standard-library or
gno.land/... imports. See
Go vs Gno compatibility for the
full list.
This page walks you from zero to a working local chain and your first on-chain transaction. For just the commands, see Quick Start.
Try the Playground to write Gno in your browser.
Install
This tutorial uses three of the toolchain binaries:
| Binary | What it is |
|---|---|
gno | the Gno language toolchain (format, test, run, mod init) |
gnokey | key management for interacting with the network |
gnodev | local development environment with hot reload + web UI |
Install all three with the one-line installer:
curl -fsSL https://raw.githubusercontent.com/gnolang/gno/master/misc/install.sh | sh
Binaries land in $HOME/.gno/bin. The installer supports Linux and
macOS; on Windows use WSL. See the Installation page for
source builds, Docker, version pinning with --version <tag>, or a full
validator node with --full.
Verify the toolchain is on your $PATH:
gno version && gnokey version && gnodev --help
gno: command not found? Add $HOME/.gno/bin to your $PATH:
export PATH="$PATH:$HOME/.gno/bin".
For autocompletion, diagnostics, and formatting in your editor, see Editor Setup.
Build and run locally
The fastest way to learn Gno is to run a chain on your own machine.
gnodev boots a local devnet with hot reload, a built-in web UI,
and pre-funds accounts so you can deploy and call code right away.
You'll write a small realm, run it through the local toolchain, boot the chain, then call it with a real transaction. Everything in this section runs offline.
1. Declare the module path
In a new directory for your realm, run this command to set its on-chain path:
gno mod init gno.land/r/myname/myrealm
It writes a gnomod.toml:
module = "gno.land/r/myname/myrealm"
gno = "0.9"
The gno line is the language version it targets. Use gno.land/r/…
for realms (stateful) or gno.land/p/… for pure packages (stateless).
Note that pure packages cannot import realms; see
import rules.
2. Write Gno code
Add .gno files next to the freshly created gnomod.toml. We'll
build a counter, a realm that stores a number and exposes a function to
increment it:
package myrealm
import "strconv"
var count int
func Increment(_ realm) int {
count++
return count
}
func Render(path string) string {
return "Count: " + strconv.Itoa(count)
}
var count int: top-level variables are automatically persisted to chain state after each transaction.Increment: the_ realmfirst parameter makes it a crossing function. Making the call crossing withIncrement(cross(cur))changes the scope of the transaction to this realm, which allows it to update the realm's state. The test calls it that way; fromgnokeyit happens automatically.Render: gnoweb calls this to display your realm in the browser. It takes the URL path as astringand returns a Markdownstring, which gnoweb renders into the page.
Add a test file alongside it, myrealm_test.gno:
package myrealm
import "testing"
func TestIncrement(cur realm, t *testing.T) {
val := Increment(cross(cur))
if val != 1 {
t.Fatalf("expected 1, got %d", val)
}
}
For deeper test patterns see the Testing guide; a
fuller version of this counter lives at
examples/gno.land/r/demo/counter.
3. Format, lint, and test
The gno CLI ships the same toolchain you'd expect from Go: a formatter
that rewrites code in canonical style, a linter that catches common
mistakes (unused imports and variables, misuse of cross/realm), and
a test runner that executes _test.gno files. Run them from the package
directory before every commit:
gno fmt ./... # rewrite .gno files in canonical style
gno lint ./... # static checks for common mistakes
gno test ./... # run _test.gno files
4. Create a key
To deploy your realm or call Increment, you sign each action with a
key. A key is a private/public keypair managed locally by gnokey:
the private side signs your transactions, and the public side derives
the g1… address that identifies you on chain.
Create one:
gnokey add alice
It prompts for an encryption password and prints a 24-word mnemonic.
Store it somewhere safe to recover the key later. List your keys
to see the derived g1... address:
gnokey list
0. alice (local) - addr: g1abc...xyz pub: gpub1pgf..., path: <nil>
That g1... address is your on-chain identity. It owns funds, signs
transactions, and forms the base of your address-based namespace
when you deploy to a shared network.
Keys created this way are development-only. Do not reuse the mnemonic for real funds.
For key import, derivation, and the full keybase reference, see Interact with gnokey.
5. Run a local chain
Once the code passes tests, boot a devnet from the package directory.
gnodev starts a single-node chain, loads the realm at its declared
package path, and serves gnoweb in the foreground:
gnodev .
Open http://localhost:8888, where gnoweb shows your realm. Click into it
to see the Render output ("Count: 0"), browse exported functions and
source code, and view prefunded account balances. Every key in your
local gnokey keybase is auto-funded at startup, so alice already has
GNOT to spend. No faucet needed.
Save a .gno file and the chain reloads automatically. Pass
additional directories on the command line to load several packages
at once.
6. Call Increment
Every realm page in gnoweb has three tabs in the top header:
Content (the Render output you've already seen), Source
(the .gno files), and Actions. The Actions tab introspects the
realm's exported crossing functions and, for each one, gives you both
a form to call it from the browser via a connected wallet such as
Adena, and the equivalent gnokey
command pre-filled with the values you've typed, copy-pasteable and
ready to run.
For Increment, the command looks like this:
gnokey maketx call \
-pkgpath "gno.land/r/myname/myrealm" \
-func "Increment" \
-gas-fee 1000000ugnot -gas-wanted 1000000000 \
-chainid dev -remote http://localhost:26657 \
alice
-pkgpath is the realm's on-chain path, the same one you passed to
gno mod init. -gas-wanted is the maximum units the transaction
may consume; -gas-fee is the price per unit, in ugnot, the smallest
GNOT denomination. Together they cap what you'll pay. See
Gas fees for estimation and tuning.
The signer at the end is the alice key you just created. You'll
reuse it in the staging and testnet sections below.
On success you'll see:
(1 int)
OK!
GAS WANTED: 1000000000
GAS USED: 234567
HEIGHT: 42
EVENTS: []
TX HASH: gQP9fJYrZMTK3GgRiio3/V35smzg/jJ62q7t4TLpdV4=
The leading (1 int) is Increment's return value. Reload the realm
page and Render flips from "Count: 0" to "Count: 1"; re-run to keep
incrementing.
For more options, see Running a local dev node.
Deploy to a shared network
Now publish your realm to a shared network so others can reach it.
Two things change compared to gnodev: keys aren't auto-funded, so
you'll need test ugnot from a faucet, and each deploy is one
explicit addpkg transaction instead of hot-reload on file save.
Pick a target network now and use it consistently. The faucet's
network dropdown and every gnokey command's -remote and
-chainid flags must match:
| Network | -chainid | -remote |
|---|---|---|
| Local | dev | http://localhost:26657 |
| Staging | staging | https://rpc.staging.gno.land:443 |
| Testnet | testN | https://rpc.<testN>.testnets.gno.land:443 |
Replace testN with the current testnet chainid. See
Networks for the live list,
including mainnet status.
Examples below use staging because it resets on a short cadence,
fine for a throwaway first deploy. For anything you want to keep around,
use the current testnet instead; staging wipes regularly and your
realm will disappear with it. Betanet (gnoland1) is the production
network. There's no open faucet; funds can be granted case-by-case via a
manually reviewed interest form.
1. Get test tokens
Deploys cost gas, paid in ugnot. Get them
from the faucet: go to faucet.gno.land,
paste your g1… address, pick a network, and submit. Tokens arrive in
seconds. The
faucet is rate-limited per address; wait out the cooldown if a
re-request is rejected.
2. Query on-chain
Confirm the funds landed before spending them on a deploy:
gnokey query bank/balances/<your-g1-addr> -remote https://rpc.staging.gno.land:443
Response shows your balance as <amount>ugnot, where 1 GNOT is
1,000,000 ugnot. Read-only queries like this don't need a chainid or a
key; they hit the RPC endpoint directly.
3. Before you deploy
Two things to know before publishing your first package:
Namespaces. The simplest path is your own address-based namespace,
gno.land/r/<your-g1-addr>/…. Every account gets one and only that
account can deploy under it, so there's nothing to register. A named
namespace like gno.land/r/<name>/… requires registering the name
on-chain first. See Users and Teams.
CLA. Some networks require contributors to acknowledge and sign a
Contributor License Agreement
before deploying. It is currently off on every network; check
betanet or
staging to confirm. If addpkg ever
fails with has not signed the required CLA, sign once at
r/sys/cla and retry.
4. Deploy your package
addpkg uploads your package directory and its gnomod.toml to the
network as a single package. Deploy yours:
gnokey maketx addpkg \
-pkgpath "gno.land/r/<your-g1-addr>/myrealm" \
-pkgdir . \
-gas-fee 1000000ugnot -gas-wanted 20000000 \
-chainid staging -remote https://rpc.staging.gno.land:443 \
alice
-pkgdir is the local directory to upload, where . means the current
directory. The alice key at the end signs the transaction. On success
you'll see:
OK!
GAS WANTED: 20000000
GAS USED: 3456789
HEIGHT: 12345
EVENTS: []
TX HASH: Ni8Oq5dP0leoT/IRkKUKT18iTv8KLL3bH8OFZiV79kM=
The package is now live and browsable at
https://staging.gno.land/r/<your-g1-addr>/myrealm. On the current
testnet the URL is https://<testN>.testnets.gno.land/r/... instead.
Two optional flags are worth knowing about:
-send <amount>ugnot: transfer GNOT to the realm with the deploy.-max-deposit <amount>ugnot: cap the storage deposit the chain may lock; the transaction fails if the cap is exceeded.
For the full flag list, see
addpkg in Interact with gnokey.
You can also deploy via the Playground with a browser
wallet like Adena.
5. Call Increment
Same shape as the local call earlier, with two changes: the package
path uses your address-based namespace, and -gas-wanted is tuned to
a realistic value. Call it:
gnokey maketx call \
-pkgpath "gno.land/r/<your-g1-addr>/myrealm" \
-func "Increment" \
-gas-fee 1000000ugnot -gas-wanted 2000000 \
-chainid staging -remote https://rpc.staging.gno.land:443 \
alice
On success the response leads with the return value, then the tx receipt:
(1 int)
OK!
GAS WANTED: 2000000
GAS USED: 234567
HEIGHT: 12346
EVENTS: []
TX HASH: gQP9fJYrZMTK3GgRiio3/V35smzg/jJ62q7t4TLpdV4=
To read the state without spending gas, query the realm's render:
gnokey query vm/qrender \
-pkgpath "gno.land/r/<your-g1-addr>/myrealm" -data "" \
-remote https://rpc.staging.gno.land:443
This returns the Render output ("Count: 1"), a free, read-only
view of your realm's state. For the full maketx call and gnokey
reference, see Interact with gnokey.
Next steps
- r/docs: on-chain tour
- Effective Gno: idiomatic patterns
- Example: the
minisocialdApp: end-to-end with deploy - Gas fees: pricing, estimation, and the "out of gas" fix
- Storage deposit: how on-chain storage is paid for, and how to cap it with
-max-deposit
Getting help
- Discord: community chat.
- Gno Forum: long-form questions and proposals, on-chain.
- GitHub issues: bugs, feature requests, roadmap.
- @_gnoland on X: announcements.