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
Before it executes a subscription operation, Router generates a unique ID to represent that operation.
Router sends a GraphQL subscription operation to Subgraph via HTTP POST using the standard GraphQL request payload
withAccept
header containingapplication/json;callbackSpec=1.0
:
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 tosubscriptionId
: The generated unique ID for the subscription operationverifier
: A string that Emitter will include in all HTTP callback requests to verify its identityheartbeatIntervalMs
: The number of milliseconds a heartbeat has to be sent to the callback endpoint for the subscription operation, if0
it means the heartbeat is disabled
Before Subgraph responds to Router's request, it sends a
check
message with callback protocol version header (subscription-protocol: callback/1.0
) to the providedcallbackUrl
for confirmation from Router:
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.
Router validates the
check
message using itsid
andverifier
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.
If validation succeeds, Subgraph spawns a background process or notifies a separate system (Emitter) to begin listening for subscription events.
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 ofheartbeatIntervalMs
milliseconds, Emitter must send acheck
message to Router to confirm that Router is still listening. (Note, the value ofheartbeatIntervalMs
is set by the initial payload sent inextensions
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 theerrors
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 theerrors
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:
1{
2 "kind": "subscription",
3 "action": "check",
4 "id": "c4a9d1b8-dc57-44ab-9e5a-6e6189b2b945",
5 "verifier": "XXX"
6}
Property | Description |
---|---|
| This value is currently always subscription . |
| The message type, which is one of the following:
check message.Router should respond with an error if this value is not one of the above. |
| 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. |
| 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:
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:
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):
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/orverifier
) 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 everyheartbeatIntervalMs
milliseconds (value coming from the initial payload sent inextensions
from the Router), causing Router to terminate that subscription.