cluster_test.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418
  1. // Copyright 2016 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 integration
  15. import (
  16. "context"
  17. "math/rand"
  18. "reflect"
  19. "strings"
  20. "testing"
  21. "time"
  22. "go.etcd.io/etcd/integration"
  23. "go.etcd.io/etcd/pkg/testutil"
  24. "go.etcd.io/etcd/pkg/types"
  25. )
  26. func TestMemberList(t *testing.T) {
  27. defer testutil.AfterTest(t)
  28. clus := integration.NewClusterV3(t, &integration.ClusterConfig{Size: 3})
  29. defer clus.Terminate(t)
  30. capi := clus.RandClient()
  31. resp, err := capi.MemberList(context.Background())
  32. if err != nil {
  33. t.Fatalf("failed to list member %v", err)
  34. }
  35. if len(resp.Members) != 3 {
  36. t.Errorf("number of members = %d, want %d", len(resp.Members), 3)
  37. }
  38. }
  39. func TestMemberAdd(t *testing.T) {
  40. defer testutil.AfterTest(t)
  41. clus := integration.NewClusterV3(t, &integration.ClusterConfig{Size: 3})
  42. defer clus.Terminate(t)
  43. capi := clus.RandClient()
  44. urls := []string{"http://127.0.0.1:1234"}
  45. resp, err := capi.MemberAdd(context.Background(), urls)
  46. if err != nil {
  47. t.Fatalf("failed to add member %v", err)
  48. }
  49. if !reflect.DeepEqual(resp.Member.PeerURLs, urls) {
  50. t.Errorf("urls = %v, want %v", urls, resp.Member.PeerURLs)
  51. }
  52. }
  53. func TestMemberAddWithExistingURLs(t *testing.T) {
  54. defer testutil.AfterTest(t)
  55. clus := integration.NewClusterV3(t, &integration.ClusterConfig{Size: 3})
  56. defer clus.Terminate(t)
  57. capi := clus.RandClient()
  58. resp, err := capi.MemberList(context.Background())
  59. if err != nil {
  60. t.Fatalf("failed to list member %v", err)
  61. }
  62. existingURL := resp.Members[0].PeerURLs[0]
  63. _, err = capi.MemberAdd(context.Background(), []string{existingURL})
  64. expectedErrKeywords := "Peer URLs already exists"
  65. if err == nil {
  66. t.Fatalf("expecting add member to fail, got no error")
  67. }
  68. if !strings.Contains(err.Error(), expectedErrKeywords) {
  69. t.Errorf("expecting error to contain %s, got %s", expectedErrKeywords, err.Error())
  70. }
  71. }
  72. func TestMemberRemove(t *testing.T) {
  73. defer testutil.AfterTest(t)
  74. clus := integration.NewClusterV3(t, &integration.ClusterConfig{Size: 3})
  75. defer clus.Terminate(t)
  76. capi := clus.Client(1)
  77. resp, err := capi.MemberList(context.Background())
  78. if err != nil {
  79. t.Fatalf("failed to list member %v", err)
  80. }
  81. rmvID := resp.Members[0].ID
  82. // indexes in capi member list don't necessarily match cluster member list;
  83. // find member that is not the client to remove
  84. for _, m := range resp.Members {
  85. mURLs, _ := types.NewURLs(m.PeerURLs)
  86. if !reflect.DeepEqual(mURLs, clus.Members[1].ServerConfig.PeerURLs) {
  87. rmvID = m.ID
  88. break
  89. }
  90. }
  91. _, err = capi.MemberRemove(context.Background(), rmvID)
  92. if err != nil {
  93. t.Fatalf("failed to remove member %v", err)
  94. }
  95. resp, err = capi.MemberList(context.Background())
  96. if err != nil {
  97. t.Fatalf("failed to list member %v", err)
  98. }
  99. if len(resp.Members) != 2 {
  100. t.Errorf("number of members = %d, want %d", len(resp.Members), 2)
  101. }
  102. }
  103. func TestMemberUpdate(t *testing.T) {
  104. defer testutil.AfterTest(t)
  105. clus := integration.NewClusterV3(t, &integration.ClusterConfig{Size: 3})
  106. defer clus.Terminate(t)
  107. capi := clus.RandClient()
  108. resp, err := capi.MemberList(context.Background())
  109. if err != nil {
  110. t.Fatalf("failed to list member %v", err)
  111. }
  112. urls := []string{"http://127.0.0.1:1234"}
  113. _, err = capi.MemberUpdate(context.Background(), resp.Members[0].ID, urls)
  114. if err != nil {
  115. t.Fatalf("failed to update member %v", err)
  116. }
  117. resp, err = capi.MemberList(context.Background())
  118. if err != nil {
  119. t.Fatalf("failed to list member %v", err)
  120. }
  121. if !reflect.DeepEqual(resp.Members[0].PeerURLs, urls) {
  122. t.Errorf("urls = %v, want %v", urls, resp.Members[0].PeerURLs)
  123. }
  124. }
  125. func TestMemberAddUpdateWrongURLs(t *testing.T) {
  126. defer testutil.AfterTest(t)
  127. clus := integration.NewClusterV3(t, &integration.ClusterConfig{Size: 1})
  128. defer clus.Terminate(t)
  129. capi := clus.RandClient()
  130. tt := [][]string{
  131. // missing protocol scheme
  132. {"://127.0.0.1:2379"},
  133. // unsupported scheme
  134. {"mailto://127.0.0.1:2379"},
  135. // not conform to host:port
  136. {"http://127.0.0.1"},
  137. // contain a path
  138. {"http://127.0.0.1:2379/path"},
  139. // first path segment in URL cannot contain colon
  140. {"127.0.0.1:1234"},
  141. // URL scheme must be http, https, unix, or unixs
  142. {"localhost:1234"},
  143. }
  144. for i := range tt {
  145. _, err := capi.MemberAdd(context.Background(), tt[i])
  146. if err == nil {
  147. t.Errorf("#%d: MemberAdd err = nil, but error", i)
  148. }
  149. _, err = capi.MemberUpdate(context.Background(), 0, tt[i])
  150. if err == nil {
  151. t.Errorf("#%d: MemberUpdate err = nil, but error", i)
  152. }
  153. }
  154. }
  155. func TestMemberAddForLearner(t *testing.T) {
  156. defer testutil.AfterTest(t)
  157. clus := integration.NewClusterV3(t, &integration.ClusterConfig{Size: 3})
  158. defer clus.Terminate(t)
  159. capi := clus.RandClient()
  160. urls := []string{"http://127.0.0.1:1234"}
  161. resp, err := capi.MemberAddAsLearner(context.Background(), urls)
  162. if err != nil {
  163. t.Fatalf("failed to add member %v", err)
  164. }
  165. if !resp.Member.IsLearner {
  166. t.Errorf("Added a member as learner, got resp.Member.IsLearner = %v", resp.Member.IsLearner)
  167. }
  168. numberOfLearners := 0
  169. for _, m := range resp.Members {
  170. if m.IsLearner {
  171. numberOfLearners++
  172. }
  173. }
  174. if numberOfLearners != 1 {
  175. t.Errorf("Added 1 learner node to cluster, got %d", numberOfLearners)
  176. }
  177. }
  178. func TestMemberPromote(t *testing.T) {
  179. defer testutil.AfterTest(t)
  180. clus := integration.NewClusterV3(t, &integration.ClusterConfig{Size: 3})
  181. defer clus.Terminate(t)
  182. // member promote request can be sent to any server in cluster,
  183. // the request will be auto-forwarded to leader on server-side.
  184. // This test explicitly includes the server-side forwarding by
  185. // sending the request to follower.
  186. leaderIdx := clus.WaitLeader(t)
  187. followerIdx := (leaderIdx + 1) % 3
  188. capi := clus.Client(followerIdx)
  189. urls := []string{"http://127.0.0.1:1234"}
  190. memberAddResp, err := capi.MemberAddAsLearner(context.Background(), urls)
  191. if err != nil {
  192. t.Fatalf("failed to add member %v", err)
  193. }
  194. if !memberAddResp.Member.IsLearner {
  195. t.Fatalf("Added a member as learner, got resp.Member.IsLearner = %v", memberAddResp.Member.IsLearner)
  196. }
  197. learnerID := memberAddResp.Member.ID
  198. numberOfLearners := 0
  199. for _, m := range memberAddResp.Members {
  200. if m.IsLearner {
  201. numberOfLearners++
  202. }
  203. }
  204. if numberOfLearners != 1 {
  205. t.Fatalf("Added 1 learner node to cluster, got %d", numberOfLearners)
  206. }
  207. // learner is not started yet. Expect learner progress check to fail.
  208. // As the result, member promote request will fail.
  209. _, err = capi.MemberPromote(context.Background(), learnerID)
  210. expectedErrKeywords := "can only promote a learner member which is in sync with leader"
  211. if err == nil {
  212. t.Fatalf("expecting promote not ready learner to fail, got no error")
  213. }
  214. if !strings.Contains(err.Error(), expectedErrKeywords) {
  215. t.Fatalf("expecting error to contain %s, got %s", expectedErrKeywords, err.Error())
  216. }
  217. // create and launch learner member based on the response of V3 Member Add API.
  218. // (the response has information on peer urls of the existing members in cluster)
  219. learnerMember := clus.MustNewMember(t, memberAddResp)
  220. clus.Members = append(clus.Members, learnerMember)
  221. if err := learnerMember.Launch(); err != nil {
  222. t.Fatal(err)
  223. }
  224. // retry until promote succeed or timeout
  225. timeout := time.After(5 * time.Second)
  226. for {
  227. select {
  228. case <-time.After(500 * time.Millisecond):
  229. case <-timeout:
  230. t.Fatalf("failed all attempts to promote learner member, last error: %v", err)
  231. }
  232. _, err = capi.MemberPromote(context.Background(), learnerID)
  233. // successfully promoted learner
  234. if err == nil {
  235. break
  236. }
  237. // if member promote fails due to learner not ready, retry.
  238. // otherwise fails the test.
  239. if !strings.Contains(err.Error(), expectedErrKeywords) {
  240. t.Fatalf("unexpected error when promoting learner member: %v", err)
  241. }
  242. }
  243. }
  244. // TestMemberPromoteMemberNotLearner ensures that promoting a voting member fails.
  245. func TestMemberPromoteMemberNotLearner(t *testing.T) {
  246. defer testutil.AfterTest(t)
  247. clus := integration.NewClusterV3(t, &integration.ClusterConfig{Size: 3})
  248. defer clus.Terminate(t)
  249. // member promote request can be sent to any server in cluster,
  250. // the request will be auto-forwarded to leader on server-side.
  251. // This test explicitly includes the server-side forwarding by
  252. // sending the request to follower.
  253. leaderIdx := clus.WaitLeader(t)
  254. followerIdx := (leaderIdx + 1) % 3
  255. cli := clus.Client(followerIdx)
  256. resp, err := cli.MemberList(context.Background())
  257. if err != nil {
  258. t.Fatalf("failed to list member %v", err)
  259. }
  260. if len(resp.Members) != 3 {
  261. t.Fatalf("number of members = %d, want %d", len(resp.Members), 3)
  262. }
  263. // promoting any of the voting members in cluster should fail
  264. expectedErrKeywords := "can only promote a learner member"
  265. for _, m := range resp.Members {
  266. _, err = cli.MemberPromote(context.Background(), m.ID)
  267. if err == nil {
  268. t.Fatalf("expect promoting voting member to fail, got no error")
  269. }
  270. if !strings.Contains(err.Error(), expectedErrKeywords) {
  271. t.Fatalf("expect error to contain %s, got %s", expectedErrKeywords, err.Error())
  272. }
  273. }
  274. }
  275. // TestMemberPromoteMemberNotExist ensures that promoting a member that does not exist in cluster fails.
  276. func TestMemberPromoteMemberNotExist(t *testing.T) {
  277. defer testutil.AfterTest(t)
  278. clus := integration.NewClusterV3(t, &integration.ClusterConfig{Size: 3})
  279. defer clus.Terminate(t)
  280. // member promote request can be sent to any server in cluster,
  281. // the request will be auto-forwarded to leader on server-side.
  282. // This test explicitly includes the server-side forwarding by
  283. // sending the request to follower.
  284. leaderIdx := clus.WaitLeader(t)
  285. followerIdx := (leaderIdx + 1) % 3
  286. cli := clus.Client(followerIdx)
  287. resp, err := cli.MemberList(context.Background())
  288. if err != nil {
  289. t.Fatalf("failed to list member %v", err)
  290. }
  291. if len(resp.Members) != 3 {
  292. t.Fatalf("number of members = %d, want %d", len(resp.Members), 3)
  293. }
  294. // generate an random ID that does not exist in cluster
  295. var randID uint64
  296. for {
  297. randID = rand.Uint64()
  298. notExist := true
  299. for _, m := range resp.Members {
  300. if m.ID == randID {
  301. notExist = false
  302. break
  303. }
  304. }
  305. if notExist {
  306. break
  307. }
  308. }
  309. expectedErrKeywords := "member not found"
  310. _, err = cli.MemberPromote(context.Background(), randID)
  311. if err == nil {
  312. t.Fatalf("expect promoting voting member to fail, got no error")
  313. }
  314. if !strings.Contains(err.Error(), expectedErrKeywords) {
  315. t.Errorf("expect error to contain %s, got %s", expectedErrKeywords, err.Error())
  316. }
  317. }
  318. // TestMaxLearnerInCluster verifies that the maximum number of learners allowed in a cluster is 1
  319. func TestMaxLearnerInCluster(t *testing.T) {
  320. defer testutil.AfterTest(t)
  321. // 1. start with a cluster with 3 voting member and 0 learner member
  322. clus := integration.NewClusterV3(t, &integration.ClusterConfig{Size: 3})
  323. defer clus.Terminate(t)
  324. // 2. adding a learner member should succeed
  325. resp1, err := clus.Client(0).MemberAddAsLearner(context.Background(), []string{"http://127.0.0.1:1234"})
  326. if err != nil {
  327. t.Fatalf("failed to add learner member %v", err)
  328. }
  329. numberOfLearners := 0
  330. for _, m := range resp1.Members {
  331. if m.IsLearner {
  332. numberOfLearners++
  333. }
  334. }
  335. if numberOfLearners != 1 {
  336. t.Fatalf("Added 1 learner node to cluster, got %d", numberOfLearners)
  337. }
  338. // 3. cluster has 3 voting member and 1 learner, adding another learner should fail
  339. _, err = clus.Client(0).MemberAddAsLearner(context.Background(), []string{"http://127.0.0.1:2345"})
  340. if err == nil {
  341. t.Fatalf("expect member add to fail, got no error")
  342. }
  343. expectedErrKeywords := "too many learner members in cluster"
  344. if !strings.Contains(err.Error(), expectedErrKeywords) {
  345. t.Fatalf("expecting error to contain %s, got %s", expectedErrKeywords, err.Error())
  346. }
  347. // 4. cluster has 3 voting member and 1 learner, adding a voting member should succeed
  348. _, err = clus.Client(0).MemberAdd(context.Background(), []string{"http://127.0.0.1:3456"})
  349. if err != nil {
  350. t.Errorf("failed to add member %v", err)
  351. }
  352. }