Skip to content
Snippets Groups Projects
Select Git revision
  • e0cd7febbe3db191a894e6a2700725d451bc1c1e
  • without_tipselection default
  • develop protected
  • fix/grafana-local-dashboard
  • wasp
  • fix/dashboard-explorer-freeze
  • master
  • feat/timerqueue
  • test/sync_debug_and_650
  • feat/sync_revamp_inv
  • wip/sync
  • tool/db-recovery
  • portcheck/fix
  • fix/synchronization
  • feat/new-dashboard-analysis
  • feat/refactored-analysis-dashboard
  • feat/new-analysis-dashboard
  • test/demo-prometheus-fpc
  • prometheus_metrics
  • wip/analysis-server
  • merge/fpc-test-value-transfer
  • v0.2.2
  • v0.2.1
  • v0.2.0
  • v0.1.3
  • v0.1.2
  • v0.1.1
  • v0.1.0
28 results

neighbor_test.go

Blame
  • user avatar
    Hans Moog authored and GitHub committed
    d177b4b4
    History
    neighbor_test.go 3.13 KiB
    package gossip
    
    import (
    	"net"
    	"sync"
    	"sync/atomic"
    	"testing"
    	"time"
    
    	"github.com/iotaledger/hive.go/autopeering/peer"
    	"github.com/iotaledger/hive.go/autopeering/peer/service"
    	"github.com/iotaledger/hive.go/crypto/ed25519"
    	"github.com/iotaledger/hive.go/events"
    	"github.com/iotaledger/hive.go/identity"
    	"github.com/stretchr/testify/assert"
    	"github.com/stretchr/testify/require"
    )
    
    var testData = []byte("foobar")
    
    func TestNeighborClose(t *testing.T) {
    	a, _, teardown := newPipe()
    	defer teardown()
    
    	n := newTestNeighbor("A", a)
    	n.Listen()
    	require.NoError(t, n.Close())
    }
    
    func TestNeighborCloseTwice(t *testing.T) {
    	a, _, teardown := newPipe()
    	defer teardown()
    
    	n := newTestNeighbor("A", a)
    	n.Listen()
    	require.NoError(t, n.Close())
    	require.NoError(t, n.Close())
    }
    
    func TestNeighborWrite(t *testing.T) {
    	a, b, teardown := newPipe()
    	defer teardown()
    
    	neighborA := newTestNeighbor("A", a)
    	defer neighborA.Close()
    	neighborA.Listen()
    
    	neighborB := newTestNeighbor("B", b)
    	defer neighborB.Close()
    
    	var count uint32
    	neighborB.Events.ReceiveMessage.Attach(events.NewClosure(func(data []byte) {
    		assert.Equal(t, testData, data)
    		atomic.AddUint32(&count, 1)
    	}))
    	neighborB.Listen()
    
    	_, err := neighborA.Write(testData)
    	require.NoError(t, err)
    
    	assert.Eventually(t, func() bool { return atomic.LoadUint32(&count) == 1 }, time.Second, 10*time.Millisecond)
    }
    
    func TestNeighborParallelWrite(t *testing.T) {
    	a, b, teardown := newPipe()
    	defer teardown()
    
    	neighborA := newTestNeighbor("A", a)
    	defer neighborA.Close()
    	neighborA.Listen()
    
    	neighborB := newTestNeighbor("B", b)
    	defer neighborB.Close()
    
    	var count uint32
    	neighborB.Events.ReceiveMessage.Attach(events.NewClosure(func(data []byte) {
    		assert.Equal(t, testData, data)
    		atomic.AddUint32(&count, 1)
    	}))
    	neighborB.Listen()
    
    	var (
    		wg       sync.WaitGroup
    		expected uint32
    	)
    	wg.Add(2)
    
    	// Writer 1
    	go func() {
    		defer wg.Done()
    		for i := 0; i < neighborQueueSize; i++ {
    			l, err := neighborA.Write(testData)
    			if err == ErrNeighborQueueFull || l == 0 {
    				continue
    			}
    			assert.NoError(t, err)
    			atomic.AddUint32(&expected, 1)
    		}
    	}()
    	// Writer 2
    	go func() {
    		defer wg.Done()
    		for i := 0; i < neighborQueueSize; i++ {
    			l, err := neighborA.Write(testData)
    			if err == ErrNeighborQueueFull || l == 0 {
    				continue
    			}
    			assert.NoError(t, err)
    			atomic.AddUint32(&expected, 1)
    		}
    	}()
    
    	wg.Wait()
    
    	done := func() bool {
    		actual := atomic.LoadUint32(&count)
    		return expected == actual
    	}
    	assert.Eventually(t, done, time.Second, 10*time.Millisecond)
    }
    
    func newTestNeighbor(name string, conn net.Conn) *Neighbor {
    	return NewNeighbor(newTestPeer(name, conn), conn, log.Named(name))
    }
    
    func newTestPeer(name string, conn net.Conn) *peer.Peer {
    	services := service.New()
    	services.Update(service.PeeringKey, conn.LocalAddr().Network(), 0)
    	services.Update(service.GossipKey, conn.LocalAddr().Network(), 0)
    
    	var publicKey ed25519.PublicKey
    	copy(publicKey[:], name)
    
    	return peer.NewPeer(identity.New(publicKey), net.IPv4zero, services)
    }
    
    func newPipe() (net.Conn, net.Conn, func()) {
    	a, b := net.Pipe()
    	teardown := func() {
    		_ = a.Close()
    		_ = b.Close()
    	}
    	return a, b, teardown
    }