mirror of
https://github.com/etcd-io/etcd.git
synced 2024-09-27 06:25:44 +00:00

Use snapshotSender to send v3 snapshot message. It puts raft snapshot message and v3 snapshot into request body, then sends it to the target peer. When it receives http.StatusNoContent, it knows the message has been received and processed successfully. As receiver, snapHandler saves v3 snapshot and then processes the raft snapshot message, then respond with http.StatusNoContent.
171 lines
4.0 KiB
Go
171 lines
4.0 KiB
Go
// Copyright 2015 CoreOS, Inc.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package rafthttp
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/coreos/etcd/etcdserver/stats"
|
|
"github.com/coreos/etcd/pkg/httputil"
|
|
"github.com/coreos/etcd/pkg/pbutil"
|
|
"github.com/coreos/etcd/pkg/types"
|
|
"github.com/coreos/etcd/raft"
|
|
"github.com/coreos/etcd/raft/raftpb"
|
|
)
|
|
|
|
const (
|
|
connPerPipeline = 4
|
|
// pipelineBufSize is the size of pipeline buffer, which helps hold the
|
|
// temporary network latency.
|
|
// The size ensures that pipeline does not drop messages when the network
|
|
// is out of work for less than 1 second in good path.
|
|
pipelineBufSize = 64
|
|
)
|
|
|
|
var errStopped = errors.New("stopped")
|
|
|
|
type pipeline struct {
|
|
from, to types.ID
|
|
cid types.ID
|
|
|
|
tr http.RoundTripper
|
|
picker *urlPicker
|
|
status *peerStatus
|
|
fs *stats.FollowerStats
|
|
r Raft
|
|
errorc chan error
|
|
|
|
msgc chan raftpb.Message
|
|
// wait for the handling routines
|
|
wg sync.WaitGroup
|
|
stopc chan struct{}
|
|
}
|
|
|
|
func newPipeline(tr http.RoundTripper, picker *urlPicker, from, to, cid types.ID, status *peerStatus, fs *stats.FollowerStats, r Raft, errorc chan error) *pipeline {
|
|
p := &pipeline{
|
|
from: from,
|
|
to: to,
|
|
cid: cid,
|
|
tr: tr,
|
|
picker: picker,
|
|
status: status,
|
|
fs: fs,
|
|
r: r,
|
|
errorc: errorc,
|
|
stopc: make(chan struct{}),
|
|
msgc: make(chan raftpb.Message, pipelineBufSize),
|
|
}
|
|
p.wg.Add(connPerPipeline)
|
|
for i := 0; i < connPerPipeline; i++ {
|
|
go p.handle()
|
|
}
|
|
return p
|
|
}
|
|
|
|
func (p *pipeline) stop() {
|
|
close(p.msgc)
|
|
close(p.stopc)
|
|
p.wg.Wait()
|
|
}
|
|
|
|
func (p *pipeline) handle() {
|
|
defer p.wg.Done()
|
|
for m := range p.msgc {
|
|
start := time.Now()
|
|
err := p.post(pbutil.MustMarshal(&m))
|
|
if err == errStopped {
|
|
return
|
|
}
|
|
end := time.Now()
|
|
|
|
if err != nil {
|
|
reportSentFailure(pipelineMsg, m)
|
|
p.status.deactivate(failureType{source: pipelineMsg, action: "write"}, err.Error())
|
|
if m.Type == raftpb.MsgApp && p.fs != nil {
|
|
p.fs.Fail()
|
|
}
|
|
p.r.ReportUnreachable(m.To)
|
|
if isMsgSnap(m) {
|
|
p.r.ReportSnapshot(m.To, raft.SnapshotFailure)
|
|
}
|
|
} else {
|
|
p.status.activate()
|
|
if m.Type == raftpb.MsgApp && p.fs != nil {
|
|
p.fs.Succ(end.Sub(start))
|
|
}
|
|
if isMsgSnap(m) {
|
|
p.r.ReportSnapshot(m.To, raft.SnapshotFinish)
|
|
}
|
|
reportSentDuration(pipelineMsg, m, time.Since(start))
|
|
}
|
|
}
|
|
}
|
|
|
|
// post POSTs a data payload to a url. Returns nil if the POST succeeds,
|
|
// error on any failure.
|
|
func (p *pipeline) post(data []byte) (err error) {
|
|
u := p.picker.pick()
|
|
req := createPostRequest(u, RaftPrefix, bytes.NewBuffer(data), "application/protobuf", p.from, p.cid)
|
|
|
|
var stopped bool
|
|
defer func() {
|
|
if stopped {
|
|
// rewrite to errStopped so the caller goroutine can stop itself
|
|
err = errStopped
|
|
}
|
|
}()
|
|
done := make(chan struct{}, 1)
|
|
cancel := httputil.RequestCanceler(p.tr, req)
|
|
go func() {
|
|
select {
|
|
case <-done:
|
|
case <-p.stopc:
|
|
waitSchedule()
|
|
stopped = true
|
|
cancel()
|
|
}
|
|
}()
|
|
|
|
resp, err := p.tr.RoundTrip(req)
|
|
done <- struct{}{}
|
|
if err != nil {
|
|
p.picker.unreachable(u)
|
|
return err
|
|
}
|
|
b, err := ioutil.ReadAll(resp.Body)
|
|
if err != nil {
|
|
p.picker.unreachable(u)
|
|
return err
|
|
}
|
|
resp.Body.Close()
|
|
|
|
err = checkPostResponse(resp, b, req, p.to)
|
|
// errMemberRemoved is a critical error since a removed member should
|
|
// always be stopped. So we use reportCriticalError to report it to errorc.
|
|
if err == errMemberRemoved {
|
|
reportCriticalError(err, p.errorc)
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
|
|
// waitSchedule waits other goroutines to be scheduled for a while
|
|
func waitSchedule() { time.Sleep(time.Millisecond) }
|