The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT", "SHOULD", "SHOULD NOT", "RECOMMENDED", "MAY", and "OPTIONAL" in this document are to be interpreted as described in IETF BCP14 (RFC2119 & RFC8174)
Copyright (c) 2023 General Motors GTO LLC Licensed to the Apache Software Foundation (ASF) under one or more contributor license agreements. See the NOTICE file distributed with this work for additional information regarding copyright ownership. The ASF licenses this file to you 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. SPDX-FileType: DOCUMENTATION SPDX-FileCopyrightText: 2023 General Motors GTO LLC SPDX-License-Identifier: Apache-2.0
uE’s (Software Entities) will be deployed dynamically on devices that are scattered throughout the uProtocol network. In order for these dynamically deployed uE’s to be able to know about what is available on their own device or on other devices, we need the ability to discover not only uEs but also what uEs serve (resources, methods, messages, etc…) as well as properties about devices. There exists today a number of Internet protocols such as DNS, Zeroconf, and others, to discover devices and of services running on said devices. the goal is to reuse as much as possible, existing Internet standards for discovering of uThings as well.
The purpose of this proposal is to define the discovery architecture so that uEs can dynamically discovery uThings (other uEs, uDevices, uDomains, uResources, uMethods, etc…), for example what topics a service in another device shall publish, properties of a service, etc…
Term | Definition |
---|---|
DNS |
Domain Name System covered in various IETF RFCs |
CDS |
Central Discovery Service implementation of uDiscovery |
LDS |
Local (to a device) implementation of uDiscovery |
Local Node |
Node who is local to the device (local information) |
Remote Node |
Node that is fetched from the CDS and cached in the LDS |
uDiscovery provides the ability to discovery stuff that is addressable using UUri notation (devices, services, topics/methods) as well as properties of said stuff.
Note
|
Data model and interface is described in udiscovery.proto |
Note
|
Components described in uDiscovery Software Entities below all implement udiscovery.proto |
Component | Description | Requirements |
---|---|---|
Local uDiscovery Service (LDS) |
Service to provide discovery of uThings per device. Every LDS has-a database that stores all of its local Nodes as well as caches node information for remote devices. |
|
Domain uDiscovery Service (uDDS) |
Storage for discovery information for all devices within its domain |
|
Central uDiscovery Service (CDS) |
Central repository storing all deployed uDevice information in the network of connected devices residing in the Cloud. It is a superset of the local uDiscovery database content |
|
Function specific requirements shall be covered later in this document.
The following section will elaborate on the uProtocol hierarchy elaborating how information is organized, referenced, and structured. This is the system classification specification used to identify and define "things" along with the storage of information (properties) of the uThings in the registry.
Hierarchical Classification below introduces node and properties that we will define in this section. The diagram illustrates the hierarchy (taxonomy) for things in the network.
Nodes are addressable uThings like device, service, topics, etc…. Each node has-a list of properties as well as 0-n child nodes. The declarations of nodes and properties can be found in the udiscovery.proto, the snippet is below:
// Typedef for a node properties. A node property can be any one of the types
// defined below
message PropertyValue {
oneof attr {
bool u_boolean = 1; // Boolean
int32 u_integer = 2; // Integer
string u_string = 3; // String
bytes u_bytes = 4; // Raw Bytes
string u_uri = 5; // A URI
google.protobuf.Timestamp u_timestamp = 6; // Timestamp
}
}
// Node can be domain, device, service, resource, method, etc...
message Node {
// URI pointing to this node
string uri = 1;
// List of child nodes under this node
repeated Node nodes = 2;
// List of node properties
map <string, PropertyValue> properties = 3;
// The node type
Type type = 4;
// What is the uThing (stored in Node) type. This is used to more easily
// identify the Node rather than parsing from uri and inferring the type
enum Type {
INVALID = 0; // Invalid node type
DOMAIN = 1; // uDomain
DEVICE = 2; // uDevice
ENTITY = 3; // uEntity (uE)
VERSION = 9; // uE Version
TOPIC = 4; // uE Topic
METHOD = 5; // uE Method
MESSAGE = 6; // uE Message
RESOURCE = 7; // uE Resource
USER = 8; // User Information
}
}
-
The Node
uri
field MUST follow the URI specifications defined in uProtocol Specifications-
UE_VERSION MUST contain MAJOR
-
UE_VERSION MUST NOT contain MINOR and PATCH
-
Table below lists example URIs for the various node types in the database hierarchy.
Node Type | Example |
---|---|
domain |
up://UDOMAIN |
device |
up://UDEVICE.UDOMAIN |
ue |
up://UDEVICE.UDOMAIN/UE_NAME |
ue_version |
up://UDEVICE.UDOMAIN/UE_NAME/UE_VERSION |
topic |
up://UDEVICE.UDOMAIN/UE_NAME/UE_VERSION/RESOURCE#MESSAGE |
resource |
up://UDEVICE.UDOMAIN/UE_NAME/UE_VERSION/RESOURCE |
message |
up://UDEVICE.UDOMAIN/UE_NAME/UE_VERSION/#MESSAGE |
method |
up://UDEVICE.UDOMAIN/UE_NAME/UE_VERSION/rpc.METHOD |
-
YAML SHALL be used as the standard format for human-readable files (defining resources, services, properties, etc…)
-
JSON SHALL be used as the runtime (machine-readable) markup language
-
Identifiers nodes, and service names SHALL use lowercase a-z with underscore between words
-
The service and resource names SHALL use lowercase a-z with underscore between words
-
Interface (APIs) and event names SHALL use camel case notation starting with a capital letter. It is recommended to use only A-Z, a-z and 0-9 in node names
-
Resources SHALL have a singular name (ex door, sunroof, etc.)
Note
|
Please see Protobuf Style Guide for more details |
A property is a name-value pair of information that is declared using Protobuf Options. There are two types of properties:
-
uProtocol Properties: Required properties that all services must set, these are defined uprotocol-options.proto
-
uService Specific Properties: Properties that are declared in their respective service proto.
Services can declare any non-reserved identifier in their own proto files.
Note
|
It is STRONGLY RECOMMENDED to scope your property names to avoid namespace collision |
The uDiscovery service, through the query and update APIs, allows any uE to discover or change the contents of the local and central databases. Given that we obviously do not want any uE to access anything in the database, we need to build in safety checks that are validated in both the local and central discovery service.
Table below outlines the database access policies written like network firewall rules (top to bottom). The rules will be broken down for specific rules for the LDS vs CDS.
-
MUST allow local uE to read or write its own Node as well as its children Nodes
-
MUST allow local uE to read Nodes that it has associated permissions to do so
-
Fetched Nodes MUST be cached per ttl requirements
-
MUST allow uLDS to read or write its own Node or its children Nodes
-
Fetched Nodes MUST be cached per ttl requirements
Node metadata are stored outside the Node structure and describe the Node itself (freshness, etc…).
Attribute | Type | RFC2119 | Description |
---|---|---|---|
ttl |
int32 |
REQUIRED |
Time-to-Live. How long (in milliseconds) the Node is valid for before it is outdated and needs to be refreshed. When the value is -1 the Node is considered to be valid forever. A Node is expired when the following is true: \begin{array}{l}\displaystyle expired = t_{current} > t_{last_updated} + ttl\end{array} |
last_updated |
Timestamp |
REQUIRED |
Last time the content of the Node has changed (been written) |
last_accessed |
Timestamp |
OPTIONAL |
The last time the Node was read (accessed) from a FindNodes() API call |
inactive |
bool |
REQUIRED |
The Node has been tagged as inactive through the DeleteNode() API call |
API requirements related to Node metadata shall be covered in the subsequent section.
In the following section we will explain the various APIs and interfaces that are defined in uDiscovery and their requirements. Interface definitions (input and output parameters, etc…) are covered in the udiscovery.proto.
Notifications are used for replicating information between uLDS, uDDS, and uCDS, and to notify local uEs if/when the content of the database has changed for various reasons such as:
-
Installation of a new service version
-
Change in property values
-
Updates to device configurations
-
etc…
In order for uEs to receive notifications, the uE calls the uDiscovery API RegisterForNotifications()
passing NotificationsRequest
message that includes the list of URIs that it would like to be notified of changes, and a value of how deep in the tree should the change notification be sent. When the depth field is set to -1 or not present, the notifications shall be sent for changes to all children nodes. Below are a few high level requirements for uDiscovery notifications:
-
uE MUST be permitted to receive the notification (access the node). Permission is granted if the node is public or per [Appendix: Code-Based uE Access Permissions (CAPs)]
-
Notifications MUST be sent on the topic
/core.udiscovery/3/nodes#Notification
-
uCDS MUST only allow notification registration from uDDS, and uDDS registration from uLDS
Note
|
uCDS or uDDS MAY allow local notification registration when it is also acting as a uLDS for the local device |
-
uLDS MUST only accept registration for Node Updates from local uEs or from the CDS and MUST NOT accept notification registration from other uDevices uEs
Note
|
Dynamic discovery of the CDS is out of scope at the time of writing of this specification and as such the CDS authority is known to the LDS. The CDS does not need to call RegisterForNotifications() , the LDS simply sends the notification event to the CDS.
|
Query APIs are used to lookup content in the database, either to resolve URIs (to be used by applications) or to fetch content of a database.
-
Any uE MAY call the query APIs defined in the sections below
-
MUST NOT return Nodes that are flagged as
inactive
-
Remote Nodes that have
expired
MUST be refreshed to the CDS -
Locally
expired
Nodes MUST NOT be returned in a query
Used by any uEs to find service instance location, and its current version. What is returned is a list of Uri strings of fully qualified uris. The lookup searches the node database to find instances that match the search criteria.
Figure below illustrates the flows for performing a query to the LDS. An empty node, shown in the figure below, is a node with only the URI populated and is returned from LDS and CDS when the node is not found. The empty node is used by the LDS to know that a node does not exist in the CDS and we do not need to keep querying the same node.
-
MUST update
last_accessed
Node attribute when API is called
uDiscovery includes a set of APIs that allows uEs to change the content of the database. We will explain each APIs functions in the following section.
-
MUST only allow uEs to update their own Node
-
When
ttl
is not specified, MUST assume -1 (live forever) -
MUST verify caller has write permission to update, add, or delete a Node
-
MUST verify caller has write permission the parent node when adding or deleting a Node
-
MUST set the Node’s
last_updated
to the current time when a write API is called
Additional CDS Requirements:
-
LDS SHALL ONLY be permitted to update Node information for which the uDevice that the LDS runs on is in the list of ancestors of the Node.
Below we shall give an example of a service called uOTA
that will install a new service called uService
to Device1
illustrating how the AddNodes()
API could be used. We will also show how the Update notification is sent to two observers; local uApp and the CDS (a remote observer).
DeleteNode() API informs uDiscovery to tag a Node to be inactive. that the Node is no longer active Below shall provide an example of a service called uOTA
that shall remove a service called uService
from Device1
illustrating how the DeleteNode()
API can be used to remove a uE. We will also show how the Update notification is sent to two observers; a local uApp and the CDS (remote observer).
ResolveUri()
is used to lookup names from ids or vice versa meaning to go to/from Long Uri from/to micro Uri.
For portability between SDKs, Long Form URIs shall be serialized to String and Short form Uris shall be serialized to Bytes per URI Specifications.
-
MUST be passed either a Long or Short form URI and return both the Micro and Long form URI if successful