watchtower/wtmock/peer: create mock net.Conn using bidi MockPeer

This commit is contained in:
Conner Fromknecht 2019-03-15 02:32:53 -07:00
parent 87e8700c5d
commit 81497eceaf
No known key found for this signature in database
GPG Key ID: E7D737B67FA592C7

@ -13,6 +13,8 @@ import (
type MockPeer struct { type MockPeer struct {
remotePub *btcec.PublicKey remotePub *btcec.PublicKey
remoteAddr net.Addr remoteAddr net.Addr
localPub *btcec.PublicKey
localAddr net.Addr
IncomingMsgs chan []byte IncomingMsgs chan []byte
OutgoingMsgs chan []byte OutgoingMsgs chan []byte
@ -20,6 +22,7 @@ type MockPeer struct {
writeDeadline <-chan time.Time writeDeadline <-chan time.Time
readDeadline <-chan time.Time readDeadline <-chan time.Time
RemoteQuit chan struct{}
Quit chan struct{} Quit chan struct{}
} }
@ -28,22 +31,62 @@ func NewMockPeer(pk *btcec.PublicKey, addr net.Addr, bufferSize int) *MockPeer {
return &MockPeer{ return &MockPeer{
remotePub: pk, remotePub: pk,
remoteAddr: addr, remoteAddr: addr,
localAddr: &net.TCPAddr{
IP: net.IP{0x32, 0x31, 0x30, 0x29},
Port: 36723,
},
IncomingMsgs: make(chan []byte, bufferSize), IncomingMsgs: make(chan []byte, bufferSize),
OutgoingMsgs: make(chan []byte, bufferSize), OutgoingMsgs: make(chan []byte, bufferSize),
Quit: make(chan struct{}), Quit: make(chan struct{}),
} }
} }
// NewMockConn establishes a bidirectional connection between two MockPeers.
func NewMockConn(localPk, remotePk *btcec.PublicKey,
localAddr, remoteAddr net.Addr,
bufferSize int) (*MockPeer, *MockPeer) {
localPeer := &MockPeer{
remotePub: remotePk,
remoteAddr: remoteAddr,
localPub: localPk,
localAddr: localAddr,
IncomingMsgs: make(chan []byte, bufferSize),
OutgoingMsgs: make(chan []byte, bufferSize),
Quit: make(chan struct{}),
}
remotePeer := &MockPeer{
remotePub: localPk,
remoteAddr: localAddr,
localPub: remotePk,
localAddr: remoteAddr,
IncomingMsgs: localPeer.OutgoingMsgs,
OutgoingMsgs: localPeer.IncomingMsgs,
Quit: make(chan struct{}),
}
localPeer.RemoteQuit = remotePeer.Quit
remotePeer.RemoteQuit = localPeer.Quit
return localPeer, remotePeer
}
// Write sends the raw bytes as the next full message read to the remote peer. // Write sends the raw bytes as the next full message read to the remote peer.
// The write will fail if either party closes the connection or the write // The write will fail if either party closes the connection or the write
// deadline expires. The passed bytes slice is copied before sending, thus the // deadline expires. The passed bytes slice is copied before sending, thus the
// bytes may be reused once the method returns. // bytes may be reused once the method returns.
func (p *MockPeer) Write(b []byte) (n int, err error) { func (p *MockPeer) Write(b []byte) (n int, err error) {
bb := make([]byte, len(b))
copy(bb, b)
select { select {
case p.OutgoingMsgs <- b: case p.OutgoingMsgs <- bb:
return len(b), nil return len(b), nil
case <-p.writeDeadline: case <-p.writeDeadline:
return 0, fmt.Errorf("write timeout expired") return 0, fmt.Errorf("write timeout expired")
case <-p.RemoteQuit:
return 0, fmt.Errorf("remote closed connected")
case <-p.Quit: case <-p.Quit:
return 0, fmt.Errorf("connection closed") return 0, fmt.Errorf("connection closed")
} }
@ -69,6 +112,8 @@ func (p *MockPeer) ReadNextMessage() ([]byte, error) {
return b, nil return b, nil
case <-p.readDeadline: case <-p.readDeadline:
return nil, fmt.Errorf("read timeout expired") return nil, fmt.Errorf("read timeout expired")
case <-p.RemoteQuit:
return nil, fmt.Errorf("remote closed connected")
case <-p.Quit: case <-p.Quit:
return nil, fmt.Errorf("connection closed") return nil, fmt.Errorf("connection closed")
} }
@ -112,6 +157,25 @@ func (p *MockPeer) RemoteAddr() net.Addr {
return p.remoteAddr return p.remoteAddr
} }
// LocalAddr returns the local net address of the peer.
func (p *MockPeer) LocalAddr() net.Addr {
return p.localAddr
}
// Read is not implemented.
func (p *MockPeer) Read(dst []byte) (int, error) {
panic("not implemented")
}
// SetDeadline is not implemented.
func (p *MockPeer) SetDeadline(t time.Time) error {
panic("not implemented")
}
// Compile-time constraint ensuring the MockPeer implements the wserver.Peer // Compile-time constraint ensuring the MockPeer implements the wserver.Peer
// interface. // interface.
var _ wtserver.Peer = (*MockPeer)(nil) var _ wtserver.Peer = (*MockPeer)(nil)
// Compile-time constraint ensuring the MockPeer implements the net.Conn
// interface.
var _ net.Conn = (*MockPeer)(nil)