Setting the Context
Recently, on February 28th, 2025; my university concluded it’s first open-source fest - ACM Amrita’s Winter of Code : S1. A program that brought in 15 projects, 10 maintainers and 50 active contributors with a bounty total amassing upto 7k over a span of two months.
In this article, I am going to dive into the tech behind our leaderboard
and
bounty-dispatcher
and how it operated for 2 months at 0 infrastructure-cost!
Why A Bounty Program ?
When Ashwin Narayanan S and Abhinav Ramakrishnan invited me to join the the organizing committee for an open source program, there were no plans of having a leaderboard, let alone a bounty program. We borrowed ideas from the very popular Hacktoberfest and decided that it would be better if the rewards could be dispatched amongst all contributors as opposed to keeping it only for the top 3 finishers.
Thus, came the challenge of maintaining a leaderboard, updating it LIVE, dispatching bounty points to each contributor and keeping it cost-effective.
Tools in The Toolbox
-
With previous experience in the MERN stack and having dealt with Next.js codebases, it was a no-brainer that the website would be built with it and hosted without any cost on Vercel.
-
Our database requirements were met using Neon - Serverless PostgreSQL which offers a very generous free tier of hosting a PostgreSQL node at 0.25 vCPU and 112 concurrent connections every month for a lifetime.
Interestingly, with the sporadic nature of distributing bounties and marking issues as “acceptable” from the Winter of Code program, our choices came down to writing a full Express.js backend or using a serverless platform.
For serverless platforms, our options spanned across
- Firebase - Google’s Mobile and App Dev Platform
- Appwrite - Build Like a Team of 100s
- Supabase - Build in a Weekend, Scale to Millions
Each one of these came with a database
and an oAuth
integration too. The
only problem with these three were that we would not require their extensive
capabilities. Additionally, we didn’t trust ourselves enough to not write
something incredibly stupid like -
const API_LIMITS = 1000000
for (let i = 0; i < API_LIMITS; i++){
await fetch("api.url.here", { methods: "GET" })
}
// If you know, you know :)))
Jokes apart, I have never been a big fan of the pay-as-you-go model which Firebase employed. Supabase and Appwrite were more approachable in that regard but time and development speed were of essence, and I had seen what Cloudflare could do!
1,00,000
free api invocations per-day,
logging and metrics dashboard, and an average deployment time of ~3 seconds
using their wrangler-cli
. Perfect place for messing up at-scale :)
Alternatively, we did consider self-hosting an Express server on a Raspberry Pi, courtsey of Abhinav Ramakrishnan’s home-lab shenanigans :) but it was ruled out in-favour of Cloudflare for reasons yet-to-be-discovered.
- Hence, final piece of the puzzle, Cloudflare Workers written in the serverless framework Hono.js that would interface with GitHub and Neon to capture all bounties and update them in the database.
Off to The Code Editors (VSCode / Neovim)
- For the UI, we quickly laid the scope that we would only build a leaderboard and display all the projects. There would not be anything fancy other than a pile of snow and snowflakes.
Auth.js (prev. NextAuth)
andGitHub oAuth Provider
were our go-tos for handling authentication and session management inside ofNext.js
- The database schema was whipped and migrated in less than 20 minutes using
Prisma
, with enough seed-data to help out with all our testing procedures.
GitHub WebHooks to Dispatch Bounty, Handle Issues and Pull Requests
With our fallback plan being a google-sheet
where each maintainer manually
records the bounty and we update the database every night to display the updated
stats of the leader, we started looking for better solutions.
At this stage, we came across GitHub’s event-driven architecture and how they have a webhook integration service where you could setup URLs and every time an event occured, GitHub would make an API call to your backend.
We decided to generate a cuid
for each of our repos and hardcoded a
record-map.
const Hooks: Record<string, string[]> = {
// QSE
"c147g7ek100009l2001": ['Abhinav-ark', 'Ashrockzzz2003'],
// Amrita PYQ
"c147g7ek100009l2002": ['Abhinav-ark', 'Ashrockzzz2003'],
// Amrita Map
"c147g7ek100009l2003": ['Abhinav-ark', 'Ashrockzzz2003'],
// ... +10 more
}
Our webhook URL looked like riteshkoushik39.cloudflare.dev/:unique-cuid-here
Each repository was configured to send requests to their unique URLs and handle the following events :
"ping" // Onboarding a repo to the program
"issue.labelled" // For marking an issue with "AMWOC" tag
"issue.unlabelled" // For removing an issue's "AMWOC" tag
"issue.assigned" // For handling assignments to contributors
"issue.unassigned" // For removing assignments to contributors with no progress
"issue.closed"
"issue.reopened"
"issue.issue_comment" // For capturing "/bounty <amount> @username"
"pull_request.labeled" // For marking PR with "AMWOC-accepted" tag
"pull_request.unlabelled" // For removing PR's "AMWOC-accepted" tag
Here, we made use of the above created Hooks
record-map in TypeScript to
map the maintainers who are allowed access to each repos’ actions such as
marking an issue as completed, assigning issues and dispatching bounties.
All database calls were writting in raw-SQL with transactions and rollback
mechanisms to handle errors. Cloudflare’s logging dashboard came in handy to
test things post-deployment as all console.logs()
were directly visible without
the need of a separate logging library.
While we did run into some JSON serialization bugs from time to time with the
dreadful [Object object]
popping up occassionally, those bugs were squashed
with an hour.
And then, distaster struck …
Meta didn’t add support for Recoil in Next.js v15 :((
We had previously worked with recoil
the popular and React friendly
state management library by Meta (prev. Facebook)
and were incredibly confident
of using it for the leaderboard. However, during development of the leaderboard,
Next.js v15
was being used as it had come out 2-3 weeks prior and it
had react19-canary-release
as a dependency. The recoil
project had been
in a state of mess where no efforts to migrate the package to support React-19
were being made and we, the naive developers reamined blissfully unaware of it.
Check out the issue: React 19 support #2318
Well, we were not the ones to give-up.
- Post dinner, we picked up Zustand, a new and optimized state management library and rewrote all of our in-browser data stores in it.
- With 30+ hours of building, testing and prettifying under the belt.
Vijay SB and Kiran Rajeev had the platform ready and deployed at
11:52PM
. - Interestingly, this was way-past Ashwin’s bedtime but I was obviously gonna torture him to wake up, onboard the projects and all the participants to the platform.
1:12AM
at night, we are online and ready to send out a university wide notification during the early hours of the day.
Inner peace.
Footnotes for our Successors
Some obvious improvements required in the future iterations of this program:
- Use
<image>
tags instead ofnext/image
as we ran out of Vercel’s - Image Optimization Limits on the final day of the program. This led to profile photos not loading for anyone who cleared the browser-cache or loaded the website for the first time. - Cache out requests in
cloudflare kv
or any other service such that calls to the entire database could be reduced. While it affects, the instant updation of the leaderboard; it is a good trade-off to save compute resources. - Setup CI/CD for the serverless function instead of manually depoloying via
wrangler-cli
.
Here are the repositories which contains the code to our leaderboard
and
bounty-dispatcher
service.
Credits to the batch of 2023-27 for building the platform:
Until next time. RK out!