WebSocket protocol support
Note
This feature is in the Preview stage.
To connect to an API gateway via WebSocket, client applications need to send a GET request
x-yc-apigateway-websocket-connect
: Opening a connection.x-yc-apigateway-websocket-message
: Sending messages via a web socket.x-yc-apigateway-websocket-disconnect
: Closing a connection.
x-yc-apigateway-websocket-connect operation
The operation is performed when a new connection is established. API Gateway invokes an integration when performing the operation. If the integration is invoked successfully, the client gets a response with the 101 Switching Protocol
HTTP code, after which a web socket is considered open as per RFC
For each new web socket, a unique connection ID is generated and returned to the client in the X-Yc-Apigateway-Websocket-Connection-Id
header. The connection ID is provided when an integration is invoked. To manage an established connection with the API, e.g., send data to the client side or close the connection, save the received ID, for example, to Yandex Managed Service for YDB.
Below, you can find a list of headers that are additionally provided in an HTTP request to the integration:
X-Yc-Apigateway-Websocket-Connection-Id
: Connection ID.X-Yc-Apigateway-Websocket-Event-Type
: Event type; in our case, it isCONNECT
.X-Yc-Apigateway-Websocket-Connected-At
: Connection time.
If a Yandex Cloud Functions function is used as an integration, the above websocket details are provided as individual fields within requestContext
in the JSON structure of the request to the function.
For this operation, you can set up authorization using a function. If authorization fails, the connection will not be established and the client will get a response with the 401
or 403
HTTP code.
Clients can use the RFCSec-WebSocket-Protocol
header to request sub-protocol support from the API gateway. The API gateway provides this header in an HTTP request to an integration.
This operation is not required. If the operation is not defined in the specification, the 101 Switching Protocol
HTTP code is returned by default after connecting to the client. Add integrations to this operation if you need to:
- Implement specific sub-protocols to enable the communication between the client and the API gateway.
- Know when a connection is opened and closed.
- Based on authorization, manage who can and cannot connect using WebSocket.
- Send messages to the client side through the connection managing API.
- Save the connection ID and other details to databases.
x-yc-apigateway-websocket-message operation
The operation is run when a message is sent from the client side. API Gateway invokes an integration when performing the operation. Data from a websocket is provided in the body of an HTTP request to the integration. Text (UTF-8) and RFCContent-Type
header gets the application/json
value, while for binary data, it gets application/octet-stream
. If a Cloud Functions function is used as an integration, the binary message is Base64-encoded; the resulting string value is written to the body
field of the request’s JSON structure, while the isBase64Encoded
flag is set to true
.
The body of the integration response is sent to a web socket at the client side as an individual message. If the Content-Type
header in the integration response has the application/json
value or starts with the text/
prefix, a text message will be sent. Otherwise, this will be a binary message.
The message cannot be larger than 128 KB, while the frame cannot be larger than 32 KB. If a message is larger than 32 KB, split it into multiple frames. If a message or frame exceeds the limit, the connection is closed with the 1009
code.
A unique ID is generated for each message. The message ID is provided in a special header when the integration is invoked. The alphabetic message ID order is time-based.
Below, you can find a list of headers that are additionally provided in an HTTP request to the integration:
X-Yc-Apigateway-Websocket-Connection-Id
: Connection ID.X-Yc-Apigateway-Websocket-Event-Type
: Event type; in our case, it isMESSAGE
.X-Yc-Apigateway-Websocket-Message-Id
: Message ID.
If a Cloud Functions function is used as an integration, the above message details are provided as individual fields within requestContext
in the JSON structure of the request to the function.
This is a required operation. Otherwise, the relevant path in the API gateway's OpenAPI specification will not support a WebSocket connection.
x-yc-apigateway-websocket-disconnect operation
The operation is run after a connection is closed as per RFC
This operation is optional. If you intend to do something while closing a connection, e.g., delete data about it from the database, we recommend doing so in the integration invoked when executing the operation.
Below, you can find a list of headers that are additionally provided in an HTTP request to the integration:
X-Yc-Apigateway-Websocket-Connection-Id
: Connection ID.X-Yc-Apigateway-Websocket-Event-Type
: Event type; in our case, it isDISCONNECT
.X-Yc-Apigateway-Websocket-Disconnect-Status-Code
: Status code. For the list of all possible codes, see the RFC protocol .X-Yc-Apigateway-Websocket-Disconnect-Reason
: Text description of the reason for closing the connection.
If a Cloud Functions function is used as an integration, the above details about a closed connection are provided as individual fields within requestContext
in the JSON structure of the request to the function.
Maximum connection lifetime is 60 minutes. A websocket is considered idle if no messages are received through it after 10 minutes. Then the connection closes. You can occasionally send RFC-specified ping frames
Since the connection can be closed for the reasons mentioned above or others, when writing client apps, it is recommended that you enable automatic reconnection.
Extension specification
Example of a specification with a static response:
/ws:
x-yc-apigateway-websocket-message:
x-yc-apigateway-integration:
type: dummy
content:
'*': Got new message!
http_code: 200
http_headers:
Content-Type: text/plain
Example of a specification with a function call:
/ws:
x-yc-apigateway-websocket-connect:
x-yc-apigateway-integration:
type: cloud_functions
function_id: b095c95ic**********
tag: "$latest"
service_account_id: ajehfe56h**********
x-yc-apigateway-websocket-message:
x-yc-apigateway-integration:
type: cloud_functions
function_id: b095c95ic**********
tag: "$latest"
service_account_id: ajehfe56h**********
x-yc-apigateway-websocket-disconnect:
x-yc-apigateway-integration:
type: cloud_functions
function_id: b095c95ic**********
tag: "$latest"
service_account_id: ajehfe56h**********