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

This change makes the etcd package compatible with the existing Go ecosystem for module versioning. Used this tool to update package imports: https://github.com/KSubedi/gomove
95 lines
2.5 KiB
Go
95 lines
2.5 KiB
Go
// Copyright 2019 The etcd Authors
|
|
//
|
|
// 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 rafttest
|
|
|
|
import (
|
|
"bufio"
|
|
"fmt"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/cockroachdb/datadriven"
|
|
"go.etcd.io/etcd/v3/raft/raftpb"
|
|
)
|
|
|
|
func (env *InteractionEnv) handleStabilize(t *testing.T, d datadriven.TestData) error {
|
|
var idxs []int
|
|
for _, id := range ints(t, d) {
|
|
idxs = append(idxs, id-1)
|
|
}
|
|
return env.Stabilize(idxs...)
|
|
}
|
|
|
|
// Stabilize repeatedly runs Ready handling on and message delivery to the set
|
|
// of nodes specified via the idxs slice until reaching a fixed point.
|
|
func (env *InteractionEnv) Stabilize(idxs ...int) error {
|
|
var nodes []Node
|
|
for _, idx := range idxs {
|
|
nodes = append(nodes, env.Nodes[idx])
|
|
}
|
|
if len(nodes) == 0 {
|
|
nodes = env.Nodes
|
|
}
|
|
|
|
withIndent := func(f func()) {
|
|
orig := env.Output.Builder
|
|
env.Output.Builder = &strings.Builder{}
|
|
f()
|
|
|
|
scanner := bufio.NewScanner(strings.NewReader(env.Output.Builder.String()))
|
|
for scanner.Scan() {
|
|
orig.WriteString(" " + scanner.Text() + "\n")
|
|
}
|
|
env.Output.Builder = orig
|
|
}
|
|
|
|
for {
|
|
done := true
|
|
for _, rn := range nodes {
|
|
if rn.HasReady() {
|
|
done = false
|
|
idx := int(rn.Status().ID - 1)
|
|
fmt.Fprintf(env.Output, "> %d handling Ready\n", idx+1)
|
|
withIndent(func() { env.ProcessReady(idx) })
|
|
}
|
|
}
|
|
for _, rn := range nodes {
|
|
id := rn.Status().ID
|
|
// NB: we grab the messages just to see whether to print the header.
|
|
// DeliverMsgs will do it again.
|
|
if msgs, _ := splitMsgs(env.Messages, id); len(msgs) > 0 {
|
|
fmt.Fprintf(env.Output, "> %d receiving messages\n", id)
|
|
withIndent(func() { env.DeliverMsgs(Recipient{ID: id}) })
|
|
done = false
|
|
}
|
|
}
|
|
if done {
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
|
|
func splitMsgs(msgs []raftpb.Message, to uint64) (toMsgs []raftpb.Message, rmdr []raftpb.Message) {
|
|
// NB: this method does not reorder messages.
|
|
for _, msg := range msgs {
|
|
if msg.To == to {
|
|
toMsgs = append(toMsgs, msg)
|
|
} else {
|
|
rmdr = append(rmdr, msg)
|
|
}
|
|
}
|
|
return toMsgs, rmdr
|
|
}
|