-
Notifications
You must be signed in to change notification settings - Fork 397
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: Add NFT Event Ticketing System #3509
base: master
Are you sure you want to change the base?
Conversation
🛠 PR Checks SummaryAll Automated Checks passed. ✅ Manual Checks (for Reviewers):
Read More🤖 This bot helps streamline PR reviews by verifying automated checks and providing guidance for contributors and reviewers. ✅ Automated Checks (for Contributors):🟢 Maintainers must be able to edit this pull request (more info) ☑️ Contributor Actions:
☑️ Reviewer Actions:
📚 Resources:Debug
|
Hey, check out this issue, it might be interesting: #3482 This is something that is will be a priority for the Gno team and you might be able to help. |
Also, seems you also added your home realm in this PR, which I assume was a mistake. Please check the CI and see why it's failing :) |
(Tried to sneak it in) Yes beginner's mistake, the branch I created already contained my home and I forgot to remove it from the cache before pushing. Ill check the reason why its failing now |
Thanks for pointing it , I'll look into in |
|
Check the CI jobs and their outputs; they'll tell you why they're red |
Codecov ReportAll modified and coverable lines are covered by tests ✅ 📢 Thoughts on this report? Let us know! |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could you remove the whitespace formatting changes from this PR in gnovm/* ? They are a bit distracting from the realm code for this review.
107b8e9
to
1cb4241
Compare
Sorry for unnecessary mess, I ran accidentally tidy and fmt on whole repo instead of my folder |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hey, this looks cool, but there are some things that should be changed/improved. Please take a look at the comments 🙏
} | ||
|
||
func CreateEvent(name, description string, dateStr string, maxTickets int, price uint64) uint64 { | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
random newline
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Still here
events = avl.NewTree() | ||
} | ||
|
||
func CreateEvent(name, description string, dateStr string, maxTickets int, price uint64) uint64 { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
seems the first three args are strings, you can shorten this
amount := sent.AmountOf("ugnot") | ||
if amount != int64(event.price) { | ||
panic(ufmt.Sprintf("Please send exactly %d ugnot", event.price)) | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What do you think about making it possible to pay with any GRC20 token as well? Ie, the event creator could specify how many tokens would be needed for a ticket.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The CreateEvent
function now accepts a paymentToken
parameter, and BuyTicket
validates the payment by checking for that token denomination.
tickets.Mint(caller, tokenId) | ||
|
||
event.ticketsSold++ | ||
setEvent(eventId, event) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
would be good to emit events here and in other functions, ie std.Emit
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I’ve added std.Emit
calls in key places (e.g., after event creation and ticket purchase)
setEvent(eventId, event) | ||
} | ||
|
||
// all events and ticket info |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is not godoc
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Removed unnecessary and not formatted comments
func Render(path string) string { | ||
output := "# Event Ticketing System\n\n" | ||
|
||
events.Iterate("", "", func(key string, value interface{}) bool { | ||
id, _ := strconv.ParseUint(key, 10, 64) | ||
event := value.(Event) | ||
|
||
output += ufmt.Sprintf("## Event #%d: %s\n", id, event.name) | ||
output += ufmt.Sprintf("Description: %s\n", event.description) | ||
output += ufmt.Sprintf("Date: %s\n", event.date.Format("2006-01-02 15:04:05")) | ||
output += ufmt.Sprintf("Tickets: %d/%d\n", event.ticketsSold, event.maxTickets) | ||
output += ufmt.Sprintf("Price: %d ugnot\n\n", event.price) | ||
|
||
if event.ticketsSold < event.maxTickets { | ||
output += ufmt.Sprintf("[Buy Ticket](/r/jjoptimist/eventix/BuyTicket?eventId=%d)\n", id) | ||
} else { | ||
output += "**SOLD OUT**\n" | ||
} | ||
output += "---\n\n" | ||
return false | ||
}) | ||
|
||
return output | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let's use p/demo/avl/pager here
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have switched to using p/demo/avl/pager
in the Render
function as suggested.
if eventId != 1 { | ||
t.Errorf("Expected first event ID to be 1, got %d", eventId) | ||
} | ||
|
||
event, exists := getEvent(eventId) | ||
if !exists { | ||
t.Error("Event was not created") | ||
} | ||
|
||
if event.name != "Test Event" { | ||
t.Errorf("Expected event name 'Test Event', got '%s'", event.name) | ||
} | ||
|
||
// Test invalid date format | ||
defer func() { | ||
if r := recover(); r == nil { | ||
t.Error("Expected panic with invalid date format") | ||
} | ||
}() | ||
CreateEvent("Test", "Test", "invalid-date", 100, 1000000) | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let's use uassert
& urequire
. Please read up on how assert & require work in go, and check out the two packages I mentioned to see how they match
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I’ve updated the tests to use uassert
and urequire
for assertions,
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hey, left a few more comments that need to be resolved before merging. Seeing all the conversions you're doing with the ID, just switch to seqid
; it will give you sorting by creation time by default and you won't have to do conversions so often.
Also seems the fmt & tests are failing
var ( | ||
events = avl.NewTree() | ||
eventCounter uint64 = 0 | ||
tickets = grc721.NewBasicNFT("Event Ticket", "EVTIX") | ||
) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Run gno fmt
on the package
date time.Time | ||
maxTickets int | ||
price uint64 // price per ticket (in tokens) | ||
paymentToken string // payment token symbol (e.g., ugnot or any GRC20 token) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
GRC20s don't work this way; your current implementation handles only native coins. After updating the code, please add a test to validate it 🙏
} | ||
|
||
func CreateEvent(name, description string, dateStr string, maxTickets int, price uint64) uint64 { | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Still here
} | ||
} | ||
|
||
pg := pager.NewPager(events, 10, true).GetPage(page) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Try using pager.MustGetPageByPath()
Seems like CI is still failing |
Yes i fixed most of the things and got the tests running properly but i didn't merge the main for some time. Looks like i need to do changes stated here |
@JJOptimist can you fix the CI and we can merge? |
This PR adds a NFT-based event ticketing system.

Dependencies:
The system demonstrates: