2016-10-25 15:20:06 +00:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httputil"
|
|
|
|
"net/url"
|
|
|
|
|
|
|
|
"github.com/influxdata/chronograf"
|
|
|
|
)
|
|
|
|
|
2016-10-28 16:27:06 +00:00
|
|
|
// ValidProxyRequest checks if queries specify a command.
|
2016-10-25 15:20:06 +00:00
|
|
|
func ValidProxyRequest(p chronograf.Query) error {
|
|
|
|
if p.Command == "" {
|
|
|
|
return fmt.Errorf("query field required")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type postProxyResponse struct {
|
|
|
|
Results interface{} `json:"results"` // results from influx
|
|
|
|
}
|
|
|
|
|
2016-10-28 16:27:06 +00:00
|
|
|
// Proxy proxies requests to infludb.
|
|
|
|
func (h *Service) Proxy(w http.ResponseWriter, r *http.Request) {
|
2016-10-25 15:20:06 +00:00
|
|
|
id, err := paramID("id", r)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, http.StatusUnprocessableEntity, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var req chronograf.Query
|
|
|
|
if err = json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
|
|
invalidJSON(w)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if err = ValidProxyRequest(req); err != nil {
|
|
|
|
invalidData(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx := r.Context()
|
2016-10-28 16:27:06 +00:00
|
|
|
src, err := h.SourcesStore.Get(ctx, id)
|
2016-10-25 15:20:06 +00:00
|
|
|
if err != nil {
|
|
|
|
notFound(w, id)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = h.TimeSeries.Connect(ctx, &src); err != nil {
|
|
|
|
msg := fmt.Sprintf("Unable to connect to source %d", id)
|
|
|
|
Error(w, http.StatusBadRequest, msg)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
response, err := h.TimeSeries.Query(ctx, req)
|
|
|
|
if err != nil {
|
|
|
|
if err == chronograf.ErrUpstreamTimeout {
|
|
|
|
msg := "Timeout waiting for Influx response"
|
|
|
|
Error(w, http.StatusRequestTimeout, msg)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// TODO: Here I want to return the error code from influx.
|
|
|
|
Error(w, http.StatusBadRequest, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
res := postProxyResponse{
|
|
|
|
Results: response,
|
|
|
|
}
|
|
|
|
encodeJSON(w, http.StatusOK, res, h.Logger)
|
|
|
|
}
|
|
|
|
|
2016-10-28 16:27:06 +00:00
|
|
|
// KapacitorProxy proxies requests to kapacitor using the path query parameter.
|
|
|
|
func (h *Service) KapacitorProxy(w http.ResponseWriter, r *http.Request) {
|
2016-10-25 15:20:06 +00:00
|
|
|
srcID, err := paramID("id", r)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, http.StatusUnprocessableEntity, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
id, err := paramID("kid", r)
|
|
|
|
if err != nil {
|
|
|
|
Error(w, http.StatusUnprocessableEntity, err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
path := r.URL.Query().Get("path")
|
|
|
|
if path == "" {
|
|
|
|
Error(w, http.StatusUnprocessableEntity, "path query parameter required")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx := r.Context()
|
|
|
|
srv, err := h.ServersStore.Get(ctx, id)
|
|
|
|
if err != nil || srv.SrcID != srcID {
|
|
|
|
notFound(w, id)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
u, err := url.Parse(srv.URL)
|
|
|
|
if err != nil {
|
|
|
|
msg := fmt.Sprintf("Error parsing kapacitor url: %v", err)
|
|
|
|
Error(w, http.StatusUnprocessableEntity, msg)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if srv.Username != "" && srv.Password != "" {
|
|
|
|
u.User = url.UserPassword(srv.Username, srv.Password)
|
|
|
|
}
|
|
|
|
u.Path = path
|
|
|
|
|
|
|
|
director := func(req *http.Request) {
|
|
|
|
req.URL = u
|
|
|
|
}
|
|
|
|
proxy := &httputil.ReverseProxy{
|
|
|
|
Director: director,
|
|
|
|
}
|
|
|
|
proxy.ServeHTTP(w, r)
|
|
|
|
}
|
|
|
|
|
2016-10-28 16:27:06 +00:00
|
|
|
// KapacitorProxyPost proxies POST to kapacitor
|
|
|
|
func (h *Service) KapacitorProxyPost(w http.ResponseWriter, r *http.Request) {
|
2016-10-25 15:20:06 +00:00
|
|
|
h.KapacitorProxy(w, r)
|
|
|
|
}
|
|
|
|
|
2016-10-28 16:27:06 +00:00
|
|
|
// KapacitorProxyPatch proxies PATCH to kapacitor
|
|
|
|
func (h *Service) KapacitorProxyPatch(w http.ResponseWriter, r *http.Request) {
|
2016-10-25 15:20:06 +00:00
|
|
|
h.KapacitorProxy(w, r)
|
|
|
|
}
|
|
|
|
|
2016-10-28 16:27:06 +00:00
|
|
|
// KapacitorProxyGet proxies GET to kapacitor
|
|
|
|
func (h *Service) KapacitorProxyGet(w http.ResponseWriter, r *http.Request) {
|
2016-10-25 15:20:06 +00:00
|
|
|
h.KapacitorProxy(w, r)
|
|
|
|
}
|
|
|
|
|
2016-10-28 16:27:06 +00:00
|
|
|
// KapacitorProxyDelete proxies DELETE to kapacitor
|
|
|
|
func (h *Service) KapacitorProxyDelete(w http.ResponseWriter, r *http.Request) {
|
2016-10-25 15:20:06 +00:00
|
|
|
h.KapacitorProxy(w, r)
|
|
|
|
}
|