Introduction
The most recent articles discussed some issues and design patterns for machine interaction using hypermedia controls, and presented an experimental open source software platform and demonstrator for an example machine-to-machine hypermedia system.This article will summarize the progress on this topic over the last year and focus on the subject of asynchronous and realtime interactions using REST design principles and, in particular, hypermedia controls.
HSML: Hypermedia Content Formats for Machine Interaction
Since the last article, there is a new Internet Draft describing a content format for hypermedia collections on constrained networks. HSML is a simple representation and interaction model for collections and associated links, forms, and items, using CoRE Link-Format and SenML.https://datatracker.ietf.org/doc/draft-koster-t2trg-hsml/
Reusable Content Format
HSML is an updated version of the content format described in the earlier articles, and is kept aligned with the current versions of SenML and CoRE Link-Format.HSML representations include CoRE Link-Format and SenML, with some extensions to the Link-Format and SenML keyword vocabularies. In addition, there is a format which includes both Link-Format and SenML elements, to facilitate interactions with items and their link parameters in a collection atomically.
HSML provides for use of the Interface Type parameter (if) in links and request parameters, and is used to select representations of a resource as defined in CoRE Interfaces.
Common Transfer Layer
The HSML draft discusses a common abstract transfer layer using CRUD + Observe/Notify, also known as CRUD+N, which is used to map various concrete protocols, including HTTP, CoAP, and MQTT.This supports a common model for resource state interaction, upon which a consistent hypermedia interaction model may be constructed, and from there be extended.
CRUD, REST, and Hypermedia
This discussion is based on REST system design in the context of distributed state machine design.In REST system design, a distributed system is represented as a state transition graph. The nodes of the graphs represent states of the system, and the arcs of the graph represent possible state transitions.
The common resource design pattern known as CRUD exposes the system state contained in the nodes of the graph. Resource state may be Created, Retrieved, Updated, or Deleted.
Hypermedia controls expose the system state transition model at the edges of the graph. Hypermedia controls are the hyperlinks and submission forms, or their machine equivalents, that describe the available state transitions offered by the system, and how, from the client, they may be effected.
Hypermedia Controls Describe the Available State Transitions |
Without hypermedia controls, CRUD clients and servers will generally need to be more tightly coupled. These systems tend to use early binding of clients to servers, with schemas and introspection, code generation, and related techniques to drive client state machines to consume resources.
With hypermedia controls to dynamically describe state transitions, REST clients can consume the hypermedia controls in order to understand how to drive system state transitions, in a more loosely coupled, separately evolvable, late binding way.
Hypermedia controls describe possible state transitions based on the current state of the system, rather than pre-determined by a static resource design.
Interactive and Reactive Hypermedia: Actions and Link Bindings
In the last article, a class of hypermedia controls was described that facilitates asynchronous and realtime interaction between hypermedia clients and REST servers. These belong to a general class of interactive, and reactive, Hypermedia controls, that can direct the transfer and processing of dynamically changing data and asynchronous events.This article describes two types of this class of Interactive Hypermedia controls, Actions and Link Bindings, in more depth, with examples and terminology consistent with the HSML Internet Draft published in the IRTF Thing to Thing research group (T2TRG):
Problem statement and scope
Asynchronous interaction over REST involves use cases for asynchronous state transfer in both directions; from client to server, and from server to client.In the client to server case, the problem is in describing incoming state transitions which may or may not have immediate results. This enables the client to interactively direct the server processing of actions through the REST model.
In the server to client case, the problem includes describing how to use asynchronous communication to communicate resource state changes outgoing, in real time, to clients in order to enable reactive client processing of server state changes and events.
A third case is when a server device is to be controlled by another device, which also exposes a server role. The problem in this case is how to describe the client role involved in state transfer between two or more server resources.
Actions and Forms
Incoming (from the client to the server) state transitions are state transitions that a client wishes to make on a resource, such as turning a power switch on or opening a garage door.For incoming state transitions, the problem is how to inform the client what transitions are available and how to construct requests which change the resource state of the system in some way that may be more or less indirectly related to the control input. It is not always a simple and immediate CRUD state update. The state of the resource may need to be asynchronously monitored to complete the client state transition at some time in the future.
For example, an actuator may take some time to complete its action, like closing a motorized gate or garage door. The outcome may be uncertain, as in the case of power interruptions or mechanical blockage of the gate. These conditions result in a number of possible eventual outcomes which need to be returned to the client for processing and client state update.
A meta-model using the concept of an "action" is presented in the HSML draft. An Action describes and accepts a representation for a desired state change to a resource. The conceptual model is that the desired state change is "created" in the context of an Action resource, which then affects the state of the linked resource(s), and which can then be monitored for its eventual outcome.
An Action link is indicated by the value of "rel" including "action", and works in analogous fashion to a form in HTML; it describes what the action is expected to accomplish, like "turn on the light" or "add a post to the blog", and includes information on how to accomplish the action, like what method to use, which resource type, which content format, schema information to describe the payload, and descriptions of expected responses.
Action links can describe simple action operations that use REST state updates, or more complex actions which create representations of abstract action descriptions in collections of action instances. Such collections may be used to sequence submitted actions in FIFO or priority order, and for clients to track long running or pending actions which have not yet completed.
Actions may be performed directly on resources they are intended to update the state of, or they may be performed on proxy resources, as in the likely example of action instances crreated in a collection.
In the case that Actions are performed on a proxy resource, the resource may be indicated in the context of the affected resource(s) by including a link with the relation type including the value "alternate" and with a resource type indicating the type of action.
Link Bindings
For outgoing (from the server to the client) state transitions, there is the additional problem of describing how a client is expected to asynchronously obtain the state transitions as they occur, perhaps with some additional QoS parameters specified as system constraints.The concept of a Link Binding is presented in the HSML draft. A link binding, in this context and in the context of the CoRE dynlink Internet Draft, is a hyperlnk that defines a dynamic data update from one resource to another.
The description of link bindings assumes that there is a generalized REST-hook mechanism in place for the source of resource state updates, which can be used to trigger the link binding transfer operation. CoAP Observe may be used if available, or HTTP EventSource, or MQTT Subscribe.
Link Binding |
The link binding can be located with the information source and cause information to be pushed or published to the destination resource, or it may be located with the destination resource and cause the source resource to be subscribed, observed, or polled.
Asynchronous Notification using the "monitor" relation |
Link bindings may use different relation types to characterize transfer direction. In the examples above, the "monitor" relation type is used to describe a target resouce (monitor) which can be used to be informed of state changes in the context (monitored) resource. Another relation type is "boundto", as described in CoRE dynlink, which defines state transfer from the target resource to the context.
Link bindings may be defined with separate source and destination controls for transfer layer parameters like methods, content formats, and resource type queries. This enables link bindings to be generally used for resource state transfer, for triggering Actions, or for generating abstract events.
Link bindings may have different source and destination schemes, enabling link bindings to be used to convert protocols. For example, updates to a REST resource may be published to an MQTT broker using a link binding. Since the content format and other resource information can be obtained using HTTP, there is less need to augment the MQTT system with meta-data when used in this way.
For schemes that don't offer subscribe or observe, for example HTTP, RESThooks can be created and used in a structured way. Similarly, for systems using MQTT subscribe or CoAP Observe, link bindings enable the orchestration and inspection of dynamic resource interactions using graph techniques and tools.
Link bindings provide a means for conditional notification, using parameters to control the time period and value/change threshold for notification messages to be generated.
An extension to link bindings may be created which enables a link binding to consume a hypermedia action. This would enable cross-protocol adaptation without proxies, by using dynamic adaptation code in the libraries that consume the hypermedia controls.
System architecture examples
Interactive hypermedia controls enable the dynamic orchestration of client-server interactions by discovering actions and adding dynamic link bindings to connect resources, event sources, and event handlers. The following are some examples of system-level orchestration using interactive hypermedia.Actions in hypermedia clients
The Action control enables a generic client to adapt to the state machine of the server when performing actuation and invoking actions. The client consumes the hypermedia control, using the information obtained to select an action, marshal its parameters, and apply the action to the resource using the correct method and content format.Typically, the hypermedia control will be annotated with semantic identifiers to indicate the type(s) of action offered and the types of parameters expected, as well as any special response that may be returned.
Resouce servers may offer action design patterns optimized for the particular situation, and clients may adapt to a variety of patterns without a-priori knowledge, using hypermedia action controls.
Actions are indicated with a link relation "rel=action", as per the HSML draft.
In the examples below, a hypermedia application only needs to know the semantic meanings of the terms. The hypermedia client can adapt to the different server designs based on consuming the hypermedia action control.
Client wants the light to move to 77% brightness with a ramp time of 10 seconds
action = moveto ({targetbrightness = 77, units = %}, {ramptime = 10, units = seconds})
The client would perform a link query to discover a moveto action and verify that it know how to express the target parameters in the available units.
In this example, the client is looking for a brightness control with a "moveto" action type
GET /example/light/?if=core.ll
[
{
"href": "",
"rel": "self",
"rt": "urn:example:luminaire",
"content-format": 50,
"if": "core.ll"
},
{
"href": "brightness/",
"rt": "urn:example:brightness",
"content-format": 50,
"if": ["core.ll", "core.b"]
},
{
"anchor": "brightness/",
"href": "moveto/",
"rel": ["alternate", "action"],
"rt": "urn:example:moveto",
"method": "create",
"accept-format": 50,
"if": "core.b"
}
]
The resource at /example/light/moveto is an alternate resource for the collection at /example/light/brightness and offers action capability. The resource type (rt) is "moveto" which for action resources indicates the type of action.
Seeing the readable parameter interface on the brightness, the client could check the units and adapt if needed.
GET /example/light/brightness/?if=core.ll
[
{
"href": "",
"rel": "self",
"rt": "urn:example:brightness",
"content-format": 50,
"if": ["core.ll", "core.b"]
},
{
"href": "value",
"rt": "urn:example:value",
"content-format": 50,
"if": "core.rp"
},
{
"href": "ramptime",
"rt": "urn:example:ramptime",
"content-format": 50,
"if": "core.rp"
},
{
"href": "moveto/",
"rel": ["alternate", "action"],
"rt": "urn:example:moveto",
"method": "create",
"accept-format": 50,
"if": "core.b"
}
]
GET /example/light/brightness/value?if=core.rp
[
{
"n": "value",
"u": "%",
"vmin": 0,
"vmax": 100,
"vres": 0.1
}
]
GET /example/light/brightness/ramptime?if=core.rp
[
{
"n": "ramptime",
"u": "s",
"vmin": 0,
"vmax": 100,
"vres": 0.1
}
]
The client knows how to invoke the action at this point:
POST /example/light/brightness/moveto/?if=core.b
[
{
"n": "value",
"v": 77
},
{
"n": "ramptime",
"v": 10
}
]
Response:
2.04 CREATED
location: "77f3ac66"
The client knows that the location can be monitored for changes in the status of the action instance that was created.
GET /example/light/moveto/77f3ac66
[
{
"n": "value",
"v": 77
},
{
"n": "ramptime",
"v": 10
},
{
"n": "status",
"vs": "pending"
},
{
"n": "remainingtime",
"v": 10
}
]
Link bindings used to monitor resource state using REST callback
Perhaps the simplest case of a link binding involves monitoring the state of a resource and sending changes to a client using PUT to a REST API endpoint. This is often called a "web callback" pattern.The resource being monitored is the source of updates and is the context of the "monitor" link binding, and the target is the destination resource, in this example the callback URI.
The link binding would look like this:
[
{
"anchor": "/example/temperature/value",
"rel": "monitor",
"href": "https://example.com:1880/my-callback-uri"
}
]
Whenever the local resource at /example/temperature/value is updated, the representation of the resource will be pushed to the URI https://example.com:1880/my-callback-uri
By default, the source would use Observe or a REST hook and the destination (transfer method) would use update, by default PUT.
Using pubsub communication with REST resources
Link bindings can specify the source and target to be differnet protocols. For example, changes to the temperature value could be published to an MQTT broker, on an arbitrary topic name.[
{
"anchor": "/example/temperature/value",
"rel": "monitor",
"href": "mqtt://example.com:1883/temperature-topic",
"pmin": 10,
"pmax": 600,
"st": 1
}
]
The broker could be a reachable service in the cloud, and the temperature sensor could be in a sleepy device behind a NAT firewall, or in a mobile phone.
In this example, the conditional notification parameters indicate that the representation of the resource will be published when the value changes by 1 unit or more from the most recent publication, will be published no more frequently than once every 10 seconds no matter what the change, and will be published at least once every 600 seconds even if no change occurs.
Device-to-device orchestration using link Bindings
Link bindings may be used to orchestrate asynchronous updates from one device to another.For example, an on-off switch could be push updates to a controlled device upon changes of state.
[
{
"anchor": "/switch/onoff/value",
"rel": "monitor",
"href": "coap://[fdfd::9]/light/onoff/value"
}
]
References
Roy Fielding's Reference Work on RESThttps://www.ics.uci.edu/~fielding/pubs/dissertation/top.htm
Roy Fielding's note on Hypertext and REST APIs
http://roy.gbiv.com/untangled/2008/rest-apis-must-be-hypertext-driven
IRTF Thing to Thing Research Group (T2TRG)
https://datatracker.ietf.org/rg/t2trg/charter/
HSML
https://datatracker.ietf.org/doc/draft-koster-t2trg-hsml/
CoRE Link-Format RFC 6690
https://tools.ietf.org/html/rfc6690
CoRE Links in JSON and CBOR
https://tools.ietf.org/html/draft-ietf-core-links-json-07
CoRE SenML
https://tools.ietf.org/html/draft-ietf-core-senml-05
CoRE Dynlink
https://tools.ietf.org/html/draft-ietf-core-dynlink-03
CoRE Interfaces
https://tools.ietf.org/html/draft-ietf-core-interfaces-09
Event Models for RESTful APIs
http://iot-datamodels.blogspot.com/2013/05/event-models-for-restful-apis.html
REST to MQTT Bridge slides
https://www.slideshare.net/michaeljohnkoster/mqtt-rest-bridge
Comments
Post a Comment