mock.go 3.05 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11
// Copyright 2020 The Swarm Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.

package mock

import (
	"context"
	"sync"

	"github.com/ethersphere/bee/pkg/swarm"
12
	"github.com/ethersphere/bee/pkg/topology"
13 14
)

15
type mock struct {
16 17 18
	peers           []swarm.Address
	closestPeer     swarm.Address
	closestPeerErr  error
19
	addPeersErr     error
20 21
	marshalJSONFunc func() ([]byte, error)
	mtx             sync.Mutex
22 23
}

24 25 26 27 28 29
func WithPeers(peers ...swarm.Address) Option {
	return optionFunc(func(d *mock) {
		d.peers = peers
	})
}

30
func WithAddPeersErr(err error) Option {
31
	return optionFunc(func(d *mock) {
32
		d.addPeersErr = err
33
	})
34 35
}

36 37 38 39
func WithClosestPeer(addr swarm.Address) Option {
	return optionFunc(func(d *mock) {
		d.closestPeer = addr
	})
40 41
}

42 43 44 45 46 47
func WithClosestPeerErr(err error) Option {
	return optionFunc(func(d *mock) {
		d.closestPeerErr = err
	})
}

48 49 50 51 52 53
func WithMarshalJSONFunc(f func() ([]byte, error)) Option {
	return optionFunc(func(d *mock) {
		d.marshalJSONFunc = f
	})
}

54 55 56 57 58 59 60 61
func NewTopologyDriver(opts ...Option) topology.Driver {
	d := new(mock)
	for _, o := range opts {
		o.apply(d)
	}
	return d
}

62 63 64 65 66 67 68 69 70
func (d *mock) AddPeers(_ context.Context, addrs ...swarm.Address) error {
	if d.addPeersErr != nil {
		return d.addPeersErr
	}

	for _, addr := range addrs {
		d.mtx.Lock()
		d.peers = append(d.peers, addr)
		d.mtx.Unlock()
71 72 73 74
	}

	return nil
}
75

76
func (d *mock) Connected(ctx context.Context, addr swarm.Address) error {
77
	return d.AddPeers(ctx, addr)
78 79 80 81 82
}

func (d *mock) Disconnected(swarm.Address) {
	panic("todo")
}
83

84
func (d *mock) Peers() []swarm.Address {
85 86
	return d.peers
}
87

88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117
func (d *mock) ClosestPeer(_ swarm.Address, skipPeers ...swarm.Address) (peerAddr swarm.Address, err error) {
	if len(skipPeers) == 0 {
		return d.closestPeer, d.closestPeerErr
	}

	d.mtx.Lock()
	defer d.mtx.Unlock()

	skipPeer := false

	for _, p := range d.peers {
		for _, a := range skipPeers {
			if a.Equal(p) {
				skipPeer = true
				break
			}
		}
		if skipPeer {
			skipPeer = false
			continue
		}

		peerAddr = p
	}

	if peerAddr.IsZero() {
		return peerAddr, topology.ErrNotFound
	}

	return peerAddr, nil
118 119
}

120 121 122 123
func (d *mock) SubscribePeersChange() (c <-chan struct{}, unsubscribe func()) {
	return c, unsubscribe
}

124
func (*mock) NeighborhoodDepth() uint8 {
125 126 127 128
	return 0
}

// EachPeer iterates from closest bin to farthest
129 130 131 132 133 134 135 136 137 138 139 140
func (d *mock) EachPeer(f topology.EachPeerFunc) (err error) {
	d.mtx.Lock()
	defer d.mtx.Unlock()

	for i, p := range d.peers {
		_, _, err = f(p, uint8(i))
		if err != nil {
			return
		}
	}

	return nil
141 142 143
}

// EachPeerRev iterates from farthest bin to closest
144 145 146 147 148 149 150 151 152 153 154 155
func (d *mock) EachPeerRev(f topology.EachPeerFunc) (err error) {
	d.mtx.Lock()
	defer d.mtx.Unlock()

	for i := len(d.peers) - 1; i >= 0; i-- {
		_, _, err = f(d.peers[i], uint8(i))
		if err != nil {
			return
		}
	}

	return nil
156 157
}

158 159 160 161
func (d *mock) MarshalJSON() ([]byte, error) {
	return d.marshalJSONFunc()
}

162 163 164 165
func (d *mock) Close() error {
	return nil
}

166 167 168 169 170 171 172
type Option interface {
	apply(*mock)
}

type optionFunc func(*mock)

func (f optionFunc) apply(r *mock) { f(r) }