-
Notifications
You must be signed in to change notification settings - Fork 2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
defer/stream: split incremental delivery into new entry points #3703
defer/stream: split incremental delivery into new entry points #3703
Conversation
Hi @glasser, I'm @github-actions bot happy to help you with this PR 👋 Supported commandsPlease post this commands in separate comments and only one per comment:
|
d1cd417
to
be87bf4
Compare
55ac8df
to
172c853
Compare
4205fd2
to
dfc882e
Compare
Updated this PR to be less of an experiment and more of an actual proposal. Updated the PR description too. I think the |
752061d
to
f786f49
Compare
This seems to have another tiny advantage besides typings: If we return a generator that includes/emits the first payload (current behavior) rather than returning the first payload separately (as in this PR), then even if the first payload is produced synchronously, the client will have to await it. Even if it’s produced asynchronously, with current behavior, the client will have to await the generator (which is not produced until the first payload is) and then again have to await the first payload emitted by the generator. Perhaps this pattern also makes sense generally for every generator that is known to be non-empty… |
172c853
to
feb203a
Compare
Fixes the bug demonstrated in graphql#3709 (which has already been incorporated into the defer-stream branch). This fix is extracted from graphql#3703, which also updates the typing and API around execute. This particular change doesn't affect the API (other than making the `subscribe` return type more honest, as its returned generator could yield AsyncExecutionResult before this change as well).
Fixes the bug demonstrated in graphql#3709 (which has already been incorporated into the defer-stream branch). This fix is extracted from graphql#3703, which also updates the typing and API around execute. This particular change doesn't affect the API, other than making the `subscribe` return type more honest, as its returned generator could yield AsyncExecutionResult before this change as well. (The reason the previous version built is because every ExecutionResult is actually an AsyncExecutionResult; fixing that fact is part of what graphql#3703 does.)
Fixes the bug demonstrated in graphql#3709 (which has already been incorporated into the defer-stream branch). This fix is extracted from graphql#3703, which also updates the typing and API around execute. This particular change doesn't affect the API, other than making the `subscribe` return type more honest, as its returned generator could yield AsyncExecutionResult before this change as well. (The reason the previous version built is because every ExecutionResult is actually an AsyncExecutionResult; fixing that fact is part of what graphql#3703 does.)
Fixes the bug demonstrated in #3709 (which has already been incorporated into the defer-stream branch). This fix is extracted from #3703, which also updates the typing and API around execute. This particular change doesn't affect the API, other than making the `subscribe` return type more honest, as its returned generator could yield AsyncExecutionResult before this change as well. (The reason the previous version built is because every ExecutionResult is actually an AsyncExecutionResult; fixing that fact is part of what #3703 does.)
f786f49
to
715cc20
Compare
@robrichard @IvanGoncharov I've rebased this on top of #3659 with the #3710 fix incorporated, and I've updated the PR description to be more informative and less "experimental". |
715cc20
to
8e0289c
Compare
After discussion with @IvanGoncharov I've updated this to move incremental delivery out of |
src/execution/execute.ts
Outdated
return result; | ||
} | ||
// Always return a Promise for a consistent API. | ||
return Promise.resolve({ singleResult: result }); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Question: Is the benefit of always returning a promise definitely worth it?
-
If the first result is synchronous,
returning the generator could also be synchronous and then we save a tick. -
This change actually sort of makes the API inconsistent in another sense as non incremental delivery can be either value or Promise, so maybe shouldn't incremental.
-
I believe we recently changed in v17 subscribe to allow returning a generator synchronously. Looks like now we r headed in the other direction. Either way, should they be aligned?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@yaacovCR Agree, with can make this function return PromiseOrValue<ExperimentalExecuteIncrementallyResults>
src/execution/execute.ts
Outdated
* Helper for the other execute functions. Returns an ExecutionResult | ||
* synchronously if all resolvers return non-Promises | ||
*/ | ||
function executeSyncOrIncrementally( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do we need this combination?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I agree, with @yaacovCR on that. It's DRY and saves a few lines. But given all the planned changes to execute
API, it would be better to just duplicate those lines in both functions.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@yaacovCR Also, it's a non-exported function, just a utility one to be DRY.
But I still think it's somewhat confusing and don't worth the DRY benefits it gives.
export function graphql( | ||
args: GraphQLArgs, | ||
): Promise<ExecutionResult | AsyncGenerator<AsyncExecutionResult, void, void>> { | ||
export function graphql(args: GraphQLArgs): Promise<ExecutionResult> { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We don't want to export a version of graphql
that supports incremental delivery?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this was @IvanGoncharov 's idea but I might be misinterpreting. Probably planning to change when not "experimental"?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@robrichard Yes, @glasser is right.
The idea is to finally merge it into main
having an "experiment" prefix to everything.
Changing the prototype of graphql
makes it a breaking change, so better to avoid it before we are sure that the shape of the response is final.
So this PR has two things inside:
- Stuff @glasser learned by implementing this feature in ApolloServer (changes to
execute
typings). - Moving everything under the "experiment" prefix and cutting non-essential functionality (e.g.
graphql
) with the end goal of merging steam/defer tomain
.
This comment has been minimized.
This comment has been minimized.
@glasser The latest changes of this PR are available on NPM as Also you can depend on latest version built from this PR: |
export interface InitialIncrementalExecutionResult< | ||
TData = ObjMap<unknown>, | ||
TExtensions = ObjMap<unknown>, | ||
> extends ExecutionResult<TData, TExtensions> { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This should have incremental
as well, according to the spec, even though graphql-js doesn't write that.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fixed.
This comment has been minimized.
This comment has been minimized.
@glasser The latest changes of this PR are available on NPM as Also you can depend on latest version built from this PR: |
I should apply the same "un-transformation" to |
Fixes the bug demonstrated in #3709 (which has already been incorporated into the defer-stream branch). This fix is extracted from #3703, which also updates the typing and API around execute. This particular change doesn't affect the API, other than making the `subscribe` return type more honest, as its returned generator could yield AsyncExecutionResult before this change as well. (The reason the previous version built is because every ExecutionResult is actually an AsyncExecutionResult; fixing that fact is part of what #3703 does.)
aa830a4
to
efda9be
Compare
This PR changes the `execute` and `graphql` APIs on the `defer-stream` branch back to having the same API as on `main`: they can only produce a single `ExecutionResult` and do not support incremental delivery. Incremental delivery is now provided by the new entry point `experimentalExecuteIncrementally`. (We will remove "experimental" once the proposal has been merged into the GraphQL spec.) This function always returns a Promise containing an object that is *either* a single result *or* an initial result plus a generator for subsequent results. This will make upgrading to graphql@17 easier, and increases the clarity of return types. Use distinct types for "the only result of a single-payload execution", "the first payload of a multi-payload execution", and "subsequent payloads of a multi-payload execution", since the data structures are different. (Multi-payload executions *always* have at least one payload, and the structure differs, which is why the new types separate the initial result from subsequent results.) Note that with the previous types, you actually had to use a function with a type guard like `isAsyncIterable`: you couldn't just write `if (Symbol.asyncIterator in result)`. I think both explicitly separating the first (differently-typed) element from the rest of the elements *and* making it possible to differentiate the types with a simple `in` check are improvements. Additionally, somebody using the API in JavaScript who doesn't realize that there is new functionality available who is confused that the result no longer has `data`/`errors` fields can just print out the result and see and search for the terms `initialResult` and `subsequentResults` instead of only seeing iterator internals. Fix `Formatted*Result` types to use `GraphQLFormattedError` rather than `GraphQLError` for `errors` nested inside `incremental`.
4f4e2a1
to
7fe995e
Compare
@@ -51,6 +51,7 @@ export class SimplePubSub<T> { | |||
emptyQueue(); | |||
return Promise.resolve({ value: undefined, done: true }); | |||
}, | |||
/* c8 ignore next 4 */ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
why do we need to ignore these?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good call. We don't any more! I will check any other additional c8 ignores I added.
@@ -113,7 +113,7 @@ describe('Execute: Accepts async iterables as list value', () => { | |||
}, | |||
}), | |||
}); | |||
return execute({ | |||
return experimentalExecuteIncrementally({ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The tests in this file don't use defer or stream. Why did you change it to use experimentalExecuteIncrementally
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good catch. It doesn't need to be, and my PR will now remove the | AsyncGenerator
from completeObjectList
's return type, making it even more clear that the tests in this file don't use defer or stream.
Revert subscribe to v16 API, add new experimentalSubscribeIncrementally
Today's updates:
I encourage squash-and-merge using the current PR description as the commit message. |
This comment has been minimized.
This comment has been minimized.
@glasser The latest changes of this PR are available on NPM as Also you can depend on latest version built from this PR: |
This PR changes the
execute
,subscribe
, andgraphql
APIs on thedefer-stream
branch back to having the same API as onmain
: they can onlyproduce a single
ExecutionResult
(or a stream ofExecutionResult
s forsubscribe
) which never has ahasNext
field, and do not support incrementaldelivery.
Incremental delivery is now provided by the new entry points
experimentalExecuteIncrementally
andexperimentalSubscribeIncrementally
. (Wewill remove "experimental" once the proposal has been merged into the GraphQL
spec.)
This will make upgrading to graphql@17 easier, and increases the clarity
of return types.
Use distinct types for "the only result of a single-payload execution",
"the first payload of a multi-payload execution", and "subsequent
payloads of a multi-payload execution", since the data structures are
different. (Multi-payload executions always have at least one payload,
and the structure differs, which is why the new types separate the
initial result from subsequent results.)
Namely, single results have no
hasNext
field; initial results havehasNext
and may combine
data
/errors
withincremental
; and subsequent results havehasNext
andincremental
.Note that with the previous types, you actually had to use a function
with a type guard like
isAsyncIterable
: you couldn't just writeif (Symbol.asyncIterator in result)
. I think both explicitly separatingthe first (differently-typed) element from the rest of the elements
and making it possible to differentiate the types with a simple
in
check are improvements. Additionally, somebody using the API in
JavaScript who doesn't realize that there is new functionality available
who is confused that the result no longer has
data
/errors
fields canjust print out the result and see and search for the terms
initialResult
andsubsequentResults
instead of only seeing iteratorinternals.
Fix
Formatted*Result
types to useGraphQLFormattedError
rather thanGraphQLError
forerrors
nested insideincremental
.