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
 | 
						|
}
 |