name: pump-token-incentives description: "Volume-based PUMP token reward system with day-indexed epochs, pro-rata distribution, accumulator tracking, and cross-program claiming on Solana." metadata: openclaw: homepage: https://github.com/nirholas/pump-fun-sdk requires: env: - SOLANA_RPC_URL
Token Incentives — Volume-Based PUMP Token Rewards
Implement and maintain the token incentive system that rewards traders with PUMP governance tokens based on their SOL trading volume, using a day-based epoch system with pro-rata distribution.
Context
The Pump protocol incentivizes trading activity by distributing PUMP tokens to users proportional to their SOL trading volume. The system tracks volume in day-long epochs, with each day having a pre-configured token supply pool.
Day-Based Epoch System
Volume tracking operates in fixed-length epochs defined by the GlobalVolumeAccumulator:
startTime— epoch system start timestampsecondsInADay— epoch length (typically 86,400 seconds)solVolumes[]— array of total SOL volume per daytotalTokenSupply[]— array of PUMP tokens available per day
dayIndex = Math.floor((currentTimestamp - startTime) / secondsInADay)
Pro-Rata Reward Formula
$$\text{tokens} = \frac{\text{userSolVolume} \times \text{dayTokenSupply}}{\text{globalSolVolume}}$$
Account Lifecycle
- Init (
initUserVolumeAccumulator) — creates the user's volume accumulator PDA - Sync (
syncUserVolumeAccumulator) — updates the accumulator with latest volume data - Claim (
claimTokenIncentives) — claims accumulated PUMP token rewards - Close (
closeUserVolumeAccumulator) — closes the account, reclaims rent
BothPrograms Aggregation
Since users trade on both the bonding curve (Pump) and AMM (PumpAMM):
fetchUserVolumeAccumulatorTotalStats(user)— sums across both programsgetTotalUnclaimedTokensBothPrograms(user)— combined unclaimed rewardsclaimTokenIncentivesBothPrograms(user, payer)— claims from bothsyncUserVolumeAccumulatorBothPrograms(user)— syncs both
Edge Cases
| Case | Behavior |
|---|---|
| Zero global volume for a day | No tokens distributed (division by zero guarded) |
| User never synced | Only totalUnclaimedTokens from account state returned |
| Day index beyond arrays | No additional rewards computed |
| User updated same day | currentDayTokens returns preview, totalUnclaimedTokens excludes it |
Patterns to Follow
- Pure functions in
tokenIncentives.ts— no side effects, no RPC calls - Accept optional
currentTimestampparameter for testability - Always use
BNarithmetic — never convert to JavaScriptnumber - Sync before claiming to ensure the latest volume is reflected
Common Pitfalls
totalUnclaimedTokensdoes NOT include the current day's rewards — only finalized dayscurrentDayTokensreturns 0 if the user's last update was on a different day (sync first)- Day indices are zero-based from
startTime, not from epoch 0 - Volume accumulator PDAs differ between Pump and PumpAMM programs