Contributors: Paul Borman, Marcus Hines, Carl Lebsack, Chris Morrow, Anees Shaikh, Rob Shakir
Date: January 30, 2018
Version: 0.6.0
This document defines a gRPC-based protocol for the modification and retrieval of configuration from a target device, as well as the control and generation of telemetry streams from a target device to a data collection system. The intention is that a single gRPC service definition can cover both configuration and telemetry - allowing a single implementation on the target, as well as a single NMS element to interact with the device via telemetry and configuration RPCs.
All messages within the gRPC service definition are defined as protocol buffers (specifically proto3). gRPC service definitions are expected to be described using the relevant features of the protobuf IDL. The protobuf definition of gNMI is maintained in the openconfig/gnmi GitHub repository.
The service defined within this document is assumed to carry payloads that contain data instances of OpenConfig YANG schemas, but can be used for any data with the following characteristics:
Currently, values may be serialised to a scalar object through encoding as a JSON string or a Protobuf type - although the definition of new serialisations is possible.
Throughout this specification the following terminology is used:
When a target wishes to communicate data relating to the state of its internal
database to an interested client, it does so via means of a common
Notification
message. Notification
messages are reused in other
higher-layer messages for various purposes. The exact use of the Notification
message is described on a per-RPC basis.
The fields of the Notification message are as follows:
timestamp
- The time at which the data was collected by the device from
the underlying source, or the time that the target generated the
Notification message (in the case that the data does not reflect an
underlying data source). This value is always represented according to the
definition in 2.2.1.prefix
- a prefix which is applied to all path fields (encoded as per
2.2.2) included in the Notification
message. The paths
expressed within the message are formed by the concatenation of prefix + path
. The prefix
always precedes the path
elements. Further semantics
of prefixes are described in 2.4.1.alias
- a string providing an alias for the prefix specified within the
notification message. The encoding of an alias, and the procedure for their
creation is described in 2.4.2.update
- a list of update messages that indicate changes in the underlying
data of the target. Both modification and creation of data is expressed
through the update message.
Update
message has three subfields:
path
- a path encoded as per 2.2.2.val
- a value encoded as per 2.2.3.duplicates
- a counter value that indicates the number of
coalesced duplicates. If a client is unable to keep up with the
server, coalescion can occur on a per update (i.e., per path) basis
such that the server can discard previous values for a given update
and return only the latest. In this case the server SHOULD increment
a count associated with the update such that a client can detect
that transitions in the state of the path have occurred, but were
suppressed due to its inability to keep up.delete
- a list of paths (encoded as per 2.2.2) that
indicate the deletion of data nodes on the target.The creator of a Notification message MUST include the timestamp
field. All
other fields are optional.
Timestamp values MUST be represented as the number of nanoseconds since the Unix
epoch (January 1st 1970 00:00:00 UTC). The value MUST be encoded as a signed
64-bit integer (int64
)1.
Paths are represented according to gNMI Path
Conventions,
which defines a structured format for path elements, and any associated key
values. Rather than utilising a single string to represent the path - with the
/
character separating each element of the path, the path is represented by an
ordered list of PathElem
messages, starting at the root node, and ending at
the most specific path element. Each PathElem
message contains the name of the
node within the data tree, along with any associated keys/attributes that may be
required for it. Keys are contained within a map<string, string>
where the key
of the map is the name of the key element, and the value is the key's value
encoded as a string.
A path is represented by the Path
message with the following fields:
origin
- a field which MAY be used to disambiguate the path if necessary.
For example, the origin may be used to indicate which organization defined
the schema to which the path belongs.elem
- an array of PathElem
messages, each containing the name of the
element, and any associated keys.target
- the name of the target for which the path is a member. Only set
in prefix
for a path. Elaboration of this field is in
2.2.2.1.Each Path
element should correspond to a node in the data tree. For example,
the path /a/b/c/d
is encoded as:
path: <
elem: <
name: "a"
>
elem: <
name: "b"
>
elem: <
name: "c"
>
elem: <
name: "d"
>
>
Where attributes are to be specified, these are encoded alongside the node name
within the path element, for example a node specified by /a/e[key=k1]/f/g
would have the path encoded as:
path: <
elem: <
name: "a"
>
elem: <
name: "e"
key: <
key: "key"
value: "k1"
>
>
elem: <
name: "f"
>
elem: <
name: "g"
>
>
The root node (/
) is encoded as a zero-length array (slice) of PathElem
messages within the elem
field.
path: <
>
Paths (defined to be the concatenation of the prefix
and path
within the
message) specified within a message MUST be absolute - no messages with relative
paths should be generated.
The target
field within a path is name for the target. This field MUST only
ever be present on prefix
paths in the corresponding request and response
messages. This field is optional for clients. When set in the prefix
in a
request, GetRequest
, SetRequest
or SubscribeRequest
, the field MUST be
reflected in the prefix
of the corresponding GetResponse
, SetResponse
or
SubscribeResponse
by a server. This field is used to allow a name to be
associated with all the data for a given stream if requested by a client. If a
client does not set this field in the prefix
of a request, it MUST NOT be set
in the prefix
of the corresponding response messages. The value for target
is tied to the context of a client RPC and not persisted or shared among
multiple clients.
The value of a data node (or subtree) is encoded in a TypedValue
message as a
oneof
field to allow selection of the data type by setting exactly one of the member
fields. The possible data types include:
Several native scalar protobuf types are included in the TypedValue
message:
string
in the string_val
field - defined to carry all string values
(including enumerated values represented as a string)int64
in the int_val
field (storing all signed integer types - i.e.,
int8
, int16
, int32
, int64
).uint64
in the uint_val
field (used to store all unsigned integer types -
i.e., uint8
, uint16
, uint32
, uint64
).bool
in the bool_val
field, used to store boolean values.bytes
(see 2.3.2)float
in the float_val
field, used to store floating-point values (i.e.,
float32
, float64
).Additional defined data types include:
Decimal64
in the decimal_val
field -- a message encoding a fixed-precision
decimal number consisting of a total number of digits and a precision
indicating the number of digits following the decimal point. It has two
subfields:
digits
- the set of digitsprecision
- number of digits following the decimal pointScalarArray
in the leaflist_val
field -- a message encoding a mixed-type
scalar array; contains single repeated field:
element
-- a TypedValue
element within the array. The type of each
element MUST be a scalar type (i.e., one of the scalar types or
Decimal64
).The remaining fields in the TypedValue
message define structured data types.
Section 2.3 describes these further.
When structured data is sent by the client or the target in an Update
message,
it MUST be serialized according to one of the supported encodings listed in the
Encoding
enumeration. The table below lists the supported encodings and their
corresponding TypedValue
fields, followed by further details on each encoding
type. It should be noted that the target never utilises the Encoding
enumeration to declare to the client the type of encoding utilised, hence the
client must infer the encoding from the populated TypedValue
field.
Name | Description |
TypedValue field |
Encoding Value |
---|---|---|---|
JSON | A JSON encoded string as per 2.3.1. | json_val |
0 |
Bytes | An arbitrary sequence of bytes as per 2.3.2. | bytes_val |
1 |
Proto | A Protobuf encoded message using protobuf.any , as per 2.3.3. |
any_val |
2 |
ASCII | An ASCII encoded string representing text formatted according to a target-defined convention (described in Section 2.3.4). | ascii_val |
3 |
JSON_IETF | A JSON encoded string as per 2.3.1 using JSON encoding compatible with RFC 7951. | json_ietf_val |
4 |
The JSON
type indicates that the value is encoded as a JSON string. This
format utilises the specification in
RFC7159. Additional types (e.g.,
JSON_IETF
) are utilised to indicate specific additional characteristics of the
encoding of the JSON data (particularly where they relate to serialisation of
YANG-modeled data).
For any JSON encoding:
val
field contains a
serialised JSON entity (object or array) corresponding to the referenced
item.Using the following example data tree:
root +
|
+-- a +
|
+-- b[name=b1] +
|
+-- c +
|
+-- d (string)
+-- e (uint32)
The following serialisations would be used (note that the examples below follow the conventions for textproto, and Go-style backticks are used for string literals that would otherwise require escaping):
For /a/b[name=b1]/c/d
:
update: <
path: <
elem: <
name: "a"
>
elem: <
name: "b"
key: <
key: "name"
value: "b1"
>
>
elem: <
name: "c"
>
elem: <
name: "d"
>
>
val: <
json_val: "AStringValue"
>
>
For /a/b[name=b1]/c/e
:
update: <
path: <
elem: <
name: "a"
>
elem: <
name: "b"
key: <
key: "name"
value: "b1"
>
>
elem: <
name: "c"
>
elem: <
name: "e"
>
>
val: <
json_val: 10042 // decoded byte array
>
>
For /a/b[name=b1]/c
:
update: <
path: <
elem: <
name: "a"
>
elem: <
name: "b"
key: <
key: "name"
value: "b1"
>
>
elem: <
name: "c"
>
>
val: <
json_val: `{ "d": "AStringValue", "e": 10042 }`
>
>
For /a
:
update: <
path: <
elem: <
name: "a'
>
>
val: <
json_ietf_val: `{ "b": [
{
"name": "b1",
"c": {
"d": "AStringValue",
"e": 10042
}
}
]
}`
>
>
Note that all JSON values MUST be valid JSON. That is to say, while a value or object may be included in the message, the relevant quoting according to the JSON specification in RFC7159 must be used. This results in quoted string values, and unquoted number values.
JSON_IETF
encoded data MUST conform with the rules for JSON serialisation
described in RFC7951. Data specified with
a type of JSON MUST be valid JSON, but no additional constraints are placed upon
it. An implementation MUST NOT serialise data with mixed JSON
and JSON_IETF
encodings.
Both the client and target MUST support the JSON
encoding as a minimum.
Data encoded using the BYTES
type (i.e., with the byte_val
field) contains a
byte sequence whose semantics is opaque to the protocol.
Data encoded using the PROTOBUF
type (i.e., within the any_val
field)
contains a serialised protobuf message using
protobuf.Any.
Note that in the case that the sender utilises this type, the receiver must
understand the schema (and hence the type of protobuf message that is
serialised) in order to decode the value. Such agreement is not guaranteed by
the protocol and hence must be established out-of-band.
The ASCII
type indicates that the data contains system-formatted ASCII encoded
text. For configuration data, for example, this may consist of semi-structured
CLI configuration data formatted according to the target platform. The gNMI
protocol does not define the format of the text – this must be established
out-of-band.
In a number of messages, a prefix can be specified to reduce the lengths of path
fields within the message. In this case, a prefix
field is specified within a
message - comprising of a valid path encoded according to Section
2.2.2. In the case that a prefix is specified, the absolute path
is comprised of the concatenation of the list of path elements representing the
prefix and the list of path elements in the path
field.
For example, again considering the data tree shown in Section
2.3.1 if a Notification
message updating values, a
prefix could be used to refer to the /a/b[name=b1]/c/d
and /a/b[name=b1]/c/e
data nodes:
notification: <
timestamp: (timestamp) // timestamp as int64
prefix: <
elem: <
name: "a"
>
elem: <
name: "b"
key: <
key: "name"
value: "b1"
>
>
elem: <
name: "c"
>
>
update: <
path: <
elem: <
name: "d"
>
>
value: <
val: <
json_val: "AStringValue"
>
>
>
update: <
path: <
elem: <
name: "e"
>
>
val: <
json_val: 10042 // converted to int representation
>
>
>
In some cases, a client or target MAY desire to utilise aliases for a particular path - such that subsequent messages can be compressed by utilising the alias, rather than using a complete representation of the path. Doing so reduces total message length, by ensuring that redundant information can be removed.
Support for path aliases MAY be provided by a target. In a case where a target does not support aliases, the maximum message length SHOULD be considered, especially in terms of bandwidth utilisation, and the efficiency of message generation.
A path alias is encoded as a string. In order to avoid valid data paths clashing
with aliases (e.g., a
in the above example), an alias name MUST be prefixed
with a #
character.
The means by which an alias is created is defined on a per-RPC basis. In order to delete an alias, the alias name is sent with the path corresponding to the alias empty.
Aliases MUST be specified as a fully expanded path, and hence MUST NOT reference other aliases within their definition, such that a single alias lookup is sufficient to resolve the absolute path.
When a client specifies a path within an RPC message which indicates a read, or retrieval of data, the path MUST be interpreted such that it refers to the node directly corresponding with the path and all its children. The path refers to the direct node and all descendent branches which originate from the node, recursively down to each leaf element. If specific nodes are expected to be excluded then an RPC MAY provide means to filter nodes, such as regular-expression based filtering, lists of excluded paths, or metadata-based filtering (based on annotations of the data schema being manipulated, should such annotations be available and understood by both client and target).
For example, consider the following data tree:
root +
|
+-- childA +
| |
| +-- leafA1
| +-- leafA2
| +-- childA3 --+
| |
| +-- leafA31
| +-- leafA32
|
+-- childB +
|
+-- leafB1
+-- leafB2
A path referring to "root" (which is represented by a path
consisting of an
empty set of elements) should result in the nodes childA
and childB
and all
of their children (leafA1, leafA2, leafB1, leafB2, childA3, leafA31
and
leafA32
) being considered by the relevant operation.
In the case that the RPC is modifying the state of data (i.e., a write operation), such recursion is not required - rather the modification operation should be considered to be targeted at the node within the schema that is specified by the path, and the value should be deserialized such that it modifies the content of any child nodes if required to do so.
Errors MUST be represented by a canonical gRPC error code
(Java,
Go,
C++). The entity
generating the error MUST specify a free-text string which indicates the context
of the error, allowing the receiving entity to generate log entries that allow a
human operator to understand the exact error that occurred, and its context, and
MAY supply structured data encoded as protobuf.Any
. Each RPC defines the
meaning of the relevant canonical error codes within the context of the
operation it performs.
The canonical error code that is chosen MUST consider the expected behavior of the client on receipt of the message. For example, error codes which indicate that a client may subsequently retry SHOULD only be used where retrying the RPC is expected to result in a different outcome.
Where the client or target wishes to indicate an error, it MUST use the Status
message in the RPC return trailers.
The Status
message consists of three fields:
code
- a 32-bit integer value corresponding to the canonical gRPC error
codemessage
- a human-readable string describing the error condition. This
string is not expected to be machine-parsable, but rather provide contextual
information which may be passed to upstream systems.details
- a repeated field of protobuf.Any
messages that carry error
details.The data tree supported by the target is expected to be defined by a set of
schemas. The definition and format of these models is out of scope of this
specification (YANG-modeled data is one example). In the case that such schema
definitions are used, the client should be able to determine the models that are
supported by the target, so that it can generate valid modifications to the data
tree, and interpret the data returned by Get
and Subscribe
RPC calls.
Additionally, the client may wish to restrict the set of models that are utilised by the target so that it can validate the data returned to it against a specific set of data models. This is particularly relevant where the target may otherwise add new values to restricted value data elements (e.g., those representing an enumerated type), or augment new data elements into the data tree.
In order to allow the client to restrict the set of data models to be used when
interacting with the target, the client MAY discover the set of models that are
supported by the target using the Capabilities
RPC described in Section
3.2. For subsequent Get
and Subscribe
RPCs, the
client MAY specify the models to be used by the target. The set of models to
use is expressed as a ModelData
message, as specified in Section
2.6.1.
If the client specifies a set of models in a Get
or Subscribe
RPC, the
target MUST NOT utilize data tree elements that are defined in schema modules
outside the specified set. In addition, where there are data tree elements that
have restricted value sets (e.g., enumerated types), and the set is extended by
a module which is outside of the set, such values MUST NOT be used in data
instances that are sent to the client. Where there are other elements of the
schema that depend on the existence of such enumerated values, the target MUST
NOT include such values in data instances sent to the client.
The ModelData
message describes a specific model that is supported by the
target and used by the client. The fields of the ModelData
message identify a
data model registered in a model catalog, as described in
[MODEL_CATALOG_DOC]
(the schema of the catalog itself - expressed in YANG - is described in
[MODEL_CATALOG_YANG]).
Each model specified by a ModelData
message may refer to a specific schema
module, a bundle of modules, or an augmentation or deviation, as described by
the catalog entry.
Each ModelData
message contains the following fields:
name
- name of the model expressed as a string.organization
- the organization publishing the model, expressed as a
string.version
- the supported (or requested) version of the model, expressed as
a string which represents the semantic version of the catalog entry.The combination of name
, organization
, and version
uniquely identifies an entry in the model catalog.
Each top-level RPC message (e.g., SubscribeRequest
and SubscribeResponse
for
the Subscribe
RPC) defines an extensions
field which can be used to carry
additional parameters for a gNMI RPC.
[GNMI-EXT]
defines the mechanisms to define such extensions. It must be noted that the base
operation of the RPCs as described in this specification MUST NOT be modified by
an extension.
A single gRPC service is defined - future revisions of this specification MAY result in additional services being introduced, and hence an implementation MUST NOT make assumptions that limit to a single service definition.
The service consists of the following RPCs:
Capabilities
- defined in Section 3.2 and used
by the client and target as an initial handshake to exchange capability
informationGet
- defined in Section
3.3, used to retrieve
snapshots of the data on the target by the client.Set
- defined in Section 3.4 and used by the client
to modify the state of the target.Subscribe
- defined in Section 3.5
and used to control subscriptions to data on the target by the client.A target implementing the gNMI service SHOULD register against the gRPC server reflection service to allow clients to determine that gNMI is available on the target.
The session between the client and server MUST be encrypted using TLS - and a target or client MUST NOT fall back to unencrypted sessions. The target and client SHOULD implement TLS >= 1.2.
New connections are mutually authenticated -- each entity validates the X.509 certificate of the remote entity to ensure that the remote entity is both known, and authorized to connect to the local system.
If the target is expected to authenticate an RPC operation, the client MUST
supply a username and password in the metadata of the RPC message (e.g.,
SubscribeRequest
, GetRequest
or SetRequest
). If the client supplies
username/password credentials, the target MUST authenticate the RPC per its
local authentication functionality.
Authorization is also performed per-RPC by the server, through validating client-provided metadata. The client MAY include the appropriate AAA metadata, which MUST contain a username, and MAY include a password in the context of each RPC call it generates. If the client includes both username and password, the target MUST authenticate and authorize the request. If the client only supplies the username, the target MUST authorize the RPC request.
A more detailed discussion of the requirements for authentication and encryption used for gNMI is in [GNMI-AUTH].
A client MAY discover the capabilities of the target using the Capabilities
RPC. The CapabilityRequest
message is sent by the client to interrogate the
target. The target MUST reply with a CapabilityResponse
message that
includes its gNMI service version, the versioned data models it supports, and
the supported data encodings. This information is used in subsequent RPC
messages from the client to indicate the set of models that the client will use
(for Get
, Subscribe
as described in Section
2.6) , and the encoding to be used for the data.
When the client does not specify the models it is using, the target SHOULD use all data schema modules that it supports when considering the data tree to be addressed. If the client does not specify the encoding in an RPC message, it MUST send JSON encoded values (the default encoding).
The CapabilityRequest
message is sent by the client to request capability
information from the target. The CapabilityRequest
message carries a single
repeated extension
field, which is used as per the definition in Section
2.7.
The CapabilityResponse
message has the following fields:
supported_models
- a set of ModelData
messages (as defined in Section
2.6.1) describing each of the models supported
by the targetsupported_encodings
- an enumeration field describing the data encodings
supported by the target, as described in Section
2.3.gNMI_version
- the semantic version of the gNMI service supported by the
target, specified as a string. The version should be interpreted as per
[OPENCONFIG-SEMVER].extension
- a repeated field to carry gNMI extensions, which is used as
per the definition in Section 2.7.In some cases, a client may require a snapshot of the state that exists on the target. In such cases, a client desires some subtree of the data tree to be serialized by the target and transmitted to it. It is expected that the values that are retrieved (whether writeable by the client or not) are collected immediately and provided to the client.
The Get
RPC provides an interface by which a client can request a set of paths
to be serialized and transmitted to it by the target. The client sends a
GetRequest
message to the target, specifying the data that is to be retrieved.
The fields of the GetRequest
message are described in Section
3.3.1.
Upon reception of a GetRequest
, the target serializes the requested paths, and
returns a GetResponse
message. The target MUST reflect the values of the
specified leaves at a particular collection time, which MAY be different for
each path specified within the GetRequest
message.
The target closes the Get
RPC following the transmission of the GetResponse
message.
The GetRequest
message contains the following fields:
prefix
- a path (specified as per Section 2.2.2), and used
as described in Section 2.4.1. The prefix is applied
to all paths within the GetRequest
message.path
- a set of paths (expressed as per Section 2.2.2) for
which the client is requesting a data snapshot from the target. The path
specified MAY utilize wildcards. In the case that the path specified is not
valid, the target MUST return an RPC response indicating an error code of
InvalidArgument
and SHOULD provide information about the invalid path in
the error message or details.type
- the type of data that is requested from the target. The valid
values for type are described below.encoding
- the encoding that the target should utilise to serialise the
subtree of the data tree requested. The type MUST be one of the encodings
specified in Section 2.3. If the
Capabilities
RPC has been utilised, the client SHOULD use an encoding
advertised as supported by the target. If the encoding is not specified,
JSON MUST be used. If the target does not support the specified encoding,
the target MUST return an error of Unimplemented
. The error message MUST
indicate that the specified encoding is unsupported.use_models
- a set of ModelData
messages (defined in Section
2.6.1) indicating the schema definition modules
that define the data elements that should be returned in response to the Get
RPC call. The semantics of the use_models
field are defined in Section
2.6.extension
- a repeated field to carry gNMI extensions, used as per the
definition in Section 2.7.Since the data tree stored by the target may consist of different types of data (e.g., values that are operational in nature, such as protocol statistics) - the client MAY specify that a subset of values in the tree are of interest. In order for such filtering to be implemented, the data schema on the target MUST be annotated in a manner which specifies the type of data for individual leaves, or subtrees of the data tree.
The types of data currently defined are:
CONFIG
- specified to be data that the target considers to be read/write.
If the data schema is described in YANG, this corresponds to the "config
true" set of leaves on the target.STATE
- specified to be the read-only data on the target. If the data
schema is described in YANG, STATE
data is the "config false" set of
leaves on the target.OPERATIONAL
- specified to be the read-only data on the target that is
related to software processes operating on the device, or external
interactions of the device.If the type
field is not specified, the target MUST return CONFIG, STATE and
OPERATIONAL data fields in the tree resulting from the client's query.
The GetResponse
message consists of:
notification
- a set of Notification
messages, as defined in Section
2.1. The target MUST generate a
Notification
message for each path specified in the client's GetRequest
,
and hence MUST NOT collapse data from multiple paths into a single
Notification
within the response. The timestamp
field of the
Notification
message MUST be set to the time at which the target's
snapshot of the relevant path was taken.extension
- a repeated field used to carry gNMI extensions, as per the
description in Section 2.7.The Get
RPC is intended for clients to retrieve relatively small sets of data
as complete objects, for example a part of the configuration. Such requests are
not expected to put a significant resource burden on the target. Since the
target is expected to return the entire snapshot in the GetResponse
message,
Get
is not well-suited for retrieving very large data sets, such as the full
contents of the routing table, or the entire component inventory. For such
operations, the Subscribe
RPC is the recommended mechanism, e.g. using the
ONCE
mode as described in Section 3.5.
Another consideration for Get
is that the timestamp returned is associated
with entire set of data requested, although individual data items may have been
sampled by the target at different times. If the client requires higher
accuracy for individual data items, the Subscribe
RPC is recommended to
request a telemetry stream (see Section
3.5.2).
Modifications to the state of the target are made through the Set
RPC. A
client sends a SetRequest
message to the target indicating the modifications
it desires.
A target receiving a SetRequest
message processes the operations specified
within it - which are treated as a transaction (see Section
3.4.3). The server MUST process deleted paths (within the
delete
field of the SetRequest
), followed by paths to be replaced (within
the replace
field), and finally updated paths (within the update
field). The
order of the replace and update fields MUST be treated as significant within a
single SetRequest
message. If a single path is specified multiple times for a
single operation (i.e., within update
or replace
), then the state of the
target MUST reflect the application of all of the operations in order, even if
they overwrite each other. A SetRequest
specifying an empty set of paths MUST
NOT be treated as an error by the target. For example, a SetRequest
message
carrying only extensions is valid.
In response to a SetRequest
, the target MUST respond with a SetResponse
message. For each operation specified in the SetRequest
message, an
UpdateResult
message MUST be included in the response field of the
SetResponse
. The order in which the operations are applied MUST be maintained
such that UpdateResult
messages can be correlated to the SetRequest
operations. In the case of a failure of an operation, the status of the
UpdateResult
message MUST be populated with error information as per the
specification in Section 3.4.7. In addition, the status
of the SetResponse
message MUST be populated with an error message indicating
the success or failure of the set of operations within the SetRequest
message
(again using the error handling behavior defined in Section
3.4.7).
A SetRequest
message consists of the following fields:
prefix
- specified as per Section 2.4.1. The prefix
specified is applied to all paths defined within other fields of the
message.delete
- A set of paths, specified as per Section 2.2.2,
which are to be removed from the data tree. A specification of the behavior
of a delete is defined in Section 3.4.6.replace
- A set of Update
messages indicating elements of the data tree
whose content is to be replaced.update
- A set of Update
messages indicating elements of the data tree
whose content is to be updated.extension
- a repeated field used to carry gNMI extensions, as per the
description in Section 2.7.The semantics of "updating" versus "replacing" content are defined in Section 3.4.4.
A re-usable Update
message is utilised to indicate changes to paths where a
new value is required. The Update
message contains two fields:
path
- a path encoded as per Section 2.2.2 indicating the
path of the element to be modified.value
- a value encoded as per Section 2.2.3
indicating the value applied to the specified node. The semantics of how the
node is updated is dependent upon the context of the update message, as
specified in Section 3.4.4.A SetResponse
consists of the following fields:
prefix
- specified as per Section 2.4.1. The prefix
specified is applied to all paths defined within other fields of the
message. The target MUST return an RPC response when the change is
successfully applied. In cases where an update was not successfully applied,
the contents of the status message used in the RPC response MUST be
specified as per Section 3.4.7.response
- containing a list of responses, one per operation specified
within the SetRequest
message. Each response consists of an UpdateResult
message with the following fields:
timestamp
- a timestamp (encoded as per Section
2.2.1) at which the set request message was accepted
by the system.path
- the path (encoded as per Section 2.2.2) specified
within the SetRequest
. In the case that a common prefix was not used
within the SetRequest
, the target MAY specify a prefix
to reduce
repetition of path elements within multiple UpdateResult
messages in
the request
field.op
- the operation corresponding to the path. This value MUST be one
of DELETE
, REPLACE
, or UPDATE
.message
- a status message (as specified in Section
2.5). This field follows the same rules as the
status field returned with the SetResponse
message specified above.extension
- a repeated field used to carry gNMI extensions, as per the
description in Section 2.7.All changes to the state of the target that are included in an individual
SetRequest
message are considered part of a transaction. That is, either all
modifications within the request are applied, or the target MUST rollback the
state changes to reflect its state before any changes were applied. The state of
the target MUST NOT appear to be changed until such time as all changes have
been accepted successfully. Hence, telemetry update messages MUST NOT reflect
a change in state until such time as the intended modifications have been
accepted.
As per the specification in Section 3.4, within an
individual transaction (SetRequest
) the order of operations is delete
,
replace
, update
.
As the scope of a "transaction" is a single SetRequest
message, a client
desiring a set of changes to be applied together MUST ensure that they are
encapsulated within a single SetRequest
message.
Changes to read-write values on the target are applied based on the replace
and update
fields of the SetRequest
message.
For both replace and update operations, if the path specified does not exist,
the target MUST create the data tree element and populate it with the data in
the Update
message, provided the path is valid according to the data tree
schema. If invalid values are specified, the target MUST cease processing
updates within the SetRequest
method, return the data tree to the state prior
to any changes, and return a SetResponse
status indicating the error
encountered.
For replace operations:
For example, consider the following tree, where a schema defines the boolean b
as having a default value of True and boolean c
as having no default value.
Both b
and c
have been previously set as False.
root +
|
+ a --+
| |
| +-- b
| |
| +-- c
|
|
+ d --+
+-- e
|
+-- f
A replace
operation is issued where e
and f
are set, and all other
elements are omitted. In this scenario, b
MUST be reverted to its default
setting of True and the configuration of c
MUST be deleted from the tree,
and returned to its original un-configured setting.
For update
operations, only the value of those data elements that are
specified explicitly should be treated as changed.
The path convention defined in Section 2.2.2 allows nodes in the
data tree to be identified by a unique set of node names (e.g., /a/b/c/d
) or
paths that consist of node names coupled with attributes (e.g., /a/e[key=10]
).
In the case where where a node name plus attribute name is required to uniquely
identify an element (i.e., the path within the schema represents a list, map, or
array), the following considerations apply:
/a/f[k1=10][k2=20]
- and a replace or update
operation's path specifies a subset of the attributes (e.g., /a/f[k1=10]
)
then this MUST be considered an error by the target system - and an status
code of InvalidArgument (3)
specified.SetRequest
. An update operation MUST be considered to add new entries to
the collection if they do not exist.InvalidArgument (3)
.For example, consider a tree corresponding to the examples above, as illustrated below.
root +
|
+ a --+
|
+-- f[k1=10][k2=20] --+
| |
| +-- k1 = 10
| +-- k2 = 20
|
+-- f[k1=10][k2=21] --+
|
+-- k1 = 10
+-- k2 = 21
In this case, nodes k1
and k2
are standalone nodes within the schema, but
also correspond to attribute values for the node "f
". In this case, an update
or replace message specifying a path of /a/f[k1=10][k2=20]
setting the value
of k1
to 100 MUST be considered erroneous, and a status code of
InvalidArgument (3)
specified.
Where a path is contained within the delete
field of the SetRequest
message,
it should be removed from the target's data tree. In the case that the path
specified is to an element that has children, these children MUST be recursively
deleted. If a wildcard path is utilised, the wildcards MUST be expanded by the
target, and the corresponding elements of the data tree deleted. Such wildcards
MUST support paths specifying a subset of attributes required to identify
entries within a collection (list, array, or map) of the data schema.
In the case that a path specifies an element within the data tree that does not exist, these deletes MUST be silently accepted.
When a client issues a SetRequest
, and the target is unable to apply the
specified changes, an error status MUST be reported to the client. The error is
specified in multiple places:
SetResponse
message indicates the completion
status of the entire transaction.UpdateResult
message, where the message field indicates the
completion status of the individual operation.The RPC status supplied with the SetResponse
message MUST reflect the overall
result of the transaction.
In the case that any operation within the SetRequest
message fails, then (as
per Section 3.4.3), the target MUST NOT apply any of the
specified changes, and MUST consider the transaction as failed. The target
SHOULD set the status code of the SetResponse
message to Aborted (10)
, along
with an appropriate error message, and MUST set the message
field of the
UpdateResult
corresponding to the failed operation to an Error
message
indicating failure. In the case that the processed operation is not the only
operation within the SetRequest
the target MUST set the message
field of the
UpdateResult
messages for all other operations, setting the code field to
Aborted (10)
.
For the operation that the target is unable to process, the code indicated in the status message returned within the RPC response MUST be set to a specific error code indicating the reason for failure based on the following mappings to canonical gRPC error codes:
Unauthenticated (16)
.PermissionDenied (7)
.InvalidArgument (3)
. In this case, the message
field of the returned
status specified SHOULD specify human-readable text indicating that the path
could not be parsed.NotFound (5)
. In this case the message
field of
the returned status message SHOULD specify human-readable text indicating
the path that was invalid.Update
message specified - InvalidArgument (3)
.
This error SHOULD be used in cases where the payload specifies scalar values
that do not correspond to the correct schema type.Unimplemented (12)
SHOULD be used to
indicate that the encoding is not supported.When a client wishes to receive updates relating to the state of data instances
on a target, it creates a subscription via the Subscribe
RPC. A subscription
consists of one or more paths, with a specified subscription mode. The mode of
each subscription determines the triggers for updates for data sent from the
target to the client.
All requests for new subscriptions are encapsulated within a SubscribeRequest
message - which itself has a mode which describes the longevity of the
subscription. A client may create a subscription which has a dedicated stream to
return one-off data (ONCE
); a subscription that utilizes a stream to
periodically request a set of data (POLL
); or a long-lived subscription that
streams data according to the triggers specified within the individual
subscription's mode (STREAM
).
The target generates messages according to the type of subscription that has been created, at the frequency requested by the client. The methods to create subscriptions are described in Section 3.5.1.
Subscriptions are created for a set of paths - which cannot be modified
throughout the lifetime of the subscription. In order to cancel a subscription,
the client cancels the Subscribe
RPC associated with the subscription, or
terminates the entire gRPC session.
Subscriptions are fundamentally a set of independent update messages relating to the state of the data tree. That is, it is not possible for a client requesting a subscription to assume that the set of update messages received represent a snapshot of the data tree at a particular point in time. Subscriptions therefore allow a client to:
STREAM
subscription), a client creating a subscription
receives an initial set of updates, terminated by a message indicating that
initial synchronisation has completed, and then receives subsequent updates
indicating changes to the initial state of those elements.GetRequest
for large subtrees within the
data tree. The target does not need to coalesce values into a single
snapshot view, or create an in-memory representation of the subtree at the
time of the request, and subsequently transmit this entire view to the
client.Based on the fact that subsequent update messages are considered to be independent, and to ensure that the efficiencies described above can be achieved, by default a target MUST NOT aggregate values within an update message.
In some cases, however, elements of the data tree may be known to change
together, or need to be interpreted by the subscriber together. Such data MUST
be explicitly marked in the schema as being eligible to be aggregated when being
published. Additionally, the subscribing client MUST explicitly request
aggregation of eligible schema elements for the subscription - by means of the
allow_aggregation
flag within a SubscriptionList
message. For elements
covered by a subscription that are not explicitly marked within the schema as
being eligible for aggregation the target MUST NOT coalesce these values,
regardless of the value of the allow_aggregation
flag.
When aggregation is not permitted by the client or the schema each update message MUST contain a (key, value) pair - where the key MUST be a path to a single leaf element within the data tree (encoded according to Section 2.2.2). The value MUST encode only the value of the leaf specified. In most cases, this will be a scalar value (i.e., a JSON value if a JSON encoding is utilised), but in some cases, where an individual leaf element within the schema represents an object, it MAY represent a set of values (i.e., a JSON object or Protobuf message).
Where aggregation is permitted by both the client and schema, each update message MUST contain a key value pair, where the key MUST be the path to the element within the data tree which is explicitly marked as being eligible for aggregation. The value MUST be an object which encodes the children of the data tree element specified. For JSON, the value is therefore a JSON object, and for Protobuf is a binary-encoded Protobuf message (along with any child messages).
A SubscribeRequest
message is sent by a client to request updates from the
target for a specified set of paths.
The fields of the SubscribeRequest
are as follows:
SubscribeRequest
relates to. These are:
subscribe
- a SubscriptionList
message specifying a new set of paths
that the client wishes to subscribe to.poll
- a Poll
message used to specify (on an existing RPC) that the
client wishes to receive a polled update for the paths specified within
the subscription. The semantics of the Poll
message are described in
Section 3.5.1.5.3.aliases
- used by a client to define (on an existing RPC) a new path
alias (as described in Section 2.4.2). The use of
the aliases message is described in Section
3.5.1.6.extension
- a repeated field used to carry gNMI extensions, as per the
description in Section 2.7.In order to create a new subscription a client MUST initiate a Subscribe
RPC
with a SubscribeRequest
message specifying the subscribe
field. The
SubscriptionList
may create a one-off subscription, a poll-only subscription,
or a streaming subscription. In the case of ONCE subscriptions, the RPC MUST be
closed following the initial response generation with the relevant status code.
Subscriptions are set once, and subsequently not modified by a client. If a
client wishes to subscribe to additional paths from a target, it MUST do so by
sending an additional Subscribe
RPC call, specifying a new SubscriptionList
message. In order to end an existing subscription, a client simply cancels the
Subscribe
RPC that relates to that subscription. If an RPC is initiated with
a SubscribeRequest
message that does not specify a SubscriptionList
message
with the request
field, the target MUST consider this an error. If an
additional SubscribeRequest
message specifying a SubscriptionList
is sent
via an existing RPC, the target MUST respond to this message with
SubscribeResponse
message indicating an error status, with a code of
InvalidArgument (4)
; other Subscribe
RPCs on the gRPC session MUST not be
modified or terminated.
If a client initiates a Subscribe
RPC with a SubscribeRequest
message which
does not contain a SubscriptionList
message, this is an error. A
SubscribeResponse
message with the status indicating a error code of
InvalidArgument
MUST be sent, and the RPC closed. The status message SHOULD
indicate that an out-of-order operation was requested on a non-existent
subscription.
A SubscriptionList
message is used to indicate a set of paths for which common
subscription behavior are required. The fields of the message are:
subscription
- a set of Subscription
messages that indicate the set of
paths associated with the subscription list.mode
- the type of subscription that is being created. This may be ONCE
(described in 3.5.1.5.1); STREAM
(described
in 3.5.1.5.2); or POLL
(described in
3.5.1.5.3). The default value for the mode
field is STREAM
.prefix
- a common prefix that is applied to all paths specified within the
message as per the definition in Section 2.4.1. The
default prefix is null.use_aliases
- a boolean flag indicating whether the client accepts target
aliases via the subscription RPC. In the case that such aliases are
accepted, the logic described in Section 2.4.2 is
utilised. By default, path aliases created by the target are not supported.qos
- a field describing the packet marking that is to be utilised for the
responses to the subscription that is being created. This field has a single
sub-value, marking
, which indicates the DSCP value as a 32-bit unsigned
integer. If the qos
field is not specified, the device should export
telemetry traffic using its default DSCP marking for management-plane
traffic.allow_aggregation
- a boolean value used by the client to allow schema
elements that are marked as eligible for aggregation to be combined into
single telemetry update messages. By default, aggregation MUST NOT be used.use_models
- a ModelData
message (as specified in Section
2.6.1) specifying the schema definition modules
that the target should use when creating a subscription. When specified, the
target MUST only consider data elements within the defined set of schema
models as defined in Section 2.6. When
use_models
is not specified, the target MUST consider all data elements
that are defined in all schema modules that it supports.updates_only
- a boolean that causes the server to send only updates to
the current state. When set to true, the target MUST not transmit the
current state of the paths that the client has subscribed to, but rather
should send only updates to them. For STREAM
subscriptions, an update
occurs upon the next sample (in the case of SAMPLE
subscriptions), or upon
the next value change for ON_CHANGE
subscriptions. For POLL
and ONCE
subscriptions, the target should send only the sync_response
message,
before proceeding to process poll requests (in the case of POLL
) or
closing the RPC (in the case of ONCE
)."extension
- a repeated field used to carry gNMI extensions, as per the
description in Section 2.7.A client generating a SubscriptionList
message MUST include the subscription
field - which MUST be a non-empty set of Subscription
messages, all other
fields are optional.
A Subscription
message generically describes a set of data that is to be
subscribed to by a client. It contains a path
, specified as per the definition
in Section 2.2.2.
There is no requirement that the path specified in the message must exist within the current data tree on the server. While the path within the subscription SHOULD be a valid path within the set of schema modules that the target supports, subscribing to any syntactically valid path within such modules MUST be allowed. In the case that a particular path does not (yet) exist, the target MUST NOT close the RPC, and instead should continue to monitor for the existence of the path, and transmit telemetry updates should it exist in the future.
For POLL
and STREAM
subscriptions, a client may optionally specify
additional parameters within the Subscription
message. The semantics of these
additional fields are described in the relevant section of this document.
A SubscribeResponse
message is transmitted by a target to a client over an
established Subscribe
RPC. The message contains the following fields:
response
fields, only one of which can
be specified per SubscribeResponse
message:
update
- a Notification
message providing an update value for a
subscribed data entity as described in Section
3.5.2.3. The update
field is also
utilised when a target wishes to create an alias within a subscription,
as described in Section
3.5.2.2.sync_response
- a boolean field indicating that all data values
corresponding to the path(s) specified in the SubscriptionList
has
been transmitted at least once, used for ONCE
, POLL
and STREAM
subscriptions.extension
- a repeated field used to carry gNMI extensions, as per the
description in Section 2.7.SubscribeResponse
messages with the update
field set to a Notification
are
not required to be sent by the target in timestamp order, and client
implementations MUST NOT assume that these messages will be received in
timestamp order.
A subscription operating in the ONCE
mode acts as a single request/response
channel. The target creates the relevant update messages, transmits them, and
subsequently closes the RPC.
In order to create a one-off subscription, a client sends a SubscribeRequest
message to the target. The subscribe
field within this message specifies a
SubscriptionList
with the mode field set to ONCE
. Updates corresponding to
the subscription are generated as per the semantics described in Section
3.5.2.
Following the transmission of all updates which correspond to data items within
the set of paths specified within the subscription list, a SubscribeResponse
message with the sync_response
field set to true
MUST be transmitted, and
the RPC via which the SubscribeRequest
was received MUST be closed.
Stream subscriptions are long-lived subscriptions which continue to transmit updates relating to the set of paths that are covered within the subscription indefinitely.
A STREAM
subscription is created by sending a SubscribeRequest
message with
the subscribe field containing a SubscriptionList
message with the type
specified as STREAM
. Each entry within the Subscription
message is specified
with one of the following modes
:
ON_CHANGE
) - when a subscription is defined to be "on
change", data updates are only sent when the value of the data item changes.
ON_CHANGE
subscriptions, the target MUST first generate updates
for all paths that match the subscription path(s), and transmit them.
Following this initial set of updates, updated values SHOULD only be
transmitted when their value changes.SAMPLE
) - a subscription that is defined to be sampled MUST
be specified along with a sample_interval
encoded as an unsigned 64-bit
integer representing nanoseconds between samples. The value of the data
item(s) MUST be sent once per sample interval to the client. If the target
is unable to support the desired sample_interval
it MUST reject the
subscription by closing the Subscribe
RPC specifying an InvalidArgument (3)
error code. If the sample_interval
is set to 0, the target MUST
create the subscription and send the data with the lowest interval possible
for the target.
suppress_redundant
field of the Subscription
message
may be set for a sampled subscription. In the case that it is set to
true
, the target SHOULD NOT generate a telemetry update message unless
the value of the path being reported on has changed since the last
update was generated. Updates MUST only be generated for those
individual leaf nodes in the subscription that have changed. That is to
say that for a subscription to /a/b
- where there are leaves c
and
d
branching from the b
node - if the value of c
has changed, but
d
remains unchanged, an update for d
MUST NOT be generated, whereas
an update for c
MUST be generated.heartbeat_interval
MAY be specified to modify the behavior of
suppress_redundant
in a sampled subscription. In this case, the
target MUST generate one telemetry update per heartbeat interval,
regardless of whether the suppress_redundant
flag is set to true
.
This value is specified as an unsigned 64-bit integer in nanoseconds.(TARGET_DEFINED)
- when a client creates a subscription
specifying the target defined mode, the target MUST determine the best type
of subscription to be created on a per-leaf basis. That is to say, if the
path specified within the message refers to some leaves which are event
driven (e.g., the changing of state of an entity based on an external
trigger) then an ON_CHANGE
subscription may be created, whereas if other
data represents counter values, a SAMPLE
subscription may be created.Polling subscriptions are used for on-demand retrieval of data items via
long-lived RPCs. A poll subscription relates to a certain set of subscribed
paths, and is initiated by sending a SubscribeRequest
message with
encapsulated SubscriptionList
. Subscription
messages contained within the
SubscriptionList
indicate the set of paths that are of interest to the polling
client.
To retrieve data from the target, a client sends a SubscribeRequest
message
to the target, containing a poll
field, specified to be an empty Poll
message. On reception of such a message, the target MUST generate updates for
all the corresponding paths within the SubscriptionList
. Updates MUST be
generated according to Section 3.5.2.3.
When a client wishes to create an alias that a target should use for a path, the
client should send a SubscribeRequest
message specifying the aliases
field.
The aliases
field consists of an AliasList
message. An AliasList
specifies
a list of aliases, each of which consists of:
path
- the target path for the alias - encoded as per Section
2.2.2.alias
- the (client-defined) alias for the path, encoded as per Section
2.4.2.Where a target is unable to support a client-defined alias it SHOULD respond
with a SubscribeResponse
message with the status code indicating an error of
the following types:
InvalidArgument (3)
where the specified alias is not acceptable to the
target.AlreadyExists (6)
where the alias defined is a duplicate of an existing
alias for the client.ResourceExhausted (8)
where the target has insufficient memory or
processing resources to support the alias.Unknown (2)
in all other cases.Thus, for a client to create an alias corresponding to the path /a/b/c/d[id=10]/e
with the name shortPath
, it sends a SubscribeRequest
message with the following fields specified:
subscriberequest: <
aliases: <
alias: <
path: <
elem: <
name: "a"
>
elem: <
name: "b"
>
elem: <
name: "c"
>
elem: <
name: "d"
key: <
name: "id"
value: "10"
>
elem: <
name: "e"
>
>
alias: "#shortPath"
>
>
>
If the alias is acceptable to the target, subsequent updates are transmitted
using the #shortPath
alias in the same manner as described in Section
3.5.2.2.
Since multiple Update
messages can be included in the Notification
message
of a SubscribeResponse
, it is possible for a target to bundle messages such
that fewer messages are sent to the client. The advantage of such bundling is
clearly to reduce the number of bytes on the wire (caused by message overhead);
however, since only Notification
messages contain the timestamp at which an
event occurred, or a sample was taken, such bundling assigns a single timestamp
for all bundled Update
values. As such, it has the downside of negatively
affecting the sample accuracy and freshness to the client, and as a result, on
the client's ability to react to events on the target.
Since it is not possible for the target to infer whether its clients are
sensitive to the latency introduced by bundling, if a target implements
optimizations such that multiple Update
messages are bundled together,
it MUST provide an ability to disable this functionality within the
configuration of the gNMI service. Additionally, a target SHOULD provide means
by which the operator can control the maximum number of updates that are to be
bundled into a single message, This configuration is expected to be implemented
out-of-band to the gNMI protocol itself.
Where the use_aliases
field of a SubscriptionList
message has been set to
true
, a target MAY create aliases for paths within a subscription. A
target-defined alias MUST be created separately from an update to the
corresponding data item(s).
To create a target-defined alias, a SubscribeResponse
message is generated
with the update
field set to a Notification
message. The Notification
message specifies the aliased path within the prefix
field, and a non-null
alias
field, specified according to Section 2.4.2.
Thus, a target wishing to create an alias relating to the path /a/b/c[id=10]
and subsequently update children of the c[id=10]
entity must:
SubscribeResponse
message and transmit it over the RPC to the
client:subscriberesponse: <
update: <
timestamp: (timestamp)
prefix: <
elem: <
name: "a"
>
elem: <
name: "b"
>
elem: <
name: "c"
key: <
name: "id"
value: "10"
>
>
>
alias: "#42"
>
>
child1
leaf
corresponding to /a/b/c[id=10]/child1
:subscriberesponse: <
update: <
timestamp: (timestamp)
prefix: <
element: "#42"
>
update: <
path: <
elem: <
name: "child1"
>
>
value: <
val: <
int_val: 32
>
>
>
>
>
When an update for a subscribed telemetry path is to be sent, a
SubscribeResponse
message is sent from the target to the client, on the RPC
associated with the subscription. The update
field of the message contains a
Notification
message as per the description in Section
2.1. The timestamp
field of the
Notification
message MUST be set to the time at which the value of the path
that is being updated was collected from the underlying data source, or the
event being reported on (in the case of ON_CHANGE
occurred).
Where a leaf node's value has changed, or a new node has been created, an Update
message specifying the path and value for the updated data item MUST be appended to the update
field of the message.
Where a node within the subscribed paths has been removed, the delete
field of
the Notification
message MUST have the path of the node that has been removed
appended to it.
To replace the contents of an entire node within the tree, the target populates
the delete
field with the path of the node being removed, along with the new
contents within the update
field.
When the target has transmitted the initial updates for all paths specified
within the subscription, a SubscribeResponse
message with the sync_response
field set to true
MUST be transmitted to the client to indicate that the
initial transmission of updates has concluded. This provides an indication to
the client that all of the existing data for the subscription has been sent at
least once. For STREAM
subscriptions, such messages are not required for
subsequent updates. For POLL
subscriptions, after each set of updates for
individual poll request, a SubscribeResponse
message with the sync_response
field set to true
MUST be generated.
In the case where the updates_only
field in the SubscribeRequest
message has
been set, a sync_response
is sent as the first message on the stream, followed
by any updates representing subsequent changes to current state. For a POLL
or
ONCE
mode, this means that only a sync_response
will be sent. The
updates_only
field allows a client to only watch for changes, e.g. an update
to configuration.
The latest Protobuf IDL gNMI specification is found in GitHub at openconfig/gnmi.
Copyright 2016 Google Inc.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License
v0.6.0: January 25, 2018
extension
fields to the top-level RPCs of the gNMI service.v0.5.0: November 14, 2017
target
field within Path
message with description in Section
2.2.2.1
updates_only
field within SubscribeRequest
message.duplicates
field withing Update
message.v0.4.0: June 16, 2017
value
field within the Update
message, in favour of
the new TypedValue
field.google/rpc/status.proto
rather than
embedded messages within successful response codes.path
message's encoding of the root path.Path
to use the PathElem
message rather than to use a
repeated string
.v0.2.2: March 7, 2017
ON_CHANGE
subscriptions and the requirement for an
initial sync of matching subscription paths.Unimplemented
(fixes #36).v0.2.1: November 10, 2016
Correct reference to TEXT
vs. ASCII
encoding type.
Ensure that the encodings enumeration is numbered consistently.
Fix broken links.
This matches the types that Go UnixNano and Java TimeUnit toNanos return, and hence is used rather than an unsigned integer. ↩