Theres no need to declare separate vars for the testNode and then create it later, it can just be created up-front.
277 lines
6.7 KiB
Go
277 lines
6.7 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 (
|
|
"bytes"
|
|
"context"
|
|
"crypto/ed25519"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"math/rand"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"os"
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/decred/dcrd/chaincfg/v3"
|
|
dcrdtypes "github.com/decred/dcrd/rpc/jsonrpc/types/v3"
|
|
"github.com/decred/slog"
|
|
"github.com/decred/vspd/database"
|
|
"github.com/gin-gonic/gin"
|
|
)
|
|
|
|
const (
|
|
// hexCharset is a list of all valid hexadecimal characters.
|
|
hexCharset = "1234567890abcdef"
|
|
// sigCharset is a list of all valid request/response signature characters
|
|
// (base64 encoding).
|
|
sigCharset = "0123456789ABCDEFGHJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz+/="
|
|
testDb = "test.db"
|
|
backupDb = "test.db-backup"
|
|
)
|
|
|
|
var (
|
|
seededRand = rand.New(rand.NewSource(time.Now().UnixNano()))
|
|
feeXPub = "feexpub"
|
|
maxVoteChangeRecords = 3
|
|
api *Server
|
|
)
|
|
|
|
// randBytes returns a byte slice of size n filled with random bytes.
|
|
func randBytes(n int) []byte {
|
|
slice := make([]byte, n)
|
|
if _, err := seededRand.Read(slice); err != nil {
|
|
panic(err)
|
|
}
|
|
return slice
|
|
}
|
|
|
|
func TestMain(m *testing.M) {
|
|
// Set test logger to stdout.
|
|
backend := slog.NewBackend(os.Stdout)
|
|
log = backend.Logger("test")
|
|
log.SetLevel(slog.LevelTrace)
|
|
|
|
// Set up some global params.
|
|
cfg := Config{
|
|
NetParams: chaincfg.MainNetParams(),
|
|
}
|
|
_, signPrivKey, _ := ed25519.GenerateKey(seededRand)
|
|
|
|
// Create a database to use.
|
|
// Ensure we are starting with a clean environment.
|
|
os.Remove(testDb)
|
|
os.Remove(backupDb)
|
|
|
|
// Create a new blank database for all tests.
|
|
var err error
|
|
var wg sync.WaitGroup
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
err = database.CreateNew(testDb, feeXPub)
|
|
if err != nil {
|
|
panic(fmt.Errorf("error creating test database: %w", err))
|
|
}
|
|
db, err := database.Open(ctx, &wg, testDb, time.Hour, maxVoteChangeRecords)
|
|
if err != nil {
|
|
panic(fmt.Errorf("error opening test database: %w", err))
|
|
}
|
|
|
|
api = &Server{
|
|
cfg: cfg,
|
|
signPrivKey: signPrivKey,
|
|
db: db,
|
|
}
|
|
|
|
// Run tests.
|
|
exitCode := m.Run()
|
|
|
|
// Request database shutdown and wait for it to complete.
|
|
cancel()
|
|
wg.Wait()
|
|
db.Close()
|
|
os.Remove(testDb)
|
|
os.Remove(backupDb)
|
|
|
|
os.Exit(exitCode)
|
|
}
|
|
|
|
// randString randomly generates a string of the requested length, using only
|
|
// characters from the provided charset.
|
|
func randString(length int, charset string) string {
|
|
b := make([]byte, length)
|
|
for i := range b {
|
|
b[i] = charset[seededRand.Intn(len(charset))]
|
|
}
|
|
return string(b)
|
|
}
|
|
|
|
// Ensure that testNode satisfies Node.
|
|
var _ Node = (*testNode)(nil)
|
|
|
|
type testNode struct {
|
|
canTicketVote bool
|
|
canTicketVoteErr error
|
|
getRawTransactionErr error
|
|
}
|
|
|
|
func (n *testNode) CanTicketVote(_ *dcrdtypes.TxRawResult, _ *chaincfg.Params) (bool, error) {
|
|
return n.canTicketVote, n.canTicketVoteErr
|
|
}
|
|
|
|
func (n *testNode) GetRawTransaction(txHash string) (*dcrdtypes.TxRawResult, error) {
|
|
return nil, n.getRawTransactionErr
|
|
}
|
|
|
|
func TestSetAltSignAddress(t *testing.T) {
|
|
const testAddr = "DsVoDXNQqyF3V83PJJ5zMdnB4pQuJHBAh15"
|
|
tests := []struct {
|
|
name string
|
|
dcrdClientErr bool
|
|
vspClosed bool
|
|
deformReq int
|
|
addr string
|
|
node *testNode
|
|
isExistingAltSignAddr bool
|
|
wantCode int
|
|
}{{
|
|
name: "ok",
|
|
addr: testAddr,
|
|
node: &testNode{
|
|
canTicketVote: true,
|
|
},
|
|
wantCode: http.StatusOK,
|
|
}, {
|
|
name: "vsp closed",
|
|
vspClosed: true,
|
|
wantCode: http.StatusBadRequest,
|
|
}, {
|
|
name: "dcrd client error",
|
|
dcrdClientErr: true,
|
|
wantCode: http.StatusInternalServerError,
|
|
}, {
|
|
|
|
name: "bad request",
|
|
deformReq: 1,
|
|
wantCode: http.StatusBadRequest,
|
|
}, {
|
|
name: "bad addr",
|
|
addr: "xxx",
|
|
wantCode: http.StatusBadRequest,
|
|
}, {
|
|
name: "addr wrong type",
|
|
addr: "DkM3ZigNyiwHrsXRjkDQ8t8tW6uKGW9g61qEkG3bMqQPQWYEf5X3J",
|
|
wantCode: http.StatusBadRequest,
|
|
}, {
|
|
name: "error getting raw tx from dcrd client",
|
|
addr: testAddr,
|
|
node: &testNode{
|
|
getRawTransactionErr: errors.New("get raw transaction error"),
|
|
},
|
|
wantCode: http.StatusInternalServerError,
|
|
}, {
|
|
name: "error getting can vote from dcrd client",
|
|
addr: testAddr,
|
|
node: &testNode{
|
|
canTicketVoteErr: errors.New("can ticket vote error"),
|
|
},
|
|
wantCode: http.StatusInternalServerError,
|
|
}, {
|
|
name: "ticket can't vote",
|
|
addr: testAddr,
|
|
node: &testNode{
|
|
canTicketVote: false,
|
|
},
|
|
wantCode: http.StatusBadRequest,
|
|
}, {
|
|
name: "hist at max",
|
|
addr: testAddr,
|
|
isExistingAltSignAddr: true,
|
|
wantCode: http.StatusBadRequest,
|
|
}}
|
|
|
|
for _, test := range tests {
|
|
t.Run(test.name, func(t *testing.T) {
|
|
ticketHash := randString(64, hexCharset)
|
|
req := &setAltSignAddrRequest{
|
|
Timestamp: time.Now().Unix(),
|
|
TicketHash: ticketHash,
|
|
TicketHex: randString(504, hexCharset),
|
|
ParentHex: randString(504, hexCharset),
|
|
AltSignAddress: test.addr,
|
|
}
|
|
reqSig := randString(504, hexCharset)
|
|
b, err := json.Marshal(req)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if test.isExistingAltSignAddr {
|
|
data := &database.AltSignAddrData{
|
|
AltSignAddr: test.addr,
|
|
Req: string(b),
|
|
ReqSig: reqSig,
|
|
Resp: string(randBytes(1000)),
|
|
RespSig: randString(96, sigCharset),
|
|
}
|
|
if err := api.db.InsertAltSignAddr(ticketHash, data); err != nil {
|
|
t.Fatalf("unable to insert alt sign addr: %v", err)
|
|
}
|
|
}
|
|
|
|
api.cfg.VspClosed = test.vspClosed
|
|
|
|
w := httptest.NewRecorder()
|
|
c, r := gin.CreateTestContext(w)
|
|
|
|
var dcrdErr error
|
|
if test.dcrdClientErr {
|
|
dcrdErr = errors.New("error")
|
|
}
|
|
|
|
handle := func(c *gin.Context) {
|
|
c.Set(dcrdKey, test.node)
|
|
c.Set(dcrdErrorKey, dcrdErr)
|
|
c.Set(requestBytesKey, b[test.deformReq:])
|
|
api.setAltSignAddr(c)
|
|
}
|
|
|
|
r.POST("/", handle)
|
|
|
|
c.Request, err = http.NewRequest(http.MethodPost, "/", nil)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
c.Request.Header.Set("VSP-Client-Signature", reqSig)
|
|
|
|
r.ServeHTTP(w, c.Request)
|
|
|
|
if test.wantCode != w.Code {
|
|
t.Errorf("expected status %d, got %d", test.wantCode, w.Code)
|
|
}
|
|
|
|
altsig, err := api.db.AltSignAddrData(ticketHash)
|
|
if err != nil {
|
|
t.Fatalf("unable to get alt sign addr data: %v", err)
|
|
}
|
|
|
|
if test.wantCode != http.StatusOK && !test.isExistingAltSignAddr {
|
|
if altsig != nil {
|
|
t.Fatalf("expected no alt sign addr saved for errored state")
|
|
}
|
|
return
|
|
}
|
|
|
|
if !bytes.Equal(b, []byte(altsig.Req)) || altsig.ReqSig != reqSig {
|
|
t.Fatalf("expected alt sign addr data different than actual")
|
|
}
|
|
})
|
|
}
|
|
}
|