* Remove helper func getCommitmentAddress. This function is only used in one place so removing it does not introduce any duplication. Removing the func also removes the need for errors.Is comparison, which will be very useful in upcoming changes. * Rename apiError to ErrorCode. * Don't use inline type for API errors. * Export webapi errors. * Export webapi request/responses. * Add types module for API requests/responses/errors
236 lines
6.9 KiB
Go
236 lines
6.9 KiB
Go
// Copyright (c) 2020-2022 The Decred developers
|
|
// Use of this source code is governed by an ISC
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package webapi
|
|
|
|
import (
|
|
"sync"
|
|
"time"
|
|
|
|
"decred.org/dcrwallet/v2/wallet/txrules"
|
|
"github.com/decred/dcrd/dcrutil/v4"
|
|
"github.com/decred/vspd/database"
|
|
"github.com/decred/vspd/rpc"
|
|
"github.com/decred/vspd/types"
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/gin-gonic/gin/binding"
|
|
)
|
|
|
|
// addrMtx protects getNewFeeAddress.
|
|
var addrMtx sync.Mutex
|
|
|
|
// getNewFeeAddress gets a new address from the address generator, and updates
|
|
// the last used address index in the database. In order to maintain consistency
|
|
// between the internal counter of address generator and the database, this func
|
|
// uses a mutex to ensure it is not run concurrently.
|
|
func (s *Server) getNewFeeAddress() (string, uint32, error) {
|
|
addrMtx.Lock()
|
|
defer addrMtx.Unlock()
|
|
|
|
addr, idx, err := s.addrGen.NextAddress()
|
|
if err != nil {
|
|
return "", 0, err
|
|
}
|
|
|
|
err = s.db.SetLastAddressIndex(idx)
|
|
if err != nil {
|
|
return "", 0, err
|
|
}
|
|
|
|
return addr, idx, nil
|
|
}
|
|
|
|
// getCurrentFee returns the minimum fee amount a client should pay in order to
|
|
// register a ticket with the VSP at the current block height.
|
|
func (s *Server) getCurrentFee(dcrdClient *rpc.DcrdRPC) (dcrutil.Amount, error) {
|
|
bestBlock, err := dcrdClient.GetBestBlockHeader()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
sDiff, err := dcrutil.NewAmount(bestBlock.SBits)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
// Using a hard-coded amount for relay fee is acceptable here because this
|
|
// amount is never actually used to construct or broadcast transactions. It
|
|
// is only used to calculate the fee charged for adding a ticket to the VSP.
|
|
const defaultMinRelayTxFee = dcrutil.Amount(1e4)
|
|
|
|
isDCP0010Active, err := dcrdClient.IsDCP0010Active()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
fee := txrules.StakePoolTicketFee(sDiff, defaultMinRelayTxFee,
|
|
int32(bestBlock.Height), s.cfg.VSPFee, s.cfg.NetParams, isDCP0010Active)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return fee, nil
|
|
}
|
|
|
|
// feeAddress is the handler for "POST /api/v3/feeaddress".
|
|
func (s *Server) feeAddress(c *gin.Context) {
|
|
|
|
const funcName = "feeAddress"
|
|
|
|
// Get values which have been added to context by middleware.
|
|
ticket := c.MustGet(ticketKey).(database.Ticket)
|
|
knownTicket := c.MustGet(knownTicketKey).(bool)
|
|
commitmentAddress := c.MustGet(commitmentAddressKey).(string)
|
|
dcrdClient := c.MustGet(dcrdKey).(*rpc.DcrdRPC)
|
|
dcrdErr := c.MustGet(dcrdErrorKey)
|
|
if dcrdErr != nil {
|
|
s.log.Errorf("%s: Could not get dcrd client: %v", funcName, dcrdErr.(error))
|
|
s.sendError(types.ErrInternalError, c)
|
|
return
|
|
}
|
|
reqBytes := c.MustGet(requestBytesKey).([]byte)
|
|
|
|
if s.cfg.VspClosed {
|
|
s.sendError(types.ErrVspClosed, c)
|
|
return
|
|
}
|
|
|
|
var request types.FeeAddressRequest
|
|
if err := binding.JSON.BindBody(reqBytes, &request); err != nil {
|
|
s.log.Warnf("%s: Bad request (clientIP=%s): %v", funcName, c.ClientIP(), err)
|
|
s.sendErrorWithMsg(err.Error(), types.ErrBadRequest, c)
|
|
return
|
|
}
|
|
|
|
ticketHash := request.TicketHash
|
|
|
|
// Respond early if we already have the fee tx for this ticket.
|
|
if knownTicket &&
|
|
(ticket.FeeTxStatus == database.FeeReceieved ||
|
|
ticket.FeeTxStatus == database.FeeBroadcast ||
|
|
ticket.FeeTxStatus == database.FeeConfirmed) {
|
|
s.log.Warnf("%s: Fee tx already received (clientIP=%s, ticketHash=%s)",
|
|
funcName, c.ClientIP(), ticket.Hash)
|
|
s.sendError(types.ErrFeeAlreadyReceived, c)
|
|
return
|
|
}
|
|
|
|
// Get ticket details.
|
|
rawTicket, err := dcrdClient.GetRawTransaction(ticketHash)
|
|
if err != nil {
|
|
s.log.Errorf("%s: dcrd.GetRawTransaction for ticket failed (ticketHash=%s): %v", funcName, ticketHash, err)
|
|
s.sendError(types.ErrInternalError, c)
|
|
return
|
|
}
|
|
|
|
// Ensure this ticket is eligible to vote at some point in the future.
|
|
canVote, err := canTicketVote(rawTicket, dcrdClient, s.cfg.NetParams)
|
|
if err != nil {
|
|
s.log.Errorf("%s: canTicketVote error (ticketHash=%s): %v", funcName, ticketHash, err)
|
|
s.sendError(types.ErrInternalError, c)
|
|
return
|
|
}
|
|
if !canVote {
|
|
s.log.Warnf("%s: Unvotable ticket (clientIP=%s, ticketHash=%s)",
|
|
funcName, c.ClientIP(), ticketHash)
|
|
s.sendError(types.ErrTicketCannotVote, c)
|
|
return
|
|
}
|
|
|
|
// VSP already knows this ticket and has already issued it a fee address.
|
|
if knownTicket {
|
|
|
|
// If the expiry period has passed we need to issue a new fee.
|
|
now := time.Now()
|
|
if ticket.FeeExpired() {
|
|
newFee, err := s.getCurrentFee(dcrdClient)
|
|
if err != nil {
|
|
s.log.Errorf("%s: getCurrentFee error (ticketHash=%s): %v", funcName, ticket.Hash, err)
|
|
s.sendError(types.ErrInternalError, c)
|
|
return
|
|
}
|
|
ticket.FeeExpiration = now.Add(feeAddressExpiration).Unix()
|
|
ticket.FeeAmount = int64(newFee)
|
|
|
|
err = s.db.UpdateTicket(ticket)
|
|
if err != nil {
|
|
s.log.Errorf("%s: db.UpdateTicket error, failed to update fee expiry (ticketHash=%s): %v",
|
|
funcName, ticket.Hash, err)
|
|
s.sendError(types.ErrInternalError, c)
|
|
return
|
|
}
|
|
s.log.Debugf("%s: Expired fee updated (newFeeAmt=%s, ticketHash=%s)",
|
|
funcName, newFee, ticket.Hash)
|
|
}
|
|
s.sendJSONResponse(types.FeeAddressResponse{
|
|
Timestamp: now.Unix(),
|
|
Request: reqBytes,
|
|
FeeAddress: ticket.FeeAddress,
|
|
FeeAmount: ticket.FeeAmount,
|
|
Expiration: ticket.FeeExpiration,
|
|
}, c)
|
|
|
|
return
|
|
}
|
|
|
|
// Beyond this point we are processing a new ticket which the VSP has not
|
|
// seen before.
|
|
|
|
fee, err := s.getCurrentFee(dcrdClient)
|
|
if err != nil {
|
|
s.log.Errorf("%s: getCurrentFee error (ticketHash=%s): %v", funcName, ticketHash, err)
|
|
s.sendError(types.ErrInternalError, c)
|
|
return
|
|
}
|
|
|
|
newAddress, newAddressIdx, err := s.getNewFeeAddress()
|
|
if err != nil {
|
|
s.log.Errorf("%s: getNewFeeAddress error (ticketHash=%s): %v", funcName, ticketHash, err)
|
|
s.sendError(types.ErrInternalError, c)
|
|
return
|
|
}
|
|
|
|
now := time.Now()
|
|
expire := now.Add(feeAddressExpiration).Unix()
|
|
|
|
// Only set purchase height if the ticket already has 6 confs, otherwise its
|
|
// purchase height may change due to reorgs.
|
|
confirmed := false
|
|
purchaseHeight := int64(0)
|
|
if rawTicket.Confirmations >= requiredConfs {
|
|
confirmed = true
|
|
purchaseHeight = rawTicket.BlockHeight
|
|
}
|
|
|
|
dbTicket := database.Ticket{
|
|
Hash: ticketHash,
|
|
PurchaseHeight: purchaseHeight,
|
|
CommitmentAddress: commitmentAddress,
|
|
FeeAddressIndex: newAddressIdx,
|
|
FeeAddress: newAddress,
|
|
Confirmed: confirmed,
|
|
FeeAmount: int64(fee),
|
|
FeeExpiration: expire,
|
|
FeeTxStatus: database.NoFee,
|
|
}
|
|
|
|
err = s.db.InsertNewTicket(dbTicket)
|
|
if err != nil {
|
|
s.log.Errorf("%s: db.InsertNewTicket failed (ticketHash=%s): %v", funcName, ticketHash, err)
|
|
s.sendError(types.ErrInternalError, c)
|
|
return
|
|
}
|
|
|
|
s.log.Debugf("%s: Fee address created for new ticket: (tktConfirmed=%t, feeAddrIdx=%d, "+
|
|
"feeAddr=%s, feeAmt=%s, ticketHash=%s)",
|
|
funcName, confirmed, newAddressIdx, newAddress, fee, ticketHash)
|
|
|
|
s.sendJSONResponse(types.FeeAddressResponse{
|
|
Timestamp: now.Unix(),
|
|
Request: reqBytes,
|
|
FeeAddress: newAddress,
|
|
FeeAmount: int64(fee),
|
|
Expiration: expire,
|
|
}, c)
|
|
}
|