HTTP Callback Protocol for GraphQL Subscriptions

Enable clients to receive real-time updates via HTTP callback


This reference describes a protocol for GraphQL servers (or subgraphs) to send subscription data to a subscribing graph router (such as the GraphOS Router) via HTTP callbacks. Use this reference if you're adding support for this protocol to a GraphQL server library or other system.

For routers with many simultaneous open subscriptions, this protocol scales better than WebSocket-based protocols, which require long-lasting open connections.

The GraphOS Router provides support for this protocol as part of its support for federated subscriptions:

Protocol flow

All actions described in these steps are performed by one of Router, Subgraph, or Emitter.

Emitter is a system that sends new subscription data to Router. Emitter is commonly also Subgraph, but it doesn't have to be.

Initialization

  1. Before it executes a subscription operation, Router generates a unique ID to represent that operation.

  2. Router sends a GraphQL subscription operation to Subgraph via HTTP POST using the standard GraphQL request payload

    with Accept header containing application/json;callbackSpec=1.0:

JSON
1{
2  "query": "subscription { userWasCreated { name reviews { body } } }",
3  "extensions": {
4    "subscription": {
5      "callbackUrl": "http://localhost:4000/callback/c4a9d1b8-dc57-44ab-9e5a-6e6189b2b945",
6      "subscriptionId": "c4a9d1b8-dc57-44ab-9e5a-6e6189b2b945",
7      "verifier": "XXX",
8      "heartbeatIntervalMs": 5000
9    }
10  }
11}

The extensions property of this payload includes a subscription object with the following:

  • callbackUrl: The URL that Emitter will send subscription data to

  • subscriptionId: The generated unique ID for the subscription operation

  • verifier: A string that Emitter will include in all HTTP callback requests to verify its identity

  • heartbeatIntervalMs: The number of milliseconds a heartbeat has to be sent to the callback endpoint for the subscription operation, if 0 it means the heartbeat is disabled

  1. Before Subgraph responds to Router's request, it sends a check message with callback protocol version header (subscription-protocol: callback/1.0) to the provided callbackUrl for confirmation from Router:

JSON
1{
2  "kind": "subscription",
3  "action": "check",
4  "id": "c4a9d1b8-dc57-44ab-9e5a-6e6189b2b945",
5  "verifier": "XXX"
6}

All messages sent to callbackUrl are HTTP POST requests. Message types are documented below.

This helps ensure that Subgraph is able to send callbacks successfully, and that the id and verifier fields are correct.

  1. Router validates the check message using its id and verifier fields.

Again, Subgraph has not yet responded to Router's original request!

  • If the check message is valid, Router responds with the following details:

    • A 204 HTTP status code

    • An empty response body

    • The response header that specifies the maximum supported protocol version subscription-protocol: callback/1.0

  • If the check message is invalid, Router responds with a status code besides 204 (400-level recommended).

    • If this occurs, Subgraph then responds to Router's request with a 400-level status code, and the subscription is canceled.

  1. If validation succeeds, Subgraph spawns a background process or notifies a separate system (Emitter) to begin listening for subscription events.

  2. Subgraph finally responds to Router with a 200-level status code and empty data GraphQL response ({ "data": null }). This indicates to Router that the subscription has been initialized.

With initialization complete, the protocol commences its main loop.

Main loop

The protocol's main loop remains active for the duration of the subscription. During the main loop, all of the following occur:

  • If heartbeats are enabled (heartbeatIntervalMs > 0), within every period of heartbeatIntervalMs milliseconds, Emitter must send a check message to Router to confirm that Router is still listening. (Note, the value of heartbeatIntervalMs is set by the initial payload sent in extensions from the Router.)

  • Whenever new subscription data is available, Emitter sends a next message to Router containing the new data.

  • If an error occurs and the subscription must be terminated, Emitter sends a complete message to Router and includes the errors field.

  • If the subscription reaches the end of its stream and no new data is forthcoming, Emitter sends a complete message to Router and omits the errors field.

  • If Router terminates a particular subscription, it should return a 404 status code for all future HTTP callbacks sent for that subscription.

    • Relatedly, if Emitter receives a 404 status code from Router for an HTTP callback, it should consider the associated subscription terminated.

Message types

During the protocol flow, Subgraph and Emitter send various messages to Router's callback URL via HTTP POST requests. All HTTP requests should include callback protocol version header (subscription-protocol: callback/1.0).

All of these messages include the following base properties in their JSON body:

JSON
1{
2  "kind": "subscription",
3  "action": "check",
4  "id": "c4a9d1b8-dc57-44ab-9e5a-6e6189b2b945",
5  "verifier": "XXX"
6}
Property Description
kind
This value is currently always subscription.
action
The message type, which is one of the following:
  • check
  • next
  • complete
The example body above is for a check message.Router should respond with an error if this value is not one of the above.
id
The identifier for the message's associated subscription, generated by Router during protocol initialization.Router should respond with a 404 status code if this value does not match the ID of an active subscription.
verifier
A string value provided by Router during initialization so it can validate callback requests from Subgraph and Emitter.Router should respond with an error if this does not match the value it provided when initializing the subscription with the corresponding id.

Fields and behaviors specific to individual message types are documented below.

check

During protocol initialization, Subgraph sends a synchronous check message to Router to help ensure that it can send callbacks successfully, and that the id and verifier fields provided by Router are correct.

As long as subscription is active and heartbeat is enabled (heartbeatIntervalMs > 0), Emitter must send a check message to Router every heartbeatIntervalMs milliseconds (value coming from the initial payload sent in extensions from the Router). This enables Emitter to confirm both that it can still reach Router's callback endpoint, and that subscription is still active.

A check message includes only base message fields:

JSON
1{
2  "kind": "subscription",
3  "action": "check",
4  "id": "c4a9d1b8-dc57-44ab-9e5a-6e6189b2b945",
5  "verifier": "XXX"
6}

If id and verifier both match Router's provided values, Router should respond with the following details:

  • A 204 HTTP status code

  • An empty response body

  • The response header that specifies the maximum supported protocol version subscription-protocol: callback/1.0

Otherwise, Router should respond with an error and subgraph should terminate the associated subscription.

During initialization, Subgraph must send this message synchronously. That's because it sends this message to Router to confirm a subscription request from Router, before responding to that request.

next

Whenever a new subscription event occurs, Emitter sends the associated data to Router in a next message.

The next message includes a payload field, which contains the subscription data in standard GraphQL JSON response format:

JSON
1{
2  "kind": "subscription",
3  "action": "next",
4  "id": "c4a9d1b8-dc57-44ab-9e5a-6e6189b2b945",
5  "verifier": "XXX",
6  "payload": {
7    "data": {
8      "getLivePriceUpdates": {
9        "__typename": "Stock",
10        "price": 5
11      }
12    }
13  }
14}

As this is a recent feature, if you are using a library that does not yet support subscription callbacks (or you are rolling your own implementation), you will need to make the payload look exactly how the response to a regular query would.

  • The root level key needs to be the name of the subscription operation.

  • The initial subscription query will contain the exact fields that the router expects to be included. Specifically, if there are requests for __typename, those need to be respected, otherwise the subscription may not behave as intended.

Both bullets above are things that are typically handled by server libraries automatically, so they may not be obvious when implementing the protocol manually.

complete

Emitter sends a complete message to Router to terminate an active subscription. Emitter might terminate a subscription for the following reasons:

  • The subscription has reached the end of its stream and no new data is forthcoming.

  • An Emitter error occurred that caused the subscription to fail.

A complete message can include an errors field containing an array of GraphQL errors. This field is required if the subscription failed and optional if it completed successfully (it's usually an empty list in this case):

TypeScript
1{
2    "kind": "subscription",
3    "action": "complete",
4    "id": "c4a9d1b8-dc57-44ab-9e5a-6e6189b2b945",
5    "verifier": "XXX",
6    "errors": [{ // Optional if subscription completed successfully
7      "message": "Something went wrong"
8    }]
9}

On receiving a complete message, Router terminates the associated subscription.

Error states

The following are common error states that can occur with this protcol:

  • Emitter can't communicate with Router's callback endpoint, either because the endpoint isn't available or because its provided credentials (id and/or verifier) are invalid.

  • Emitter receives an error HTTP status code from Router's callback endpoint.

    • If the error code is 404, Emitter should consider the associated subscription terminated by Router.

    • In other error cases, Emitter should consider the subscription terminated due to an unexpected error.

  • Emitter fails to send Router a check message for an active subscription every heartbeatIntervalMs milliseconds (value coming from the initial payload sent in extensions from the Router), causing Router to terminate that subscription.

Feedback

Edit on GitHub

Forums