Skip to content

Conversation

@robrichard
Copy link
Contributor

@robrichard robrichard commented Mar 6, 2025

Follow up from #1135, Based on top of #1148

This creates a definition for Response Map, defined as the map normally returned by GraphQL queries and mutations containing data and/or errors. The term response map is already being use so I formally defined it and used the defined term throughout the rest of the spec. I also have an alternative proposal, Response Payload, as I think that fits better in the direction Incremental Delivery will be going.

The definition of Response is updated to include both Response Map, and Response Stream (for subscriptions).

If we go with response map, I think the end result with Incremental Delivery will be something like:

type ResponseMap = {
  data?: object
  errors?: Error[]
  extensions?: object
}
type ResponseStream = ResponseMap[]
type IncrementalStream = [InitialResponseMap, ...SubsequentResponseMap];
type Response = ResponseMap | ResponseStream | IncrementalStream;

I went through every mention of response and most of the remaining usages are around response key or response field, referring to the object property name (ie for aliases and error paths). This could also likely be standardized in a follow up. See https://github.com/graphql/graphql-spec/pull/1147/files

@robrichard robrichard requested review from benjie and leebyron March 6, 2025 20:04
@netlify
Copy link

netlify bot commented Mar 6, 2025

Deploy Preview for graphql-spec-draft ready!

Name Link
🔨 Latest commit 6c6f765
🔍 Latest deploy log https://app.netlify.com/sites/graphql-spec-draft/deploys/67ee764231d68900086eaf48
😎 Deploy Preview https://deploy-preview-1143--graphql-spec-draft.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

benjie

This comment was marked as resolved.

@robrichard robrichard force-pushed the robrichard/response-map branch from 9e0af97 to 00fe58d Compare March 6, 2025 20:55
Copy link
Member

@benjie benjie left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks good to me 👍

(I have not done a full review of the spec to ensure we caught everything.)

@robrichard robrichard force-pushed the robrichard/response-map branch from dc300d1 to 78a2553 Compare March 13, 2025 17:02
@robrichard robrichard force-pushed the robrichard/response-map branch 2 times, most recently from d127655 to d906813 Compare March 13, 2025 17:51
@benjie benjie added the ✏️ Editorial PR is non-normative or does not influence implementation label Mar 13, 2025
@benjie
Copy link
Member

benjie commented Mar 13, 2025

I prefer #1149 - "response payload" is much clearer IMO.

@martinbonnin
Copy link
Contributor

I like payload better too 👍

Alternatively, did we already discuss reusing the graphql-js terminology?

/**
 * The result of GraphQL execution.
 *
 *   - `errors` is included when any errors occurred as a non-empty array.
 *   - `data` is the result of a successful execution of the query.
 *   - `hasNext` is true if a future payload is expected.
 *   - `extensions` is reserved for adding non-standard properties.
 *   - `incremental` is a list of the results from defer/stream directives.
 */
export interface ExecutionResult {
  errors?: ReadonlyArray<GraphQLError>;
  data?: TData | null;
  extensions?: TExtensions;
}

@benjie
Copy link
Member

benjie commented Mar 14, 2025

I like that, but it would involve renaming a few other things too. In particular ResponseStream being a stream of ExecutionResult is less clear. Since we're introducing IncrementalStream, I wonder if just calling it SubscriptionStream would be better? Then we'd end up with:

type Response = ExecutionResult | SubscriptionStream | IncrementalStream;
type ExecutionResult = {
  data?: object
  errors?: Error[]
  extensions?: object
}
type SubscriptionStream = ExecutionResult[]
type IncrementalStream = [InitialExecutionResult, ...SubsequentExecutionResult];

type SubsequentExecutionResult = {
  incremental?: IncrementalResult[]
  pending?: Pending[]
  completed?: CompletedResult[],
  hasNext?: boolean;
}

type InitialExecutionResult =
  ExecutionResult &
  Partial<IncrementalExecutionResult> &
  { hasNext: true };

type IncrementalResult = IncrementalObjectResult | IncrementalListResult

This kind of works?

@martinbonnin
Copy link
Contributor

martinbonnin commented Mar 14, 2025

That'd work for me 👍 . As I was explaining in the other pr, I expect ${Foo}Stream to be a stream of Foo. If we wanted to keep things a bit more lightweight, we could also remove the Execution from ExecutionResult.

So maybe ResultStream/IncrementalResultStream instead of SubscriptionStream/IncrementalStream?

type Response = Result | ResultStream | IncrementalResultStream;
type Result = {
  data?: object
  errors?: Error[]
  extensions?: object
}
type ResultStream = Result[]
type IncrementalResultStream = [InitialResult, ...SubsequentResult];

type SubsequentResult = {
  incremental?: IncrementalResult[]
  pending?: Pending[]
  completed?: CompletedResult[],
  hasNext?: boolean;
}

type InitialResult =
  Result &
  Partial<IncrementalResult> &
  { hasNext: true };

type IncrementalResult = IncrementalObjectResult | IncrementalListResult

@benjie
Copy link
Member

benjie commented Mar 14, 2025

If we wanted to keep things a bit more lightweight, we could also remove the Execution from ExecutionResult.

We use the term "result" throughout the GraphQL spec, so I think that calling it "result" would be dangerous. ("response" is not used so generically, so doesn't suffer the same issue.)

I expect ${Foo}Stream to be a stream of Foo

That's how it's used currently, but I think it's also reasonable for it to be a name for the stream - e.g. "SubscriptionStream" is clearly to me "a stream for a subscription" rather than "a stream of Subscription". We often use prefixes to name things to disambiguate them, e.g. in CollectFields() we use fragmentGroupedFieldSet to mean "the groupedFieldSet related to this fragment".

@leebyron
Copy link
Collaborator

I think we should keep "Response" as the general concept and qualify it with each variant. Response Map, Incremental Response Map, Response Stream

This will present longer names, but generally we should aim to avoid long term ambiguity if possible

@martinbonnin
Copy link
Contributor

@leebyron "Response Stream" is really ambiguous to me. Is it a response or is it a stream of responses? Or both?

@martinbonnin
Copy link
Contributor

Also interesting question is how we look at this in the context of GraphQL over HTTP.

Do we consider that application/graphql-response+json may contain incremental/subscription responses (feels quite weird IMO).

Incremental responses could potentially use jsonl format (application/graphql-response+jsonl) but not having the media type discriminate what type of response is expected feels a bit weird.

This creates a definition for Response Map, defined as the map normally returned by graphql queries and mutations containing data and/or errors.

The definition of Response is updated to include both Response Map, and Response Stream (for subscriptions).

For Incremental Delivery I plan on updating Response to also include _Incremental Stream_.

Address feedback

Update spec/Section 7 -- Response.md

Co-authored-by: Benjie <[email protected]>
@robrichard robrichard force-pushed the robrichard/response-map branch from d906813 to 181e7ea Compare April 3, 2025 11:46
@robrichard
Copy link
Contributor Author

Superseded by #1159

@robrichard robrichard closed this Apr 17, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

✏️ Editorial PR is non-normative or does not influence implementation

Projects

None yet

Development

Successfully merging this pull request may close these issues.

4 participants