From 5e5f4c94571bd225fdb49dff78d0dcfa188c1b16 Mon Sep 17 00:00:00 2001 From: Philemon Ukane Date: Sun, 11 Dec 2022 02:07:46 +0100 Subject: [PATCH 1/4] calculate stake rewards This implements a Decred stake calculator, where users can provide how much to stake and get an estimate of their rewards over the specified duration. - New Endpoints added ("GET /stake-reward" and "POST /stake-reward"). - New Stake Calculator Page added. --- cmd/dcrdata/internal/explorer/explorer.go | 102 ++++++------ .../internal/explorer/explorerroutes.go | 157 ++++++++++++++++++ cmd/dcrdata/internal/explorer/templates.go | 11 +- cmd/dcrdata/main.go | 2 + cmd/dcrdata/views/extras.tmpl | 1 + cmd/dcrdata/views/home.tmpl | 3 +- cmd/dcrdata/views/stake_reward.tmpl | 150 +++++++++++++++++ 7 files changed, 369 insertions(+), 57 deletions(-) create mode 100644 cmd/dcrdata/views/stake_reward.tmpl diff --git a/cmd/dcrdata/internal/explorer/explorer.go b/cmd/dcrdata/internal/explorer/explorer.go index 823862378..3a99bb829 100644 --- a/cmd/dcrdata/internal/explorer/explorer.go +++ b/cmd/dcrdata/internal/explorer/explorer.go @@ -366,7 +366,8 @@ func New(cfg *ExplorerConfig) *explorerUI { "rawtx", "status", "parameters", "agenda", "agendas", "charts", "sidechains", "disapproved", "ticketpool", "visualblocks", "statistics", "windows", "timelisting", "addresstable", "proposals", "proposal", - "market", "insight_root", "attackcost", "treasury", "treasurytable", "verify_message"} + "market", "insight_root", "attackcost", "treasury", "treasurytable", + "verify_message", "stake_reward"} for _, name := range tmpls { if err := exp.templates.addTemplate(name); err != nil { @@ -569,9 +570,9 @@ func (exp *explorerUI) Store(blockData *blockdata.BlockData, msgBlock *wire.MsgB // Simulate the annual staking rate. go func(height int64, sdiff float64, supply int64) { - ASR, _ := exp.simulateASR(1000, false, stakePerc, + ASR, _ := exp.simulateStakeReturn(1000, false, stakePerc, dcrutil.Amount(supply).ToCoin(), - float64(height), sdiff) + float64(height), sdiff, 365 /* for a year */) p.Lock() p.HomeInfo.ASR = ASR p.Unlock() @@ -673,13 +674,12 @@ func (exp *explorerUI) addRoutes() { exp.Mux.Get("/stats", redirect("statistics")) } -// Simulate ticket purchase and re-investment over a full year for a given -// starting amount of DCR and calculation parameters. Generate a TEXT table of -// the simulation results that can optionally be used for future expansion of -// dcrdata functionality. -func (exp *explorerUI) simulateASR(StartingDCRBalance float64, IntegerTicketQty bool, - CurrentStakePercent float64, ActualCoinbase float64, CurrentBlockNum float64, - ActualTicketPrice float64) (ASR float64, ReturnTable string) { +// simulateStakeReturn simulates ticket purchase and re-investment over the +// specified durationInDays for a given starting amount of DCR and calculation +// parameters. +func (exp *explorerUI) simulateStakeReturn(startingDCRBalance float64, integerTicketQty bool, + currentStakePercent float64, actualCoinbase float64, currentBlockNum float64, + actualTicketPrice float64, durationInDays float64) (stakeReturn float64, returnTable string) { // Calculations are only useful on mainnet. Short circuit calculations if // on any other version of chain params. @@ -687,18 +687,18 @@ func (exp *explorerUI) simulateASR(StartingDCRBalance float64, IntegerTicketQty return 0, "" } - BlocksPerDay := 86400 / exp.ChainParams.TargetTimePerBlock.Seconds() - BlocksPerYear := 365 * BlocksPerDay - TicketsPurchased := float64(0) + blocksPerDay := 86400 / exp.ChainParams.TargetTimePerBlock.Seconds() + totalBlocksInDuration := durationInDays * blocksPerDay + ticketsPurchased := float64(0) votesPerBlock := exp.ChainParams.VotesPerBlock() - StakeRewardAtBlock := func(blocknum float64) float64 { - Subsidy := exp.dataSource.BlockSubsidy(int64(blocknum), votesPerBlock) - return dcrutil.Amount(Subsidy.PoS / int64(votesPerBlock)).ToCoin() + stakeRewardAtBlock := func(blocknum float64) float64 { + subsidy := exp.dataSource.BlockSubsidy(int64(blocknum), votesPerBlock) + return dcrutil.Amount(subsidy.PoS / int64(votesPerBlock)).ToCoin() } - MaxCoinSupplyAtBlock := func(blocknum float64) float64 { + maxCoinSupplyAtBlock := func(blocknum float64) float64 { // 4th order poly best fit curve to Decred mainnet emissions plot. // Curve fit was done with 0 Y intercept and Pre-Mine added after. @@ -709,72 +709,70 @@ func (exp *explorerUI) simulateASR(StartingDCRBalance float64, IntegerTicketQty 1680000) // Premine 1.68M } - CoinAdjustmentFactor := ActualCoinbase / MaxCoinSupplyAtBlock(CurrentBlockNum) + coinAdjustmentFactor := actualCoinbase / maxCoinSupplyAtBlock(currentBlockNum) - TheoreticalTicketPrice := func(blocknum float64) float64 { - ProjectedCoinsCirculating := MaxCoinSupplyAtBlock(blocknum) * CoinAdjustmentFactor * CurrentStakePercent - TicketPoolSize := (float64(exp.MeanVotingBlocks) + float64(exp.ChainParams.TicketMaturity) + + theoreticalTicketPrice := func(blocknum float64) float64 { + projectedCoinsCirculating := maxCoinSupplyAtBlock(blocknum) * coinAdjustmentFactor * currentStakePercent + ticketPoolSize := (float64(exp.MeanVotingBlocks) + float64(exp.ChainParams.TicketMaturity) + float64(exp.ChainParams.CoinbaseMaturity)) * float64(exp.ChainParams.TicketsPerBlock) - return ProjectedCoinsCirculating / TicketPoolSize + return projectedCoinsCirculating / ticketPoolSize } - TicketAdjustmentFactor := ActualTicketPrice / TheoreticalTicketPrice(CurrentBlockNum) + ticketAdjustmentFactor := actualTicketPrice / theoreticalTicketPrice(currentBlockNum) // Prepare for simulation - simblock := CurrentBlockNum - TicketPrice := ActualTicketPrice - DCRBalance := StartingDCRBalance + simblock := currentBlockNum + ticketPrice := actualTicketPrice + dcrBalance := startingDCRBalance - ReturnTable += "\n\nBLOCKNUM DCR TICKETS TKT_PRICE TKT_REWRD ACTION\n" - ReturnTable += fmt.Sprintf("%8d %9.2f %8.1f %9.2f %9.2f INIT\n", - int64(simblock), DCRBalance, TicketsPurchased, - TicketPrice, StakeRewardAtBlock(simblock)) + returnTable = "\n\nBLOCKNUM DCR TICKETS TKT_PRICE TKT_REWRD ACTION\n" + returnTable += fmt.Sprintf("%8d %9.2f %8.1f %9.2f %9.2f INIT\n", + int64(simblock), dcrBalance, ticketsPurchased, + ticketPrice, stakeRewardAtBlock(simblock)) - for simblock < (BlocksPerYear + CurrentBlockNum) { + for simblock < (totalBlocksInDuration + currentBlockNum) { // Simulate a Purchase on simblock - TicketPrice = TheoreticalTicketPrice(simblock) * TicketAdjustmentFactor + ticketPrice = theoreticalTicketPrice(simblock) * ticketAdjustmentFactor - if IntegerTicketQty { + if integerTicketQty { // Use this to simulate integer qtys of tickets up to max funds - TicketsPurchased = math.Floor(DCRBalance / TicketPrice) + ticketsPurchased = math.Floor(dcrBalance / ticketPrice) } else { // Use this to simulate ALL funds used to buy tickets - even fractional tickets // which is actually not possible - TicketsPurchased = (DCRBalance / TicketPrice) + ticketsPurchased = (dcrBalance / ticketPrice) } - DCRBalance -= (TicketPrice * TicketsPurchased) - ReturnTable += fmt.Sprintf("%8d %9.2f %8.1f %9.2f %9.2f BUY\n", - int64(simblock), DCRBalance, TicketsPurchased, - TicketPrice, StakeRewardAtBlock(simblock)) + dcrBalance -= (ticketPrice * ticketsPurchased) + returnTable += fmt.Sprintf("%8d %9.2f %8.1f %9.2f %9.2f BUY\n", + int64(simblock), dcrBalance, ticketsPurchased, + ticketPrice, stakeRewardAtBlock(simblock)) // Move forward to average vote simblock += (float64(exp.ChainParams.TicketMaturity) + float64(exp.MeanVotingBlocks)) - ReturnTable += fmt.Sprintf("%8d %9.2f %8.1f %9.2f %9.2f VOTE\n", - int64(simblock), DCRBalance, TicketsPurchased, - (TheoreticalTicketPrice(simblock) * TicketAdjustmentFactor), StakeRewardAtBlock(simblock)) + returnTable += fmt.Sprintf("%8d %9.2f %8.1f %9.2f %9.2f VOTE\n", + int64(simblock), dcrBalance, ticketsPurchased, + (theoreticalTicketPrice(simblock) * ticketAdjustmentFactor), stakeRewardAtBlock(simblock)) // Simulate return of funds - DCRBalance += (TicketPrice * TicketsPurchased) + dcrBalance += (ticketPrice * ticketsPurchased) // Simulate reward - DCRBalance += (StakeRewardAtBlock(simblock) * TicketsPurchased) - TicketsPurchased = 0 + dcrBalance += (stakeRewardAtBlock(simblock) * ticketsPurchased) + ticketsPurchased = 0 // Move forward to coinbase maturity simblock += float64(exp.ChainParams.CoinbaseMaturity) - ReturnTable += fmt.Sprintf("%8d %9.2f %8.1f %9.2f %9.2f REWARD\n", - int64(simblock), DCRBalance, TicketsPurchased, - (TheoreticalTicketPrice(simblock) * TicketAdjustmentFactor), StakeRewardAtBlock(simblock)) + returnTable += fmt.Sprintf("%8d %9.2f %8.1f %9.2f %9.2f REWARD\n", + int64(simblock), dcrBalance, ticketsPurchased, + (theoreticalTicketPrice(simblock) * ticketAdjustmentFactor), stakeRewardAtBlock(simblock)) // Need to receive funds before we can use them again so add 1 block simblock++ } - // Scale down to exactly 365 days - SimulationReward := ((DCRBalance - StartingDCRBalance) / StartingDCRBalance) * 100 - ASR = (BlocksPerYear / (simblock - CurrentBlockNum)) * SimulationReward - ReturnTable += fmt.Sprintf("ASR over 365 Days is %.2f.\n", ASR) + simulationReward := ((dcrBalance - startingDCRBalance) / startingDCRBalance) * 100 + stakeReturn = (totalBlocksInDuration / (simblock - currentBlockNum)) * simulationReward return } diff --git a/cmd/dcrdata/internal/explorer/explorerroutes.go b/cmd/dcrdata/internal/explorer/explorerroutes.go index ea58fe109..f09377dc3 100644 --- a/cmd/dcrdata/internal/explorer/explorerroutes.go +++ b/cmd/dcrdata/internal/explorer/explorerroutes.go @@ -2854,3 +2854,160 @@ func (exp *explorerUI) VerifyMessageHandler(w http.ResponseWriter, r *http.Reque } displayPage("", true) } + +type stakeReward struct { + Reward float64 + RewardInDcr float64 + RewardDurationInDays float64 + TotalTicketsCost float64 + Amount string + StartDate string + EndDate string + Error string +} + +// StakeReward is the page handler for the "GET /stake-reward" path. +func (exp *explorerUI) StakeReward(w http.ResponseWriter, r *http.Request) { + voteReward := exp.pageData.HomeInfo.NBlockSubsidy.PoS / int64(exp.ChainParams.VotesPerBlock()) + str, err := exp.templates.exec("stake_reward", struct { + *CommonPageData + VoteReward float64 + CurrentTicketPrice float64 + TicketReward float64 + MinimumRewardPeriod string + ExchangeRate *types.Conversion + StakeReward *stakeReward + }{ + VoteReward: toFloat64Amount(voteReward), + CurrentTicketPrice: exp.pageData.HomeInfo.StakeDiff, + TicketReward: exp.pageData.HomeInfo.TicketReward, + MinimumRewardPeriod: exp.pageData.HomeInfo.RewardPeriod, + ExchangeRate: exp.pageData.HomeInfo.ExchangeRate, + CommonPageData: exp.commonData(r), + }) + if err != nil { + log.Errorf("Template execute failure: %v", err) + exp.StatusPage(w, defaultErrorCode, defaultErrorMessage, "", ExpStatusError) + return + } + + w.Header().Set("Content-Type", "text/html") + w.WriteHeader(http.StatusOK) + io.WriteString(w, str) +} + +// CalculateStakeReward is the handler for "POST /stake-reward" path. +func (exp *explorerUI) CalculateStakeReward(w http.ResponseWriter, r *http.Request) { + startDateStr := r.PostFormValue("startDate") + endDateStr := r.PostFormValue("endDate") + amountStr := r.PostFormValue("amount") + + var reward, rewardInDcr, totalTicketCost float64 + var durationInDays float64 + var err error + exchangeRate := exp.pageData.HomeInfo.ExchangeRate + currentTicketPrice := exp.pageData.HomeInfo.StakeDiff + homeInfo := exp.pageData.HomeInfo + voteReward := exp.pageData.HomeInfo.NBlockSubsidy.PoS / int64(exp.ChainParams.VotesPerBlock()) + + displayPage := func(errMsg string) { + str, err := exp.templates.exec("stake_reward", struct { + *CommonPageData + VoteReward float64 + CurrentTicketPrice float64 + TicketReward float64 + MinimumRewardPeriod string + ExchangeRate *types.Conversion + StakeReward *stakeReward + }{ + CommonPageData: exp.commonData(r), + VoteReward: toFloat64Amount(voteReward), + CurrentTicketPrice: currentTicketPrice, + TicketReward: homeInfo.TicketReward, + MinimumRewardPeriod: homeInfo.RewardPeriod, + ExchangeRate: exchangeRate, + StakeReward: &stakeReward{ + Reward: reward, + Amount: amountStr, + RewardDurationInDays: durationInDays, + TotalTicketsCost: totalTicketCost, + StartDate: startDateStr, + RewardInDcr: rewardInDcr, + EndDate: endDateStr, + Error: errMsg, + }, + }) + + if err != nil { + log.Errorf("Template execute failure: %v", err) + exp.StatusPage(w, defaultErrorCode, defaultErrorMessage, "", ExpStatusError) + return + } + + w.Header().Set("Content-Type", "text/html") + w.WriteHeader(http.StatusOK) + io.WriteString(w, str) + } + + startDate, err := time.Parse("2006-01-02", startDateStr) + if err != nil { + displayPage("invalid start date") + return + } + + now := time.Now() + if startDate.Before(now) { + displayPage("staking start date must be in the future") + return + } + + endDate, err := time.Parse("2006-01-02", endDateStr) + if err != nil { + displayPage("invalid end date") + return + } + + if endDate.Before(now) { + displayPage("staking end date must be in the future") + return + } + + if startDate.After(endDate) { + displayPage("staking start date cannot be before specified end date") + return + } + + // Parse and ensure amount provided for tickets is sane. + amount, err := strconv.ParseInt(amountStr, 10, 64) + if err != nil { + displayPage(err.Error()) + return + } + + amountInDCR := float64(amount) / exchangeRate.Value + if amountInDCR < currentTicketPrice { + displayPage(fmt.Sprintf("%s (%s) cannot buy a single ticket, need at least %.2f (%s)", + amountStr, exchangeRate.Index, (currentTicketPrice * exchangeRate.Value), exchangeRate.Index)) + return + } + + durationInDays = endDate.Sub(startDate).Hours() / 24 + minimumRewardDurationInDays := ((float64(exp.ChainParams.TicketMaturity) + + float64(exp.MeanVotingBlocks) + + float64(exp.ChainParams.CoinbaseMaturity)) * exp.ChainParams.TargetTimePerBlock.Hours()) / 24 + + if durationInDays < minimumRewardDurationInDays { + displayPage(fmt.Sprintf("minimum stake reward duration is %.2f days", minimumRewardDurationInDays)) + return + } + + poolPercent := homeInfo.PoolInfo.Percentage / 100 + reward, _ = exp.simulateStakeReturn(amountInDCR, true, poolPercent, + dcrutil.Amount(homeInfo.CoinSupply).ToCoin(), float64(exp.pageData.BlockInfo.Height), + currentTicketPrice, durationInDays) + + rewardInDcr = amountInDCR * reward / 100 + totalTicketCost = math.Floor(amountInDCR/currentTicketPrice) * currentTicketPrice + + displayPage("") +} diff --git a/cmd/dcrdata/internal/explorer/templates.go b/cmd/dcrdata/internal/explorer/templates.go index 562a919aa..67a0b336f 100644 --- a/cmd/dcrdata/internal/explorer/templates.go +++ b/cmd/dcrdata/internal/explorer/templates.go @@ -325,6 +325,10 @@ func formattedDuration(duration time.Duration, str *periodMap) string { return i(durationsec) + pl(str.s, durationsec) } +func toFloat64Amount(intAmount int64) float64 { + return dcrutil.Amount(intAmount).ToCoin() +} + func makeTemplateFuncMap(params *chaincfg.Params) template.FuncMap { netTheme := "theme-" + strings.ToLower(netName(params)) @@ -363,6 +367,9 @@ func makeTemplateFuncMap(params *chaincfg.Params) template.FuncMap { "divideFloat": func(n, d float64) float64 { return n / d }, + "float64Multiply": func(x, y float64) float64 { + return x * y + }, "multiply": func(a, b int64) int64 { return a * b }, @@ -404,9 +411,7 @@ func makeTemplateFuncMap(params *chaincfg.Params) template.FuncMap { "amountAsDecimalParts": func(v int64, useCommas bool) []string { return float64Formatting(dcrutil.Amount(v).ToCoin(), 8, useCommas) }, - "toFloat64Amount": func(intAmount int64) float64 { - return dcrutil.Amount(intAmount).ToCoin() - }, + "toFloat64Amount": toFloat64Amount, "dcrPerKbToAtomsPerByte": func(amt dcrutil.Amount) int64 { return int64(math.Round(float64(amt) / 1e3)) }, diff --git a/cmd/dcrdata/main.go b/cmd/dcrdata/main.go index ae53a6827..efbc6c74e 100644 --- a/cmd/dcrdata/main.go +++ b/cmd/dcrdata/main.go @@ -786,6 +786,8 @@ func _main(ctx context.Context) error { r.Get("/attack-cost", explore.AttackCost) r.Get("/verify-message", explore.VerifyMessagePage) r.With(mw.Tollbooth(limiter)).Post("/verify-message", explore.VerifyMessageHandler) + r.Get("/stake-reward", explore.StakeReward) + r.With(mw.Tollbooth(limiter)).Post("/stake-reward", explore.CalculateStakeReward) }) // Configure a page for the bare "/insight" path. This mounts the static diff --git a/cmd/dcrdata/views/extras.tmpl b/cmd/dcrdata/views/extras.tmpl index e8e20a655..70a1a95f1 100644 --- a/cmd/dcrdata/views/extras.tmpl +++ b/cmd/dcrdata/views/extras.tmpl @@ -112,6 +112,7 @@ Treasury Decode/Broadcast Tx Verify Message + Calculate Stake Reward {{- if eq .NetName "Mainnet"}} Switch To Testnet {{- else}} diff --git a/cmd/dcrdata/views/home.tmpl b/cmd/dcrdata/views/home.tmpl index a40fa9107..aa0f39ccb 100644 --- a/cmd/dcrdata/views/home.tmpl +++ b/cmd/dcrdata/views/home.tmpl @@ -120,8 +120,7 @@ {{threeSigFigs .Mempool.LikelyMineable.RegularTotal}} DCR - {{- /* TODO: pe-SM, pe-MD, ETC CAN GO */ -}} -
+
{{.Mempool.NumTickets}} tickets
diff --git a/cmd/dcrdata/views/stake_reward.tmpl b/cmd/dcrdata/views/stake_reward.tmpl new file mode 100644 index 000000000..15ed04334 --- /dev/null +++ b/cmd/dcrdata/views/stake_reward.tmpl @@ -0,0 +1,150 @@ +{{define "stake_reward" -}} + + +{{template "html-head" headData .CommonPageData "Decred Staking Reward Calculator"}} +{{template "navbar" . }} +
+

Calculate Decred Staking Reward

+
+

Use this Decred Staking Calculator to estimate your potential stake reward to see how much you'll earn by staking Decred. + The rewards predicted by this calculator are only an estimate, It does not consider network fees or fees paid to a VSP. + Staking Decred provides stake holders with rewards in addition to the potential market price gains. +

+

The mimimum duration required for a ticket to recieve a stake reward of {{printf "%.2f" .TicketReward}}% is {{ .MinimumRewardPeriod}}. The more Decred you stake, the more rewards you can earn. + Learn how to stake Decred. +

+

Disclaimer: This calculator only predicts an estimate of rewards. The actual amount of Decred earned may vary and will depend on several factors, + including actual stake reward or changes to network parameters. + Decred provides this calculator for guidance only and accepts no responsibility for any discrepancy between estimated and actual rewards. +

+ +
+
+
+
+ +
+ +
+
+
+ +
+ +
+
+
+ +
+ +
+
+ + {{with .StakeReward}} + {{- if .Error -}} + Error calculating stake reward: {{.Error}} + {{end}} + {{end}} +
+ + {{$conv := .ExchangeRate}} +
+
+
+ +
+ {{template "decimalParts" (float64AsDecimalParts .CurrentTicketPrice 8 false 2)}} + DCR +
+ {{if $conv}} +
+ {{printf "%.2f" (float64Multiply .CurrentTicketPrice $conv.Value)}} {{$conv.Index}} +
+ {{- end}} +
+
+
Vote Reward
+
+ + {{template "decimalParts" (float64AsDecimalParts .VoteReward 8 true 2)}} + + DCR/vote +
+ {{if $conv}} +
+ {{printf "%.2f" (float64Multiply .VoteReward $conv.Value)}} {{$conv.Index}} +
+ {{- end}} +
+
+ {{with .StakeReward -}} + {{- if not .Error -}} +
Estimated Stake Rewards
+
+
+
+ Reward + + +
+
+ + {{template "decimalParts" (float64AsDecimalParts .RewardInDcr 8 true 2)}} + + DCR +
+ {{if $conv}} +
+ {{printf "%.2f" (float64Multiply .RewardInDcr $conv.Value)}} {{$conv.Index}} +
+ {{- end}} +
+ ~{{printf "%.2f" .Reward}}% in ~{{.RewardDurationInDays}} days +
+
+
+ {{$nTickets := divideFloat .TotalTicketsCost $.CurrentTicketPrice}} +
+ Total Cost + +
+
+ + {{template "decimalParts" (float64AsDecimalParts .TotalTicketsCost 8 true 2)}} + + DCR +
+ {{if $conv}} +
+ {{printf "%.2f" (float64Multiply .TotalTicketsCost $conv.Value)}} {{$conv.Index}} +
+ {{- end}} +
+ {{printf "%.f" $nTickets}} {{if gt $nTickets 1.0}}Tickets{{else}}Ticket{{end}} +
+
+
+ {{- end -}} + {{- end -}} +
+
+{{ template "footer" . }} + + +{{- end}} From 9fc739a1beb034a59e15cfbee534821bc804a091 Mon Sep 17 00:00:00 2001 From: Philemon Ukane Date: Mon, 19 Dec 2022 23:23:51 +0100 Subject: [PATCH 2/4] refactor explorerUI.simulateStakeReturn --- cmd/dcrdata/internal/explorer/explorer.go | 30 ++++--------------- .../internal/explorer/explorerroutes.go | 2 +- 2 files changed, 7 insertions(+), 25 deletions(-) diff --git a/cmd/dcrdata/internal/explorer/explorer.go b/cmd/dcrdata/internal/explorer/explorer.go index 3a99bb829..f87c86b41 100644 --- a/cmd/dcrdata/internal/explorer/explorer.go +++ b/cmd/dcrdata/internal/explorer/explorer.go @@ -570,7 +570,7 @@ func (exp *explorerUI) Store(blockData *blockdata.BlockData, msgBlock *wire.MsgB // Simulate the annual staking rate. go func(height int64, sdiff float64, supply int64) { - ASR, _ := exp.simulateStakeReturn(1000, false, stakePerc, + ASR := exp.simulateStakeReturn(1000, false, stakePerc, dcrutil.Amount(supply).ToCoin(), float64(height), sdiff, 365 /* for a year */) p.Lock() @@ -679,17 +679,16 @@ func (exp *explorerUI) addRoutes() { // parameters. func (exp *explorerUI) simulateStakeReturn(startingDCRBalance float64, integerTicketQty bool, currentStakePercent float64, actualCoinbase float64, currentBlockNum float64, - actualTicketPrice float64, durationInDays float64) (stakeReturn float64, returnTable string) { + actualTicketPrice float64, durationInDays float64) float64 { // Calculations are only useful on mainnet. Short circuit calculations if // on any other version of chain params. if exp.ChainParams.Name != "mainnet" { - return 0, "" + return 0 } blocksPerDay := 86400 / exp.ChainParams.TargetTimePerBlock.Seconds() totalBlocksInDuration := durationInDays * blocksPerDay - ticketsPurchased := float64(0) votesPerBlock := exp.ChainParams.VotesPerBlock() @@ -721,17 +720,12 @@ func (exp *explorerUI) simulateStakeReturn(startingDCRBalance float64, integerTi // Prepare for simulation simblock := currentBlockNum - ticketPrice := actualTicketPrice dcrBalance := startingDCRBalance - returnTable = "\n\nBLOCKNUM DCR TICKETS TKT_PRICE TKT_REWRD ACTION\n" - returnTable += fmt.Sprintf("%8d %9.2f %8.1f %9.2f %9.2f INIT\n", - int64(simblock), dcrBalance, ticketsPurchased, - ticketPrice, stakeRewardAtBlock(simblock)) - for simblock < (totalBlocksInDuration + currentBlockNum) { // Simulate a Purchase on simblock - ticketPrice = theoreticalTicketPrice(simblock) * ticketAdjustmentFactor + var ticketsPurchased float64 + ticketPrice := theoreticalTicketPrice(simblock) * ticketAdjustmentFactor if integerTicketQty { // Use this to simulate integer qtys of tickets up to max funds @@ -743,37 +737,25 @@ func (exp *explorerUI) simulateStakeReturn(startingDCRBalance float64, integerTi } dcrBalance -= (ticketPrice * ticketsPurchased) - returnTable += fmt.Sprintf("%8d %9.2f %8.1f %9.2f %9.2f BUY\n", - int64(simblock), dcrBalance, ticketsPurchased, - ticketPrice, stakeRewardAtBlock(simblock)) // Move forward to average vote simblock += (float64(exp.ChainParams.TicketMaturity) + float64(exp.MeanVotingBlocks)) - returnTable += fmt.Sprintf("%8d %9.2f %8.1f %9.2f %9.2f VOTE\n", - int64(simblock), dcrBalance, ticketsPurchased, - (theoreticalTicketPrice(simblock) * ticketAdjustmentFactor), stakeRewardAtBlock(simblock)) // Simulate return of funds dcrBalance += (ticketPrice * ticketsPurchased) // Simulate reward dcrBalance += (stakeRewardAtBlock(simblock) * ticketsPurchased) - ticketsPurchased = 0 // Move forward to coinbase maturity simblock += float64(exp.ChainParams.CoinbaseMaturity) - returnTable += fmt.Sprintf("%8d %9.2f %8.1f %9.2f %9.2f REWARD\n", - int64(simblock), dcrBalance, ticketsPurchased, - (theoreticalTicketPrice(simblock) * ticketAdjustmentFactor), stakeRewardAtBlock(simblock)) - // Need to receive funds before we can use them again so add 1 block simblock++ } simulationReward := ((dcrBalance - startingDCRBalance) / startingDCRBalance) * 100 - stakeReturn = (totalBlocksInDuration / (simblock - currentBlockNum)) * simulationReward - return + return (totalBlocksInDuration / (simblock - currentBlockNum)) * simulationReward } func (exp *explorerUI) watchExchanges() { diff --git a/cmd/dcrdata/internal/explorer/explorerroutes.go b/cmd/dcrdata/internal/explorer/explorerroutes.go index f09377dc3..5f89a8c7f 100644 --- a/cmd/dcrdata/internal/explorer/explorerroutes.go +++ b/cmd/dcrdata/internal/explorer/explorerroutes.go @@ -3002,7 +3002,7 @@ func (exp *explorerUI) CalculateStakeReward(w http.ResponseWriter, r *http.Reque } poolPercent := homeInfo.PoolInfo.Percentage / 100 - reward, _ = exp.simulateStakeReturn(amountInDCR, true, poolPercent, + reward = exp.simulateStakeReturn(amountInDCR, true, poolPercent, dcrutil.Amount(homeInfo.CoinSupply).ToCoin(), float64(exp.pageData.BlockInfo.Height), currentTicketPrice, durationInDays) From 1572078a6ca12331a5c6bad2c7a62a5ff85bc743 Mon Sep 17 00:00:00 2001 From: Philemon Ukane Date: Thu, 22 Dec 2022 05:21:55 +0100 Subject: [PATCH 3/4] buy fractional tickets during stake reward simulation --- cmd/dcrdata/internal/explorer/explorerroutes.go | 9 +-------- cmd/dcrdata/views/stake_reward.tmpl | 16 +++++++++++----- 2 files changed, 12 insertions(+), 13 deletions(-) diff --git a/cmd/dcrdata/internal/explorer/explorerroutes.go b/cmd/dcrdata/internal/explorer/explorerroutes.go index 5f89a8c7f..9812747f5 100644 --- a/cmd/dcrdata/internal/explorer/explorerroutes.go +++ b/cmd/dcrdata/internal/explorer/explorerroutes.go @@ -2985,12 +2985,6 @@ func (exp *explorerUI) CalculateStakeReward(w http.ResponseWriter, r *http.Reque } amountInDCR := float64(amount) / exchangeRate.Value - if amountInDCR < currentTicketPrice { - displayPage(fmt.Sprintf("%s (%s) cannot buy a single ticket, need at least %.2f (%s)", - amountStr, exchangeRate.Index, (currentTicketPrice * exchangeRate.Value), exchangeRate.Index)) - return - } - durationInDays = endDate.Sub(startDate).Hours() / 24 minimumRewardDurationInDays := ((float64(exp.ChainParams.TicketMaturity) + float64(exp.MeanVotingBlocks) + @@ -3001,8 +2995,7 @@ func (exp *explorerUI) CalculateStakeReward(w http.ResponseWriter, r *http.Reque return } - poolPercent := homeInfo.PoolInfo.Percentage / 100 - reward = exp.simulateStakeReturn(amountInDCR, true, poolPercent, + reward = exp.simulateStakeReturn(amountInDCR, false, homeInfo.PoolInfo.Percentage/100, dcrutil.Amount(homeInfo.CoinSupply).ToCoin(), float64(exp.pageData.BlockInfo.Height), currentTicketPrice, durationInDays) diff --git a/cmd/dcrdata/views/stake_reward.tmpl b/cmd/dcrdata/views/stake_reward.tmpl index 15ed04334..61f669618 100644 --- a/cmd/dcrdata/views/stake_reward.tmpl +++ b/cmd/dcrdata/views/stake_reward.tmpl @@ -27,8 +27,8 @@
@@ -99,7 +99,7 @@ Reward
@@ -122,7 +122,7 @@ {{$nTickets := divideFloat .TotalTicketsCost $.CurrentTicketPrice}}
Total Cost - +
@@ -136,7 +136,13 @@
{{- end}}
- {{printf "%.f" $nTickets}} {{if gt $nTickets 1.0}}Tickets{{else}}Ticket{{end}} + + {{if lt $nTickets 1.0}} + Note: Stake amount is not sufficient to buy a single ticket. + {{else}} + {{printf "%.f" $nTickets}} {{if gt $nTickets 1.0}}Tickets{{else}}Ticket{{end}} + {{end}} +
From a32baf7920c642a6231ce3ec8343508615446dfe Mon Sep 17 00:00:00 2001 From: Philemon Ukane Date: Thu, 22 Dec 2022 06:54:17 +0100 Subject: [PATCH 4/4] make menu scrollable --- cmd/dcrdata/views/extras.tmpl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cmd/dcrdata/views/extras.tmpl b/cmd/dcrdata/views/extras.tmpl index 70a1a95f1..3663129b5 100644 --- a/cmd/dcrdata/views/extras.tmpl +++ b/cmd/dcrdata/views/extras.tmpl @@ -98,7 +98,7 @@ -