Skip to content

Latest commit

 

History

History
671 lines (507 loc) · 25.3 KB

spec.md

File metadata and controls

671 lines (507 loc) · 25.3 KB

Foxglove WebSocket protocol v1

Protocol overview

  • An application wishing to provide data for streamed consumption by Foxglove hosts a WebSocket server.

  • The client (Foxglove) will specify supported subprotocols (a standard part of the WebSocket handshake) when establishing the connection. The current version of this document corresponds to subprotocol foxglove.websocket.v1. The server must select a subprotocol with which it is compatible for the connection to continue.

    • Example client code in JavaScript:
      new WebSocket("ws://...", ["foxglove.websocket.v1"]);
  • Both text and binary messages are used on the WebSocket connection.

    • Each text message must be a JSON object having a field called op which identifies the type of message. The interpretation of the other fields depends on the opcode.

    • Similarly, each binary message starts with a 1-byte opcode identifying the type of message. The interpretation of the remaining bytes depends on the opcode.

  • Upon establishing a connection, the server must send clients a Server Info message with a list of supported capabilities.

Summary of messages

Sent by server

Sent by client

JSON messages

Each JSON message must be an object containing a field called op which identifies the type of message.

Server Info

  • This message is always sent to new clients upon connection.

Fields

  • op: string "serverInfo"
  • name: free-form information about the server which the client may optionally display or use for debugging purposes
  • capabilities: array of strings, informing the client about which optional features are supported
    • clientPublish: Allow clients to advertise channels to send data messages to the server
    • parameters: Allow clients to get & set parameters
    • parametersSubscribe: Allow clients to subscribe to parameter changes
    • time: The server may publish binary time messages
    • services: Allow clients to call services
    • connectionGraph: Allow clients to subscribe to updates to the connection graph
    • assets: Allow clients to fetch assets
  • supportedEncodings: array of strings | informing the client about which encodings may be used for client-side publishing or service call requests/responses. Only set if client publishing or services are supported.
  • metadata: optional map of key-value pairs
  • sessionId: optional string. Allows the client to understand if the connection is a re-connection or if it is connecting to a new server instance. This can for example be a timestamp or a UUID.

Example

{
  "op": "serverInfo",
  "name": "example server",
  "capabilities": ["clientPublish", "time"],
  "supportedEncodings": ["json"],
  "metadata": {
    "key": "value"
  },
  "sessionId": "1675789422160"
}

Status

  • The server may send this message at any time. Client developers may use it for debugging purposes, display it to the end user, or ignore it.

Fields

  • op: string "status"
  • level: 0 (info), 1 (warning), 2 (error)
  • message: string
  • id: string | undefined. Optional identifier for the status message. Newer status messages with the same identifier should replace previous messages. removeStatus can reference the identifier to indicate a status message is no longer valid.

Example

{
  "op": "status",
  "level": 0,
  "message": "Some info",
  "id": "status-123"
}

Remove Status

  • Informs the client that previously sent status message(s) are no longer valid.

Fields

  • op: string "removeStatus"
  • statusIds: array of string, ids of the status messages to be removed. The array must not be empty.

Example

{
  "op": "removeStatus",
  "statusIds": ["status-123"]
}

Advertise

  • Informs the client about newly available channels.
  • At least one Advertise message is always sent to new clients upon connection.

Fields

  • op: string "advertise"
  • channels: array of:
    • id: number. The server may reuse ids when channels disappear and reappear, but only if the channel keeps the exact same topic, encoding, schemaName, and schema. Clients will use this unique id to cache schema info and deserialization routines.
    • topic: string
    • encoding: string, type of encoding used for message encoding

      Note: Encodings currently supported by Foxglove are json, protobuf, ros1, and cdr (ROS 2). For more information, see the Foxglove documentation.

    • schemaName: string
    • schema: string

      Note: Foxglove expects the schema to be in a format matching the schemaEncoding or alternatively the encoding if schemaEncoding is not specified. For more information, see the Foxglove documentation.

    • schemaEncoding: string | undefined, type of encoding used for schema encoding. May be used if the schema encoding can't be uniquely deduced from the message encoding.

Example

{
  "op": "advertise",
  "channels": [
    {
      "id": 1,
      "topic": "foo",
      "encoding": "protobuf",
      "schemaName": "ExampleMsg",
      "schema": "ZXhhbXBsZSBkYXRh"
    }
  ]
}

Unadvertise

Informs the client that channels are no longer available.

Fields

  • op: string "unadvertise"
  • channelIds: array of number, corresponding to previous Advertise

Example

{
  "op": "unadvertise",
  "channelIds": [1, 2]
}

Parameter Values

Informs the client about parameters. Only supported if the server declares the parameters capability.

Fields

  • op: string "parameterValues"
  • parameters: array of:
    • name: string, name of the parameter
    • value: ParameterValue, where ParameterValue is of type number | boolean | string | ParameterValue[] | { [key: string]: ParameterValue } | undefined.
    • type: "byte_array" | "float64" | "float64_array" | undefined. If the type is byte_array, value shall be interpreted as base64 encoded string. If the type is float64, value must be a valid decimal or integer value that can be represented by a float64. If the type is float64_array, value must be an array of valid decimal or integer values.
  • id: string | undefined. Only set when the getParameters or setParameters request's id field was set

Example

{
  "op": "parameterValues",
  "parameters": [
    { "name": "/int_param", "value": 2 },
    { "name": "/float_param", "value": 1.2 },
    { "name": "/string_param", "value": "foo" },
    { "name": "/node/nested_ints_param", "value": [1, 2, 3] }
    { "name": "/byte_array_param", "value": "QUJDRA==", "type": "byte_array" },
    { "name": "/float_param_int", "value": 3, "type": "float64" },
    { "name": "/float_array_param", "value": [1.1, 2, 3.3], "type": "float64_array" },
  ],
  "id": "request-123"
}

Advertise Services

Informs the client about available services. Only supported if the server declares the services capability.

Fields

  • op: string "advertiseServices"
  • services: array of:
    • id: number. The server may reuse ids when services disappear and reappear, but only if the services keeps the exact same name, type, and schema. Clients will use this unique id to cache schema info and deserialization routines.
    • name: string, name used to identify the service.
    • type: string, type of the service. May be used to derive request & response schema names if request or response are not given.
    • request: object describing the request. Required if requestSchema is not given.
      • encoding: string, type of encoding used for request message encoding.
      • schemaName: string, name of the request schema.
      • schemaEncoding: string, type of encoding used for schema encoding.
      • schema: string, schema definition in a format matching the schemaEncoding.
    • response: object describing the response. Required if responseSchema is not given.
      • encoding: string, type of encoding used for response message encoding.
      • schemaName: string, name of the response schema.
      • schemaEncoding: string, type of encoding used for schema encoding.
      • schema: string, schema definition in a format matching the schemaEncoding.
    • requestSchema: string | undefined, request schema definition. The schema encoding will be derived from the supportedEncodings sent by the server. Required if request is not given. Field is present for backwards compatibilty, prefer using request instead.
    • responseSchema: string | undefined, response schema definition. The schema encoding will be derived from the supportedEncodings sent by the server. Required if response is not given. Field is present for backwards compatibilty, prefer using response instead.

Example

{
  "op": "advertiseServices",
  "services": [
    {
      "id": 1,
      "name": "foo",
      "type": "std_srvs/Empty",
      "requestSchema": "",
      "responseSchema": ""
    },
    {
      "id": 2,
      "name": "set_bool",
      "type": "std_srvs/SetBool",
      "request": { "encoding": "ros1", "schemaName": "std_srvs/SetBool_Request", "schemaEncoding": "ros1msg", "schema": "bool data" },
      "response": { "encoding": "ros1", "schemaName": "std_srvs/SetBool_Response", "schemaEncoding": "ros1msg", "schema": "bool success\nstring message" }
    }
  ]
}

Unadvertise Services

Informs the client about services that are no longer available. Only supported if the server declares the services capability.

Fields

  • op: string "unadvertiseServices"
  • serviceIds: array of number, corresponding to previous advertisement

Example

{
  "op": "unadvertiseServices",
  "serviceIds": [1, 2]
}

Connection graph update

Informs the client about updates to the connection graph. This is only sent to clients which have previously subscribed to connection graph updates. Only supported if the server previously declared that it has the connectionGraph capability.

Fields

  • op: string "connectionGraphUpdate"
  • publishedTopics: array of
    • name: topic name
    • publisherIds: array of string, list of publisher IDs. Replaces any previous publisher information about this topic.
  • subscribedTopics: array of
    • name: topic name
    • subscriberIds: array of string, list of subscriber IDs. Replaces any previous subscriber information about this topic.
  • advertisedServices: array of
    • name: service name
    • providerIds: array of string, list of provider IDs. Replaces any previous provider information about this service.
  • removedTopics: array of string, names of topics that have been removed
  • removedServices: array of string, names of services that have been removed

Example

{
  "op": "connectionGraphUpdate",
  "publishedTopics": [
    { "name": "/foo", "publisherIds": ["/node_1"] },
    { "name": "/bar", "publisherIds": ["/node_1"] }
  ],
  "subscribedTopics": [{ "name": "/foo", "subscriberIds": ["/node_2"] }],
  "advertisedServices": [{ "name": "/set_bool", "providerIds": ["/node_2"] }],
  "removedTopics": ["/baz"],
  "removedServices": []
}

Service Call Failure

Informs the client about failure to call a service. Only supported if the server previously declared that it has the services capability.

Fields

  • op: string "serviceCallFailure"
  • serviceId: number
  • callId: number
  • message: string

Example

{
  "op": "serviceCallFailure",
  "serviceId": 1,
  "callId": 1,
  "message": "Service does not exist",
}

Subscribe

  • Requests that the server start streaming messages on a given topic (or topics) to the client.
  • A client may only have one subscription for each channel at a time.

Fields

  • op: string "subscribe"
  • subscriptions: array of:
    • id: number chosen by the client. The client may not reuse ids across multiple active subscriptions. The server may ignore subscriptions that attempt to reuse an id (and send an error status message). After unsubscribing, the client may reuse the id.
    • channelId: number, corresponding to previous Advertise message(s)

Example

{
  "op": "subscribe",
  "subscriptions": [
    { "id": 0, "channelId": 3 },
    { "id": 1, "channelId": 5 }
  ]
}

Unsubscribe

  • Requests that the server stop streaming messages to which the client previously subscribed.

Fields

  • op: string "unsubscribe"
  • subscriptionIds: array of number, corresponding to previous Subscribe message(s)

Example

{
  "op": "unsubscribe",
  "subscriptionIds": [0, 1]
}

Client Advertise

  • Informs the server about available client channels. Note that the client is only allowed to advertise channels if the server previously declared that it has the clientPublish capability.

Fields

  • op: string "advertise"
  • channels: array of:
    • id: number chosen by the client. The client may reuse ids that have previously been unadvertised.
    • topic: string
    • encoding: string, one of the message encodings supported by the server
    • schemaName: string
    • schema: string | undefined, optional schema definition.
    • schemaEncoding: string | undefined, optional schema encoding.

Example

{
  "op": "advertise",
  "channels": [
    {
      "id": 1,
      "topic": "foo",
      "encoding": "protobuf",
      "schemaName": "ExampleMsg"
    }
  ]
}

Client Unadvertise

  • Informs the server that client channels are no longer available. Note that the client is only allowed to unadvertise channels if the server previously declared that it has the clientPublish capability.

Fields

  • op: string "unadvertise"
  • channelIds: array of number, corresponding to previous Client Advertise

Example

{
  "op": "unadvertise",
  "channelIds": [1, 2]
}

Get Parameters

Request one or more parameters. Only supported if the server previously declared that it has the parameters capability.

Fields

  • op: string "getParameters"
  • parameterNames: string[], leave empty to retrieve all currently set parameters
  • id: string | undefined, arbitrary string used for identifying the corresponding server response

Example

{
  "op": "getParameters",
  "parameterNames": [
    "/int_param",
    "/float_param",
    "/string_param",
    "/node/nested_ints_param"
  ],
  "id": "request-123"
}

Set Parameters

Set one or more parameters. Only supported if the server previously declared that it has the parameters capability.

Fields

  • op: string "setParameters"
  • parameters: array of:
    • name: string
    • value: number | boolean | string | number[] | boolean[] | string[] | undefined. If the value is not set (undefined), the parameter shall be unset (removed).
    • type: "byte_array" | "float64" | "float64_array" | undefined. If the type is byte_array, value shall be a base64 encoded string. If the type is float64, value must be a valid decimal or integer value that can be represented by a float64. If the type is float64_array, value must be an array of valid decimal or integer values.
  • id: string | undefined, arbitrary string used for identifying the corresponding server response. If this field is not set, the server may not send a response to the client.

Example

{
  "op": "setParameters",
  "parameters": [
    { "name": "/int_param", "value": 3 },
    { "name": "/float_param", "value": 4.1 },
    { "name": "/byte_array_param", "value": "QUJDRA==", "type": "byte_array" },
    { "name": "/float_param_int", "value": 3, "type": "float64" },
    { "name": "/float_array_param", "value": [1.1, 2, 3.3], "type": "float64_array" },
  ],
  "id": "request-456"
}

Subscribe Parameter Update

Subscribe to parameter updates. Only supported if the server previously declared that it has the parametersSubscribe capability.

Sending subscribeParameterUpdates multiple times will append the list of parameter subscriptions, not replace them. Note that parameters can be subscribed at most once. Hence, this operation will ignore parameters that are already subscribed. Use unsubscribeParameterUpdates to unsubscribe from existing parameter subscriptions.

Fields

  • op: string "subscribeParameterUpdates"
  • parameterNames: string[], leave empty to subscribe to all currently known parameters

Example

{
  "op": "subscribeParameterUpdates",
  "parameterNames": [
    "/int_param",
    "/float_param",
    "/string_param",
    "/node/nested_ints_param"
  ]
}

Unsubscribe Parameter Update

Unsubscribe from parameter updates. Only supported if the server previously declared that it has the parametersSubscribe capability.

Fields

  • op: string "unsubscribeParameterUpdates"
  • parameterNames: string[], leave empty to unsubscribe from all parameter updates

Example

{
  "op": "unsubscribeParameterUpdates",
  "parameterNames": [
    "/int_param",
    "/float_param",
    "/string_param",
    "/node/nested_ints_param"
  ]
}

Subscribe connection graph updates

Subscribe to connection graph updates. Only supported if the server previously declared that it has the connectionGraph capability.

Fields

  • op: string "subscribeConnectionGraph"

Example

{
  "op": "subscribeConnectionGraph"
}

UnSubscribe connection graph updates

Unsubscribe from connection graph updates. Only supported if the server previously declared that it has the connectionGraph capability.

Fields

  • op: string "unsubscribeConnectionGraph"

Example

{
  "op": "unsubscribeConnectionGraph"
}

Fetch asset

Fetch an asset from the server. Only supported if the server previously declared that it has the assets capability.

Fields

  • op: string "fetchAsset"
  • uri: string, uniform resource identifier to locate a single asset
  • requestId: number, unique 32-bit unsigned integer which is to be included in the response

Example

{
  "op": "fetchAsset",
  "uri": "package://foo/robot.urdf",
  "requestId": 123
}

Binary messages

All binary messages must start with a 1-byte opcode identifying the type of message. The interpretation of the remaining bytes depends on the opcode.

All integer types explicitly specified (uint32, uint64, etc.) in this section are encoded with little-endian byte order.

Message Data

  • Provides a raw message payload, encoded as specified in the Advertise corresponding to the channel.
  • Subscription id must correspond to a Subscribe that was previously sent.
Bytes Type Description
1 opcode 0x01
4 uint32 subscription id
8 uint64 log/receive timestamp (nanoseconds)
remaining bytes uint8[] message payload

Time

  • Inform clients about the latest server time. This allows accelerated, slowed, or stepped control over the progress of time.
  • If the server publishes time data, then timestamps of published messages must originate from the same time source
  • The server may only publish time data if it previously declared support for it via the time capability
Bytes Type Description
1 opcode 0x02
8 uint64 timestamp (nanoseconds)

Service Call Response

Bytes Type Description
1 opcode 0x03
4 uint32 service id
4 uint32 call id
4 uint32 encoding length
encoding length char[] encoding, same encoding that was used for the request
remaining bytes uint8[] response payload

Fetch Asset Response

  • Response to a fetch asset request
  • Only supported if the server previously declared the assets capability.
Bytes Type Description
1 opcode 0x04
4 uint32 request id, as given in the corresponding request
1 uint8 status enum, 0 for success and 1 for error. Values >1 are reserved for future use
4 uint32 error message length
error message length char[] error message, empty if status == 0
remaining bytes uint8[] asset data (file contents), empty if status != 0

Client Message Data

  • Sends a binary websocket message containing the raw messsage payload to the server. Note that the client is only allowed to publish messages if the server previously declared that it has the clientPublish capability.
  • The message payload is encoded as advertised in the Client Advertise operation.
Bytes Type Description
1 opcode 0x01
4 uint32 channel id
remaining bytes uint8[] message payload

Service Call Request

  • Request to call a service that has been advertised by the server.
  • Only supported if the server previously declared the services capability.
Bytes Type Description
1 opcode 0x02
4 uint32 service id
4 uint32 call id, a unique number to identify the corresponding service response
4 uint32 encoding length
encoding length char[] encoding, one of the encodings supported by the server
remaining bytes uint8[] request payload