From d019a2cf9a7b583d4f2695de206a4b55840d38e6 Mon Sep 17 00:00:00 2001 From: spetrunin Date: Mon, 11 Nov 2024 20:50:32 +0200 Subject: [PATCH 1/2] fix: variables normalizer - remove skips and operation name --- v2/pkg/astnormalization/astnormalization.go | 105 ++++++++++-------- .../astnormalization/astnormalization_test.go | 60 +++++++++- .../operation_definition_removal.go | 3 + .../variables_default_value_extraction.go | 30 +---- .../astnormalization/variables_extraction.go | 14 --- .../variables_extraction_test.go | 22 ++-- .../variables_unused_deletion.go | 38 +------ 7 files changed, 126 insertions(+), 146 deletions(-) diff --git a/v2/pkg/astnormalization/astnormalization.go b/v2/pkg/astnormalization/astnormalization.go index 2a3f9e5f01..1a667ae235 100644 --- a/v2/pkg/astnormalization/astnormalization.go +++ b/v2/pkg/astnormalization/astnormalization.go @@ -71,7 +71,6 @@ package astnormalization import ( "github.com/wundergraph/graphql-go-tools/v2/pkg/ast" "github.com/wundergraph/graphql-go-tools/v2/pkg/astvisitor" - "github.com/wundergraph/graphql-go-tools/v2/pkg/internal/unsafebytes" "github.com/wundergraph/graphql-go-tools/v2/pkg/operationreport" ) @@ -88,10 +87,10 @@ func NormalizeOperation(operation, definition *ast.Document, report *operationre func NormalizeNamedOperation(operation, definition *ast.Document, operationName []byte, report *operationreport.Report) { normalizer := NewWithOpts( + WithRemoveNotMatchingOperationDefinitions(), WithExtractVariables(), WithRemoveFragmentDefinitions(), WithInlineFragmentSpreads(), - WithRemoveNotMatchingOperationDefinitions(), WithRemoveUnusedVariables(), ) normalizer.NormalizeNamedOperation(operation, definition, operationName, report) @@ -106,8 +105,6 @@ type walkerStage struct { type OperationNormalizer struct { operationWalkers []walkerStage - variablesExtraction *variablesExtractionVisitor - variablesDefaultValuesExtraction *variablesDefaultValueExtractionVisitor removeOperationDefinitionsVisitor *removeOperationDefinitionsVisitor options options @@ -193,7 +190,21 @@ func WithNormalizeDefinition() Option { } func (o *OperationNormalizer) setupOperationWalkers() { - o.operationWalkers = make([]walkerStage, 0, 6) + o.operationWalkers = make([]walkerStage, 0, 9) + + // NOTE: normalization rules for variables relies on the fact that + // we will visit only single operation, so it is important to remove non-matching operations + if o.options.removeNotMatchingOperationDefinitions { + removeNotMatchingOperationDefinitionsWalker := astvisitor.NewWalker(2) + // this rule do not walk deep into ast, so separate walk not expensive, + // but we could not mix this walk with other rules, because they need to go deep + o.removeOperationDefinitionsVisitor = removeOperationDefinitions(&removeNotMatchingOperationDefinitionsWalker) + + o.operationWalkers = append(o.operationWalkers, walkerStage{ + name: "removeNotMatchingOperationDefinitions", + walker: &removeNotMatchingOperationDefinitionsWalker, + }) + } directivesIncludeSkip := astvisitor.NewWalker(8) directiveIncludeSkip(&directivesIncludeSkip) @@ -210,10 +221,6 @@ func (o *OperationNormalizer) setupOperationWalkers() { detectVariableUsage(&directivesIncludeSkip, del) } - if o.options.removeNotMatchingOperationDefinitions { - o.removeOperationDefinitionsVisitor = removeOperationDefinitions(&directivesIncludeSkip) - } - o.operationWalkers = append(o.operationWalkers, walkerStage{ name: "directivesIncludeSkip, removeOperationDefinitions", walker: &directivesIncludeSkip, @@ -230,7 +237,7 @@ func (o *OperationNormalizer) setupOperationWalkers() { if o.options.extractVariables { extractVariablesWalker := astvisitor.NewWalker(8) - o.variablesExtraction = extractVariables(&extractVariablesWalker) + extractVariables(&extractVariablesWalker) o.operationWalkers = append(o.operationWalkers, walkerStage{ name: "extractVariables", walker: &extractVariablesWalker, @@ -270,7 +277,7 @@ func (o *OperationNormalizer) setupOperationWalkers() { if o.options.extractVariables { variablesProcessing := astvisitor.NewWalker(8) inputCoercionForList(&variablesProcessing) - o.variablesDefaultValuesExtraction = extractVariablesDefaultValue(&variablesProcessing) + extractVariablesDefaultValue(&variablesProcessing) injectInputFieldDefaults(&variablesProcessing) o.operationWalkers = append(o.operationWalkers, walkerStage{ @@ -312,12 +319,6 @@ func (o *OperationNormalizer) NormalizeNamedOperation(operation, definition *ast } } - if o.variablesExtraction != nil { - o.variablesExtraction.operationName = operationName - } - if o.variablesDefaultValuesExtraction != nil { - o.variablesDefaultValuesExtraction.operationName = operationName - } if o.removeOperationDefinitionsVisitor != nil { o.removeOperationDefinitionsVisitor.operationName = operationName } @@ -337,46 +338,52 @@ func (o *OperationNormalizer) NormalizeNamedOperation(operation, definition *ast } type VariablesNormalizer struct { - pre *astvisitor.Walker - post *astvisitor.Walker - coerce *astvisitor.Walker - - detect *variableUsageDetector - del *deleteUnusedVariablesVisitor - extractVariables *variablesExtractionVisitor - extractDefaultVariables *variablesDefaultValueExtractionVisitor + firstDetectUnused *astvisitor.Walker + secondExtract *astvisitor.Walker + thirdDeleteUnused *astvisitor.Walker + fourthCoerce *astvisitor.Walker } func NewVariablesNormalizer() *VariablesNormalizer { - pre := astvisitor.NewWalker(8) - post := astvisitor.NewWalker(8) - coerce := astvisitor.NewWalker(0) - ex := extractVariables(&post) - def := extractVariablesDefaultValue(&post) - del := deleteUnusedVariables(&post) - det := detectVariableUsage(&pre, del) - inputCoercionForList(&coerce) + // delete unused modifying variables refs, + // so it is safer to run it sequentially with the extraction + thirdDeleteUnused := astvisitor.NewWalker(8) + del := deleteUnusedVariables(&thirdDeleteUnused) + + // register variable usage detection on the first stage + // and pass usage information to the deletion visitor + // so it keeps variables that are defined but not used at all + // ensuring that validation can still catch them + firstDetectUnused := astvisitor.NewWalker(8) + detectVariableUsage(&firstDetectUnused, del) + + secondExtract := astvisitor.NewWalker(8) + extractVariables(&secondExtract) + extractVariablesDefaultValue(&secondExtract) + + fourthCoerce := astvisitor.NewWalker(0) + inputCoercionForList(&fourthCoerce) + return &VariablesNormalizer{ - pre: &pre, - post: &post, - coerce: &coerce, - detect: det, - del: del, - extractVariables: ex, - extractDefaultVariables: def, + firstDetectUnused: &firstDetectUnused, + secondExtract: &secondExtract, + thirdDeleteUnused: &thirdDeleteUnused, + fourthCoerce: &fourthCoerce, } } -func (v *VariablesNormalizer) NormalizeNamedOperation(operation, definition *ast.Document, operationName string, report *operationreport.Report) { - operationNameBytes := unsafebytes.StringToBytes(operationName) - v.extractVariables.operationName = operationNameBytes - v.extractDefaultVariables.operationName = operationNameBytes - v.detect.operationName = operationNameBytes - v.del.operationName = operationNameBytes - v.pre.Walk(operation, definition, report) +func (v *VariablesNormalizer) NormalizeOperation(operation, definition *ast.Document, report *operationreport.Report) { + v.firstDetectUnused.Walk(operation, definition, report) + if report.HasErrors() { + return + } + v.secondExtract.Walk(operation, definition, report) + if report.HasErrors() { + return + } + v.thirdDeleteUnused.Walk(operation, definition, report) if report.HasErrors() { return } - v.post.Walk(operation, definition, report) - v.coerce.Walk(operation, definition, report) + v.fourthCoerce.Walk(operation, definition, report) } diff --git a/v2/pkg/astnormalization/astnormalization_test.go b/v2/pkg/astnormalization/astnormalization_test.go index 9217f11d36..57e8bd6c88 100644 --- a/v2/pkg/astnormalization/astnormalization_test.go +++ b/v2/pkg/astnormalization/astnormalization_test.go @@ -851,11 +851,11 @@ func TestVariablesNormalizer(t *testing.T) { normalizer := NewVariablesNormalizer() report := operationreport.Report{} - normalizer.NormalizeNamedOperation(&operationDocument, &definitionDocument, "HttpBinPost", &report) + normalizer.NormalizeOperation(&operationDocument, &definitionDocument, &report) require.False(t, report.HasErrors(), report.Error()) out := unsafeprinter.Print(&operationDocument) - require.Equal(t, `mutation HttpBinPost($bar: String!, $a: HttpBinPostInput){httpBinPost(input: $a){headers {userAgent} data {foo}}}`, out) + assert.Equal(t, `mutation HttpBinPost($bar: String!, $a: HttpBinPostInput){httpBinPost(input: $a){headers {userAgent} data {foo}}}`, out) require.Equal(t, `{"a":{"foo":"bar","bar":null}}`, string(operationDocument.Input.Variables)) } @@ -934,21 +934,69 @@ var runWithVariablesAssert = func(t *testing.T, registerVisitor func(walker *ast assert.Equal(t, expectedVariables, actualVariables) } +// runWithVariablesAssertAndPreNormalize - runs pre-normalization functions before the main normalization function +var runWithVariablesAssertAndPreNormalize = func(t *testing.T, registerVisitor func(walker *astvisitor.Walker), definition, operation, operationName, expectedOutput, variablesInput, expectedVariables string, prerequisites ...registerNormalizeFunc) { + t.Helper() + + definitionDocument := unsafeparser.ParseGraphqlDocumentString(definition) + err := asttransform.MergeDefinitionWithBaseSchema(&definitionDocument) + if err != nil { + panic(err) + } + + operationDocument := unsafeparser.ParseGraphqlDocumentString(operation) + expectedOutputDocument := unsafeparser.ParseGraphqlDocumentString(expectedOutput) + report := operationreport.Report{} + + if variablesInput != "" { + operationDocument.Input.Variables = []byte(variablesInput) + } + + additionalWalker := astvisitor.NewWalker(48) + for _, fn := range prerequisites { + fn(&additionalWalker) + } + report = operationreport.Report{} + additionalWalker.Walk(&operationDocument, &definitionDocument, &report) + if report.HasErrors() { + panic(report.Error()) + } + + initialWorker := astvisitor.NewWalker(48) + registerVisitor(&initialWorker) + initialWorker.Walk(&operationDocument, &definitionDocument, &report) + if report.HasErrors() { + panic(report.Error()) + } + + actualAST := mustString(astprinter.PrintString(&operationDocument)) + expectedAST := mustString(astprinter.PrintString(&expectedOutputDocument)) + assert.Equal(t, expectedAST, actualAST) + actualVariables := string(operationDocument.Input.Variables) + assert.Equal(t, expectedVariables, actualVariables) +} + var runWithVariablesExtraction = func(t *testing.T, normalizeFunc registerNormalizeVariablesFunc, definition, operation, operationName, expectedOutput, variablesInput, expectedVariables string, additionalNormalizers ...registerNormalizeFunc) { t.Helper() runWithVariablesAssert(t, func(walker *astvisitor.Walker) { - visitor := normalizeFunc(walker) - visitor.operationName = []byte(operationName) + normalizeFunc(walker) }, definition, operation, operationName, expectedOutput, variablesInput, expectedVariables, additionalNormalizers...) } +var runWithVariablesExtractionAndPreNormalize = func(t *testing.T, normalizeFunc registerNormalizeVariablesFunc, definition, operation, operationName, expectedOutput, variablesInput, expectedVariables string, prerequisites ...registerNormalizeFunc) { + t.Helper() + + runWithVariablesAssertAndPreNormalize(t, func(walker *astvisitor.Walker) { + normalizeFunc(walker) + }, definition, operation, operationName, expectedOutput, variablesInput, expectedVariables, prerequisites...) +} + var runWithVariablesDefaultValues = func(t *testing.T, normalizeFunc registerNormalizeVariablesDefaulValueFunc, definition, operation, operationName, expectedOutput, variablesInput, expectedVariables string) { t.Helper() runWithVariablesAssert(t, func(walker *astvisitor.Walker) { - visitor := normalizeFunc(walker) - visitor.operationName = []byte(operationName) + normalizeFunc(walker) }, definition, operation, operationName, expectedOutput, variablesInput, expectedVariables) } diff --git a/v2/pkg/astnormalization/operation_definition_removal.go b/v2/pkg/astnormalization/operation_definition_removal.go index c3d396a534..eadbc3149a 100644 --- a/v2/pkg/astnormalization/operation_definition_removal.go +++ b/v2/pkg/astnormalization/operation_definition_removal.go @@ -32,6 +32,9 @@ func (r *removeOperationDefinitionsVisitor) EnterOperationDefinition(ref int) { if !bytes.Equal(r.operation.OperationDefinitionNameBytes(ref), r.operationName) { r.operationsToRemove[ref] = struct{}{} } + + // we do not want to visit the children of the operation definition + r.Walker.SkipNode() } func (r *removeOperationDefinitionsVisitor) LeaveDocument(operation, definition *ast.Document) { diff --git a/v2/pkg/astnormalization/variables_default_value_extraction.go b/v2/pkg/astnormalization/variables_default_value_extraction.go index 8ba407d4bf..411c509144 100644 --- a/v2/pkg/astnormalization/variables_default_value_extraction.go +++ b/v2/pkg/astnormalization/variables_default_value_extraction.go @@ -28,9 +28,7 @@ type variablesDefaultValueExtractionVisitor struct { *astvisitor.Walker operation, definition *ast.Document importer astimport.Importer - operationName []byte operationRef int - skip bool variablesNamesUsedInPositionsExpectingNonNullType [][]byte variableRefsWithDefaultValuesDefined []int } @@ -39,9 +37,6 @@ func (v *variablesDefaultValueExtractionVisitor) EnterArgument(ref int) { // variable could be used in directive argument which requires non-null type // in case such variable has default value and not of non-null type, we need to make it non-null - if v.skip { - return - } if len(v.Ancestors) == 0 || v.Ancestors[0].Kind != ast.NodeKindOperationDefinition { return } @@ -62,10 +57,6 @@ func (v *variablesDefaultValueExtractionVisitor) EnterArgument(ref int) { } func (v *variablesDefaultValueExtractionVisitor) EnterField(ref int) { - if v.skip { - return - } - // find field definition from document fieldName := v.operation.FieldNameBytes(ref) fieldDefRef, ok := v.definition.NodeFieldDefinitionByName(v.EnclosingTypeDefinition, fieldName) @@ -91,10 +82,6 @@ func (v *variablesDefaultValueExtractionVisitor) EnterField(ref int) { } func (v *variablesDefaultValueExtractionVisitor) EnterVariableDefinition(ref int) { - if v.skip { - return - } - // skip when we have no default value for variable if !v.operation.VariableDefinitionHasDefaultValue(ref) { return @@ -135,23 +122,10 @@ func (v *variablesDefaultValueExtractionVisitor) EnterVariableDefinition(ref int } func (v *variablesDefaultValueExtractionVisitor) EnterOperationDefinition(ref int) { - if len(v.operationName) == 0 { - v.skip = false - return - } - operationName := v.operation.OperationDefinitionNameBytes(ref) v.operationRef = ref - v.skip = !bytes.Equal(operationName, v.operationName) - - v.variablesNamesUsedInPositionsExpectingNonNullType = make([][]byte, 0, len(v.operation.VariableDefinitions)) - v.variableRefsWithDefaultValuesDefined = make([]int, 0, len(v.operation.VariableDefinitions)) } -func (v *variablesDefaultValueExtractionVisitor) LeaveOperationDefinition(_ int) { - if v.skip { - return - } - +func (v *variablesDefaultValueExtractionVisitor) LeaveOperationDefinition(ref int) { // find and make variable not null for j := 0; j < len(v.variableRefsWithDefaultValuesDefined); j++ { variableDefRef := v.variableRefsWithDefaultValuesDefined[j] @@ -232,4 +206,6 @@ func (v *variablesDefaultValueExtractionVisitor) saveArgumentsWithTypeNotNull(op func (v *variablesDefaultValueExtractionVisitor) EnterDocument(operation, definition *ast.Document) { v.operation, v.definition = operation, definition + v.variablesNamesUsedInPositionsExpectingNonNullType = make([][]byte, 0, len(v.operation.VariableDefinitions)) + v.variableRefsWithDefaultValuesDefined = make([]int, 0, len(v.operation.VariableDefinitions)) } diff --git a/v2/pkg/astnormalization/variables_extraction.go b/v2/pkg/astnormalization/variables_extraction.go index 37d23613b9..8632e695f8 100644 --- a/v2/pkg/astnormalization/variables_extraction.go +++ b/v2/pkg/astnormalization/variables_extraction.go @@ -18,7 +18,6 @@ func extractVariables(walker *astvisitor.Walker) *variablesExtractionVisitor { } walker.RegisterEnterDocumentVisitor(visitor) walker.RegisterEnterArgumentVisitor(visitor) - walker.RegisterEnterOperationVisitor(visitor) return visitor } @@ -26,25 +25,12 @@ type variablesExtractionVisitor struct { *astvisitor.Walker operation, definition *ast.Document importer astimport.Importer - operationName []byte skip bool extractedVariables [][]byte extractedVariableTypeRefs []int } -func (v *variablesExtractionVisitor) EnterOperationDefinition(ref int) { - if len(v.operationName) == 0 { - v.skip = false - return - } - operationName := v.operation.OperationDefinitionNameBytes(ref) - v.skip = !bytes.Equal(operationName, v.operationName) -} - func (v *variablesExtractionVisitor) EnterArgument(ref int) { - if v.skip { - return - } if v.operation.Arguments[ref].Value.Kind == ast.ValueKindVariable { return } diff --git a/v2/pkg/astnormalization/variables_extraction_test.go b/v2/pkg/astnormalization/variables_extraction_test.go index f6218eee67..02b5dd2ab5 100644 --- a/v2/pkg/astnormalization/variables_extraction_test.go +++ b/v2/pkg/astnormalization/variables_extraction_test.go @@ -2,6 +2,8 @@ package astnormalization import ( "testing" + + "github.com/wundergraph/graphql-go-tools/v2/pkg/astvisitor" ) func TestVariablesExtraction(t *testing.T) { @@ -64,7 +66,7 @@ func TestVariablesExtraction(t *testing.T) { }`, `{"foo":"bar"}`, `{"a":{"foo":"bar"},"foo":"bar"}`) }) t.Run("multiple queries", func(t *testing.T) { - runWithVariablesExtraction(t, extractVariables, forumExampleSchema, ` + runWithVariablesExtractionAndPreNormalize(t, extractVariables, forumExampleSchema, ` mutation Register { createUser(input: {user: {id: "jens" username: "jens"}}){ user { @@ -92,19 +94,11 @@ func TestVariablesExtraction(t *testing.T) { username } } - } - mutation CreatePost { - createPost(input: {post: {authorId: "jens" title: "my post" body: "my body"}}){ - post { - id - title - body - userByAuthorId { - username - } - } - } - }`, ``, `{"a":{"user":{"id":"jens","username":"jens"}}}`) + }`, ``, `{"a":{"user":{"id":"jens","username":"jens"}}}`, + func(walker *astvisitor.Walker) { + rm := removeOperationDefinitions(walker) + rm.operationName = []byte("Register") + }) }) t.Run("values on directives should be ignored", func(t *testing.T) { runWithVariablesExtraction(t, extractVariables, forumExampleSchema, ` diff --git a/v2/pkg/astnormalization/variables_unused_deletion.go b/v2/pkg/astnormalization/variables_unused_deletion.go index 85dafca516..dc82b47980 100644 --- a/v2/pkg/astnormalization/variables_unused_deletion.go +++ b/v2/pkg/astnormalization/variables_unused_deletion.go @@ -1,11 +1,11 @@ package astnormalization import ( - "bytes" "fmt" "slices" "github.com/tidwall/sjson" + "github.com/wundergraph/graphql-go-tools/v2/pkg/ast" "github.com/wundergraph/graphql-go-tools/v2/pkg/astvisitor" ) @@ -16,7 +16,7 @@ func deleteUnusedVariables(walker *astvisitor.Walker) *deleteUnusedVariablesVisi } visitor.Walker.RegisterEnterDocumentVisitor(visitor) visitor.Walker.RegisterEnterArgumentVisitor(visitor) - visitor.Walker.RegisterOperationDefinitionVisitor(visitor) + visitor.Walker.RegisterLeaveOperationVisitor(visitor) return visitor } @@ -25,24 +25,9 @@ type deleteUnusedVariablesVisitor struct { operation, definition *ast.Document variableNamesUsed []string variableNamesSafeForDeletion []string - - operationName []byte - skip bool -} - -func (d *deleteUnusedVariablesVisitor) EnterOperationDefinition(ref int) { - if len(d.operationName) == 0 { - d.skip = false - return - } - operationName := d.operation.OperationDefinitionNameBytes(ref) - d.skip = !bytes.Equal(operationName, d.operationName) } func (d *deleteUnusedVariablesVisitor) LeaveOperationDefinition(ref int) { - if d.skip { - return - } var ( err error ) @@ -79,9 +64,6 @@ func (d *deleteUnusedVariablesVisitor) LeaveOperationDefinition(ref int) { } func (d *deleteUnusedVariablesVisitor) EnterArgument(ref int) { - if d.skip { - return - } d.traverseValue(d.operation.Arguments[ref].Value) } @@ -113,7 +95,6 @@ func detectVariableUsage(walker *astvisitor.Walker, deletion *deleteUnusedVariab } visitor.Walker.RegisterEnterDocumentVisitor(visitor) visitor.Walker.RegisterEnterArgumentVisitor(visitor) - visitor.Walker.RegisterEnterOperationVisitor(visitor) return visitor } @@ -121,18 +102,6 @@ type variableUsageDetector struct { *astvisitor.Walker operation, definition *ast.Document deletion *deleteUnusedVariablesVisitor - - operationName []byte - skip bool -} - -func (v *variableUsageDetector) EnterOperationDefinition(ref int) { - if len(v.operationName) == 0 { - v.skip = false - return - } - operationName := v.operation.OperationDefinitionNameBytes(ref) - v.skip = !bytes.Equal(operationName, v.operationName) } func (v *variableUsageDetector) EnterDocument(operation, definition *ast.Document) { @@ -141,9 +110,6 @@ func (v *variableUsageDetector) EnterDocument(operation, definition *ast.Documen } func (v *variableUsageDetector) EnterArgument(ref int) { - if v.skip { - return - } v.traverseValue(v.operation.Arguments[ref].Value) } From 714b83abf12a8f53ba056ac9dc34a3552f86c86d Mon Sep 17 00:00:00 2001 From: spetrunin Date: Tue, 12 Nov 2024 00:04:34 +0200 Subject: [PATCH 2/2] chore: fix execution normalization tests --- execution/engine/extractor_test.go | 3 +-- execution/graphql/normalization.go | 7 ++++-- execution/graphql/normalization_test.go | 29 ++++++------------------- 3 files changed, 13 insertions(+), 26 deletions(-) diff --git a/execution/engine/extractor_test.go b/execution/engine/extractor_test.go index c472e19c60..fa7111c2b7 100644 --- a/execution/engine/extractor_test.go +++ b/execution/engine/extractor_test.go @@ -25,9 +25,8 @@ func TestExtractor_ExtractFieldsFromRequest(t *testing.T) { graphql.NewExtractor().ExtractFieldsFromRequest(&request, schema, &report, fields) expectedFields := graphql.RequestTypes{ - "Foo": {"fooField": {}}, "Post": {"description": {}, "id": {}, "user": {}}, - "Query": {"foo": {}, "posts": {}}, + "Query": {"posts": {}}, "User": {"id": {}, "name": {}}, } diff --git a/execution/graphql/normalization.go b/execution/graphql/normalization.go index 914332f486..62253b0d61 100644 --- a/execution/graphql/normalization.go +++ b/execution/graphql/normalization.go @@ -33,11 +33,14 @@ func (r *Request) Normalize(schema *Schema, options ...astnormalization.Option) } } - normalizer := astnormalization.NewWithOpts(options...) - if r.OperationName != "" { + options = append(options, astnormalization.WithRemoveNotMatchingOperationDefinitions()) + normalizer := astnormalization.NewWithOpts(options...) normalizer.NormalizeNamedOperation(&r.document, &schema.document, []byte(r.OperationName), &report) } else { + // TODO: we should validate count of operations - to throw an error + // and do full normalization for the single anonymous operation + normalizer := astnormalization.NewWithOpts(options...) normalizer.NormalizeOperation(&r.document, &schema.document, &report) } diff --git a/execution/graphql/normalization_test.go b/execution/graphql/normalization_test.go index c3f5451483..1eaa9f26f6 100644 --- a/execution/graphql/normalization_test.go +++ b/execution/graphql/normalization_test.go @@ -31,6 +31,7 @@ func TestRequest_Normalize(t *testing.T) { t.Run("should successfully normalize request with fragments", func(t *testing.T) { schema := StarwarsSchema(t) request := StarwarsRequestForQuery(t, starwars.FileFragmentsQuery) + request.OperationName = "Fragments" documentBeforeNormalization := request.document result, err := request.Normalize(schema) @@ -125,27 +126,11 @@ func TestRequest_Normalize(t *testing.T) { request := StarwarsRequestForQuery(t, starwars.FileMultiQueriesWithArguments) request.OperationName = "GetDroid" - runNormalization(t, &request, `{"a":"1"}`, `query GetDroid($a: ID!){ + runNormalization(t, &request, `{"a":"1"}`, + `query GetDroid($a: ID!){ droid(id: $a){ name } -} - -query Search { - search(name: "C3PO"){ - ... on Droid { - name - primaryFunction - } - ... on Human { - name - height - } - ... on Starship { - name - length - } - } }`) }) @@ -154,9 +139,9 @@ query Search { request := Request{ OperationName: "charactersByIds", Variables: stringify(map[string]interface{}{"a": 1}), - Query: `query ($a: [Int]) { charactersByIds(ids: $a) { name }}`, + Query: `query charactersByIds($a: [Int]) { charactersByIds(ids: $a) { name }}`, } - runNormalizationWithSchema(t, schema, &request, `{"a":[1]}`, `query($a: [Int]){ + runNormalizationWithSchema(t, schema, &request, `{"a":[1]}`, `query charactersByIds($a: [Int]){ charactersByIds(ids: $a){ name } @@ -184,9 +169,9 @@ query Search { Variables: stringify(map[string]interface{}{ "ids": 1, }), - Query: `query($ids: [Int]) {charactersByIds(ids: $ids) { name }}`, + Query: `query charactersByIds($ids: [Int]) {charactersByIds(ids: $ids) { name }}`, } - runNormalizationWithSchema(t, schema, &request, `{"ids":[1]}`, `query($ids: [Int]){ + runNormalizationWithSchema(t, schema, &request, `{"ids":[1]}`, `query charactersByIds($ids: [Int]){ charactersByIds(ids: $ids){ name }