package apollotracing_test import ( "encoding/json" "io" "net/http" "net/http/httptest" "strings" "testing" "time" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" "github.com/vektah/gqlparser/v2/ast" "github.com/vektah/gqlparser/v2/gqlerror" "github.com/99designs/gqlgen/graphql" "github.com/99designs/gqlgen/graphql/handler/apollotracing" "github.com/99designs/gqlgen/graphql/handler/extension" "github.com/99designs/gqlgen/graphql/handler/lru" "github.com/99designs/gqlgen/graphql/handler/testserver" "github.com/99designs/gqlgen/graphql/handler/transport" ) type alwaysError struct{} func (a *alwaysError) Read(p []byte) (int, error) { return 0, io.ErrUnexpectedEOF } func TestApolloTracing(t *testing.T) { now := time.Unix(0, 0) graphql.Now = func() time.Time { defer func() { now = now.Add(100 * time.Nanosecond) }() return now } h := testserver.New() h.AddTransport(transport.POST{}) h.Use(apollotracing.Tracer{}) resp := doRequest(h, http.MethodPost, "/graphql", `{"query":"{ name }"}`) assert.Equal(t, http.StatusOK, resp.Code, resp.Body.String()) var respData struct { Extensions struct { Tracing apollotracing.TracingExtension `json:"tracing"` } `json:"extensions"` } require.NoError(t, json.Unmarshal(resp.Body.Bytes(), &respData)) tracing := &respData.Extensions.Tracing require.EqualValues(t, 1, tracing.Version) require.Zero(t, tracing.StartTime.UnixNano()) require.EqualValues(t, 900, tracing.EndTime.UnixNano()) require.EqualValues(t, 900, tracing.Duration) require.EqualValues(t, 300, tracing.Parsing.StartOffset) require.EqualValues(t, 100, tracing.Parsing.Duration) require.EqualValues(t, 500, tracing.Validation.StartOffset) require.EqualValues(t, 100, tracing.Validation.Duration) require.EqualValues(t, 700, tracing.Execution.Resolvers[0].StartOffset) require.EqualValues(t, 100, tracing.Execution.Resolvers[0].Duration) require.EqualValues(t, ast.Path{ast.PathName("name")}, tracing.Execution.Resolvers[0].Path) require.Equal(t, "Query", tracing.Execution.Resolvers[0].ParentType) require.Equal(t, "name", tracing.Execution.Resolvers[0].FieldName) require.Equal(t, "String!", tracing.Execution.Resolvers[0].ReturnType) } func TestApolloTracing_withFail(t *testing.T) { now := time.Unix(0, 0) graphql.Now = func() time.Time { defer func() { now = now.Add(100 * time.Nanosecond) }() return now } h := testserver.New() h.AddTransport(transport.POST{}) h.Use(extension.AutomaticPersistedQuery{Cache: lru.New(100)}) h.Use(apollotracing.Tracer{}) resp := doRequest(h, http.MethodPost, "/graphql", `{"operationName":"A","extensions":{"persistedQuery":{"version":1,"sha256Hash":"338bbc16ac780daf81845339fbf0342061c1e9d2b702c96d3958a13a557083a6"}}}`) assert.Equal(t, http.StatusOK, resp.Code, resp.Body.String()) b := resp.Body.Bytes() t.Log(string(b)) var respData struct { Errors gqlerror.List } require.NoError(t, json.Unmarshal(b, &respData)) require.Len(t, respData.Errors, 1) require.Equal(t, "PersistedQueryNotFound", respData.Errors[0].Message) } func TestApolloTracing_withUnexpectedEOF(t *testing.T) { h := testserver.New() h.AddTransport(transport.POST{}) h.Use(apollotracing.Tracer{}) resp := doRequestWithReader(h, http.MethodPost, "/graphql", &alwaysError{}) assert.Equal(t, http.StatusOK, resp.Code) } func doRequest(handler http.Handler, method, target, body string) *httptest.ResponseRecorder { return doRequestWithReader(handler, method, target, strings.NewReader(body)) } func doRequestWithReader(handler http.Handler, method string, target string, reader io.Reader) *httptest.ResponseRecorder { r := httptest.NewRequest(method, target, reader) r.Header.Set("Content-Type", "application/json") w := httptest.NewRecorder() handler.ServeHTTP(w, r) return w }