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

feat(govdao): better rendering #3096

Merged
merged 36 commits into from
Dec 2, 2024
Merged
Show file tree
Hide file tree
Changes from 16 commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
40cef48
update rendering
leohhhn Nov 8, 2024
19352be
rm title
leohhhn Nov 8, 2024
80bf254
rm newline
leohhhn Nov 8, 2024
c236aee
rm Title
leohhhn Nov 8, 2024
97fe8fd
add abstain
leohhhn Nov 8, 2024
c6c0e9b
fix tests
leohhhn Nov 8, 2024
84d94e0
add title to propos
leohhhn Nov 9, 2024
841c1eb
fix simpledao tests
leohhhn Nov 9, 2024
e163ddd
fix tests
leohhhn Nov 9, 2024
5396eb4
set title, add test check
leohhhn Nov 9, 2024
62946d3
add username resolving
leohhhn Nov 9, 2024
4dc4f8e
update rendering
leohhhn Nov 9, 2024
c6f6515
rm leon/props
leohhhn Nov 9, 2024
83fc896
Merge branch 'master' into govdao2-rendering
leohhhn Nov 9, 2024
090966a
add bytes.Buffer
leohhhn Nov 10, 2024
7dca498
add bytes.Buffer
leohhhn Nov 10, 2024
1a9c7ed
typo
leohhhn Nov 12, 2024
05253fc
separete rendering
leohhhn Nov 12, 2024
6ed08b0
rm Render args
leohhhn Nov 12, 2024
0ae3f23
order imports
leohhhn Nov 12, 2024
3d6ca75
mod tidy
leohhhn Nov 12, 2024
54d37cd
update tests
leohhhn Nov 12, 2024
cc445c3
fix tests
leohhhn Nov 12, 2024
7961e92
mod tidy
leohhhn Nov 12, 2024
218efbf
organize
leohhhn Nov 12, 2024
ac7ee1d
fix
leohhhn Nov 12, 2024
50a948f
rename file
leohhhn Nov 18, 2024
1d6ad1b
Merge branch 'master' into govdao2-rendering
leohhhn Nov 18, 2024
dd53283
fix test
leohhhn Nov 18, 2024
15818dd
fix test
leohhhn Nov 18, 2024
82e301f
buffer>builder
leohhhn Nov 21, 2024
9bd9cd9
replace with simple string
leohhhn Nov 21, 2024
715a778
bold
leohhhn Nov 21, 2024
00081ed
update
leohhhn Nov 21, 2024
2516635
Merge branch 'master' into govdao2-rendering
leohhhn Nov 29, 2024
fad63ca
fix tests
leohhhn Nov 29, 2024
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
1 change: 1 addition & 0 deletions examples/gno.land/p/demo/dao/dao.gno
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ const (
// that contains the necessary information to
// log and generate a valid proposal
type ProposalRequest struct {
Title string // the title associated with the proposal
Description string // the description associated with the proposal
Executor Executor // the proposal executor
}
Expand Down
7 changes: 5 additions & 2 deletions examples/gno.land/p/demo/dao/proposals.gno
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ var (
Accepted ProposalStatus = "accepted" // proposal gathered quorum
NotAccepted ProposalStatus = "not accepted" // proposal failed to gather quorum
ExecutionSuccessful ProposalStatus = "execution successful" // proposal is executed successfully
ExecutionFailed ProposalStatus = "execution failed" // proposal is failed during execution
ExecutionFailed ProposalStatus = "execution failed" // proposal has failed during execution
)

func (s ProposalStatus) String() string {
Expand All @@ -42,6 +42,9 @@ type Proposal interface {
// Author returns the author of the proposal
Author() std.Address

// Title returns the title of the proposal
Title() string

// Description returns the description of the proposal
Description() string

Expand All @@ -58,5 +61,5 @@ type Proposal interface {
IsExpired() bool

// Render renders the proposal in a readable format
Render() string
Render(idx int, authorUsername string) string
leohhhn marked this conversation as resolved.
Show resolved Hide resolved
}
8 changes: 8 additions & 0 deletions examples/gno.land/p/demo/simpledao/dao.gno
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ package simpledao
import (
"errors"
"std"
"strings"

"gno.land/p/demo/avl"
"gno.land/p/demo/dao"
Expand All @@ -12,6 +13,7 @@ import (

var (
ErrInvalidExecutor = errors.New("invalid executor provided")
ErrInvalidTitle = errors.New("invalid proposal title provided")
ErrInsufficientProposalFunds = errors.New("insufficient funds for proposal")
ErrInsufficientExecuteFunds = errors.New("insufficient funds for executing proposal")
ErrProposalExecuted = errors.New("proposal already executed")
Expand Down Expand Up @@ -47,6 +49,11 @@ func (s *SimpleDAO) Propose(request dao.ProposalRequest) (uint64, error) {
return 0, ErrInvalidExecutor
}

// Make sure the title is set
if strings.TrimSpace(request.Title) == "" {
return 0, ErrInvalidTitle
}

var (
caller = getDAOCaller()
sentCoins = std.GetOrigSend() // Get the sent coins, if any
Expand All @@ -61,6 +68,7 @@ func (s *SimpleDAO) Propose(request dao.ProposalRequest) (uint64, error) {
// Create the wrapped proposal
prop := &proposal{
author: caller,
title: request.Title,
description: request.Description,
executor: request.Executor,
status: dao.Active,
Expand Down
49 changes: 49 additions & 0 deletions examples/gno.land/p/demo/simpledao/dao_test.gno
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,50 @@ func TestSimpleDAO_Propose(t *testing.T) {
)
})

t.Run("invalid title", func(t *testing.T) {
t.Parallel()

var (
called = false
cb = func() error {
called = true

return nil
}
ex = &mockExecutor{
executeFn: cb,
}

sentCoins = std.NewCoins(
std.NewCoin(
"ugnot",
minProposalFeeValue,
),
)

ms = &mockMemberStore{
isMemberFn: func(_ std.Address) bool {
return false
},
}
s = New(ms)
)

std.TestSetOrigSend(sentCoins, std.Coins{})

_, err := s.Propose(dao.ProposalRequest{
Executor: ex,
Title: "", // Set invalid title
})
uassert.ErrorIs(
t,
err,
ErrInvalidTitle,
)

uassert.False(t, called)
})

t.Run("caller cannot cover fee", func(t *testing.T) {
t.Parallel()

Expand All @@ -58,6 +102,7 @@ func TestSimpleDAO_Propose(t *testing.T) {
ex = &mockExecutor{
executeFn: cb,
}
title = "Proposal title"

sentCoins = std.NewCoins(
std.NewCoin(
Expand All @@ -80,6 +125,7 @@ func TestSimpleDAO_Propose(t *testing.T) {

_, err := s.Propose(dao.ProposalRequest{
Executor: ex,
Title: title,
})
uassert.ErrorIs(
t,
Expand All @@ -105,6 +151,7 @@ func TestSimpleDAO_Propose(t *testing.T) {
executeFn: cb,
}
description = "Proposal description"
title = "Proposal title"

proposer = testutils.TestAddress("proposer")
sentCoins = std.NewCoins(
Expand All @@ -129,6 +176,7 @@ func TestSimpleDAO_Propose(t *testing.T) {

// Make sure the proposal was added
id, err := s.Propose(dao.ProposalRequest{
Title: title,
Description: description,
Executor: ex,
})
Expand All @@ -141,6 +189,7 @@ func TestSimpleDAO_Propose(t *testing.T) {

uassert.Equal(t, proposer.String(), prop.Author().String())
uassert.Equal(t, description, prop.Description())
uassert.Equal(t, title, prop.Title())
uassert.Equal(t, dao.Active.String(), prop.Status().String())

stats := prop.Stats()
Expand Down
1 change: 1 addition & 0 deletions examples/gno.land/p/demo/simpledao/gno.mod
Original file line number Diff line number Diff line change
Expand Up @@ -9,4 +9,5 @@ require (
gno.land/p/demo/uassert v0.0.0-latest
gno.land/p/demo/ufmt v0.0.0-latest
gno.land/p/demo/urequire v0.0.0-latest
gno.land/p/moul/txlink v0.0.0-latest
)
63 changes: 49 additions & 14 deletions examples/gno.land/p/demo/simpledao/propstore.gno
Original file line number Diff line number Diff line change
@@ -1,12 +1,16 @@
package simpledao

import (
"bytes"
"errors"
"std"
"strconv"
"strings"

"gno.land/p/demo/dao"
"gno.land/p/demo/seqid"
"gno.land/p/demo/ufmt"
"gno.land/p/moul/txlink"
)

var ErrMissingProposal = errors.New("proposal is missing")
Expand All @@ -18,6 +22,7 @@ const maxRequestProposals = 10
// proposal is the internal simpledao proposal implementation
type proposal struct {
author std.Address // initiator of the proposal
title string // title of the proposal
description string // description of the proposal

executor dao.Executor // executor for the proposal
Expand All @@ -31,6 +36,10 @@ func (p *proposal) Author() std.Address {
return p.author
}

func (p *proposal) Title() string {
return p.title
}

func (p *proposal) Description() string {
return p.description
}
Expand Down Expand Up @@ -59,19 +68,26 @@ func (p *proposal) IsExpired() bool {
return false // this proposal never expires
}

func (p *proposal) Render() string {
func (p *proposal) Render(idx int, authorUsername string) string {
// Fetch the voting stats
stats := p.Stats()

output := ""
output += ufmt.Sprintf("Author: %s", p.Author().String())
output += "\n\n"
output += p.Description()
output += "\n\n"
output += ufmt.Sprintf("Status: %s", p.Status().String())
output += "\n\n"
output += ufmt.Sprintf(
"Voting stats: YES %d (%d%%), NO %d (%d%%), ABSTAIN %d (%d%%), MISSING VOTE %d (%d%%)",
var out bytes.Buffer

out.WriteString(ufmt.Sprintf("# Proposal #%d - %s\n\n", idx, p.title))

out.WriteString("## Description\n\n")
leohhhn marked this conversation as resolved.
Show resolved Hide resolved
if strings.TrimSpace(p.description) != "" {
out.WriteString(ufmt.Sprintf("%s\n\n", p.description))
} else {
out.WriteString("No description provided.\n\n")
}

out.WriteString("## Proposal information\n\n")
out.WriteString(ufmt.Sprintf("#### Status: %s\n\n", strings.ToUpper(p.Status().String())))

out.WriteString(ufmt.Sprintf(
"#### Voting stats:\n- YES %d (%d%%)\n- NO %d (%d%%)\n- ABSTAIN %d (%d%%)\n- MISSING VOTE %d (%d%%)\n",
leohhhn marked this conversation as resolved.
Show resolved Hide resolved
stats.YayVotes,
stats.YayPercent(),
stats.NayVotes,
Expand All @@ -80,11 +96,30 @@ func (p *proposal) Render() string {
stats.AbstainPercent(),
stats.MissingVotes(),
stats.MissingVotesPercent(),
)
output += "\n\n"
output += ufmt.Sprintf("Threshold met: %t", stats.YayVotes > (2*stats.TotalVotingPower)/3)
))
out.WriteString("\n\n")
thresholdOut := strings.ToUpper(ufmt.Sprintf("%t", stats.YayVotes > (2*stats.TotalVotingPower)/3))

out.WriteString(ufmt.Sprintf("#### Threshold met: %s\n\n", thresholdOut))

if authorUsername != "" {
out.WriteString(ufmt.Sprintf("#### Author: [%s](/r/demo/users:%s)\n\n", authorUsername, authorUsername))
} else {
out.WriteString(ufmt.Sprintf("#### Author: %s\n\n", p.author.String()))
}

out.WriteString("## Actions\n\n")
if p.status == dao.Active {
out.WriteString(ufmt.Sprintf("[Vote YES](%s) - [Vote NO](%s) - [Vote ABSTAIN](%s)",
txlink.URL("VoteOnProposal", "id", strconv.Itoa(idx), "option", "YES"),
txlink.URL("VoteOnProposal", "id", strconv.Itoa(idx), "option", "NO"),
txlink.URL("VoteOnProposal", "id", strconv.Itoa(idx), "option", "ABSTAIN"),
))
} else {
out.WriteString("The voting period for this proposal is over.\n\n")
}

return output
return out.String()
}

// addProposal adds a new simpledao proposal to the store
Expand Down
62 changes: 43 additions & 19 deletions examples/gno.land/r/gov/dao/v2/dao.gno
Original file line number Diff line number Diff line change
@@ -1,13 +1,17 @@
package govdao

import (
"bytes"
"std"
"strconv"
"strings"

"gno.land/p/demo/dao"
"gno.land/p/demo/membstore"
"gno.land/p/demo/simpledao"
"gno.land/p/demo/ufmt"

"gno.land/r/demo/users"
)

var (
Expand Down Expand Up @@ -69,33 +73,50 @@ func GetMembStore() membstore.MemberStore {
}

func Render(path string) string {
var out bytes.Buffer

if path == "" {
out.WriteString("# GovDAO Proposals\n\n")
numProposals := d.Size()

if numProposals == 0 {
return "No proposals found :(" // corner case
out.WriteString("No proposals found :(") // corner case
return out.String()
}

output := ""

offset := uint64(0)
if numProposals >= 10 {
offset = uint64(numProposals) - 10
}

// Fetch the last 10 proposals
for idx, prop := range d.Proposals(offset, uint64(10)) {
output += ufmt.Sprintf(
"- [Proposal #%d](%s:%d) - (**%s**)(by %s)\n",
idx,
"/r/gov/dao/v2",
idx,
prop.Status().String(),
prop.Author().String(),
)
proposals := d.Proposals(offset, uint64(10))
petar-dambovaliev marked this conversation as resolved.
Show resolved Hide resolved
for i := len(proposals) - 1; i >= 0; i-- {
prop := proposals[i]

title := prop.Title()
if len(title) > 40 {
title = title[:40] + "..."
}

propID := offset + uint64(i)
out.WriteString(ufmt.Sprintf("## [Prop #%d - %s](/r/gov/dao/v2:%d)\n\n", propID, title, propID))
out.WriteString(ufmt.Sprintf("#### Status: %s\n\n", strings.ToUpper(prop.Status().String())))

user := users.GetUserByAddress(prop.Author())
authorUsername := prop.Author().String()
if user != nil {
authorUsername = ufmt.Sprintf("[%s](/r/demo/users:%s)", user.Name, user.Name)
}

out.WriteString(ufmt.Sprintf("#### Author: %s\n\n", authorUsername))

if i != 0 {
out.WriteString("---\n\n")
}
}

return output
return out.String()
}

// Display the detailed proposal
Expand All @@ -110,12 +131,15 @@ func Render(path string) string {
return ufmt.Sprintf("unable to fetch proposal, %s", err.Error())
}

authorUsername := ""
user := users.GetUserByAddress(prop.Author())
if user != nil {
authorUsername = user.Name
}

// Render the proposal
output := ""
output += ufmt.Sprintf("# Prop #%d", idx)
output += "\n\n"
output += prop.Render()
output += "\n\n"
out.WriteString(prop.Render(idx, authorUsername))
out.WriteString("\n\n")

return output
return out.String()
}
1 change: 1 addition & 0 deletions examples/gno.land/r/gov/dao/v2/gno.mod
Original file line number Diff line number Diff line change
Expand Up @@ -7,4 +7,5 @@ require (
gno.land/p/demo/simpledao v0.0.0-latest
gno.land/p/demo/ufmt v0.0.0-latest
gno.land/p/gov/executor v0.0.0-latest
gno.land/r/demo/users v0.0.0-latest
)
Loading
Loading