v2_server.go 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  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 etcdserver
  15. import (
  16. "context"
  17. "time"
  18. pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
  19. "github.com/coreos/etcd/etcdserver/v2store"
  20. )
  21. type RequestV2 pb.Request
  22. type RequestV2Handler interface {
  23. Post(ctx context.Context, r *RequestV2) (Response, error)
  24. Put(ctx context.Context, r *RequestV2) (Response, error)
  25. Delete(ctx context.Context, r *RequestV2) (Response, error)
  26. QGet(ctx context.Context, r *RequestV2) (Response, error)
  27. Get(ctx context.Context, r *RequestV2) (Response, error)
  28. Head(ctx context.Context, r *RequestV2) (Response, error)
  29. }
  30. type reqV2HandlerEtcdServer struct {
  31. reqV2HandlerStore
  32. s *EtcdServer
  33. }
  34. type reqV2HandlerStore struct {
  35. store v2store.Store
  36. applier ApplierV2
  37. }
  38. func NewStoreRequestV2Handler(s v2store.Store, applier ApplierV2) RequestV2Handler {
  39. return &reqV2HandlerStore{s, applier}
  40. }
  41. func (a *reqV2HandlerStore) Post(ctx context.Context, r *RequestV2) (Response, error) {
  42. return a.applier.Post(r), nil
  43. }
  44. func (a *reqV2HandlerStore) Put(ctx context.Context, r *RequestV2) (Response, error) {
  45. return a.applier.Put(r), nil
  46. }
  47. func (a *reqV2HandlerStore) Delete(ctx context.Context, r *RequestV2) (Response, error) {
  48. return a.applier.Delete(r), nil
  49. }
  50. func (a *reqV2HandlerStore) QGet(ctx context.Context, r *RequestV2) (Response, error) {
  51. return a.applier.QGet(r), nil
  52. }
  53. func (a *reqV2HandlerStore) Get(ctx context.Context, r *RequestV2) (Response, error) {
  54. if r.Wait {
  55. wc, err := a.store.Watch(r.Path, r.Recursive, r.Stream, r.Since)
  56. return Response{Watcher: wc}, err
  57. }
  58. ev, err := a.store.Get(r.Path, r.Recursive, r.Sorted)
  59. return Response{Event: ev}, err
  60. }
  61. func (a *reqV2HandlerStore) Head(ctx context.Context, r *RequestV2) (Response, error) {
  62. ev, err := a.store.Get(r.Path, r.Recursive, r.Sorted)
  63. return Response{Event: ev}, err
  64. }
  65. func (a *reqV2HandlerEtcdServer) Post(ctx context.Context, r *RequestV2) (Response, error) {
  66. return a.processRaftRequest(ctx, r)
  67. }
  68. func (a *reqV2HandlerEtcdServer) Put(ctx context.Context, r *RequestV2) (Response, error) {
  69. return a.processRaftRequest(ctx, r)
  70. }
  71. func (a *reqV2HandlerEtcdServer) Delete(ctx context.Context, r *RequestV2) (Response, error) {
  72. return a.processRaftRequest(ctx, r)
  73. }
  74. func (a *reqV2HandlerEtcdServer) QGet(ctx context.Context, r *RequestV2) (Response, error) {
  75. return a.processRaftRequest(ctx, r)
  76. }
  77. func (a *reqV2HandlerEtcdServer) processRaftRequest(ctx context.Context, r *RequestV2) (Response, error) {
  78. data, err := ((*pb.Request)(r)).Marshal()
  79. if err != nil {
  80. return Response{}, err
  81. }
  82. ch := a.s.w.Register(r.ID)
  83. start := time.Now()
  84. err = a.s.r.Propose(ctx, data)
  85. if err != nil {
  86. proposalsFailed.Inc()
  87. a.s.w.Trigger(r.ID, nil)
  88. return Response{}, err
  89. }
  90. proposalsPending.Inc()
  91. defer proposalsPending.Dec()
  92. select {
  93. case x := <-ch:
  94. resp := x.(Response)
  95. return resp, resp.Err
  96. case <-ctx.Done():
  97. proposalsFailed.Inc()
  98. a.s.w.Trigger(r.ID, nil) // GC wait
  99. return Response{}, a.s.parseProposeCtxErr(ctx.Err(), start)
  100. case <-a.s.stopping:
  101. }
  102. return Response{}, ErrStopped
  103. }
  104. func (s *EtcdServer) Do(ctx context.Context, r pb.Request) (Response, error) {
  105. r.ID = s.reqIDGen.Next()
  106. h := &reqV2HandlerEtcdServer{
  107. reqV2HandlerStore: reqV2HandlerStore{
  108. store: s.v2store,
  109. applier: s.applyV2,
  110. },
  111. s: s,
  112. }
  113. rp := &r
  114. resp, err := ((*RequestV2)(rp)).Handle(ctx, h)
  115. resp.Term, resp.Index = s.Term(), s.CommittedIndex()
  116. return resp, err
  117. }
  118. // Handle interprets r and performs an operation on s.store according to r.Method
  119. // and other fields. If r.Method is "POST", "PUT", "DELETE", or a "GET" with
  120. // Quorum == true, r will be sent through consensus before performing its
  121. // respective operation. Do will block until an action is performed or there is
  122. // an error.
  123. func (r *RequestV2) Handle(ctx context.Context, v2api RequestV2Handler) (Response, error) {
  124. if r.Method == "GET" && r.Quorum {
  125. r.Method = "QGET"
  126. }
  127. switch r.Method {
  128. case "POST":
  129. return v2api.Post(ctx, r)
  130. case "PUT":
  131. return v2api.Put(ctx, r)
  132. case "DELETE":
  133. return v2api.Delete(ctx, r)
  134. case "QGET":
  135. return v2api.QGet(ctx, r)
  136. case "GET":
  137. return v2api.Get(ctx, r)
  138. case "HEAD":
  139. return v2api.Head(ctx, r)
  140. }
  141. return Response{}, ErrUnknownMethod
  142. }
  143. func (r *RequestV2) String() string {
  144. rpb := pb.Request(*r)
  145. return rpb.String()
  146. }