http.go 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345
  1. package etcdhttp
  2. import (
  3. "encoding/binary"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "io"
  8. "io/ioutil"
  9. "log"
  10. "net/http"
  11. "net/url"
  12. "strconv"
  13. "strings"
  14. "time"
  15. crand "crypto/rand"
  16. "github.com/coreos/etcd/elog"
  17. etcdErr "github.com/coreos/etcd/error"
  18. "github.com/coreos/etcd/etcdserver"
  19. "github.com/coreos/etcd/etcdserver/etcdserverpb"
  20. "github.com/coreos/etcd/raft/raftpb"
  21. "github.com/coreos/etcd/store"
  22. "github.com/coreos/etcd/third_party/code.google.com/p/go.net/context"
  23. )
  24. const (
  25. keysPrefix = "/v2/keys"
  26. machinesPrefix = "/v2/machines"
  27. raftPrefix = "/raft"
  28. DefaultTimeout = 500 * time.Millisecond
  29. )
  30. var errClosed = errors.New("etcdhttp: client closed connection")
  31. // NewHandler generates a muxed http.Handler with the given parameters.
  32. func NewHandler(server etcdserver.Server, peers Peers, timeout time.Duration) http.Handler {
  33. sh := &serverHandler{
  34. timeout: timeout,
  35. server: server,
  36. peers: peers,
  37. }
  38. if sh.timeout == 0 {
  39. sh.timeout = DefaultTimeout
  40. }
  41. mux := http.NewServeMux()
  42. mux.HandleFunc(raftPrefix, sh.serveRaft)
  43. mux.HandleFunc(keysPrefix, sh.serveKeys)
  44. mux.HandleFunc(keysPrefix+"/", sh.serveKeys)
  45. // TODO: dynamic configuration may make this outdated. take care of it.
  46. // TODO: dynamic configuration may introduce race also.
  47. mux.HandleFunc(machinesPrefix, sh.serveMachines)
  48. mux.HandleFunc("/", http.NotFound)
  49. return mux
  50. }
  51. // serverHandler provides http.Handlers for etcd client and raft communication.
  52. type serverHandler struct {
  53. timeout time.Duration
  54. server etcdserver.Server
  55. peers Peers
  56. }
  57. func (h serverHandler) serveKeys(w http.ResponseWriter, r *http.Request) {
  58. if !allowMethod(w, r.Method, "GET", "PUT", "POST", "DELETE") {
  59. return
  60. }
  61. ctx, cancel := context.WithTimeout(context.Background(), h.timeout)
  62. defer cancel()
  63. rr, err := parseRequest(r, genID())
  64. if err != nil {
  65. writeError(w, err)
  66. return
  67. }
  68. resp, err := h.server.Do(ctx, rr)
  69. if err != nil {
  70. writeError(w, err)
  71. return
  72. }
  73. var ev *store.Event
  74. switch {
  75. case resp.Event != nil:
  76. ev = resp.Event
  77. case resp.Watcher != nil:
  78. if ev, err = waitForEvent(ctx, w, resp.Watcher); err != nil {
  79. http.Error(w, err.Error(), http.StatusGatewayTimeout)
  80. return
  81. }
  82. default:
  83. writeError(w, errors.New("received response with no Event/Watcher!"))
  84. return
  85. }
  86. if err = writeEvent(w, ev); err != nil {
  87. // Should never be reached
  88. log.Println("error writing event: %v", err)
  89. }
  90. }
  91. // serveMachines responds address list in the format '0.0.0.0, 1.1.1.1'.
  92. // TODO: rethink the format of machine list because it is not json format.
  93. func (h serverHandler) serveMachines(w http.ResponseWriter, r *http.Request) {
  94. if !allowMethod(w, r.Method, "GET", "HEAD") {
  95. return
  96. }
  97. endpoints := h.peers.Endpoints()
  98. w.Write([]byte(strings.Join(endpoints, ", ")))
  99. }
  100. func (h serverHandler) serveRaft(w http.ResponseWriter, r *http.Request) {
  101. if !allowMethod(w, r.Method, "POST") {
  102. return
  103. }
  104. b, err := ioutil.ReadAll(r.Body)
  105. if err != nil {
  106. log.Println("etcdhttp: error reading raft message:", err)
  107. http.Error(w, "error reading raft message", http.StatusBadRequest)
  108. return
  109. }
  110. var m raftpb.Message
  111. if err := m.Unmarshal(b); err != nil {
  112. log.Println("etcdhttp: error unmarshaling raft message:", err)
  113. http.Error(w, "error unmarshaling raft message", http.StatusBadRequest)
  114. return
  115. }
  116. log.Printf("etcdhttp: raft recv message from %#x: %+v", m.From, m)
  117. if err := h.server.Process(context.TODO(), m); err != nil {
  118. log.Println("etcdhttp: error processing raft message:", err)
  119. writeError(w, err)
  120. return
  121. }
  122. w.WriteHeader(http.StatusNoContent)
  123. }
  124. // genID generates a random id that is: n < 0 < n.
  125. func genID() int64 {
  126. for {
  127. b := make([]byte, 8)
  128. if _, err := io.ReadFull(crand.Reader, b); err != nil {
  129. panic(err) // really bad stuff happened
  130. }
  131. n := int64(binary.BigEndian.Uint64(b))
  132. if n != 0 {
  133. return n
  134. }
  135. }
  136. }
  137. // parseRequest converts a received http.Request to a server Request,
  138. // performing validation of supplied fields as appropriate.
  139. // If any validation fails, an empty Request and non-nil error is returned.
  140. func parseRequest(r *http.Request, id int64) (etcdserverpb.Request, error) {
  141. emptyReq := etcdserverpb.Request{}
  142. err := r.ParseForm()
  143. if err != nil {
  144. return emptyReq, etcdErr.NewRequestError(
  145. etcdErr.EcodeInvalidForm,
  146. err.Error(),
  147. )
  148. }
  149. if !strings.HasPrefix(r.URL.Path, keysPrefix) {
  150. return emptyReq, etcdErr.NewRequestError(
  151. etcdErr.EcodeInvalidForm,
  152. "incorrect key prefix",
  153. )
  154. }
  155. p := r.URL.Path[len(keysPrefix):]
  156. var pIdx, wIdx, ttl uint64
  157. if pIdx, err = getUint64(r.Form, "prevIndex"); err != nil {
  158. return emptyReq, etcdErr.NewRequestError(
  159. etcdErr.EcodeIndexNaN,
  160. `invalid value for "prevIndex"`,
  161. )
  162. }
  163. if wIdx, err = getUint64(r.Form, "waitIndex"); err != nil {
  164. return emptyReq, etcdErr.NewRequestError(
  165. etcdErr.EcodeIndexNaN,
  166. `invalid value for "waitIndex"`,
  167. )
  168. }
  169. if ttl, err = getUint64(r.Form, "ttl"); err != nil {
  170. return emptyReq, etcdErr.NewRequestError(
  171. etcdErr.EcodeTTLNaN,
  172. `invalid value for "ttl"`,
  173. )
  174. }
  175. var rec, sort, wait bool
  176. if rec, err = getBool(r.Form, "recursive"); err != nil {
  177. return emptyReq, etcdErr.NewRequestError(
  178. etcdErr.EcodeInvalidField,
  179. `invalid value for "recursive"`,
  180. )
  181. }
  182. if sort, err = getBool(r.Form, "sorted"); err != nil {
  183. return emptyReq, etcdErr.NewRequestError(
  184. etcdErr.EcodeInvalidField,
  185. `invalid value for "sorted"`,
  186. )
  187. }
  188. if wait, err = getBool(r.Form, "wait"); err != nil {
  189. return emptyReq, etcdErr.NewRequestError(
  190. etcdErr.EcodeInvalidField,
  191. `invalid value for "wait"`,
  192. )
  193. }
  194. // prevExists is nullable, so leave it null if not specified
  195. var pe *bool
  196. if _, ok := r.Form["prevExists"]; ok {
  197. bv, err := getBool(r.Form, "prevExists")
  198. if err != nil {
  199. return emptyReq, etcdErr.NewRequestError(
  200. etcdErr.EcodeInvalidField,
  201. "invalid value for prevExists",
  202. )
  203. }
  204. pe = &bv
  205. }
  206. rr := etcdserverpb.Request{
  207. Id: id,
  208. Method: r.Method,
  209. Path: p,
  210. Val: r.FormValue("value"),
  211. PrevValue: r.FormValue("prevValue"),
  212. PrevIndex: pIdx,
  213. PrevExists: pe,
  214. Recursive: rec,
  215. Since: wIdx,
  216. Sorted: sort,
  217. Wait: wait,
  218. }
  219. if pe != nil {
  220. rr.PrevExists = pe
  221. }
  222. // TODO(jonboulle): use fake clock instead of time module
  223. // https://github.com/coreos/etcd/issues/1021
  224. if ttl > 0 {
  225. expr := time.Duration(ttl) * time.Second
  226. rr.Expiration = time.Now().Add(expr).UnixNano()
  227. }
  228. return rr, nil
  229. }
  230. // getUint64 extracts a uint64 by the given key from a Form. If the key does
  231. // not exist in the form, 0 is returned. If the key exists but the value is
  232. // badly formed, an error is returned. If multiple values are present only the
  233. // first is considered.
  234. func getUint64(form url.Values, key string) (i uint64, err error) {
  235. if vals, ok := form[key]; ok {
  236. i, err = strconv.ParseUint(vals[0], 10, 64)
  237. }
  238. return
  239. }
  240. // getBool extracts a bool by the given key from a Form. If the key does not
  241. // exist in the form, false is returned. If the key exists but the value is
  242. // badly formed, an error is returned. If multiple values are present only the
  243. // first is considered.
  244. func getBool(form url.Values, key string) (b bool, err error) {
  245. if vals, ok := form[key]; ok {
  246. b, err = strconv.ParseBool(vals[0])
  247. }
  248. return
  249. }
  250. // writeError logs and writes the given Error to the ResponseWriter
  251. // If Error is an etcdErr, it is rendered to the ResponseWriter
  252. // Otherwise, it is assumed to be an InternalServerError
  253. func writeError(w http.ResponseWriter, err error) {
  254. if err == nil {
  255. return
  256. }
  257. log.Println(err)
  258. if e, ok := err.(*etcdErr.Error); ok {
  259. e.Write(w)
  260. } else {
  261. http.Error(w, "Internal Server Error", http.StatusInternalServerError)
  262. }
  263. }
  264. // writeEvent serializes the given Event and writes the resulting JSON to the
  265. // given ResponseWriter
  266. func writeEvent(w http.ResponseWriter, ev *store.Event) error {
  267. if ev == nil {
  268. return errors.New("cannot write empty Event!")
  269. }
  270. w.Header().Set("Content-Type", "application/json")
  271. w.Header().Add("X-Etcd-Index", fmt.Sprint(ev.Index()))
  272. if ev.IsCreated() {
  273. w.WriteHeader(http.StatusCreated)
  274. }
  275. return json.NewEncoder(w).Encode(ev)
  276. }
  277. // waitForEvent waits for a given Watcher to return its associated
  278. // event. It returns a non-nil error if the given Context times out
  279. // or the given ResponseWriter triggers a CloseNotify.
  280. func waitForEvent(ctx context.Context, w http.ResponseWriter, wa store.Watcher) (*store.Event, error) {
  281. // TODO(bmizerany): support streaming?
  282. defer wa.Remove()
  283. var nch <-chan bool
  284. if x, ok := w.(http.CloseNotifier); ok {
  285. nch = x.CloseNotify()
  286. }
  287. select {
  288. case ev := <-wa.EventChan():
  289. return ev, nil
  290. case <-nch:
  291. elog.TODO()
  292. return nil, errClosed
  293. case <-ctx.Done():
  294. return nil, ctx.Err()
  295. }
  296. }
  297. // allowMethod verifies that the given method is one of the allowed methods,
  298. // and if not, it writes an error to w. A boolean is returned indicating
  299. // whether or not the method is allowed.
  300. func allowMethod(w http.ResponseWriter, m string, ms ...string) bool {
  301. for _, meth := range ms {
  302. if m == meth {
  303. return true
  304. }
  305. }
  306. w.Header().Set("Allow", strings.Join(ms, ","))
  307. http.Error(w, "Method Not Allowed", http.StatusMethodNotAllowed)
  308. return false
  309. }