cluster_test.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736
  1. // Copyright 2015 The etcd Authors
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package membership
  15. import (
  16. "encoding/json"
  17. "fmt"
  18. "path"
  19. "reflect"
  20. "testing"
  21. "go.etcd.io/etcd/etcdserver/api/v2store"
  22. "go.etcd.io/etcd/pkg/mock/mockstore"
  23. "go.etcd.io/etcd/pkg/testutil"
  24. "go.etcd.io/etcd/pkg/types"
  25. "go.etcd.io/etcd/raft/raftpb"
  26. "go.uber.org/zap"
  27. )
  28. func TestClusterMember(t *testing.T) {
  29. membs := []*Member{
  30. newTestMember(1, nil, "node1", nil),
  31. newTestMember(2, nil, "node2", nil),
  32. }
  33. tests := []struct {
  34. id types.ID
  35. match bool
  36. }{
  37. {1, true},
  38. {2, true},
  39. {3, false},
  40. }
  41. for i, tt := range tests {
  42. c := newTestCluster(membs)
  43. m := c.Member(tt.id)
  44. if g := m != nil; g != tt.match {
  45. t.Errorf("#%d: find member = %v, want %v", i, g, tt.match)
  46. }
  47. if m != nil && m.ID != tt.id {
  48. t.Errorf("#%d: id = %x, want %x", i, m.ID, tt.id)
  49. }
  50. }
  51. }
  52. func TestClusterMemberByName(t *testing.T) {
  53. membs := []*Member{
  54. newTestMember(1, nil, "node1", nil),
  55. newTestMember(2, nil, "node2", nil),
  56. }
  57. tests := []struct {
  58. name string
  59. match bool
  60. }{
  61. {"node1", true},
  62. {"node2", true},
  63. {"node3", false},
  64. }
  65. for i, tt := range tests {
  66. c := newTestCluster(membs)
  67. m := c.MemberByName(tt.name)
  68. if g := m != nil; g != tt.match {
  69. t.Errorf("#%d: find member = %v, want %v", i, g, tt.match)
  70. }
  71. if m != nil && m.Name != tt.name {
  72. t.Errorf("#%d: name = %v, want %v", i, m.Name, tt.name)
  73. }
  74. }
  75. }
  76. func TestClusterMemberIDs(t *testing.T) {
  77. c := newTestCluster([]*Member{
  78. newTestMember(1, nil, "", nil),
  79. newTestMember(4, nil, "", nil),
  80. newTestMember(100, nil, "", nil),
  81. })
  82. w := []types.ID{1, 4, 100}
  83. g := c.MemberIDs()
  84. if !reflect.DeepEqual(w, g) {
  85. t.Errorf("IDs = %+v, want %+v", g, w)
  86. }
  87. }
  88. func TestClusterPeerURLs(t *testing.T) {
  89. tests := []struct {
  90. mems []*Member
  91. wurls []string
  92. }{
  93. // single peer with a single address
  94. {
  95. mems: []*Member{
  96. newTestMember(1, []string{"http://192.0.2.1"}, "", nil),
  97. },
  98. wurls: []string{"http://192.0.2.1"},
  99. },
  100. // single peer with a single address with a port
  101. {
  102. mems: []*Member{
  103. newTestMember(1, []string{"http://192.0.2.1:8001"}, "", nil),
  104. },
  105. wurls: []string{"http://192.0.2.1:8001"},
  106. },
  107. // several members explicitly unsorted
  108. {
  109. mems: []*Member{
  110. newTestMember(2, []string{"http://192.0.2.3", "http://192.0.2.4"}, "", nil),
  111. newTestMember(3, []string{"http://192.0.2.5", "http://192.0.2.6"}, "", nil),
  112. newTestMember(1, []string{"http://192.0.2.1", "http://192.0.2.2"}, "", nil),
  113. },
  114. wurls: []string{"http://192.0.2.1", "http://192.0.2.2", "http://192.0.2.3", "http://192.0.2.4", "http://192.0.2.5", "http://192.0.2.6"},
  115. },
  116. // no members
  117. {
  118. mems: []*Member{},
  119. wurls: []string{},
  120. },
  121. // peer with no peer urls
  122. {
  123. mems: []*Member{
  124. newTestMember(3, []string{}, "", nil),
  125. },
  126. wurls: []string{},
  127. },
  128. }
  129. for i, tt := range tests {
  130. c := newTestCluster(tt.mems)
  131. urls := c.PeerURLs()
  132. if !reflect.DeepEqual(urls, tt.wurls) {
  133. t.Errorf("#%d: PeerURLs = %v, want %v", i, urls, tt.wurls)
  134. }
  135. }
  136. }
  137. func TestClusterClientURLs(t *testing.T) {
  138. tests := []struct {
  139. mems []*Member
  140. wurls []string
  141. }{
  142. // single peer with a single address
  143. {
  144. mems: []*Member{
  145. newTestMember(1, nil, "", []string{"http://192.0.2.1"}),
  146. },
  147. wurls: []string{"http://192.0.2.1"},
  148. },
  149. // single peer with a single address with a port
  150. {
  151. mems: []*Member{
  152. newTestMember(1, nil, "", []string{"http://192.0.2.1:8001"}),
  153. },
  154. wurls: []string{"http://192.0.2.1:8001"},
  155. },
  156. // several members explicitly unsorted
  157. {
  158. mems: []*Member{
  159. newTestMember(2, nil, "", []string{"http://192.0.2.3", "http://192.0.2.4"}),
  160. newTestMember(3, nil, "", []string{"http://192.0.2.5", "http://192.0.2.6"}),
  161. newTestMember(1, nil, "", []string{"http://192.0.2.1", "http://192.0.2.2"}),
  162. },
  163. wurls: []string{"http://192.0.2.1", "http://192.0.2.2", "http://192.0.2.3", "http://192.0.2.4", "http://192.0.2.5", "http://192.0.2.6"},
  164. },
  165. // no members
  166. {
  167. mems: []*Member{},
  168. wurls: []string{},
  169. },
  170. // peer with no client urls
  171. {
  172. mems: []*Member{
  173. newTestMember(3, nil, "", []string{}),
  174. },
  175. wurls: []string{},
  176. },
  177. }
  178. for i, tt := range tests {
  179. c := newTestCluster(tt.mems)
  180. urls := c.ClientURLs()
  181. if !reflect.DeepEqual(urls, tt.wurls) {
  182. t.Errorf("#%d: ClientURLs = %v, want %v", i, urls, tt.wurls)
  183. }
  184. }
  185. }
  186. func TestClusterValidateAndAssignIDsBad(t *testing.T) {
  187. tests := []struct {
  188. clmembs []*Member
  189. membs []*Member
  190. }{
  191. {
  192. // unmatched length
  193. []*Member{
  194. newTestMember(1, []string{"http://127.0.0.1:2379"}, "", nil),
  195. },
  196. []*Member{},
  197. },
  198. {
  199. // unmatched peer urls
  200. []*Member{
  201. newTestMember(1, []string{"http://127.0.0.1:2379"}, "", nil),
  202. },
  203. []*Member{
  204. newTestMember(1, []string{"http://127.0.0.1:4001"}, "", nil),
  205. },
  206. },
  207. {
  208. // unmatched peer urls
  209. []*Member{
  210. newTestMember(1, []string{"http://127.0.0.1:2379"}, "", nil),
  211. newTestMember(2, []string{"http://127.0.0.2:2379"}, "", nil),
  212. },
  213. []*Member{
  214. newTestMember(1, []string{"http://127.0.0.1:2379"}, "", nil),
  215. newTestMember(2, []string{"http://127.0.0.2:4001"}, "", nil),
  216. },
  217. },
  218. }
  219. for i, tt := range tests {
  220. ecl := newTestCluster(tt.clmembs)
  221. lcl := newTestCluster(tt.membs)
  222. if err := ValidateClusterAndAssignIDs(zap.NewExample(), lcl, ecl); err == nil {
  223. t.Errorf("#%d: unexpected update success", i)
  224. }
  225. }
  226. }
  227. func TestClusterValidateAndAssignIDs(t *testing.T) {
  228. tests := []struct {
  229. clmembs []*Member
  230. membs []*Member
  231. wids []types.ID
  232. }{
  233. {
  234. []*Member{
  235. newTestMember(1, []string{"http://127.0.0.1:2379"}, "", nil),
  236. newTestMember(2, []string{"http://127.0.0.2:2379"}, "", nil),
  237. },
  238. []*Member{
  239. newTestMember(3, []string{"http://127.0.0.1:2379"}, "", nil),
  240. newTestMember(4, []string{"http://127.0.0.2:2379"}, "", nil),
  241. },
  242. []types.ID{3, 4},
  243. },
  244. }
  245. for i, tt := range tests {
  246. lcl := newTestCluster(tt.clmembs)
  247. ecl := newTestCluster(tt.membs)
  248. if err := ValidateClusterAndAssignIDs(zap.NewExample(), lcl, ecl); err != nil {
  249. t.Errorf("#%d: unexpect update error: %v", i, err)
  250. }
  251. if !reflect.DeepEqual(lcl.MemberIDs(), tt.wids) {
  252. t.Errorf("#%d: ids = %v, want %v", i, lcl.MemberIDs(), tt.wids)
  253. }
  254. }
  255. }
  256. func TestClusterValidateConfigurationChange(t *testing.T) {
  257. cl := NewCluster(zap.NewExample(), "")
  258. cl.SetStore(v2store.New())
  259. for i := 1; i <= 4; i++ {
  260. attr := RaftAttributes{PeerURLs: []string{fmt.Sprintf("http://127.0.0.1:%d", i)}}
  261. cl.AddMember(&Member{ID: types.ID(i), RaftAttributes: attr})
  262. }
  263. cl.RemoveMember(4)
  264. attr := RaftAttributes{PeerURLs: []string{fmt.Sprintf("http://127.0.0.1:%d", 1)}}
  265. ctx, err := json.Marshal(&Member{ID: types.ID(5), RaftAttributes: attr})
  266. if err != nil {
  267. t.Fatal(err)
  268. }
  269. attr = RaftAttributes{PeerURLs: []string{fmt.Sprintf("http://127.0.0.1:%d", 5)}}
  270. ctx5, err := json.Marshal(&Member{ID: types.ID(5), RaftAttributes: attr})
  271. if err != nil {
  272. t.Fatal(err)
  273. }
  274. attr = RaftAttributes{PeerURLs: []string{fmt.Sprintf("http://127.0.0.1:%d", 3)}}
  275. ctx2to3, err := json.Marshal(&Member{ID: types.ID(2), RaftAttributes: attr})
  276. if err != nil {
  277. t.Fatal(err)
  278. }
  279. attr = RaftAttributes{PeerURLs: []string{fmt.Sprintf("http://127.0.0.1:%d", 5)}}
  280. ctx2to5, err := json.Marshal(&Member{ID: types.ID(2), RaftAttributes: attr})
  281. if err != nil {
  282. t.Fatal(err)
  283. }
  284. tests := []struct {
  285. cc raftpb.ConfChange
  286. werr error
  287. }{
  288. {
  289. raftpb.ConfChange{
  290. Type: raftpb.ConfChangeRemoveNode,
  291. NodeID: 3,
  292. },
  293. nil,
  294. },
  295. {
  296. raftpb.ConfChange{
  297. Type: raftpb.ConfChangeAddNode,
  298. NodeID: 4,
  299. },
  300. ErrIDRemoved,
  301. },
  302. {
  303. raftpb.ConfChange{
  304. Type: raftpb.ConfChangeRemoveNode,
  305. NodeID: 4,
  306. },
  307. ErrIDRemoved,
  308. },
  309. {
  310. raftpb.ConfChange{
  311. Type: raftpb.ConfChangeAddNode,
  312. NodeID: 1,
  313. },
  314. ErrIDExists,
  315. },
  316. {
  317. raftpb.ConfChange{
  318. Type: raftpb.ConfChangeAddNode,
  319. NodeID: 5,
  320. Context: ctx,
  321. },
  322. ErrPeerURLexists,
  323. },
  324. {
  325. raftpb.ConfChange{
  326. Type: raftpb.ConfChangeRemoveNode,
  327. NodeID: 5,
  328. },
  329. ErrIDNotFound,
  330. },
  331. {
  332. raftpb.ConfChange{
  333. Type: raftpb.ConfChangeAddNode,
  334. NodeID: 5,
  335. Context: ctx5,
  336. },
  337. nil,
  338. },
  339. {
  340. raftpb.ConfChange{
  341. Type: raftpb.ConfChangeUpdateNode,
  342. NodeID: 5,
  343. Context: ctx,
  344. },
  345. ErrIDNotFound,
  346. },
  347. // try to change the peer url of 2 to the peer url of 3
  348. {
  349. raftpb.ConfChange{
  350. Type: raftpb.ConfChangeUpdateNode,
  351. NodeID: 2,
  352. Context: ctx2to3,
  353. },
  354. ErrPeerURLexists,
  355. },
  356. {
  357. raftpb.ConfChange{
  358. Type: raftpb.ConfChangeUpdateNode,
  359. NodeID: 2,
  360. Context: ctx2to5,
  361. },
  362. nil,
  363. },
  364. }
  365. for i, tt := range tests {
  366. err := cl.ValidateConfigurationChange(tt.cc)
  367. if err != tt.werr {
  368. t.Errorf("#%d: validateConfigurationChange error = %v, want %v", i, err, tt.werr)
  369. }
  370. }
  371. }
  372. func TestClusterGenID(t *testing.T) {
  373. cs := newTestCluster([]*Member{
  374. newTestMember(1, nil, "", nil),
  375. newTestMember(2, nil, "", nil),
  376. })
  377. cs.genID()
  378. if cs.ID() == 0 {
  379. t.Fatalf("cluster.ID = %v, want not 0", cs.ID())
  380. }
  381. previd := cs.ID()
  382. cs.SetStore(mockstore.NewNop())
  383. cs.AddMember(newTestMember(3, nil, "", nil))
  384. cs.genID()
  385. if cs.ID() == previd {
  386. t.Fatalf("cluster.ID = %v, want not %v", cs.ID(), previd)
  387. }
  388. }
  389. func TestNodeToMemberBad(t *testing.T) {
  390. tests := []*v2store.NodeExtern{
  391. {Key: "/1234", Nodes: []*v2store.NodeExtern{
  392. {Key: "/1234/strange"},
  393. }},
  394. {Key: "/1234", Nodes: []*v2store.NodeExtern{
  395. {Key: "/1234/raftAttributes", Value: stringp("garbage")},
  396. }},
  397. {Key: "/1234", Nodes: []*v2store.NodeExtern{
  398. {Key: "/1234/attributes", Value: stringp(`{"name":"node1","clientURLs":null}`)},
  399. }},
  400. {Key: "/1234", Nodes: []*v2store.NodeExtern{
  401. {Key: "/1234/raftAttributes", Value: stringp(`{"peerURLs":null}`)},
  402. {Key: "/1234/strange"},
  403. }},
  404. {Key: "/1234", Nodes: []*v2store.NodeExtern{
  405. {Key: "/1234/raftAttributes", Value: stringp(`{"peerURLs":null}`)},
  406. {Key: "/1234/attributes", Value: stringp("garbage")},
  407. }},
  408. {Key: "/1234", Nodes: []*v2store.NodeExtern{
  409. {Key: "/1234/raftAttributes", Value: stringp(`{"peerURLs":null}`)},
  410. {Key: "/1234/attributes", Value: stringp(`{"name":"node1","clientURLs":null}`)},
  411. {Key: "/1234/strange"},
  412. }},
  413. }
  414. for i, tt := range tests {
  415. if _, err := nodeToMember(tt); err == nil {
  416. t.Errorf("#%d: unexpected nil error", i)
  417. }
  418. }
  419. }
  420. func TestClusterAddMember(t *testing.T) {
  421. st := mockstore.NewRecorder()
  422. c := newTestCluster(nil)
  423. c.SetStore(st)
  424. c.AddMember(newTestMember(1, nil, "node1", nil))
  425. wactions := []testutil.Action{
  426. {
  427. Name: "Create",
  428. Params: []interface{}{
  429. path.Join(StoreMembersPrefix, "1", "raftAttributes"),
  430. false,
  431. `{"peerURLs":null}`,
  432. false,
  433. v2store.TTLOptionSet{ExpireTime: v2store.Permanent},
  434. },
  435. },
  436. }
  437. if g := st.Action(); !reflect.DeepEqual(g, wactions) {
  438. t.Errorf("actions = %v, want %v", g, wactions)
  439. }
  440. }
  441. func TestClusterMembers(t *testing.T) {
  442. cls := &RaftCluster{
  443. members: map[types.ID]*Member{
  444. 1: {ID: 1},
  445. 20: {ID: 20},
  446. 100: {ID: 100},
  447. 5: {ID: 5},
  448. 50: {ID: 50},
  449. },
  450. }
  451. w := []*Member{
  452. {ID: 1},
  453. {ID: 5},
  454. {ID: 20},
  455. {ID: 50},
  456. {ID: 100},
  457. }
  458. if g := cls.Members(); !reflect.DeepEqual(g, w) {
  459. t.Fatalf("Members()=%#v, want %#v", g, w)
  460. }
  461. }
  462. func TestClusterRemoveMember(t *testing.T) {
  463. st := mockstore.NewRecorder()
  464. c := newTestCluster(nil)
  465. c.SetStore(st)
  466. c.RemoveMember(1)
  467. wactions := []testutil.Action{
  468. {Name: "Delete", Params: []interface{}{MemberStoreKey(1), true, true}},
  469. {Name: "Create", Params: []interface{}{RemovedMemberStoreKey(1), false, "", false, v2store.TTLOptionSet{ExpireTime: v2store.Permanent}}},
  470. }
  471. if !reflect.DeepEqual(st.Action(), wactions) {
  472. t.Errorf("actions = %v, want %v", st.Action(), wactions)
  473. }
  474. }
  475. func TestClusterUpdateAttributes(t *testing.T) {
  476. name := "etcd"
  477. clientURLs := []string{"http://127.0.0.1:4001"}
  478. tests := []struct {
  479. mems []*Member
  480. removed map[types.ID]bool
  481. wmems []*Member
  482. }{
  483. // update attributes of existing member
  484. {
  485. []*Member{
  486. newTestMember(1, nil, "", nil),
  487. },
  488. nil,
  489. []*Member{
  490. newTestMember(1, nil, name, clientURLs),
  491. },
  492. },
  493. // update attributes of removed member
  494. {
  495. nil,
  496. map[types.ID]bool{types.ID(1): true},
  497. nil,
  498. },
  499. }
  500. for i, tt := range tests {
  501. c := newTestCluster(tt.mems)
  502. c.removed = tt.removed
  503. c.UpdateAttributes(types.ID(1), Attributes{Name: name, ClientURLs: clientURLs})
  504. if g := c.Members(); !reflect.DeepEqual(g, tt.wmems) {
  505. t.Errorf("#%d: members = %+v, want %+v", i, g, tt.wmems)
  506. }
  507. }
  508. }
  509. func TestNodeToMember(t *testing.T) {
  510. n := &v2store.NodeExtern{Key: "/1234", Nodes: []*v2store.NodeExtern{
  511. {Key: "/1234/attributes", Value: stringp(`{"name":"node1","clientURLs":null}`)},
  512. {Key: "/1234/raftAttributes", Value: stringp(`{"peerURLs":null}`)},
  513. }}
  514. wm := &Member{ID: 0x1234, RaftAttributes: RaftAttributes{}, Attributes: Attributes{Name: "node1"}}
  515. m, err := nodeToMember(n)
  516. if err != nil {
  517. t.Fatalf("unexpected nodeToMember error: %v", err)
  518. }
  519. if !reflect.DeepEqual(m, wm) {
  520. t.Errorf("member = %+v, want %+v", m, wm)
  521. }
  522. }
  523. func newTestCluster(membs []*Member) *RaftCluster {
  524. c := &RaftCluster{lg: zap.NewExample(), members: make(map[types.ID]*Member), removed: make(map[types.ID]bool)}
  525. for _, m := range membs {
  526. c.members[m.ID] = m
  527. }
  528. return c
  529. }
  530. func stringp(s string) *string { return &s }
  531. func TestIsReadyToAddNewMember(t *testing.T) {
  532. tests := []struct {
  533. members []*Member
  534. want bool
  535. }{
  536. {
  537. // 0/3 members ready, should fail
  538. []*Member{
  539. newTestMember(1, nil, "", nil),
  540. newTestMember(2, nil, "", nil),
  541. newTestMember(3, nil, "", nil),
  542. },
  543. false,
  544. },
  545. {
  546. // 1/2 members ready, should fail
  547. []*Member{
  548. newTestMember(1, nil, "1", nil),
  549. newTestMember(2, nil, "", nil),
  550. },
  551. false,
  552. },
  553. {
  554. // 1/3 members ready, should fail
  555. []*Member{
  556. newTestMember(1, nil, "1", nil),
  557. newTestMember(2, nil, "", nil),
  558. newTestMember(3, nil, "", nil),
  559. },
  560. false,
  561. },
  562. {
  563. // 1/1 members ready, should succeed (special case of 1-member cluster for recovery)
  564. []*Member{
  565. newTestMember(1, nil, "1", nil),
  566. },
  567. true,
  568. },
  569. {
  570. // 2/3 members ready, should fail
  571. []*Member{
  572. newTestMember(1, nil, "1", nil),
  573. newTestMember(2, nil, "2", nil),
  574. newTestMember(3, nil, "", nil),
  575. },
  576. false,
  577. },
  578. {
  579. // 3/3 members ready, should be fine to add one member and retain quorum
  580. []*Member{
  581. newTestMember(1, nil, "1", nil),
  582. newTestMember(2, nil, "2", nil),
  583. newTestMember(3, nil, "3", nil),
  584. },
  585. true,
  586. },
  587. {
  588. // 3/4 members ready, should be fine to add one member and retain quorum
  589. []*Member{
  590. newTestMember(1, nil, "1", nil),
  591. newTestMember(2, nil, "2", nil),
  592. newTestMember(3, nil, "3", nil),
  593. newTestMember(4, nil, "", nil),
  594. },
  595. true,
  596. },
  597. {
  598. // empty cluster, it is impossible but should fail
  599. []*Member{},
  600. false,
  601. },
  602. }
  603. for i, tt := range tests {
  604. c := newTestCluster(tt.members)
  605. if got := c.IsReadyToAddNewMember(); got != tt.want {
  606. t.Errorf("%d: isReadyToAddNewMember returned %t, want %t", i, got, tt.want)
  607. }
  608. }
  609. }
  610. func TestIsReadyToRemoveMember(t *testing.T) {
  611. tests := []struct {
  612. members []*Member
  613. removeID uint64
  614. want bool
  615. }{
  616. {
  617. // 1/1 members ready, should fail
  618. []*Member{
  619. newTestMember(1, nil, "1", nil),
  620. },
  621. 1,
  622. false,
  623. },
  624. {
  625. // 0/3 members ready, should fail
  626. []*Member{
  627. newTestMember(1, nil, "", nil),
  628. newTestMember(2, nil, "", nil),
  629. newTestMember(3, nil, "", nil),
  630. },
  631. 1,
  632. false,
  633. },
  634. {
  635. // 1/2 members ready, should be fine to remove unstarted member
  636. // (isReadyToRemoveMember() logic should return success, but operation itself would fail)
  637. []*Member{
  638. newTestMember(1, nil, "1", nil),
  639. newTestMember(2, nil, "", nil),
  640. },
  641. 2,
  642. true,
  643. },
  644. {
  645. // 2/3 members ready, should fail
  646. []*Member{
  647. newTestMember(1, nil, "1", nil),
  648. newTestMember(2, nil, "2", nil),
  649. newTestMember(3, nil, "", nil),
  650. },
  651. 2,
  652. false,
  653. },
  654. {
  655. // 3/3 members ready, should be fine to remove one member and retain quorum
  656. []*Member{
  657. newTestMember(1, nil, "1", nil),
  658. newTestMember(2, nil, "2", nil),
  659. newTestMember(3, nil, "3", nil),
  660. },
  661. 3,
  662. true,
  663. },
  664. {
  665. // 3/4 members ready, should be fine to remove one member
  666. []*Member{
  667. newTestMember(1, nil, "1", nil),
  668. newTestMember(2, nil, "2", nil),
  669. newTestMember(3, nil, "3", nil),
  670. newTestMember(4, nil, "", nil),
  671. },
  672. 3,
  673. true,
  674. },
  675. {
  676. // 3/4 members ready, should be fine to remove unstarted member
  677. []*Member{
  678. newTestMember(1, nil, "1", nil),
  679. newTestMember(2, nil, "2", nil),
  680. newTestMember(3, nil, "3", nil),
  681. newTestMember(4, nil, "", nil),
  682. },
  683. 4,
  684. true,
  685. },
  686. }
  687. for i, tt := range tests {
  688. c := newTestCluster(tt.members)
  689. if got := c.IsReadyToRemoveMember(tt.removeID); got != tt.want {
  690. t.Errorf("%d: isReadyToAddNewMember returned %t, want %t", i, got, tt.want)
  691. }
  692. }
  693. }