2019-11-25 13:39:32 +00:00
|
|
|
package remote
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2019-12-04 10:05:35 +00:00
|
|
|
"context"
|
2020-01-09 02:46:29 +00:00
|
|
|
"io"
|
|
|
|
"net"
|
2019-11-25 13:39:32 +00:00
|
|
|
"testing"
|
2020-01-09 02:46:29 +00:00
|
|
|
"time"
|
2019-11-25 13:39:32 +00:00
|
|
|
|
2020-01-15 12:47:13 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/ethdb/codecpool"
|
2019-12-12 05:23:08 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
2019-12-02 13:47:00 +00:00
|
|
|
)
|
|
|
|
|
2020-01-09 02:46:29 +00:00
|
|
|
func TestReconnect(t *testing.T) {
|
2020-01-31 07:31:50 +00:00
|
|
|
assert := assert.New(t)
|
2020-01-09 02:46:29 +00:00
|
|
|
// Prepare input buffer with one command CmdVersion
|
|
|
|
var inBuf bytes.Buffer
|
2020-01-15 12:47:13 +00:00
|
|
|
encoder := codecpool.Encoder(&inBuf)
|
|
|
|
defer codecpool.Return(encoder)
|
2020-01-09 02:46:29 +00:00
|
|
|
// output buffer to receive the result of the command
|
|
|
|
var outBuf bytes.Buffer
|
2020-01-15 12:47:13 +00:00
|
|
|
decoder := codecpool.Decoder(&outBuf)
|
|
|
|
defer codecpool.Return(decoder)
|
2020-01-09 02:46:29 +00:00
|
|
|
|
|
|
|
dialCallCounter := 0
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
|
|
|
pingCh := make(chan time.Time, ClientMaxConnections)
|
2020-03-11 11:02:37 +00:00
|
|
|
opts := DefaultOpts
|
|
|
|
opts.DialFunc = func(ctx context.Context) (in io.Reader, out io.Writer, closer io.Closer, err error) {
|
|
|
|
dialCallCounter++
|
|
|
|
if dialCallCounter%2 == 0 {
|
|
|
|
return &inBuf, &outBuf, nil, net.UnknownNetworkError("Oops")
|
|
|
|
}
|
|
|
|
return &inBuf, &outBuf, nil, nil
|
|
|
|
}
|
|
|
|
|
2020-01-09 02:46:29 +00:00
|
|
|
db := &DB{
|
2020-03-11 11:02:37 +00:00
|
|
|
opts: opts,
|
2020-01-09 02:46:29 +00:00
|
|
|
connectionPool: make(chan *conn, ClientMaxConnections),
|
|
|
|
doDial: make(chan struct{}, ClientMaxConnections),
|
|
|
|
doPing: pingCh,
|
2019-12-22 15:10:46 +00:00
|
|
|
}
|
2020-01-09 02:46:29 +00:00
|
|
|
|
|
|
|
// no open connections by default
|
2020-01-31 07:31:50 +00:00
|
|
|
assert.Equal(0, dialCallCounter)
|
|
|
|
assert.Equal(0, len(db.connectionPool))
|
2020-01-09 02:46:29 +00:00
|
|
|
|
|
|
|
// open 1 connection and wait for it
|
|
|
|
db.doDial <- struct{}{}
|
|
|
|
db.autoReconnect(ctx)
|
|
|
|
<-db.connectionPool
|
2020-01-31 07:31:50 +00:00
|
|
|
assert.Equal(1, dialCallCounter)
|
|
|
|
assert.Equal(0, len(db.connectionPool))
|
2020-01-09 02:46:29 +00:00
|
|
|
|
|
|
|
// open 2nd connection - dialFunc will return err on 2nd call, but db must reconnect automatically
|
|
|
|
db.doDial <- struct{}{}
|
|
|
|
db.autoReconnect(ctx) // dial err
|
|
|
|
db.autoReconnect(ctx) // dial ok
|
|
|
|
<-db.connectionPool
|
2020-01-31 07:31:50 +00:00
|
|
|
assert.Equal(3, dialCallCounter)
|
|
|
|
assert.Equal(0, len(db.connectionPool))
|
2020-01-09 02:46:29 +00:00
|
|
|
|
|
|
|
// open conn and call ping on it
|
|
|
|
db.doDial <- struct{}{}
|
2020-01-31 07:31:50 +00:00
|
|
|
assert.Nil(encoder.Encode(ResponseOk))
|
|
|
|
assert.Nil(encoder.Encode(Version))
|
2020-01-09 02:46:29 +00:00
|
|
|
db.autoReconnect(ctx) // dial err
|
|
|
|
db.autoReconnect(ctx) // dial ok
|
2020-01-31 07:31:50 +00:00
|
|
|
assert.Equal(5, dialCallCounter)
|
|
|
|
assert.Equal(1, len(db.connectionPool))
|
2020-01-09 02:46:29 +00:00
|
|
|
pingCh <- time.Now()
|
|
|
|
db.autoReconnect(ctx)
|
|
|
|
var cmd Command
|
2020-01-31 07:31:50 +00:00
|
|
|
assert.Nil(decoder.Decode(&cmd))
|
|
|
|
assert.Equal(CmdVersion, cmd)
|
2020-01-09 02:46:29 +00:00
|
|
|
|
|
|
|
// TODO: cover case when ping receive io.EOF
|
2019-12-22 15:10:46 +00:00
|
|
|
}
|