How to write a simple dApp on Gno.land
Overview
This guide will show you how to write a complete dApp that combines both a package and a realm. Our app will allow any user to create a poll, and subsequently vote YAY or NAY for any poll that has not exceeded the voting deadline.
Defining dApp functionality
Our dApp will consist of a Poll package, which will handle all things related to the Poll struct, and a Poll Factory realm, which will handle the user-facing functionality and rendering.
For simplicity, we will define the functionality in plain text, and leave comments explaining the code.
Poll Package
- Defines a
Poll
struct - Defines a
NewPoll
constructor - Defines
Poll
field getters - Defines a
Vote
function - Defines a
HasVoted
check method - Defines a
VoteCount
getter method
package poll
import (
"std"
"gno.land/p/demo/avl"
)
// Main struct
type Poll struct {
title string
description string
deadline int64 // block height
voters *avl.Tree // addr -> yes / no (bool)
}
// Getters
func (p Poll) Title() string {
return p.title
}
func (p Poll) Description() string {
return p.description
}
func (p Poll) Deadline() int64 {
return p.deadline
}
func (p Poll) Voters() *avl.Tree {
return p.voters
}
// Poll instance constructor
func NewPoll(title, description string, deadline int64) *Poll {
return &Poll{
title: title,
description: description,
deadline: deadline,
voters: avl.NewTree(),
}
}
// Vote Votes for a user
func (p *Poll) Vote(voter std.Address, vote bool) {
p.Voters().Set(voter.String(), vote)
}
// HasVoted vote: yes - true, no - false
func (p *Poll) HasVoted(address std.Address) (bool, bool) {
vote, exists := p.Voters().Get(address.String())
if exists {
return true, vote.(bool)
}
return false, false
}
// VoteCount Returns the number of yay & nay votes
func (p Poll) VoteCount() (int, int) {
var yay int
p.Voters().Iterate("", "", func(key string, value interface{}) bool {
vote := value.(bool)
if vote == true {
yay = yay + 1
}
return false
})
return yay, p.Voters().Size() - yay
}
View this code in the Playground here.
A few remarks:
- We are using the
std
library for accessing blockchain-related functionality and types, such asstd.Address
. - Since the
map
data type is not deterministic in Go, we need to use the AVL tree structure, defined undergno.land/p/demo/avl
. It behaves similarly to a map; it maps a key of typestring
onto a value of any type -interface{}
. - We are importing the
gno.land/p/demo/avl
package directly from on-chain storage. You can find predeployed packages & libraries which provide additional Gno functionality in the Gno monorepo, under theexamples/
folder.
After testing the Poll
package, we need to deploy it in order to use it in our realm.
Check out the deployment guide to learn how to do this.
Poll Factory Realm
Moving on, we can create the Poll Factory realm.
The realm will contain the following functionality:
- An exported
NewPoll
method, to allow users to create polls - An exported
Vote
method, to allow users to pledge votes for any active poll - A
Render
function to display the realm state
package poll
import (
"bytes"
"std"
"gno.land/p/demo/avl"
"gno.land/p/demo/poll"
"gno.land/p/demo/seqid"
"gno.land/p/demo/ufmt"
)
// state variables
var (
polls *avl.Tree // id -> Poll
pollIDCounter seqid.ID
)
func init() {
polls = avl.NewTree()
}
// NewPoll - Creates a new Poll instance
func NewPoll(title, description string, deadline int64) string {
// get block height
if deadline <= std.GetHeight() {
panic("deadline has to be in the future")
}
// Generate int
id := pollIDCounter.Next().String()
p := poll.NewPoll(title, description, deadline)
// add new poll in avl tree
polls.Set(id, p)
return ufmt.Sprintf("Successfully created poll #%s!", id)
}
// Vote - vote for a specific Poll
// yes - true, no - false
func Vote(id string, vote bool) string {
// get txSender
txSender := std.GetOrigCaller()
// get specific Poll from AVL tree
pollRaw, exists := polls.Get(id)
if !exists {
panic("poll with specified doesn't exist")
}
// cast Poll into proper format
poll, _ := pollRaw.(*poll.Poll)
voted, _ := poll.HasVoted(txSender)
if voted {
panic("you've already voted!")
}
if poll.Deadline() <= std.GetHeight() {
panic("voting for this poll is closed")
}
// record vote
poll.Vote(txSender, vote)
// update Poll in tree
polls.Set(id, poll)
if vote == true {
return ufmt.Sprintf("Successfully voted YAY for poll #%s!", id)
}
return ufmt.Sprintf("Successfully voted NAY for poll #%s!", id)
}
Depending on where you deployed your Poll
package, you will have to change its
import path in the realm code.
With that we have written the core functionality of the realm, and all that is left is the Render function. Its purpose is to help us display the state of the realm in Markdown, by formatting the state into a string buffer:
func Render(path string) string {
var b bytes.Buffer
b.WriteString("# Polls!\n\n")
if polls.Size() == 0 {
b.WriteString("### No active polls currently!")
return b.String()
}
polls.Iterate("", "", func(key string, value interface{}) bool {
// cast raw data from tree into Poll struct
p := value.(*poll.Poll)
ddl := p.Deadline()
yay, nay := p.VoteCount()
yayPercent := 0
nayPercent := 0
if yay+nay != 0 {
yayPercent = yay * 100 / (yay + nay)
nayPercent = nay * 100 / (yay + nay)
}
b.WriteString(
ufmt.Sprintf(
"## Poll #%s: %s\n",
key, // poll ID
p.Title(),
),
)
dropdown := "<details>\n<summary>Poll details</summary><br>"
b.WriteString(dropdown + "Description: " + p.Description())
b.WriteString(
ufmt.Sprintf("<br>Voting until block: %d<br>Current vote count: %d",
p.Deadline(),
p.Voters().Size()),
)
b.WriteString(
ufmt.Sprintf("<br>YAY votes: %d (%d%%)", yay, yayPercent),
)
b.WriteString(
ufmt.Sprintf("<br>NAY votes: %d (%d%%)</details>", nay, nayPercent),
)
dropdown = "<br><details>\n<summary>Vote details</summary>"
b.WriteString(dropdown)
p.Voters().Iterate("", "", func(key string, value interface{}) bool {
voter := key
vote := value.(bool)
if vote == true {
b.WriteString(
ufmt.Sprintf("<br>%s voted YAY!", voter),
)
} else {
b.WriteString(
ufmt.Sprintf("<br>%s voted NAY!", voter),
)
}
return false
})
b.WriteString("</details>\n\n")
return false
})
return b.String()
}
View this code in the Playground here.
To see how to deploy this app, visit the Deployment guide.
Conclusion
That's it 🎉
You have successfully built a simple but fully-fledged dApp using Gno! Now you're ready to conquer new, more complex dApps in Gno.