224 lines
5.1 KiB
Go
224 lines
5.1 KiB
Go
// Copyright 2018 The go-ethereum Authors
|
|
// This file is part of the go-ethereum library.
|
|
//
|
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// The go-ethereum library is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package protocols
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/ethereum/go-ethereum/p2p"
|
|
"github.com/ethereum/go-ethereum/p2p/simulations/adapters"
|
|
"github.com/ethereum/go-ethereum/rlp"
|
|
)
|
|
|
|
//dummy Balance implementation
|
|
type dummyBalance struct {
|
|
amount int64
|
|
peer *Peer
|
|
}
|
|
|
|
//dummy Prices implementation
|
|
type dummyPrices struct{}
|
|
|
|
//a dummy message which needs size based accounting
|
|
//sender pays
|
|
type perBytesMsgSenderPays struct {
|
|
Content string
|
|
}
|
|
|
|
//a dummy message which needs size based accounting
|
|
//receiver pays
|
|
type perBytesMsgReceiverPays struct {
|
|
Content string
|
|
}
|
|
|
|
//a dummy message which is paid for per unit
|
|
//sender pays
|
|
type perUnitMsgSenderPays struct{}
|
|
|
|
//receiver pays
|
|
type perUnitMsgReceiverPays struct{}
|
|
|
|
//a dummy message which has zero as its price
|
|
type zeroPriceMsg struct{}
|
|
|
|
//a dummy message which has no accounting
|
|
type nilPriceMsg struct{}
|
|
|
|
//return the price for the defined messages
|
|
func (d *dummyPrices) Price(msg interface{}) *Price {
|
|
switch msg.(type) {
|
|
//size based message cost, receiver pays
|
|
case *perBytesMsgReceiverPays:
|
|
return &Price{
|
|
PerByte: true,
|
|
Value: uint64(100),
|
|
Payer: Receiver,
|
|
}
|
|
//size based message cost, sender pays
|
|
case *perBytesMsgSenderPays:
|
|
return &Price{
|
|
PerByte: true,
|
|
Value: uint64(100),
|
|
Payer: Sender,
|
|
}
|
|
//unitary cost, receiver pays
|
|
case *perUnitMsgReceiverPays:
|
|
return &Price{
|
|
PerByte: false,
|
|
Value: uint64(99),
|
|
Payer: Receiver,
|
|
}
|
|
//unitary cost, sender pays
|
|
case *perUnitMsgSenderPays:
|
|
return &Price{
|
|
PerByte: false,
|
|
Value: uint64(99),
|
|
Payer: Sender,
|
|
}
|
|
case *zeroPriceMsg:
|
|
return &Price{
|
|
PerByte: false,
|
|
Value: uint64(0),
|
|
Payer: Sender,
|
|
}
|
|
case *nilPriceMsg:
|
|
return nil
|
|
}
|
|
return nil
|
|
}
|
|
|
|
//dummy accounting implementation, only stores values for later check
|
|
func (d *dummyBalance) Add(amount int64, peer *Peer) error {
|
|
d.amount = amount
|
|
d.peer = peer
|
|
return nil
|
|
}
|
|
|
|
type testCase struct {
|
|
msg interface{}
|
|
size uint32
|
|
sendResult int64
|
|
recvResult int64
|
|
}
|
|
|
|
//lowest level unit test
|
|
func TestBalance(t *testing.T) {
|
|
//create instances
|
|
balance := &dummyBalance{}
|
|
prices := &dummyPrices{}
|
|
//create the spec
|
|
spec := createTestSpec()
|
|
//create the accounting hook for the spec
|
|
acc := NewAccounting(balance, prices)
|
|
//create a peer
|
|
id := adapters.RandomNodeConfig().ID
|
|
p := p2p.NewPeer(id, "testPeer", nil)
|
|
peer := NewPeer(p, &dummyRW{}, spec)
|
|
//price depends on size, receiver pays
|
|
msg := &perBytesMsgReceiverPays{Content: "testBalance"}
|
|
size, _ := rlp.EncodeToBytes(msg)
|
|
|
|
testCases := []testCase{
|
|
{
|
|
msg,
|
|
uint32(len(size)),
|
|
int64(len(size) * 100),
|
|
int64(len(size) * -100),
|
|
},
|
|
{
|
|
&perBytesMsgSenderPays{Content: "testBalance"},
|
|
uint32(len(size)),
|
|
int64(len(size) * -100),
|
|
int64(len(size) * 100),
|
|
},
|
|
{
|
|
&perUnitMsgSenderPays{},
|
|
0,
|
|
int64(-99),
|
|
int64(99),
|
|
},
|
|
{
|
|
&perUnitMsgReceiverPays{},
|
|
0,
|
|
int64(99),
|
|
int64(-99),
|
|
},
|
|
{
|
|
&zeroPriceMsg{},
|
|
0,
|
|
int64(0),
|
|
int64(0),
|
|
},
|
|
{
|
|
&nilPriceMsg{},
|
|
0,
|
|
int64(0),
|
|
int64(0),
|
|
},
|
|
}
|
|
checkAccountingTestCases(t, testCases, acc, peer, balance, true)
|
|
checkAccountingTestCases(t, testCases, acc, peer, balance, false)
|
|
}
|
|
|
|
func checkAccountingTestCases(t *testing.T, cases []testCase, acc *Accounting, peer *Peer, balance *dummyBalance, send bool) {
|
|
for _, c := range cases {
|
|
var err error
|
|
var expectedResult int64
|
|
//reset balance before every check
|
|
balance.amount = 0
|
|
if send {
|
|
err = acc.Send(peer, c.size, c.msg)
|
|
expectedResult = c.sendResult
|
|
} else {
|
|
err = acc.Receive(peer, c.size, c.msg)
|
|
expectedResult = c.recvResult
|
|
}
|
|
|
|
checkResults(t, err, balance, peer, expectedResult)
|
|
}
|
|
}
|
|
|
|
func checkResults(t *testing.T, err error, balance *dummyBalance, peer *Peer, result int64) {
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if balance.peer != peer {
|
|
t.Fatalf("expected Add to be called with peer %v, got %v", peer, balance.peer)
|
|
}
|
|
if balance.amount != result {
|
|
t.Fatalf("Expected balance to be %d but is %d", result, balance.amount)
|
|
}
|
|
}
|
|
|
|
//create a test spec
|
|
func createTestSpec() *Spec {
|
|
spec := &Spec{
|
|
Name: "test",
|
|
Version: 42,
|
|
MaxMsgSize: 10 * 1024,
|
|
Messages: []interface{}{
|
|
&perBytesMsgReceiverPays{},
|
|
&perBytesMsgSenderPays{},
|
|
&perUnitMsgReceiverPays{},
|
|
&perUnitMsgSenderPays{},
|
|
&zeroPriceMsg{},
|
|
&nilPriceMsg{},
|
|
},
|
|
}
|
|
return spec
|
|
}
|