influxdb/http/query_handler_test.go

688 lines
19 KiB
Go
Raw Normal View History

package http
import (
"bytes"
"context"
"encoding/json"
"fmt"
"io"
"io/ioutil"
"net/http"
"net/http/httptest"
"reflect"
"regexp"
"strings"
"testing"
"time"
"github.com/google/go-cmp/cmp"
"github.com/influxdata/flux"
"github.com/influxdata/flux/csv"
"github.com/influxdata/flux/lang"
"github.com/influxdata/influxdb"
2019-09-18 20:19:51 +00:00
platform "github.com/influxdata/influxdb"
icontext "github.com/influxdata/influxdb/context"
"github.com/influxdata/influxdb/http/metric"
"github.com/influxdata/influxdb/kit/check"
tracetesting "github.com/influxdata/influxdb/kit/tracing/testing"
kithttp "github.com/influxdata/influxdb/kit/transport/http"
influxmock "github.com/influxdata/influxdb/mock"
"github.com/influxdata/influxdb/query"
"github.com/influxdata/influxdb/query/mock"
"go.uber.org/zap/zaptest"
)
func TestFluxService_Query(t *testing.T) {
orgID, err := influxdb.IDFromString("abcdabcdabcdabcd")
if err != nil {
t.Fatal(err)
}
tests := []struct {
name string
token string
ctx context.Context
r *query.ProxyRequest
status int
want flux.Statistics
wantW string
wantErr bool
}{
{
name: "query",
ctx: context.Background(),
token: "mytoken",
r: &query.ProxyRequest{
Request: query.Request{
OrganizationID: *orgID,
Compiler: lang.FluxCompiler{
Query: "from()",
},
},
Dialect: csv.DefaultDialect(),
},
status: http.StatusOK,
want: flux.Statistics{},
wantW: "howdy\n",
},
{
name: "missing org id",
ctx: context.Background(),
token: "mytoken",
r: &query.ProxyRequest{
Request: query.Request{
Compiler: lang.FluxCompiler{
Query: "from()",
},
},
Dialect: csv.DefaultDialect(),
},
wantErr: true,
},
{
name: "error status",
token: "mytoken",
ctx: context.Background(),
r: &query.ProxyRequest{
Request: query.Request{
OrganizationID: *orgID,
Compiler: lang.FluxCompiler{
Query: "from()",
},
},
Dialect: csv.DefaultDialect(),
},
status: http.StatusUnauthorized,
wantErr: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if reqID := r.URL.Query().Get(OrgID); reqID == "" {
if name := r.URL.Query().Get(Org); name == "" {
// Request must have org or orgID.
kithttp.ErrorHandler(0).HandleHTTPError(context.TODO(), influxdb.ErrInvalidOrgFilter, w)
return
}
}
w.WriteHeader(tt.status)
_, _ = fmt.Fprintln(w, "howdy")
}))
defer ts.Close()
s := &FluxService{
Addr: ts.URL,
Token: tt.token,
}
w := &bytes.Buffer{}
got, err := s.Query(tt.ctx, w, tt.r)
if (err != nil) != tt.wantErr {
t.Errorf("FluxService.Query() error = %v, wantErr %v", err, tt.wantErr)
return
}
if diff := cmp.Diff(tt.want, got); diff != "" {
t.Errorf("FluxService.Query() = -want/+got: %v", diff)
}
if gotW := w.String(); gotW != tt.wantW {
t.Errorf("FluxService.Query() = %v, want %v", gotW, tt.wantW)
}
})
}
}
func TestFluxQueryService_Query(t *testing.T) {
2019-08-22 02:08:51 +00:00
var orgID influxdb.ID
2018-09-26 16:36:39 +00:00
orgID.DecodeFromString("aaaaaaaaaaaaaaaa")
tests := []struct {
name string
token string
ctx context.Context
r *query.Request
csv string
status int
want string
wantErr bool
}{
{
name: "error status",
token: "mytoken",
ctx: context.Background(),
r: &query.Request{
2018-09-26 16:36:39 +00:00
OrganizationID: orgID,
Compiler: lang.FluxCompiler{
Query: "from()",
},
},
status: http.StatusUnauthorized,
wantErr: true,
},
{
name: "returns csv",
token: "mytoken",
ctx: context.Background(),
r: &query.Request{
2018-09-26 16:36:39 +00:00
OrganizationID: orgID,
Compiler: lang.FluxCompiler{
Query: "from()",
},
},
status: http.StatusOK,
csv: `#datatype,string,long,dateTime:RFC3339,double,long,string,boolean,string,string,string
#group,false,false,false,false,false,false,false,true,true,true
#default,_result,,,,,,,,,
,result,table,_time,usage_user,test,mystr,this,cpu,host,_measurement
,,0,2018-08-29T13:08:47Z,10.2,10,yay,true,cpu-total,a,cpui
`,
want: toCRLF(`,_result,0,2018-08-29T13:08:47Z,10.2,10,yay,true,cpu-total,a,cpui
`),
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
2018-09-26 16:36:39 +00:00
var orgIDStr string
ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
2018-09-26 16:36:39 +00:00
orgIDStr = r.URL.Query().Get(OrgID)
w.WriteHeader(tt.status)
fmt.Fprintln(w, tt.csv)
}))
s := &FluxQueryService{
Addr: ts.URL,
Token: tt.token,
}
res, err := s.Query(tt.ctx, tt.r)
if (err != nil) != tt.wantErr {
t.Errorf("FluxQueryService.Query() error = %v, wantErr %v", err, tt.wantErr)
return
}
if res != nil && res.Err() != nil {
t.Errorf("FluxQueryService.Query() result error = %v", res.Err())
return
}
if tt.wantErr {
return
}
2018-11-08 20:49:35 +00:00
defer res.Release()
enc := csv.NewMultiResultEncoder(csv.ResultEncoderConfig{
NoHeader: true,
Delimiter: ',',
})
b := bytes.Buffer{}
n, err := enc.Encode(&b, res)
if err != nil {
t.Errorf("FluxQueryService.Query() encode error = %v", err)
return
}
if n != int64(len(tt.want)) {
t.Errorf("FluxQueryService.Query() encode result = %d, want %d", n, len(tt.want))
}
2018-09-26 16:36:39 +00:00
if orgIDStr == "" {
t.Error("FluxQueryService.Query() encoded orgID is empty")
}
if got, want := orgIDStr, tt.r.OrganizationID.String(); got != want {
t.Errorf("FluxQueryService.Query() encoded orgID = %s, want %s", got, want)
}
got := b.String()
if !reflect.DeepEqual(got, tt.want) {
t.Errorf("FluxQueryService.Query() =\n%s\n%s", got, tt.want)
}
})
}
}
func TestFluxHandler_postFluxAST(t *testing.T) {
tests := []struct {
name string
w *httptest.ResponseRecorder
r *http.Request
want string
status int
}{
{
name: "get ast from()",
w: httptest.NewRecorder(),
r: httptest.NewRequest("POST", "/api/v2/query/ast", bytes.NewBufferString(`{"query": "from()"}`)),
2019-12-13 15:58:00 +00:00
want: `{"ast":{"type":"Package","package":"main","files":[{"type":"File","location":{"start":{"line":1,"column":1},"end":{"line":1,"column":7},"source":"from()"},"metadata":"parser-type=rust","package":null,"imports":null,"body":[{"type":"ExpressionStatement","location":{"start":{"line":1,"column":1},"end":{"line":1,"column":7},"source":"from()"},"expression":{"type":"CallExpression","location":{"start":{"line":1,"column":1},"end":{"line":1,"column":7},"source":"from()"},"callee":{"type":"Identifier","location":{"start":{"line":1,"column":1},"end":{"line":1,"column":5},"source":"from"},"name":"from"}}}]}]}}
`,
status: http.StatusOK,
},
{
name: "error from bad json",
w: httptest.NewRecorder(),
r: httptest.NewRequest("POST", "/api/v2/query/ast", bytes.NewBufferString(`error!`)),
refactor: http error serialization matches the new error schema (#15196) The http error schema has been changed to simplify the outward facing API. The `op` and `error` attributes have been dropped because they confused people. The `error` attribute will likely be readded in some form in the future, but only as additional context and will not be required or even suggested for the UI to use. Errors are now output differently both when they are serialized to JSON and when they are output as strings. The `op` is no longer used if it is present. It will only appear as an optional attribute if at all. The `message` attribute for an error is always output and it will be the prefix for any nested error. When this is serialized to JSON, the message is automatically flattened so a nested error such as: influxdb.Error{ Msg: errors.New("something bad happened"), Err: io.EOF, } This would be written to the message as: something bad happened: EOF This matches a developers expectations much more easily as most programmers assume that wrapping an error will act as a prefix for the inner error. This is flattened when written out to HTTP in order to make this logic immaterial to a frontend developer. The code is still present and plays an important role in categorizing the error type. On the other hand, the code will not be output as part of the message as it commonly plays a redundant and confusing role when humans read it. The human readable message usually gives more context and a message like with the code acting as a prefix is generally not desired. But, the code plays a very important role in helping to identify categories of errors and so it is very important as part of the return response.
2019-09-19 15:06:47 +00:00
want: `{"code":"invalid","message":"invalid json: invalid character 'e' looking for beginning of value"}`,
status: http.StatusBadRequest,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
h := &FluxHandler{
HTTPErrorHandler: kithttp.ErrorHandler(0),
}
h.postFluxAST(tt.w, tt.r)
if got := tt.w.Body.String(); got != tt.want {
t.Errorf("http.postFluxAST = got\n%vwant\n%v", got, tt.want)
}
if got := tt.w.Code; got != tt.status {
t.Errorf("http.postFluxAST = got %d\nwant %d", got, tt.status)
}
})
}
}
func TestFluxService_Check(t *testing.T) {
ts := httptest.NewServer(http.HandlerFunc(HealthHandler))
defer ts.Close()
s := &FluxService{
Addr: ts.URL,
}
got := s.Check(context.Background())
want := check.Response{
Name: "influxdb",
Status: "pass",
Message: "ready for queries and writes",
Checks: check.Responses{},
}
if !cmp.Equal(want, got) {
t.Errorf("unexpected response -want/+got: " + cmp.Diff(want, got))
}
}
func TestFluxQueryService_Check(t *testing.T) {
ts := httptest.NewServer(http.HandlerFunc(HealthHandler))
defer ts.Close()
s := &FluxQueryService{
Addr: ts.URL,
}
got := s.Check(context.Background())
want := check.Response{
Name: "influxdb",
Status: "pass",
Message: "ready for queries and writes",
Checks: check.Responses{},
}
if !cmp.Equal(want, got) {
t.Errorf("unexpected response -want/+got: " + cmp.Diff(want, got))
}
}
var crlfPattern = regexp.MustCompile(`\r?\n`)
func toCRLF(data string) string {
return crlfPattern.ReplaceAllString(data, "\r\n")
}
type noopEventRecorder struct{}
func (noopEventRecorder) Record(context.Context, metric.Event) {}
var _ metric.EventRecorder = noopEventRecorder{}
// Certain error cases must be encoded as influxdb.Error so they can be properly decoded clientside.
func TestFluxHandler_PostQuery_Errors(t *testing.T) {
defer tracetesting.SetupInMemoryTracing(t.Name())()
orgSVC := newInMemKVSVC(t)
b := &FluxBackend{
HTTPErrorHandler: kithttp.ErrorHandler(0),
log: zaptest.NewLogger(t),
QueryEventRecorder: noopEventRecorder{},
OrganizationService: orgSVC,
ProxyQueryService: &mock.ProxyQueryService{
QueryF: func(ctx context.Context, w io.Writer, req *query.ProxyRequest) (flux.Statistics, error) {
return flux.Statistics{}, &influxdb.Error{
Code: influxdb.EInvalid,
Msg: "some query error",
}
},
},
}
h := NewFluxHandler(zaptest.NewLogger(t), b)
t.Run("missing authorizer", func(t *testing.T) {
ts := httptest.NewServer(h)
defer ts.Close()
resp, err := http.Post(ts.URL+"/api/v2/query", "application/json", strings.NewReader("{}"))
if err != nil {
t.Fatal(err)
}
defer resp.Body.Close()
if actual := resp.Header.Get("Trace-Id"); actual == "" {
t.Error("expected trace ID header")
}
if resp.StatusCode != http.StatusUnauthorized {
t.Errorf("expected unauthorized status, got %d", resp.StatusCode)
}
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
t.Fatal(err)
}
var ierr influxdb.Error
if err := json.Unmarshal(body, &ierr); err != nil {
t.Logf("failed to json unmarshal into influxdb.error: %q", body)
t.Fatal(err)
}
if !strings.Contains(ierr.Msg, "authorization is") {
t.Fatalf("expected error to mention authorization, got %s", ierr.Msg)
}
})
t.Run("authorizer but syntactically invalid JSON request", func(t *testing.T) {
w := httptest.NewRecorder()
req, err := http.NewRequest("POST", "/api/v2/query", strings.NewReader("oops"))
if err != nil {
t.Fatal(err)
}
authz := &influxdb.Authorization{}
req = req.WithContext(icontext.SetAuthorizer(req.Context(), authz))
h.handleQuery(w, req)
if actual := w.Header().Get("Trace-Id"); actual == "" {
t.Error("expected trace ID header")
}
if w.Code != http.StatusBadRequest {
t.Errorf("expected bad request status, got %d", w.Code)
}
body := w.Body.Bytes()
var ierr influxdb.Error
if err := json.Unmarshal(body, &ierr); err != nil {
t.Logf("failed to json unmarshal into influxdb.error: %q", body)
t.Fatal(err)
}
if !strings.Contains(ierr.Msg, "decode request body") {
t.Fatalf("expected error to mention decoding, got %s", ierr.Msg)
}
})
t.Run("valid request but executing query results in client error", func(t *testing.T) {
org := influxdb.Organization{Name: t.Name()}
if err := orgSVC.CreateOrganization(context.Background(), &org); err != nil {
t.Fatal(err)
}
req, err := http.NewRequest("POST", "/api/v2/query?orgID="+org.ID.String(), bytes.NewReader([]byte("buckets()")))
if err != nil {
t.Fatal(err)
}
authz := &influxdb.Authorization{}
req = req.WithContext(icontext.SetAuthorizer(req.Context(), authz))
req.Header.Set("Content-Type", "application/vnd.flux")
w := httptest.NewRecorder()
h.handleQuery(w, req)
if actual := w.Header().Get("Trace-Id"); actual == "" {
t.Error("expected trace ID header")
}
if w.Code != http.StatusBadRequest {
t.Errorf("expected bad request status, got %d", w.Code)
}
body := w.Body.Bytes()
t.Logf("%s", body)
var ierr influxdb.Error
if err := json.Unmarshal(body, &ierr); err != nil {
t.Logf("failed to json unmarshal into influxdb.error: %q", body)
t.Fatal(err)
}
if got, want := ierr.Code, influxdb.EInvalid; got != want {
t.Fatalf("unexpected error code -want/+got:\n\t- %v\n\t+ %v", want, got)
}
if ierr.Msg != "some query error" {
t.Fatalf("expected error message to mention 'some query error', got %s", ierr.Err.Error())
}
})
}
func TestFluxService_Query_gzip(t *testing.T) {
// orgService is just to mock out orgs by returning
// the same org every time.
orgService := &influxmock.OrganizationService{
2019-08-22 02:08:51 +00:00
FindOrganizationByIDF: func(ctx context.Context, id influxdb.ID) (*influxdb.Organization, error) {
return &influxdb.Organization{
ID: id,
Name: id.String(),
}, nil
},
2019-08-22 02:08:51 +00:00
FindOrganizationF: func(ctx context.Context, filter influxdb.OrganizationFilter) (*influxdb.Organization, error) {
return &influxdb.Organization{
ID: influxdb.ID(1),
Name: influxdb.ID(1).String(),
}, nil
},
}
// queryService is test setup that returns the same CSV for all queries.
queryService := &mock.ProxyQueryService{
QueryF: func(ctx context.Context, w io.Writer, req *query.ProxyRequest) (flux.Statistics, error) {
_, _ = w.Write([]byte(`#datatype,string,long,dateTime:RFC3339,double,long,string,boolean,string,string,string
#group,false,false,false,false,false,false,false,true,true,true
#default,_result,,,,,,,,,
,result,table,_time,usage_user,test,mystr,this,cpu,host,_measurement
,,0,2018-08-29T13:08:47Z,10.2,10,yay,true,cpu-total,a,cpui`))
return flux.Statistics{}, nil
},
}
// authService is yet more test setup that returns an operator auth for any token.
authService := &influxmock.AuthorizationService{
2019-08-22 02:08:51 +00:00
FindAuthorizationByTokenFn: func(ctx context.Context, token string) (*influxdb.Authorization, error) {
return &influxdb.Authorization{
ID: influxdb.ID(1),
OrgID: influxdb.ID(1),
Permissions: influxdb.OperPermissions(),
}, nil
},
}
fluxBackend := &FluxBackend{
HTTPErrorHandler: kithttp.ErrorHandler(0),
log: zaptest.NewLogger(t),
QueryEventRecorder: noopEventRecorder{},
OrganizationService: orgService,
ProxyQueryService: queryService,
}
fluxHandler := NewFluxHandler(zaptest.NewLogger(t), fluxBackend)
// fluxHandling expects authorization to be on the request context.
// AuthenticationHandler extracts the token from headers and places
// the auth on context.
auth := NewAuthenticationHandler(zaptest.NewLogger(t), kithttp.ErrorHandler(0))
auth.AuthorizationService = authService
auth.Handler = fluxHandler
2019-09-18 20:19:51 +00:00
auth.UserService = &influxmock.UserService{
FindUserByIDFn: func(ctx context.Context, id platform.ID) (*influxdb.User, error) {
return &influxdb.User{}, nil
},
}
ts := httptest.NewServer(auth)
defer ts.Close()
newFakeRequest := func() *http.Request {
req, err := http.NewRequest("POST", ts.URL+"/api/v2/query?orgID=0000000000000001", bytes.NewReader([]byte("buckets()")))
if err != nil {
t.Fatal(err)
}
req.Header.Set("Content-Type", "application/vnd.flux")
SetToken("not important hard coded test response", req)
return req
}
// disable any gzip compression
client := &http.Client{
Transport: &http.Transport{
MaxIdleConns: 10,
IdleConnTimeout: 30 * time.Second,
DisableCompression: true,
},
}
req := newFakeRequest()
res, err := client.Do(req)
if err != nil {
t.Fatalf("unable to POST to server: %v", err)
}
if res.StatusCode != http.StatusOK {
t.Errorf("unexpected status code %s", res.Status)
}
identityBody, _ := ioutil.ReadAll(res.Body)
_ = res.Body.Close()
// now, we try to use gzip
req = newFakeRequest()
// If we enable compression, we should get the same response.
client = &http.Client{
Transport: &http.Transport{
MaxIdleConns: 10,
IdleConnTimeout: 30 * time.Second,
DisableCompression: false,
},
}
res, err = client.Do(req)
if err != nil {
t.Fatalf("unable to POST to server: %v", err)
}
gzippedBody, _ := ioutil.ReadAll(res.Body)
_ = res.Body.Close()
if res.StatusCode != http.StatusOK {
t.Errorf("unexpected status code %s", res.Status)
}
if string(identityBody) != string(gzippedBody) {
t.Errorf("unexpected difference in identity and compressed bodies:\n%s\n%s", string(identityBody), string(gzippedBody))
}
}
func Benchmark_Query_no_gzip(b *testing.B) {
benchmarkQuery(b, true)
}
func Benchmark_Query_gzip(b *testing.B) {
benchmarkQuery(b, false)
}
func benchmarkQuery(b *testing.B, disableCompression bool) {
// orgService is just to mock out orgs by returning
// the same org every time.
orgService := &influxmock.OrganizationService{
2019-08-22 02:08:51 +00:00
FindOrganizationByIDF: func(ctx context.Context, id influxdb.ID) (*influxdb.Organization, error) {
return &influxdb.Organization{
ID: id,
Name: id.String(),
}, nil
},
2019-08-22 02:08:51 +00:00
FindOrganizationF: func(ctx context.Context, filter influxdb.OrganizationFilter) (*influxdb.Organization, error) {
return &influxdb.Organization{
ID: influxdb.ID(1),
Name: influxdb.ID(1).String(),
}, nil
},
}
// queryService is test setup that returns the same CSV for all queries.
queryService := &mock.ProxyQueryService{
QueryF: func(ctx context.Context, w io.Writer, req *query.ProxyRequest) (flux.Statistics, error) {
_, _ = w.Write([]byte(`#datatype,string,long,dateTime:RFC3339,double,long,string,boolean,string,string,string
#group,false,false,false,false,false,false,false,true,true,true
#default,_result,,,,,,,,,
,result,table,_time,usage_user,test,mystr,this,cpu,host,_measurement
,,0,2018-08-29T13:08:47Z,10.2,10,yay,true,cpu-total,a,cpui`))
return flux.Statistics{}, nil
},
}
// authService is yet more test setup that returns an operator auth for any token.
authService := &influxmock.AuthorizationService{
2019-08-22 02:08:51 +00:00
FindAuthorizationByTokenFn: func(ctx context.Context, token string) (*influxdb.Authorization, error) {
return &influxdb.Authorization{
ID: influxdb.ID(1),
OrgID: influxdb.ID(1),
Permissions: influxdb.OperPermissions(),
}, nil
},
}
fluxBackend := &FluxBackend{
HTTPErrorHandler: kithttp.ErrorHandler(0),
log: zaptest.NewLogger(b),
QueryEventRecorder: noopEventRecorder{},
OrganizationService: orgService,
ProxyQueryService: queryService,
}
fluxHandler := NewFluxHandler(zaptest.NewLogger(b), fluxBackend)
// fluxHandling expects authorization to be on the request context.
// AuthenticationHandler extracts the token from headers and places
// the auth on context.
auth := NewAuthenticationHandler(zaptest.NewLogger(b), kithttp.ErrorHandler(0))
auth.AuthorizationService = authService
auth.Handler = fluxHandler
ts := httptest.NewServer(auth)
defer ts.Close()
newFakeRequest := func() *http.Request {
req, err := http.NewRequest("POST", ts.URL+"/api/v2/query?orgID=0000000000000001", bytes.NewReader([]byte("buckets()")))
if err != nil {
b.Fatal(err)
}
req.Header.Set("Content-Type", "application/vnd.flux")
SetToken("not important hard coded test response", req)
return req
}
client := &http.Client{
Transport: &http.Transport{
MaxIdleConns: 10,
IdleConnTimeout: 30 * time.Second,
DisableCompression: disableCompression,
},
}
b.ResetTimer()
for i := 0; i < b.N; i++ {
req := newFakeRequest()
res, err := client.Do(req)
if err != nil {
b.Fatalf("unable to POST to server: %v", err)
}
if res.StatusCode != http.StatusOK {
b.Errorf("unexpected status code %s", res.Status)
}
_, _ = ioutil.ReadAll(res.Body)
_ = res.Body.Close()
}
}