Description
Issue workflow progress
Progress of the issue based on the Contributor Workflow
- 1. The issue provides a reproduction available on Github, Stackblitz or CodeSandbox
Make sure to fork this template and run
yarn generate
in the terminal.Please make sure the GraphQL Tools package versions under
package.json
matches yours. - 2. A failing test has been provided
- 3. A local solution has been provided
- 4. A pull request is pending review
Describe the bug
When making a query that original object contributes no field to selections
, the resulting object is null, and no query is being made to delegate graphql server.
This case occurs when
- there is no
ID!
field(s) required to query delegate schema, i.e. a global object, possibly related to use case of Allow using merge directive without keyArg argument if resolver takes 0 arguments #4336, and - there is no field offered by original object (no field at step 2)
To Reproduce
Steps to reproduce the behavior:
cd gateway npm i pip install awscli aws-sam-cli sam local start-api -t ./local.yml -l message.log --host 0.0.0.0 -p 3000
In the scenario, fields named zero*
is offered by 0-th schema, named one*
is offered by 1-st schema, stitched together
Query:
{
zeroUser {
oneValue
}
oneUser {
zeroValue
}
}
{
"data": {
"zeroUser": null,
"oneUser": null
}
}
Expected behavior
Correct result:
{
"data": {
"zeroUser": {
"oneValue": "1: User: query"
},
"oneUser": {
"zeroValue": "0: User: query"
}
}
}
Environment:
- OS: Linux 4.18.0-372.9.1.el8.x86_64
- "@graphql-tools/graphql-file-loader": "^7.3.14",
- "@graphql-tools/load": "^7.5.13",
- "@graphql-tools/schema": "^8.3.13",
- "@graphql-tools/stitch": "^8.6.12",
- "@graphql-yoga/node": "^2.8.0",
- "@vendia/serverless-express": "^4.8.0",
- NodeJS: v14.19.3
Additional context
graphql-tools/packages/delegate/src/finalizeGatewayRequest.ts
Lines 368 to 371 in 8b63198
When the original object (step 2) does not require an ID, and there is no field offered by object's original schema, all selections will be filtered out, resulting in selections.length === 0
for gateway schema.
In the case of schema stitching, it cannot be assumed that the node without selections at the gateway is safe to delete, the result of empty array is only due to the fact that all fields being offered by delegate schemas, thus has to continue onto step 4 onward to retrieve the correct result.
Here is a workaround for your reference.
if (selections == null ) {
return null;
} else if (selections.length === 0){
selections.push({"kind":"Field","name":{"kind":"Name","value":"__typename"},"arguments":[],"directives":[]})
}
Personally I don't consider it a fix, what it does is just adding __typename
when the array is empty.
Afaik in a valid query, a non-primitive field must at least have 1 subfield selection, thus this would not happen without stitching and filtering.
E.g.:
# Invalid
{
zeroUser {
# type User without any subfield selected
}
}
# Valid
{
zeroUser {
__typename
}
}
Again thanks @Abbywpy for testing the issue