members.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. /*
  2. Copyright 2014 CoreOS, Inc.
  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. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package client
  14. import (
  15. "bytes"
  16. "encoding/json"
  17. "fmt"
  18. "net/http"
  19. "net/url"
  20. "path"
  21. "time"
  22. "github.com/coreos/etcd/Godeps/_workspace/src/code.google.com/p/go.net/context"
  23. "github.com/coreos/etcd/etcdserver/etcdhttp/httptypes"
  24. "github.com/coreos/etcd/pkg/types"
  25. )
  26. var (
  27. DefaultV2MembersPrefix = "/v2/members"
  28. )
  29. func NewMembersAPI(tr *http.Transport, eps []string, to time.Duration) (MembersAPI, error) {
  30. c, err := newHTTPClusterClient(tr, eps)
  31. if err != nil {
  32. return nil, err
  33. }
  34. mAPI := httpMembersAPI{
  35. client: c,
  36. timeout: to,
  37. }
  38. return &mAPI, nil
  39. }
  40. type MembersAPI interface {
  41. List() ([]httptypes.Member, error)
  42. Add(peerURL string) (*httptypes.Member, error)
  43. Remove(mID string) error
  44. }
  45. type httpMembersAPI struct {
  46. client httpActionDo
  47. timeout time.Duration
  48. }
  49. func (m *httpMembersAPI) List() ([]httptypes.Member, error) {
  50. req := &membersAPIActionList{}
  51. ctx, cancel := context.WithTimeout(context.Background(), m.timeout)
  52. code, body, err := m.client.do(ctx, req)
  53. cancel()
  54. if err != nil {
  55. return nil, err
  56. }
  57. if err := assertStatusCode(http.StatusOK, code); err != nil {
  58. return nil, err
  59. }
  60. var mCollection httptypes.MemberCollection
  61. if err := json.Unmarshal(body, &mCollection); err != nil {
  62. return nil, err
  63. }
  64. return []httptypes.Member(mCollection), nil
  65. }
  66. func (m *httpMembersAPI) Add(peerURL string) (*httptypes.Member, error) {
  67. urls, err := types.NewURLs([]string{peerURL})
  68. if err != nil {
  69. return nil, err
  70. }
  71. req := &membersAPIActionAdd{peerURLs: urls}
  72. ctx, cancel := context.WithTimeout(context.Background(), m.timeout)
  73. code, body, err := m.client.do(ctx, req)
  74. cancel()
  75. if err != nil {
  76. return nil, err
  77. }
  78. if err := assertStatusCode(http.StatusCreated, code); err != nil {
  79. return nil, err
  80. }
  81. var memb httptypes.Member
  82. if err := json.Unmarshal(body, &memb); err != nil {
  83. return nil, err
  84. }
  85. return &memb, nil
  86. }
  87. func (m *httpMembersAPI) Remove(memberID string) error {
  88. req := &membersAPIActionRemove{memberID: memberID}
  89. ctx, cancel := context.WithTimeout(context.Background(), m.timeout)
  90. code, _, err := m.client.do(ctx, req)
  91. cancel()
  92. if err != nil {
  93. return err
  94. }
  95. return assertStatusCode(http.StatusNoContent, code)
  96. }
  97. type membersAPIActionList struct{}
  98. func (l *membersAPIActionList) httpRequest(ep url.URL) *http.Request {
  99. u := v2MembersURL(ep)
  100. req, _ := http.NewRequest("GET", u.String(), nil)
  101. return req
  102. }
  103. type membersAPIActionRemove struct {
  104. memberID string
  105. }
  106. func (d *membersAPIActionRemove) httpRequest(ep url.URL) *http.Request {
  107. u := v2MembersURL(ep)
  108. u.Path = path.Join(u.Path, d.memberID)
  109. req, _ := http.NewRequest("DELETE", u.String(), nil)
  110. return req
  111. }
  112. type membersAPIActionAdd struct {
  113. peerURLs types.URLs
  114. }
  115. func (a *membersAPIActionAdd) httpRequest(ep url.URL) *http.Request {
  116. u := v2MembersURL(ep)
  117. m := httptypes.MemberCreateRequest{PeerURLs: a.peerURLs}
  118. b, _ := json.Marshal(&m)
  119. req, _ := http.NewRequest("POST", u.String(), bytes.NewReader(b))
  120. req.Header.Set("Content-Type", "application/json")
  121. return req
  122. }
  123. func assertStatusCode(want, got int) (err error) {
  124. if want != got {
  125. err = fmt.Errorf("unexpected status code %d", got)
  126. }
  127. return err
  128. }
  129. // v2MembersURL add the necessary path to the provided endpoint
  130. // to route requests to the default v2 members API.
  131. func v2MembersURL(ep url.URL) *url.URL {
  132. ep.Path = path.Join(ep.Path, DefaultV2MembersPrefix)
  133. return &ep
  134. }