get_handler.go 2.4 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. package v2
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "net/http"
  6. "strconv"
  7. etcdErr "github.com/coreos/etcd/error"
  8. "github.com/coreos/etcd/log"
  9. "github.com/coreos/etcd/store"
  10. "github.com/coreos/go-raft"
  11. "github.com/gorilla/mux"
  12. )
  13. func GetHandler(w http.ResponseWriter, req *http.Request, s Server) error {
  14. var err error
  15. var event *store.Event
  16. events := make([]*store.Event, 0)
  17. vars := mux.Vars(req)
  18. key := "/" + vars["key"]
  19. // Help client to redirect the request to the current leader
  20. if req.FormValue("consistent") == "true" && s.State() != raft.Leader {
  21. leader := s.Leader()
  22. hostname, _ := s.PeerURL(leader)
  23. url := hostname + req.URL.Path
  24. log.Debugf("Redirect consistent get to %s", url)
  25. http.Redirect(w, req, url, http.StatusTemporaryRedirect)
  26. return nil
  27. }
  28. recursive := (req.FormValue("recursive") == "true")
  29. sorted := (req.FormValue("sorted") == "true")
  30. if req.FormValue("wait") == "true" { // watch
  31. // Create a command to watch from a given index (default 0).
  32. var sinceIndex uint64 = 0
  33. waitIndex := req.FormValue("waitIndex")
  34. if waitIndex != "" {
  35. sinceIndex, err = strconv.ParseUint(string(req.FormValue("waitIndex")), 10, 64)
  36. if err != nil {
  37. return etcdErr.NewError(etcdErr.EcodeIndexNaN, "Watch From Index", store.UndefIndex, store.UndefTerm)
  38. }
  39. }
  40. // Start the watcher on the store.
  41. eventChan, err := s.Store().Watch(key, recursive, sinceIndex, s.CommitIndex(), s.Term())
  42. if err != nil {
  43. return etcdErr.NewError(500, key, store.UndefIndex, store.UndefTerm)
  44. }
  45. cn, _ := w.(http.CloseNotifier)
  46. closeChan := cn.CloseNotify()
  47. eventLoop:
  48. for {
  49. select {
  50. case <-closeChan:
  51. return nil
  52. case event = <-eventChan:
  53. // for events other than expire, just one event for one watch
  54. // for expire event, we might have a stream of events
  55. // we use a nil item to terminate the expire event stream
  56. if event != nil && event.Action == store.Expire {
  57. events = append(events, event)
  58. } else {
  59. events = append(events, event)
  60. break eventLoop
  61. }
  62. }
  63. }
  64. } else { //get
  65. // Retrieve the key from the store.
  66. event, err = s.Store().Get(key, recursive, sorted, s.CommitIndex(), s.Term())
  67. if err != nil {
  68. return err
  69. }
  70. }
  71. var b []byte
  72. w.Header().Add("X-Etcd-Index", fmt.Sprint(events[0].Index))
  73. w.Header().Add("X-Etcd-Term", fmt.Sprint(events[0].Term))
  74. w.WriteHeader(http.StatusOK)
  75. b, _ = json.Marshal(events)
  76. w.Write(b)
  77. return nil
  78. }