mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-10 10:16:24 +00:00
lib/protocol: Don't send anything else before cluster config (#5741)
This commit is contained in:
parent
129df0613b
commit
3775a64d5c
@ -184,6 +184,7 @@ type rawConnection struct {
|
|||||||
|
|
||||||
inbox chan message
|
inbox chan message
|
||||||
outbox chan asyncMessage
|
outbox chan asyncMessage
|
||||||
|
clusterConfigBox chan *ClusterConfig
|
||||||
dispatcherLoopStopped chan struct{}
|
dispatcherLoopStopped chan struct{}
|
||||||
closed chan struct{}
|
closed chan struct{}
|
||||||
closeOnce sync.Once
|
closeOnce sync.Once
|
||||||
@ -230,6 +231,7 @@ func NewConnection(deviceID DeviceID, reader io.Reader, writer io.Writer, receiv
|
|||||||
awaiting: make(map[int32]chan asyncResult),
|
awaiting: make(map[int32]chan asyncResult),
|
||||||
inbox: make(chan message),
|
inbox: make(chan message),
|
||||||
outbox: make(chan asyncMessage),
|
outbox: make(chan asyncMessage),
|
||||||
|
clusterConfigBox: make(chan *ClusterConfig),
|
||||||
dispatcherLoopStopped: make(chan struct{}),
|
dispatcherLoopStopped: make(chan struct{}),
|
||||||
closed: make(chan struct{}),
|
closed: make(chan struct{}),
|
||||||
compression: compress,
|
compression: compress,
|
||||||
@ -327,9 +329,11 @@ func (c *rawConnection) Request(folder string, name string, offset int64, size i
|
|||||||
return res.val, res.err
|
return res.val, res.err
|
||||||
}
|
}
|
||||||
|
|
||||||
// ClusterConfig send the cluster configuration message to the peer and returns any error
|
// ClusterConfig sends the cluster configuration message to the peer.
|
||||||
|
// It must be called just once (as per BEP), otherwise it will panic.
|
||||||
func (c *rawConnection) ClusterConfig(config ClusterConfig) {
|
func (c *rawConnection) ClusterConfig(config ClusterConfig) {
|
||||||
c.send(&config, nil)
|
c.clusterConfigBox <- &config
|
||||||
|
close(c.clusterConfigBox)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *rawConnection) Closed() bool {
|
func (c *rawConnection) Closed() bool {
|
||||||
@ -657,6 +661,16 @@ func (c *rawConnection) send(msg message, done chan struct{}) bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (c *rawConnection) writerLoop() {
|
func (c *rawConnection) writerLoop() {
|
||||||
|
select {
|
||||||
|
case cc := <-c.clusterConfigBox:
|
||||||
|
err := c.writeMessage(cc)
|
||||||
|
if err != nil {
|
||||||
|
c.internalClose(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
case <-c.closed:
|
||||||
|
return
|
||||||
|
}
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case hm := <-c.outbox:
|
case hm := <-c.outbox:
|
||||||
|
@ -172,6 +172,48 @@ func TestCloseRace(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestClusterConfigFirst(t *testing.T) {
|
||||||
|
m := newTestModel()
|
||||||
|
|
||||||
|
c := NewConnection(c0ID, &testutils.BlockingRW{}, &testutils.NoopRW{}, m, "name", CompressAlways).(wireFormatConnection).Connection.(*rawConnection)
|
||||||
|
c.Start()
|
||||||
|
|
||||||
|
select {
|
||||||
|
case c.outbox <- asyncMessage{&Ping{}, nil}:
|
||||||
|
t.Fatal("able to send ping before cluster config")
|
||||||
|
case <-time.After(100 * time.Millisecond):
|
||||||
|
// Allow some time for c.writerLoop to setup after c.Start
|
||||||
|
}
|
||||||
|
|
||||||
|
c.ClusterConfig(ClusterConfig{})
|
||||||
|
|
||||||
|
done := make(chan struct{})
|
||||||
|
if ok := c.send(&Ping{}, done); !ok {
|
||||||
|
t.Fatal("send ping after cluster config returned false")
|
||||||
|
}
|
||||||
|
select {
|
||||||
|
case <-done:
|
||||||
|
case <-time.After(time.Second):
|
||||||
|
t.Fatal("timed out before ping was sent")
|
||||||
|
}
|
||||||
|
|
||||||
|
done = make(chan struct{})
|
||||||
|
go func() {
|
||||||
|
c.internalClose(errManual)
|
||||||
|
close(done)
|
||||||
|
}()
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-done:
|
||||||
|
case <-time.After(5 * time.Second):
|
||||||
|
t.Fatal("Close didn't return before timeout")
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := m.closedError(); err != errManual {
|
||||||
|
t.Fatal("Connection should be closed")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestMarshalIndexMessage(t *testing.T) {
|
func TestMarshalIndexMessage(t *testing.T) {
|
||||||
if testing.Short() {
|
if testing.Short() {
|
||||||
quickCfg.MaxCount = 10
|
quickCfg.MaxCount = 10
|
||||||
|
Loading…
Reference in New Issue
Block a user