mirror of
https://github.com/moparisthebest/xeps
synced 2024-11-15 13:55:03 -05:00
1757 lines
308 KiB
XML
1757 lines
308 KiB
XML
<?xml version='1.0' encoding='UTF-8'?>
|
|
<!DOCTYPE xep SYSTEM 'xep.dtd' [
|
|
<!ENTITY % ents SYSTEM 'xep.ent'>
|
|
%ents;
|
|
]>
|
|
<?xml-stylesheet type='text/xsl' href='xep.xsl'?>
|
|
<xep>
|
|
<header>
|
|
<title>Internet of Things - Provisioning</title>
|
|
<abstract>This specification describes an architecture for efficient provisioning of services, access rights and user privileges in for the Internet of Things, where
|
|
communication between Things is done using the XMPP protocol.</abstract>
|
|
&LEGALNOTICE;
|
|
<number>0324</number>
|
|
<status>Experimental</status>
|
|
<type>Standards Track</type>
|
|
<sig>Standards</sig>
|
|
<approver>Council</approver>
|
|
<dependencies>
|
|
<spec>XMPP Core</spec>
|
|
<spec>XEP-0001</spec>
|
|
<spec>XEP-0030</spec>
|
|
<spec>XEP-0323</spec>
|
|
<spec>XEP-0325</spec>
|
|
</dependencies>
|
|
<supersedes/>
|
|
<supersededby/>
|
|
<shortname>sensor-network-provisioning</shortname>
|
|
<author>
|
|
<firstname>Peter</firstname>
|
|
<surname>Waher</surname>
|
|
<email>peter.waher@clayster.com</email>
|
|
<jid>peter.waher@jabber.org</jid>
|
|
<uri>http://www.linkedin.com/in/peterwaher</uri>
|
|
</author>
|
|
<revision>
|
|
<version>0.2</version>
|
|
<date>2014-03-10</date>
|
|
<initials>pw</initials>
|
|
<remark>
|
|
<p>Corrected downloadPrivileges example.</p>
|
|
<p>Made several corrections of the language.</p>
|
|
<p>Expanded the introduction.</p>
|
|
<p>Changes "Sensor Networks" to "Internet of Things".</p>
|
|
<p>Fixed links to documents with new numbers.</p>
|
|
<p>Changed namespace urn:xmpp:sn to urn:xmpp:iot</p>
|
|
</remark>
|
|
</revision>
|
|
<revision>
|
|
<version>0.1</version>
|
|
<date>2013-04-16</date>
|
|
<initials>psa</initials>
|
|
<remark>
|
|
<p>Initial published version approved by the XMPP Council.</p>
|
|
</remark>
|
|
</revision>
|
|
<revision>
|
|
<version>0.0.5</version>
|
|
<date>2013-04-04</date>
|
|
<initials>pw</initials>
|
|
<remark>
|
|
<p>Added control use cases.</p>
|
|
<p>Grouped use cases.</p>
|
|
</remark>
|
|
</revision>
|
|
<revision>
|
|
<version>0.0.4</version>
|
|
<date>2013-04-01</date>
|
|
<initials>pw</initials>
|
|
<remark>
|
|
<p>Added altitude credentials.</p>
|
|
<p>Added resource information of original called to their corresponding JIDs.</p>
|
|
<p>Changed the return type of a rejected message.</p>
|
|
<p>Made images inline.</p>
|
|
<p>Converted the glossary into a definition list.</p>
|
|
</remark>
|
|
</revision>
|
|
<revision>
|
|
<version>0.0.3</version>
|
|
<date>2013-03-18</date>
|
|
<initials>pw</initials>
|
|
<remark>
|
|
<p>Added information about how to read sensors from large subsystems.</p>
|
|
<p>Added friend recommendation message.</p>
|
|
<p>Added client/device/service tokens.</p>
|
|
</remark>
|
|
</revision>
|
|
<revision>
|
|
<version>0.0.2</version>
|
|
<date>2013-03-12</date>
|
|
<initials>pw</initials>
|
|
<remark>
|
|
<p>Added use cases for service access rights and corresponding user privileges.</p>
|
|
</remark>
|
|
</revision>
|
|
<revision>
|
|
<version>0.0.1</version>
|
|
<date>2013-03-11</date>
|
|
<initials>pw</initials>
|
|
<remark>
|
|
<p>First draft.</p>
|
|
</remark>
|
|
</revision>
|
|
</header>
|
|
<section1 topic='Introduction' anchor='intro'>
|
|
<p>
|
|
This specification describes an architecture for efficient provisioning of services, access rights and user privileges in for the Internet of Things, where
|
|
communication between Things is done using the XMPP protocol.
|
|
</p>
|
|
<p>
|
|
Note has to be taken, that this XEP, and other Internet of Things-related XEP's, are designed for implementation in small devices, many of which have very limited
|
|
amount of memory (both RAM and ROM) or resources (processing power). Therefore, simplicity is of utmost importance. Furthermore, Internet of Things networks can
|
|
become huge, easily containing millions or billions of devices in peer-to-peer networks.
|
|
</p>
|
|
<p>
|
|
An added complexity in the provisioning case is that Things (small sensors for example) often have very limited user interface options. Therefore, this document
|
|
explains how provisioning can be done efficiently using a trusted third party with more power and options when it comes to user interface design and storage.
|
|
</p>
|
|
<p>
|
|
This document defines the following important operations to allow for efficient provisioning of services in the Internet of Things, based on XMPP:
|
|
</p>
|
|
<ul>
|
|
<li>What Things knows what Things</li>
|
|
<li>What Things can read data from what Things, and what data.</li>
|
|
<li>What Things can control what Things, and what parts.</li>
|
|
<li>Control of Users in the network.</li>
|
|
<li>Control of Services in the network.</li>
|
|
<li>Control generic boolean User Privileges in the network.</li>
|
|
</ul>
|
|
<p>
|
|
This XEP relies on &xep0323; and &xep0325; for sensor data readout and control interfaces.
|
|
Internet of Things contain many different architectures and use cases. For this reason, the IoT standards have been divided into multiple XEPs according to the following table:
|
|
</p>
|
|
<table caption='Internet of Things XEPs'>
|
|
<tr>
|
|
<th>XEP</th>
|
|
<th>Description</th>
|
|
</tr>
|
|
<tr>
|
|
<td>xep-0000-IoT-BatteryPoweredSensors</td>
|
|
<td>Defines how to handle the peculiars related to battery powered devices, and other devices intermittently available on the network.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>xep-0000-IoT-Discovery</td>
|
|
<td>Defines the peculiars of Thing discovery in Internet of Things. Apart from discovering Things by JID, it also defines how to discover Things based on location, etc.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>xep-0000-IoT-Events</td>
|
|
<td>Defines how Things send events, how event subscription, hysteresis levels, etc., are configured.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>xep-0000-IoT-Interoperability</td>
|
|
<td>Defines guidelines for how to achieve interoperability in Internet of Things, publishing interoperability interfaces for different types of devices.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>xep-0000-IoT-Multicast</td>
|
|
<td>Defines how sensor data can be multicast in efficient ways.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>xep-0000-IoT-PubSub</td>
|
|
<td>Defines how efficient publication of sensor data can be made in Internet of Things.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>xep-0000-IoT-Chat</td>
|
|
<td>Defines how human-to-machine interfaces should be constructed using chat messages to be user friendly, automatable and consistent with other IoT extensions and possible underlying architecture.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>XEP-0322</td>
|
|
<td>
|
|
Defines how to EXI can be used in XMPP to achieve efficient compression of data. Albeit not a Internet of Things specific XEP, this XEP should be considered
|
|
in all Internet of Things implementations where memory and packet size is an issue.
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>XEP-0323</td>
|
|
<td>
|
|
Provides the underlying architecture, basic operations and data structures for sensor data communication over XMPP networks.
|
|
It includes a hardware abstraction model, removing any technical detail implemented in underlying technologies. This XEP is used by all other
|
|
Internet of Things XEPs.
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>XEP-0324</td>
|
|
<td>This specification. Defines how provisioning, the management of access privileges, etc., can be efficiently and easily implemented.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>XEP-0325</td>
|
|
<td>Defines how to control actuators and other devices in Internet of Things.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>XEP-0326</td>
|
|
<td>Defines how to handle architectures containing concentrators or servers handling multiple Things.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>XEP-0331</td>
|
|
<td>Defines extensions for how color parameters can be handled, based on &xep0004;</td>
|
|
</tr>
|
|
<tr>
|
|
<td>XEP-0336</td>
|
|
<td>Defines extensions for how dynamic forms can be created, based on &xep0004;, &xep0122;, &xep0137; and &xep0141;.</td>
|
|
</tr>
|
|
</table>
|
|
</section1>
|
|
<section1 topic='Glossary' anchor='glossary'>
|
|
<p>The following table lists common terms and corresponding descriptions.</p>
|
|
<dl>
|
|
<di>
|
|
<dt>Actuator</dt>
|
|
<dd>Device containing at least one configurable property or output that can and should be controlled by some other entity or device.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Authority</dt>
|
|
<dd>Used synonymously with Provisioning Server.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Computed Value</dt>
|
|
<dd>A value that is computed instead of measured.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Concentrator</dt>
|
|
<dd>Device managing a set of devices which it publishes on the XMPP network.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Field</dt>
|
|
<dd>
|
|
One item of sensor data. Contains information about: Node, Field Name, Value, Precision, Unit, Value Type, Status, Timestamp, Localization information, etc.
|
|
Fields should be unique within the triple (Node ID, Field Name, Timestamp).
|
|
</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Field Name</dt>
|
|
<dd>Name of a field of sensor data. Examples: Energy, Volume, Flow, Power, etc.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Field Type</dt>
|
|
<dd>What type of value the field represents. Examples: Momentary Value, Status Value, Identification Value, Calculated Value, Peak Value, Historical Value, etc.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Historical Value</dt>
|
|
<dd>A value stored in memory from a previous timestamp.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Identification Value</dt>
|
|
<dd>A value that can be used for identification. (Serial numbers, meter IDs, locations, names, etc.)</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Localization information</dt>
|
|
<dd>Optional information for a field, allowing the sensor to control how the information should be presented to human viewers.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Meter</dt>
|
|
<dd>A device possible containing multiple sensors, used in metering applications. Examples: Electricity meter, Water Meter, Heat Meter, Cooling Meter, etc.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Momentary Value</dt>
|
|
<dd>A momentary value represents a value measured at the time of the read-out.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Node</dt>
|
|
<dd>
|
|
Graphs contain nodes and edges between nodes. In Internet of Things, sensors, actuators, meters, devices, gateways, etc., are often depicted as nodes whereas links between sensors (friendships)
|
|
are depicted as edges. In abstract terms, it's easier to talk about a Node, rather than list different possible node types (sensors, actuators, meters, devices, gateways, etc.).
|
|
Each Node has a Node ID.
|
|
</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Node ID</dt>
|
|
<dd>
|
|
An ID uniquely identifying a node within its corresponding context. If a globally unique ID is desired, an architecture should be used using a universally accepted
|
|
ID scheme.
|
|
</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Parameter</dt>
|
|
<dd>
|
|
Readable and/or writable property on a node/device. The XEP-0326 &xep0326; deals with reading and writing parameters
|
|
on nodes/devices. Fields are not parameters, and parameters are not fields.
|
|
</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Peak Value</dt>
|
|
<dd>A maximum or minimum value during a given period.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Provisioning Server</dt>
|
|
<dd>An application that can configure a network and provide services to users or Things. In Internet of Things, a Provisioning Server knows who knows whom,
|
|
what privileges users have, who can read what data and who can control what devices and what parts of these devices.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Precision</dt>
|
|
<dd>
|
|
In physics, precision determines the number of digits of precision. In sensor networks however, this definition is not easily applicable. Instead, precision
|
|
determines, for example, the number of decimals of precision, or power of precision. Example: 123.200 MWh contains 3 decimals of precision. All entities parsing and
|
|
delivering field information in sensor networks should always retain the number of decimals in a message.
|
|
</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Sensor</dt>
|
|
<dd>
|
|
Device measuring at least one digital value (0 or 1) or analog value (value with precision and physical unit). Examples: Temperature sensor, pressure sensor, etc.
|
|
Sensor values are reported as fields during read-out. Each sensor has a unique Node ID.
|
|
</dd>
|
|
</di>
|
|
<di>
|
|
<dt>SN</dt>
|
|
<dd>Sensor Network. A network consisting, but not limited to sensors, where transport and use of sensor data is of primary concern. A sensor network may contain actuators, network applications, monitors, services, etc.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Status Value</dt>
|
|
<dd>A value displaying status information about something.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Timestamp</dt>
|
|
<dd>Timestamp of value, when the value was sampled or recorded.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Thing</dt>
|
|
<dd>
|
|
Internet of Things basically consists of Things connected to the Internet. Things can be any device, sensor, actuator etc., that can have an
|
|
Internet connection.
|
|
</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Thing Registry</dt>
|
|
<dd>
|
|
A registry where Things can register for simple and secure discovery by the owner of the Thing. The registry can also be used as a database for meta information
|
|
about Things in the network.
|
|
</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Token</dt>
|
|
<dd>
|
|
A client, device or user can get a token from a provisioning server. These tokens can be included in requests to other entities in the network, so these entities can validate
|
|
access rights with the provisioning server.
|
|
</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Unit</dt>
|
|
<dd>Physical unit of value. Example: MWh, l/s, etc.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Value</dt>
|
|
<dd>A field value.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Value Status</dt>
|
|
<dd>Status of field value. Contains important status information for Quality of Service purposes. Examples: Ok, Error, Warning, Time Shifted, Missing, Signed, etc.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>Value Type</dt>
|
|
<dd>Can be numeric, string, boolean, Date & Time, Time Span or Enumeration.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>WSN</dt>
|
|
<dd>Wireless Sensor Network, a sensor network including wireless devices.</dd>
|
|
</di>
|
|
<di>
|
|
<dt>XMPP Client</dt>
|
|
<dd>Application connected to an XMPP network, having a JID. Note that sensors, as well as applications requesting sensor data can be XMPP clients.</dd>
|
|
</di>
|
|
</dl>
|
|
</section1>
|
|
<section1 topic='Use Cases' anchor='usecases'>
|
|
<p>
|
|
The most basic use case in sensor networks is to read out sensor data from a sensor. However, since protecting end-user integrity and system security is vital, access
|
|
rights and user privileges have to be imposed on the network.
|
|
</p>
|
|
<p>
|
|
To store access rights in all sensors might be very impractical. Not only does it consume memory, it's difficult to maintain track of the current system status, make sure
|
|
all devices have the latest configuration, distribute changes to the configuration, etc.
|
|
</p>
|
|
<p>
|
|
Furthermore, most sensors and small devices have very limited possibility to provide a rich user interface. Perhaps all it can do is to provide a small LED and a button,
|
|
useful perhaps for installing the sensor in the network, but not much more.
|
|
</p>
|
|
<p>
|
|
As an added complexity, the sensor network operator might not even have access to the XMPP Servers used, and provisioning needs to lie outside of the XMPP Server domains.
|
|
</p>
|
|
<p>
|
|
To solve this problem in an efficient manner, an architecture using distributed trusted third parties is proposed. Such third parties would:
|
|
</p>
|
|
<ul>
|
|
<li>Provide a rich user interface and configurable options to end user or back end systems.</li>
|
|
<li>Control friendships (who can communicate with whom).</li>
|
|
<li>Control content available for different friends (what can be read by whom).</li>
|
|
<li>Control operations accessible by different friends (what can be controlled/configured by whom).</li>
|
|
<li>Provide additional interoperability services to nodes in the network (for instance, unit conversion).</li>
|
|
</ul>
|
|
<section2 topic='Delegating trust'>
|
|
<section3 topic='Delegating original trust to a Provisioning Server'>
|
|
<p>
|
|
Trust is delegated to a provisioning server by a device, simply by befriending the provisioning server and asking it questions and complying with
|
|
its answers.
|
|
</p>
|
|
<p>
|
|
As an illustrative example, following is a short description of how such a trust relationship can be created in a scenario where the sensor only
|
|
has a single LED and a single button.
|
|
</p>
|
|
<ul>
|
|
<li>Somebody is installing the sensor, giving it a connection to an XMPP server and a JID, reachable from the provisioning server.</li>
|
|
<li>The provisioning server is told to create a friendship request to the new sensor.</li>
|
|
<li>The sensor flashes its LED for a given time (for example: 30 seconds).</li>
|
|
<li>Viewing the LED, the person installing the sensor presses the button.</li>
|
|
<li>Receiving the button press within the given time period, accepts the friendship request. Optionally, the device can give user feedback using the LED.</li>
|
|
<li>The device performs a service discovery of the new friend, having been a manually added friend.</li>
|
|
<li>If the new friend supports this provisioning extension, further responsibilities are delegated to this device.</li>
|
|
<li>As the last step the device asks the provisioning server for a token. This device token is later used in calls to other devices and can be used to check access rights.</li>
|
|
</ul>
|
|
<p>
|
|
The following diagram shows the general case:
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
<p>
|
|
The successful case can also be illustrated in a sequence diagram, as follows:
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
<p>
|
|
The following example shows how a device can request a device token from the provisioning server:
|
|
</p>
|
|
<example caption='Requesting a device token'>
|
|
<![CDATA[
|
|
<iq type='get'
|
|
from='device@clayster.com/device'
|
|
to='provisioning@clayster.com'
|
|
id='13'>
|
|
<getToken xmlns='urn:xmpp:iot:provisioning' deviceId='WatchamacallitDevice1'/>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='device@clayster.com/device'
|
|
id='13'>
|
|
<getTokenResponse xmlns='urn:xmpp:iot:provisioning' token='device0001'/>
|
|
</iq>]]>
|
|
</example>
|
|
<p>
|
|
The provisioning server must not return tokens that contain white space characters.
|
|
</p>
|
|
<p>
|
|
<strong>Note:</strong> In many cases, an address to a provisioning server might be preprogrammed during production of the
|
|
device. In these cases, parts of the above procedure may not be necessary. All the client needs to do, if the provisioning server is not available
|
|
in the roster of the device, is to send a subscription request to the provisioning server, to alert the server of the existence of the device,
|
|
and possibly request a device token.
|
|
</p>
|
|
<p>
|
|
The following use cases will assume such a trust relationship has been created between the corresponding device and the provisioning server.
|
|
</p>
|
|
</section3>
|
|
<section3 topic='Delegating Secondary Trust'>
|
|
<p>
|
|
The <strong>isFriendResponse</strong> element returned by the provisioning server contains an attribute <strong>secondaryTrustAllowed</strong> that is by default
|
|
set to false. If the provisioning server has no problem with allowing multiple trust to be delegated by devices in the network, it can choose to set this
|
|
attribute to true in the response. If true, the device knows it has the right to add its own friends, or to add secondary trust relationships.
|
|
</p>
|
|
<p>
|
|
The following diagram continues with the example given above, of how a sensor with a limited user interface, can allow to manually add new friends, including new
|
|
trust relationships using a single LED and a button.
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
</section3>
|
|
<section3 topic='Multiple tokens'>
|
|
<p>
|
|
When multiple trust is used, the entity (client, user, service, etc.) has one token from each provisioning server. However, when sending a token to a third party,
|
|
the sender does not know what provisioning server(s) the third party uses to check access rights and user privileges. Therefore, the client must send all tokens, separated
|
|
by a space.
|
|
</p>
|
|
<p>
|
|
When a provisioning server receives a request containing multiple tokens, the most forgiving response must be returned.
|
|
</p>
|
|
<example caption='Readout request using multiple tokens'>
|
|
<![CDATA[
|
|
<iq type='get'
|
|
from='master@clayster.com/amr'
|
|
to='device@clayster.com'
|
|
id='15'>
|
|
<req xmlns='urn:xmpp:iot:sensordata' momentary='true' serviceToken='service0001 servicio0001' userToken='user0001' seqnr='4'/>
|
|
</iq>
|
|
|
|
<iq type='get'
|
|
from='device@clayster.com/device'
|
|
to='provisioning@clayster.com'
|
|
id='16'>
|
|
<canRead xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' serviceToken='service0001 servicio0001' userToken='user0001' momentary='true'/>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='device@clayster.com/device'
|
|
id='16'>
|
|
<canReadResponse xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' momentary='true' result='true'/>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='device@clayster.com'
|
|
to='master@clayster.com/amr'
|
|
id='15'>
|
|
<accepted xmlns='urn:xmpp:iot:sensordata' seqnr='4'/>
|
|
</iq>]]>
|
|
</example>
|
|
</section3>
|
|
</section2>
|
|
<section2 topic='Friendships'>
|
|
<section3 topic='Friendship request accepted'>
|
|
<p>
|
|
The following diagram displays how a friendship request from an external party can be handled, delegating the responsibility to a trusted third party:
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
<p>
|
|
The communication between the XMPP Device and the Provisioning Server could be as follows:
|
|
</p>
|
|
<example caption='Friendship request accepted'>
|
|
<![CDATA[
|
|
<iq type='get'
|
|
from='device@clayster.com/device'
|
|
to='provisioning@clayster.com'
|
|
id='1'>
|
|
<isFriend xmlns='urn:xmpp:iot:provisioning' jid='client1@clayster.com'/>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='device@clayster.com/device'
|
|
id='1'>
|
|
<isFriendResponse xmlns='urn:xmpp:iot:provisioning' jid='client1@clayster.com' result='true'/>
|
|
</iq>]]>
|
|
</example>
|
|
<p>
|
|
<strong>Note:</strong> The provisioning server implicitly understands which two JIDs that are to be checked: The first one is the sender of the message, the second one
|
|
is the JID available in the <strong>jid</strong> attribute in the request.
|
|
</p>
|
|
<p>
|
|
<strong>Note 2:</strong> Any resource information in the JID must be ignored by the provisioning server.
|
|
</p>
|
|
</section3>
|
|
<section3 topic='Friendship request rejected'>
|
|
<p>
|
|
The following diagram displays a friendship request from an external party being rejected as a result of the trusted third party negating the friendship:
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
<p>
|
|
The communication between the XMPP Device and the Provisioning Server could be as follows:
|
|
</p>
|
|
<example caption='Friendship request rejected'>
|
|
<![CDATA[
|
|
<iq type='get'
|
|
from='device@clayster.com/device'
|
|
to='provisioning@clayster.com'
|
|
id='2'>
|
|
<isFriend xmlns='urn:xmpp:iot:provisioning' jid='client2@clayster.com'/>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='device@clayster.com/device'
|
|
id='2'>
|
|
<isFriendResponse xmlns='urn:xmpp:iot:provisioning' jid='client2@clayster.com' result='false'/>
|
|
</iq>]]>
|
|
</example>
|
|
</section3>
|
|
<section3 topic='Unfriending existing friends'>
|
|
<p>
|
|
If the provisioning server decides that two friends in the network should no longer be friends and communicate with each other, it simply sends a message to
|
|
at least one of the friends as follows:
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
<p>
|
|
The provisioning server should only send such messages to clients that have previously asked the provisioning server if friendship requests should be accepted or not.
|
|
</p>
|
|
<p>
|
|
<strong>Note:</strong> The device should only honor such messages, if the sender is the trusted third party. Such messages received from other entities not trusted should
|
|
be silently ignored.
|
|
</p>
|
|
|
|
<example caption='Unfriending existing friend'>
|
|
<![CDATA[
|
|
<message from='provisioning@clayster.com'
|
|
to='device@clayster.com'>
|
|
<unfriend xmlns='urn:xmpp:iot:provisioning' jid='client2@clayster.com'/>
|
|
</message>]]>
|
|
</example>
|
|
</section3>
|
|
<section3 topic='Recommending friendships'>
|
|
<p>
|
|
The provisioning server can, apart from accepting new friendships and rejecting old friendships, also recommend new friendships. In this case, the provisioning server
|
|
simply sends a message to one or both of the soon to be friends, as follows:
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
<example caption='Recommending friendships'>
|
|
<![CDATA[
|
|
<message from='provisioning@clayster.com'
|
|
to='device@clayster.com'>
|
|
<friend xmlns='urn:xmpp:iot:provisioning' jid='client2@clayster.com'/>
|
|
</message>]]>
|
|
</example>
|
|
<p>
|
|
Note that the receptor can still ask the provisioning server if it can form a friendship with the suggested friend, using the <strong>isFriend</strong> command.
|
|
</p>
|
|
</section3>
|
|
</section2>
|
|
<section2 topic='Device Read-out'>
|
|
<section3 topic='Rejecting read-outs'>
|
|
<p>
|
|
An important use case for provisioning in sensor networks is who gets to read out sensor data from which sensors. This use case details how communication with a
|
|
provisioning server can help the device determine if a client has sufficient access rights to read the values of the device.
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
<p>
|
|
<strong>Note:</strong> This use case is an extension of the use case 'Read-out rejected' in the XEP-0323
|
|
<link url='http://xmpp.org/extensions/xep-0323.html'>Internet of Things - Sensor Data</link>.
|
|
</p>
|
|
<p>
|
|
The following example shows the communication first between the client and the device, then between the device and the provisioning server, and last between the device and the client:
|
|
</p>
|
|
<example caption='Rejecting read-outs'>
|
|
<![CDATA[
|
|
<iq type='get'
|
|
from='master@clayster.com/amr'
|
|
to='device@clayster.com'
|
|
id='3'>
|
|
<req xmlns='urn:xmpp:iot:sensordata' momentary='true' serviceToken='service0001' userToken='user0001' seqnr='1'/>
|
|
</iq>
|
|
|
|
<iq type='get'
|
|
from='device@clayster.com/device'
|
|
to='provisioning@clayster.com'
|
|
id='4'>
|
|
<canRead xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' serviceToken='service0001' userToken='user0001' momentary='true'/>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='device@clayster.com/device'
|
|
id='4'>
|
|
<canReadResponse xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' momentary='true' result='false'/>
|
|
</iq>
|
|
|
|
<iq type='error'
|
|
from='device@clayster.com'
|
|
to='master@clayster.com/amr'
|
|
id='3'>
|
|
<rejected xmlns='urn:xmpp:iot:sensordata' seqnr='1'>
|
|
<error>Access denied.</error>
|
|
</rejected>
|
|
</iq>]]>
|
|
</example>
|
|
</section3>
|
|
<section3 topic='Restricting nodes during read-out'>
|
|
<p>
|
|
In case the device handles multiple nodes that can be read, the provisioning server has the possibility to grant read-out, but to limit the nodes that can be read out.
|
|
The provisioning server does this by returning the list of nodes that can be read.
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
<p>
|
|
<strong>Note:</strong> This use case is an extension of the use case 'Read-out of multiple devices' in the XEP-0323
|
|
<link url='http://xmpp.org/extensions/xep-0323.html'>Internet of Things - Sensor Data</link>.
|
|
</p>
|
|
<p>
|
|
<strong>Note 2:</strong> If the server responds, but without specifying a list of nodes, the device can assume that all nodes available in the original request are allowed
|
|
to be read. If no nodes in the request are allowed to be read, the provisioning server must respond with a result='false', so the device can reject the read-out request.
|
|
</p>
|
|
<p>
|
|
The following example shows the communication first between the client and the device, then between the device and the provisioning server, and last between the device and the client:
|
|
</p>
|
|
|
|
<example caption='Restricting nodes during read-out'>
|
|
<![CDATA[
|
|
<iq type='get'
|
|
from='master@clayster.com/amr'
|
|
to='device@clayster.com'
|
|
id='5'>
|
|
<req xmlns='urn:xmpp:iot:sensordata' momentary='true' serviceToken='service0001' userToken='user0001' seqnr='2'>
|
|
<node nodeId='Device02'/>
|
|
<node nodeId='Device03'/>
|
|
</req>
|
|
</iq>
|
|
|
|
<iq type='get'
|
|
from='device@clayster.com/device'
|
|
to='provisioning@clayster.com'
|
|
id='6'>
|
|
<canRead xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' momentary='true' serviceToken='service0001' userToken='user0001'>
|
|
<node nodeId='Device02'/>
|
|
<node nodeId='Device03'/>
|
|
</canRead>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='device@clayster.com/device'
|
|
id='6'>
|
|
<canReadResponse xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' momentary='true' result='true'>
|
|
<node nodeId='Device02'/>
|
|
</canReadResponse>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='device@clayster.com'
|
|
to='master@clayster.com/amr'
|
|
id='5'>
|
|
<accepted xmlns='urn:xmpp:iot:sensordata' seqnr='2'/>
|
|
</iq>]]>
|
|
</example>
|
|
<p>
|
|
Note that the provisioning server responds with a <strong>canReadResponse</strong> element, similar to the <strong>canRead</strong> element in the request, except
|
|
only the nodes allowed to be read are read. The device must only permit read-out of nodes listed in the response from the provisioning server. Other nodes available
|
|
in the request should be ignored.
|
|
</p>
|
|
</section3>
|
|
<section3 topic='Restricting fields during read-out'>
|
|
<p>
|
|
In case the provisioning server wants to limit the fields a device can send to a client, the provisioning server has the possibility to grant read-out, but
|
|
list a set of fields the device is allowed to send to the corresponding client.
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
<p>
|
|
<strong>Note:</strong> If the server responds, but without specifying a list of field names, the device can assume that all fields available in the original request are allowed
|
|
to be sent. If no fields in the request are allowed to be sent, the provisioning server must respond with a result='false', so the device can reject the read-out request.
|
|
</p>
|
|
<p>
|
|
The following example shows the communication first between the client and the device, then between the device and the provisioning server, and last between the device and the client:
|
|
</p>
|
|
|
|
<example caption='Restricting fields during read-out'>
|
|
<![CDATA[
|
|
<iq type='get'
|
|
from='master@clayster.com/amr'
|
|
to='device@clayster.com'
|
|
id='7'>
|
|
<req xmlns='urn:xmpp:iot:sensordata' momentary='true' serviceToken='service0001' userToken='user0001' seqnr='3'/>
|
|
</iq>
|
|
|
|
<iq type='get'
|
|
from='device@clayster.com/device'
|
|
to='provisioning@clayster.com'
|
|
id='8'>
|
|
<canRead xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' momentary='true' serviceToken='service0001' userToken='user0001'/>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='device@clayster.com/device'
|
|
id='8'>
|
|
<canReadResponse xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' momentary='true' result='true'>
|
|
<field name='Energy'/>
|
|
<field name='Power'/>
|
|
</canReadResponse>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='device@clayster.com'
|
|
to='master@clayster.com/amr'
|
|
id='7'>
|
|
<accepted xmlns='urn:xmpp:iot:sensordata' seqnr='3'/>
|
|
</iq>]]>
|
|
</example>
|
|
<p>
|
|
Note that the provisioning server responds with a <strong>canReadResponse</strong> element, similar to the <strong>canRead</strong> element in the request, except only
|
|
the fields allowed to be sent are listed. The client must only send fields having field names in this list.
|
|
</p>
|
|
<p>
|
|
Also note, that the provisioning server can return a list of both allowed nodes and allowed field names in the response. In this case, the device must only send allowed fields
|
|
from allowed nodes, and ignore all other fields and/or nodes.
|
|
</p>
|
|
</section3>
|
|
</section2>
|
|
<section2 topic='Device Control'>
|
|
<section3 topic='Rejecting control actions'>
|
|
<p>
|
|
An important use case for provisioning in sensor networks is who gets to control devices, and what they can control. This use case details how communication with a
|
|
provisioning server can help the device determine if a client has sufficient access rights to perform control actions on the device.
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
<!-- sensor-network-provisioning-15.png-->
|
|
</p>
|
|
<p>
|
|
The following example shows the communication first between the client and the device, then between the device and the provisioning server, and last between the device and the client:
|
|
</p>
|
|
<example caption='Rejecting control action'>
|
|
<![CDATA[
|
|
<iq type='set'
|
|
from='master@clayster.com/amr'
|
|
to='device@clayster.com'
|
|
id='18'>
|
|
<set xmlns='urn:xmpp:iot:control' xml:lang='en'>
|
|
<boolean name='Output' value='true'/>
|
|
</set>
|
|
</iq>
|
|
|
|
<iq type='get'
|
|
from='device@clayster.com/device'
|
|
to='provisioning@clayster.com'
|
|
id='1'>
|
|
<canControl xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' serviceToken='service0001' userToken='user0001'>
|
|
<parameter name='Output'/>
|
|
</canControl>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='device@clayster.com/device'
|
|
id='1'>
|
|
<canControlResponse xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' result='false'/>
|
|
</iq>
|
|
|
|
<iq type='error'
|
|
from='device@clayster.com'
|
|
to='master@clayster.com/amr'
|
|
id='18'>
|
|
<setResponse xmlns='urn:xmpp:iot:control' responseCode='InsufficientPrivileges'/>
|
|
</iq>]]>
|
|
</example>
|
|
</section3>
|
|
<section3 topic='Restricting nodes during control'>
|
|
<p>
|
|
In case the device handles multiple nodes that can be read, the provisioning server has the possibility to grant control access, but to limit the nodes that can be controlled.
|
|
The provisioning server does this by returning the list of nodes that can be controlled.
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
<!-- sensor-network-provisioning-14.png-->
|
|
</p>
|
|
<p>
|
|
<strong>Note:</strong> If the server responds, but without specifying a list of nodes, the device can assume that all nodes available in the original request are allowed
|
|
to be controlled. If no nodes in the request are allowed to be controlled, the provisioning server must respond with a result='false', so the device can reject the read-out request.
|
|
The same is true for parameters: If the provisioning server does not specify parameters in the response, the caller can assume all parameters are allowed.
|
|
</p>
|
|
<p>
|
|
The following example shows the communication first between the client and the device, then between the device and the provisioning server, and last between the device and the client:
|
|
</p>
|
|
<example caption='Restricting nodes during control'>
|
|
<![CDATA[
|
|
<iq type='set'
|
|
from='master@clayster.com/amr'
|
|
to='concentrator@clayster.com'
|
|
id='19'>
|
|
<set xmlns='urn:xmpp:iot:control' xml:lang='en'>
|
|
<node nodeId='DigitalOutput1'/>
|
|
<node nodeId='DigitalOutput2'/>
|
|
<node nodeId='DigitalOutput3'/>
|
|
<node nodeId='DigitalOutput4'/>
|
|
<boolean name='Output' value='true'/>
|
|
</set>
|
|
</iq>
|
|
|
|
<iq type='get'
|
|
from='concentrator@clayster.com/plc'
|
|
to='provisioning@clayster.com'
|
|
id='2'>
|
|
<canControl xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' serviceToken='service0001' userToken='user0001'>
|
|
<node nodeId='DigitalOutput1'/>
|
|
<node nodeId='DigitalOutput2'/>
|
|
<node nodeId='DigitalOutput3'/>
|
|
<node nodeId='DigitalOutput4'/>
|
|
<parameter name='Output'/>
|
|
</canControl>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='concentrator@clayster.com/plc'
|
|
id='2'>
|
|
<canControlResponse xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' result='true'>
|
|
<node nodeId='DigitalOutput2'/>
|
|
<node nodeId='DigitalOutput3'/>
|
|
</canControlResponse>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='concentrator@clayster.com'
|
|
to='master@clayster.com/amr'
|
|
id='19'>
|
|
<setResponse xmlns='urn:xmpp:iot:control' responseCode='OK'>
|
|
<node nodeId='DigitalOutput2'/>
|
|
<node nodeId='DigitalOutput3'/>
|
|
</setResponse>
|
|
</iq>]]>
|
|
</example>
|
|
<p>
|
|
Note that the provisioning server responds with a <strong>canControlResponse</strong> element, similar to the <strong>canControl</strong> element in the request, except
|
|
only the nodes allowed to be controlled are included. The device must only permit control of nodes listed in the response from the provisioning server. Other nodes available
|
|
in the request should be ignored.
|
|
</p>
|
|
<p>
|
|
Also note, that the restricted set of nodes and/or parameters returned from the provisioning server must be returned to the original caller, so it can
|
|
act on the information that only a partial control action was allowed and taken.
|
|
</p>
|
|
</section3>
|
|
<section3 topic='Restricting parameters during control'>
|
|
<p>
|
|
In case the provisioning server wants to limit the control parameters a client can control in a device, the provisioning server has the possibility to grant
|
|
control access, but list a set of parameters the client is allowed to control in the corresponding device.
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
<!-- sensor-network-provisioning-14.png-->
|
|
</p>
|
|
<p>
|
|
<strong>Note:</strong> If the server responds, but without specifying a list of nodes, the device can assume that all nodes available in the original request are allowed
|
|
to be controlled. If no nodes in the request are allowed to be controlled, the provisioning server must respond with a result='false', so the device can reject the read-out request.
|
|
The same is true for parameters: If the provisioning server does not specify parameters in the response, the caller can assume all parameters are allowed.
|
|
</p>
|
|
<p>
|
|
The following example shows the communication first between the client and the device, then between the device and the provisioning server, and last between the device and the client:
|
|
</p>
|
|
<example caption='Restricting parameters during control'>
|
|
<![CDATA[
|
|
<iq type='set'
|
|
from='master@clayster.com/amr'
|
|
to='plc@clayster.com'
|
|
id='20'>
|
|
<set xmlns='urn:xmpp:iot:control' xml:lang='en'>
|
|
<boolean name='DigitalOutput1' value='true'/>
|
|
<boolean name='DigitalOutput2' value='true'/>
|
|
<boolean name='DigitalOutput3' value='true'/>
|
|
<boolean name='DigitalOutput4' value='true'/>
|
|
<int name='AnalogOutput1' value='65535'/>
|
|
<int name='AnalogOutput2' value='65535'/>
|
|
<int name='AnalogOutput3' value='65535'/>
|
|
<int name='AnalogOutput4' value='65535'/>
|
|
</set>
|
|
</iq>
|
|
|
|
<iq type='get'
|
|
from='plc@clayster.com/plc'
|
|
to='provisioning@clayster.com'
|
|
id='3'>
|
|
<canControl xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' serviceToken='service0001' userToken='user0001'>
|
|
<parameter name='DigitalOutput1'/>
|
|
<parameter name='DigitalOutput2'/>
|
|
<parameter name='DigitalOutput3'/>
|
|
<parameter name='DigitalOutput4'/>
|
|
<parameter name='AnalogOutput1'/>
|
|
<parameter name='AnalogOutput2'/>
|
|
<parameter name='AnalogOutput3'/>
|
|
<parameter name='AnalogOutput4'/>
|
|
</canControl>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='plc@clayster.com/plc'
|
|
id='3'>
|
|
<canControlResponse xmlns='urn:xmpp:iot:provisioning' jid='master@clayster.com' result='true'>
|
|
<parameter name='DigitalOutput1'/>
|
|
<parameter name='DigitalOutput2'/>
|
|
<parameter name='DigitalOutput3'/>
|
|
<parameter name='DigitalOutput4'/>
|
|
</canControlResponse>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='plc@clayster.com'
|
|
to='master@clayster.com/amr'
|
|
id='20'>
|
|
<setResponse xmlns='urn:xmpp:iot:control' responseCode='OK'>
|
|
<parameter name='DigitalOutput1'/>
|
|
<parameter name='DigitalOutput2'/>
|
|
<parameter name='DigitalOutput3'/>
|
|
<parameter name='DigitalOutput4'/>
|
|
</setResponse>
|
|
</iq>]]>
|
|
</example>
|
|
<p>
|
|
Note that the provisioning server responds with a <strong>canControlResponse</strong> element, similar to the <strong>canControl</strong> element in the request, except only
|
|
the parameters allowed to be sent are listed. The device must only control parameters included in this list.
|
|
</p>
|
|
<p>
|
|
Also note, that the provisioning server can return a list of both allowed nodes and allowed parameter names in the response back to the client, so it can
|
|
act on the information that only a partial control action was allowed and taken.
|
|
</p>
|
|
</section3>
|
|
</section2>
|
|
<section2 topic='Cache'>
|
|
<section3 topic='Clear cache'>
|
|
<p>
|
|
When the provisioning server updates access rights and user privileges in the system, it will send a <strong>clearCache</strong> command to corresponding devices.
|
|
If a device was offline during the change, the provisioning server must send the <strong>clearCache</strong> message when the device comes online again. To acknowledge
|
|
the receipt of the command, the client responds with a <strong>clearCacheResponse</strong> element. This response message does not contain any information on what was
|
|
done by the client. It simply acknowledges the receipt of the command, to make sure the provisioning server does not resend the clear cache command again.
|
|
</p>
|
|
<p>
|
|
<strong>Note:</strong> The <strong>clearCache</strong> command does not include information on what has been changed, so the device needs to clear the entire cache. This
|
|
to avoid complexities in making sure updates made to the provisioning rules works in all cases, and to minimize complexity in the implementation of the protocol on the sensor side.
|
|
It is also not deemed to decrease network performance, since changing provisioning rules for a device is an exceptional event and therefore does not affect performance during
|
|
normal operation.
|
|
</p>
|
|
|
|
<example caption='Clear cache'>
|
|
<![CDATA[
|
|
<iq type='get'
|
|
from='provisioning@clayster.com/provisioning'
|
|
to='device@clayster.com'
|
|
id='17'>
|
|
<clearCache xmlns='urn:xmpp:iot:provisioning'/>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='device@clayster.com'
|
|
to='provisioning@clayster.com/provisioning'
|
|
id='17'>
|
|
<clearCacheResponse xmlns='urn:xmpp:iot:provisioning'/>
|
|
</iq>]]>
|
|
</example>
|
|
</section3>
|
|
</section2>
|
|
<section2 topic='Services'>
|
|
<section3 topic='Getting a service token' anchor='servicetoken'>
|
|
<p>
|
|
A service requesting provisioning assistance, needs to retrieve a service token from the provisioning server. The following example shows how this can be done.
|
|
</p>
|
|
<example caption='Requesting a service token'>
|
|
<![CDATA[
|
|
<iq type='get'
|
|
from='service@clayster.com/service'
|
|
to='provisioning@clayster.com'
|
|
id='14'>
|
|
<getToken xmlns='urn:xmpp:iot:provisioning' serviceId='WatchamacallitService1'/>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='service@clayster.com/service'
|
|
id='14'>
|
|
<getTokenResponse xmlns='urn:xmpp:iot:provisioning' token='service0001'/>
|
|
</iq>]]>
|
|
</example>
|
|
</section3>
|
|
<section3 topic='User access to service' anchor='usertoken'>
|
|
<p>
|
|
Provisioning in sensor networks also requires control of user access to different services in the network. This use case shows how service access rights are controlled using
|
|
a trusted provisioning server.
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
<p>
|
|
First, the user connects to the service in some way. This can be done using XMPP, HTTP, HTTPS or some other means. The service need to extract some form of identifying
|
|
credentials from the user, and provide that to the provisioning server. The provisioning server determines if the client has access rights to the service based on these
|
|
credentials, and also provides the service with a <strong>userToken</strong> that the service can use in further communication with the provisioning server regarding
|
|
the user and its privileges.
|
|
</p>
|
|
<p>
|
|
The following table lists some different types of credentials that the service can extract from the client:
|
|
</p>
|
|
<table caption='User Credentials'>
|
|
<tr>
|
|
<th>Type</th>
|
|
<th>Protocols</th>
|
|
<th>Description</th>
|
|
</tr>
|
|
<tr>
|
|
<td>JID</td>
|
|
<td>XMPP</td>
|
|
<td>Allows provisioning to be done on the JID the user has.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>IP Address</td>
|
|
<td>HTTP, HTTPS</td>
|
|
<td>Allows provisioning to be done on IP address or IP-ranges for instance.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Host Name</td>
|
|
<td>HTTP, HTTPS with DNS</td>
|
|
<td>If the service has access to the client host name, for instance in an intra network, this can be used for provisioning.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>X.509 Certificate</td>
|
|
<td>HTTPS</td>
|
|
<td>If the client provides a client certificate, such a certificate can be used to provide provisioning.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>X.509 Certificate Thumbprint</td>
|
|
<td>HTTPS</td>
|
|
<td>
|
|
The client can also choose to validate the certificate itself and only send the certificate thumbprint to the provisioning server. Even though this
|
|
provides somewhat lesser security than providing the entire certificate, it might be an option to distribute certificate validation checks across the network
|
|
to lower the work load on the provisioning server.
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>User Name</td>
|
|
<td>HTTP, HTTPS, XMPP</td>
|
|
<td>
|
|
If authenticated HTTP(S) access is implemented, the service can provide the user name as credentials. XMPP based clients can use
|
|
information in the roster to provide user name information to the provisioning server.
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Geolocation</td>
|
|
<td>HTML5 over HTTP(S), XMPP</td>
|
|
<td>
|
|
If the geographic location (longitude, latitude and possibly altitude) of the user client is known, it can be used. HTML 5 provides mechanism whereby the location
|
|
of the client can be fetched. &xep0080; provides a mechanism whereby client location can be obtained over XMPP.
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td>SSO Token</td>
|
|
<td>Intranet</td>
|
|
<td>If a single sign on token is available, such a token could be provided as credentials.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Protocol</td>
|
|
<td>Any</td>
|
|
<td>Connection protocol used to connect to the service.</td>
|
|
</tr>
|
|
</table>
|
|
<p>
|
|
The provisioning server receives these credentials, and decides if the user should have access to the service or not, based on rules configured in the provisioning service.
|
|
If the user is granted access, a <strong>userToken</strong> is generated and returned to the service.
|
|
</p>
|
|
<p>
|
|
Now, the service can determine if this access grant is sufficient or not. It can require the user to login into the service first. If so, the service should provide the provisioning
|
|
server with the user name used during login, when logged in.
|
|
</p>
|
|
<example caption='User access to service'>
|
|
<![CDATA[
|
|
<!-- user connects to service -->
|
|
|
|
<iq type='get'
|
|
from='service@clayster.com/service'
|
|
to='provisioning@clayster.com'
|
|
id='9'>
|
|
<canAccess xmlns='urn:xmpp:iot:provisioning' serviceToken='service0001'>
|
|
<credentials type='IpAddress' value='10.0.0.1'/>
|
|
<credentials type='Longitude' value='123.45'/>
|
|
<credentials type='Latitude' value='67.89'/>
|
|
</canAccess>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='service@clayster.com/service'
|
|
id='9'>
|
|
<canAccessResponse xmlns='urn:xmpp:iot:provisioning' userToken='user0001' result='true'/>
|
|
</iq>
|
|
|
|
<!-- user performs login into service -->
|
|
|
|
<message from='service@clayster.com/service'
|
|
to='provisioning@clayster.com'>
|
|
<userLoggedIn xmlns='urn:xmpp:iot:provisioning' serviceToken='service0001' userToken='user0001' userName='Kermit' />
|
|
</message>
|
|
|
|
<!-- user continues interacting with service -->]]>
|
|
</example>
|
|
</section3>
|
|
</section2>
|
|
<section2 topic='User privileges'>
|
|
<section3 topic='User privileges in service'>
|
|
<p>
|
|
When a user has been given access to a service, and properly been identified, the service can ask the provisioning service for detailed user privileges to control
|
|
different aspects of the service. This can be done using the <strong>hasPrivilege</strong> command. Here, the service sends its <strong>serviceToken</strong> and
|
|
the <strong>userToken</strong> earlier received when being granted access to the service. Furthermore a <strong>privilegeId</strong> has to be provided.
|
|
</p>
|
|
<p>
|
|
A <strong>Privilege ID</strong> is a string composed of one or a sequence of parts, delimited by period characters. The Privilege IDs form a tree of privileges,
|
|
using an invisible, but common, root privilege.
|
|
</p>
|
|
<p>
|
|
The following table suggests some examples of Privilege IDs, with suggestive descriptions. (Only used as an example.)
|
|
</p>
|
|
<table caption='Privilege ID examples'>
|
|
<tr>
|
|
<th>Privilege ID</th>
|
|
<th>Description</th>
|
|
</tr>
|
|
<tr>
|
|
<td>Databases.Energy.Select</td>
|
|
<td>Gives the user the rights to select data from the Energy database.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Databases.Energy.Insert</td>
|
|
<td>Gives the user the rights to insert data into the Energy database.</td>
|
|
</tr>
|
|
<tr>
|
|
<td>Databases.Energy.Delete</td>
|
|
<td>Gives the user the rights to delete data from the Energy database.</td>
|
|
</tr>
|
|
</table>
|
|
<p>
|
|
<strong>Note:</strong> Note that privilege IDs are local to the service. Different services are allowed to use similar or same Privilege IDs, in different contexts and
|
|
with different meanings. The provisioning server must separate Privilege IDs from different services.
|
|
</p>
|
|
<p>
|
|
The client must always provide full Privilege IDs to the provisioning server. The provisioning server however, can grant partial privilege IDs, pointing to parent
|
|
privilege nodes to a user or a role object, granting a user a specific role. If granting a parent privilege ID to a user or role, this is interpreted as
|
|
giving the corresponding user or role the privileges of the entire sub-tree defined by the parent privilege ID.
|
|
</p>
|
|
<p>
|
|
If additional control over privileges is desired, negative privileges can be assigned to the user or role. Granted or explicitly rejected privileges are specified in a
|
|
sequential list of full or partial privilege IDs. This list is then processed sequentially to determine if a privilege is granted or not.
|
|
</p>
|
|
<p>
|
|
Example: Consider the privileges above. Give a user or its corresponding role the following privileges in a sequential list:
|
|
</p>
|
|
<ul>
|
|
<li>Exclude: Databases.Energy.Delete</li>
|
|
<li>Include: Databases.Energy</li>
|
|
</ul>
|
|
<p>
|
|
This would give the user rights to select and insert data, but not to delete data from the Energy database.
|
|
</p>
|
|
<p>
|
|
<strong>Note:</strong> Care should be taken when constructing Privilege IDs, so they do not include variable data that potentially can create an infinite amount of
|
|
privilege IDs. For example: Do not include user names, sensor IDs, etc., in privilege IDs, as the number of such entities cannot be estimated or is scalable beforehand.
|
|
</p>
|
|
<p>
|
|
The following diagram shows an example of how a service asks permission from a provisioning server before an action is taken:
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
<example caption='User privileges in service'>
|
|
<![CDATA[
|
|
<!-- user wants to perform action -->
|
|
|
|
<iq type='get'
|
|
from='service@clayster.com/service'
|
|
to='provisioning@clayster.com'
|
|
id='10'>
|
|
<hasPrivilege xmlns='urn:xmpp:iot:provisioning' serviceToken='service0001' userToken='user0001' privilegeId='Sensors.View'/>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='service@clayster.com/service'
|
|
id='10'>
|
|
<hasPrivilegeResponse xmlns='urn:xmpp:iot:provisioning' result='true'/>
|
|
</iq>
|
|
|
|
<!-- user performs action -->]]>
|
|
</example>
|
|
</section3>
|
|
<section3 topic='Download all user privileges'>
|
|
<p>
|
|
To improve performance, services can download the entire set of user privileges, and perform privilege checks internally. The following diagram displays
|
|
how the above two use cases could be handled in such a case:
|
|
</p>
|
|
<p>
|
|
<img src=''/>
|
|
</p>
|
|
<p>
|
|
<strong>Note:</strong> When downloading privileges using this command, a sequential list of full or partial privilege IDs will be returned together with the
|
|
corresponding include or exclude flags. The above mentioned algorithm of determining user privileges must be implemented by the service, if this method is
|
|
to be used.
|
|
</p>
|
|
<example caption='Download all user privileges'>
|
|
<![CDATA[
|
|
<!-- user connects to service -->
|
|
|
|
<iq type='get'
|
|
from='service@clayster.com/service'
|
|
to='provisioning@clayster.com'
|
|
id='11'>
|
|
<canAccess xmlns='urn:xmpp:iot:provisioning' serviceToken='service0001'>
|
|
<credentials type='IpAddress' value='10.0.0.1'/>
|
|
<credentials type='Longitude' value='123.45'/>
|
|
<credentials type='Latitude' value='67.89'/>
|
|
</canAccess>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='service@clayster.com/service'
|
|
id='11'>
|
|
<canAccessResponse xmlns='urn:xmpp:iot:provisioning' userToken='user0001' result='true'/>
|
|
</iq>
|
|
|
|
<!-- user performs login into service -->
|
|
|
|
<message from='service@clayster.com/service'
|
|
to='provisioning@clayster.com'>
|
|
<userLoggedIn xmlns='urn:xmpp:iot:provisioning' serviceToken='service0001' userToken='user0001' userName='Kermit' />
|
|
</message>
|
|
|
|
<iq type='get'
|
|
from='service@clayster.com/service'
|
|
to='provisioning@clayster.com'
|
|
id='12'>
|
|
<downloadPrivileges xmlns='urn:xmpp:iot:provisioning' serviceToken='service0001' userToken='user0001'/>
|
|
</iq>
|
|
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='service@clayster.com/service'
|
|
id='12'>
|
|
<downloadPrivilegesResponse>
|
|
<exclude id='Sensors.Delete'/>
|
|
<include id='Sensors'/>
|
|
</downloadPrivilegesResponse>
|
|
</iq>
|
|
|
|
<!-- user performs actions without interaction with the provisioning server -->]]>
|
|
</example>
|
|
<p>
|
|
<strong>Note:</strong> If the user or service has not been correctly identified, logged in, etc., the resulting list must only include privileges
|
|
that default users that are not logged in can have. This list can be empty.
|
|
</p>
|
|
</section3>
|
|
</section2>
|
|
</section1>
|
|
<section1 topic='Determining Support' anchor='support'>
|
|
<p>If an entity supports the protocol specified herein, it MUST advertise that fact by returning a feature of "urn:xmpp:iot:provisioning" in response to &xep0030; information requests.</p>
|
|
<example caption="Service discovery information request">
|
|
<![CDATA[
|
|
<iq type='get'
|
|
from='device@clayster.com/device'
|
|
to='provisioning@clayster.com'
|
|
id='disco1'>
|
|
<query xmlns='http://jabber.org/protocol/disco#info'/>
|
|
</iq>]]>
|
|
</example>
|
|
<example caption="Service discovery information response">
|
|
<![CDATA[
|
|
<iq type='result'
|
|
from='provisioning@clayster.com'
|
|
to='device@clayster.com/device'
|
|
id='disco1'>
|
|
<query xmlns='http://jabber.org/protocol/disco#info'>
|
|
...
|
|
<feature var='urn:xmpp:iot:provisioning'/>
|
|
...
|
|
</query>
|
|
</iq>]]>
|
|
</example>
|
|
<p>
|
|
In order for an application to determine whether an entity supports this protocol, where possible it SHOULD use the dynamic, presence-based profile of service discovery defined
|
|
in &xep0115;. However, if an application has not received entity capabilities information from an entity, it SHOULD use explicit service discovery instead.
|
|
</p>
|
|
</section1>
|
|
<section1 topic='Implementation Notes' anchor='impl'>
|
|
<section2 topic='Caching and cache time' anchor='cache'>
|
|
<p>
|
|
To minimize network traffic, and optimize response time, devices should cache access rights and user privileges provided by the provisioning server. If memory is limited,
|
|
items in the cache should be ordered by last access, and items with the oldest last access timestamp should be removed first. A safety valve can optionally be implemented as well,
|
|
removing unused cache items after a certain age, even if memory is available.
|
|
</p>
|
|
<p>
|
|
The device can assume that access rights and user privileges on the provisioning server do not change over time, unless the provisioning server says so.
|
|
</p>
|
|
<p>
|
|
The provisioning server on the other hand, must keep track of when a device is online and offline, and clear the cache of the device if changes are made that affects the device.
|
|
If the device was offline when those changes occurred, the provisioning server must send the clear cache command as soon as the device comes online again.
|
|
</p>
|
|
<p>
|
|
When creating a new trust relationship, the device should always clear its cache, if it contains information from before.
|
|
</p>
|
|
<p>
|
|
To minimize stress of the provisioning server during synchronous sensor start up, for instance after a power failure, all clients should aim to persist its cache if possible. Clients
|
|
not persisting its cache may produce too much stress on the provisioning server on start-up, practically removing it from the network.
|
|
</p>
|
|
</section2>
|
|
<section2 topic='Working with multiple provisioning servers'>
|
|
<p>
|
|
When working with multiple provisioning servers, there are some things that should be considered:
|
|
</p>
|
|
<ul>
|
|
<li>
|
|
When a device requests information from the provisioning servers, this can be done in a parallel fashion. Even though the provisioning servers
|
|
were added in a sequential fashion, there is no order or priority between the servers.
|
|
</li>
|
|
<li>
|
|
When receiving different responses from different servers, with regards to access rights, privileges, relationships, etc.,
|
|
the union of rights, or the most forgiving or most accepting response should be used. If one of the servers grant a privilege, the privilege is assumed
|
|
to exist. If one of the servers grant a friendship request, the friendship request should be granted, etc. Most probably, different servers will manage
|
|
different subsets of entities on the network, and when they receive questions about unrecognized devices, they will simply deny access.
|
|
</li>
|
|
<li>
|
|
If a provisioning server requires total control of the network, it should not allow secondary trust relationships. However, if such a server enters a network
|
|
and there exist previous provisioning servers (i.e. they accept secondary trust relationships), more trust relationships are assumed to be acceptable.
|
|
</li>
|
|
<li>
|
|
Even though examples in this document only list a secondary trust relationship, there is no such limit. There may exist many different trust relationships in a network
|
|
or for a given device.
|
|
</li>
|
|
</ul>
|
|
</section2>
|
|
<section2 topic='Automatic aggregation of services, users and privileges'>
|
|
<p>
|
|
One important design consideration when implementing a provisioning server is how to handle new services, users and privileges. One option might be to automatically
|
|
ignore anything not recognized. Another option might be to dynamically add new services, user names and privileges to internal data sources, making it easier to manage
|
|
new types of services dynamically. However, adding such items automatically might also make such data sources grow beyond control.
|
|
</p>
|
|
</section2>
|
|
<section2 topic='Reading devices from large subsystems'>
|
|
<p>
|
|
All examples in this document have been simplified examples where a few devices containing a few fields have been read. However, in many cases large subsystems with
|
|
very many sensors containing many fields have to be read, as is documented in <link url='http://xmpp.org/extensions/xep-0326.html'>Internet of Things - Concentrators</link>
|
|
<link url='http://xmpp.org/extensions/xep-0326.html'>Internet of Things - Concentrators</link>. In such cases, a node may have to be specified using two or perhaps
|
|
even three ID's: a <strong>sourceId</strong> identifying the data source controlling the device, a possible <strong>cacheType</strong> narrowing down the search to
|
|
a specific kind of node, and the common <strong>nodeId</strong>. For more information about this, see
|
|
<link url='http://xmpp.org/extensions/xep-0326.html'>Internet of Things - Concentrators</link>.
|
|
</p>
|
|
<p>
|
|
<strong>Note:</strong> For cases where the <strong>nodeId</strong> is sufficient to uniquely identify the node, it is sufficient to provide this attribute in the request.
|
|
If there is ambiguity in the request, the receptor must treat the request as a request with a set of nodes, all with the corresponding <strong>nodeId</strong> as requested.
|
|
</p>
|
|
</section2>
|
|
<section2 topic='Different types of tokens'>
|
|
<p>
|
|
A small note regarding the use of different tokens. A service can get a <strong>Service Token</strong>, a device a <strong>Device Token</strong> and a user
|
|
a <strong>User Token</strong>. When delegating these tokens to third parties, a service sends its <strong>Service Token</strong>. But, if the service does this
|
|
within the context of a user action, the service sends both its <strong>Service Token</strong> and the users <strong>User Token</strong>. The same with a device.
|
|
If a device delegates its token to a third party, it sends its <strong>Device Token</strong>. But if the device performs the action in the context of a user action,
|
|
the device sends both its <strong>Device Token</strong> as well as its <strong>User Token</strong>.
|
|
</p>
|
|
</section2>
|
|
</section1>
|
|
<section1 topic='Security Considerations' anchor='security'>
|
|
<p>
|
|
Delegating trust to a third party may create a weak link in the overall security of a sensor network. Therefore, it's vitally important that the following be adhered to:
|
|
</p>
|
|
<ul>
|
|
<li>
|
|
Trust should only be delegated in a secure way. Once delegated, it should be able to lock this trust so it cannot be changed without reverting the device to
|
|
factory default settings. Such a locked delegation mode can be likened with a production mode, where vital configuration parameters should not be able to be changed.
|
|
</li>
|
|
<li>
|
|
If allowing installation using a LED, as above, make sure the LED does not light up for a long time, limiting the window of access where the sensor can be linked to a
|
|
provisioning server.
|
|
</li>
|
|
<li>
|
|
Provisioning servers should be monitored during operation, since it provides a vital link in the operation of the network.
|
|
</li>
|
|
<li>
|
|
Multiple provisioning servers could be allowed, for redundancy or for scalability. This specification does not limit the number of provisioning servers used in a network,
|
|
not used by a device. Examples in this specification use only one provisioning server for simplicity.
|
|
</li>
|
|
<li>
|
|
Generally, take care what kind of provisioning servers you allow in a network.
|
|
</li>
|
|
</ul>
|
|
</section1>
|
|
<section1 topic='IANA Considerations' anchor='iana'>
|
|
<p>This document requires no interaction with &IANA;.</p>
|
|
</section1>
|
|
<section1 topic='XMPP Registrar Considerations' anchor='registrar'>
|
|
<p>
|
|
The <link url="#schema">protocol schema</link> needs to be added to the list of <link url="http://xmpp.org/resources/schemas/">XMPP protocol schemas</link>.
|
|
</p>
|
|
</section1>
|
|
<section1 topic='XML Schema' anchor='schema'>
|
|
<code>
|
|
<![CDATA[
|
|
<?xml version='1.0' encoding='UTF-8'?>
|
|
<xs:schema
|
|
xmlns:xs='http://www.w3.org/2001/XMLSchema'
|
|
targetNamespace='urn:xmpp:iot:provisioning'
|
|
xmlns='urn:xmpp:iot:provisioning'
|
|
xmlns:sn='urn:xmpp:iot:sensordata'
|
|
elementFormDefault='qualified'>
|
|
|
|
<xs:import namespace='urn:xmpp:iot:sensordata'/>
|
|
|
|
<xs:element name='getToken' type='GetToken'/>
|
|
<xs:element name='getTokenResponse' type='GetTokenResponse'/>
|
|
|
|
<xs:element name='isFriend' type='Friend'/>
|
|
<xs:element name='isFriendResponse' type='FriendResponse'/>
|
|
<xs:element name='unfriend' type='Friend'/>
|
|
<xs:element name='Friend' type='Friend'/>
|
|
|
|
<xs:element name='canRead' type='CanRead'/>
|
|
<xs:element name='canReadResponse' type='CanReadResponse'/>
|
|
|
|
<xs:element name='canControl' type='CanControl'/>
|
|
<xs:element name='canControlResponse' type='CanControlResponse'/>
|
|
|
|
<xs:element name='clearCache'>
|
|
<xs:complexType/>
|
|
</xs:element>
|
|
|
|
<xs:element name='clearCacheResponse'>
|
|
<xs:complexType/>
|
|
</xs:element>
|
|
|
|
<xs:element name='canAccess'>
|
|
<xs:complexType>
|
|
<xs:choice minOccurs='0' maxOccurs='unbounded'>
|
|
<xs:element name='jid' type='JidCredential'/>
|
|
<xs:element name='ip4' type='Ip4Credential'/>
|
|
<xs:element name='ip6' type='Ip6Credential'/>
|
|
<xs:element name='hostName' type='HostNameCredential'/>
|
|
<xs:element name='x509Certificate' type='X509CertificateCredential'/>
|
|
<xs:element name='x509CertificateThumbprint' type='X509CertificateThumbprintCredential'/>
|
|
<xs:element name='userName' type='UserNameCredential'/>
|
|
<xs:element name='longitude' type='LongitudeCredential'/>
|
|
<xs:element name='latitude' type='LatitudeCredential'/>
|
|
<xs:element name='altitude' type='AltitudeCredential'/>
|
|
<xs:element name='sso' type='SsoCredential'/>
|
|
<xs:element name='protocol' type='ProtocolCredential'/>
|
|
</xs:choice>
|
|
<xs:attribute name='serviceToken' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
</xs:element>
|
|
|
|
<xs:element name='canAccessResponse'>
|
|
<xs:complexType>
|
|
<xs:attribute name='result' type='xs:boolean' use='required'/>
|
|
<xs:attribute name='userToken' type='xs:string' use='optional'/>
|
|
</xs:complexType>
|
|
</xs:element>
|
|
|
|
<xs:element name='userLoggedIn'>
|
|
<xs:complexType>
|
|
<xs:attribute name='serviceToken' type='xs:string' use='required'/>
|
|
<xs:attribute name='userToken' type='xs:string' use='required'/>
|
|
<xs:attribute name='userName' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
</xs:element>
|
|
|
|
<xs:element name='hasPrivilege'>
|
|
<xs:complexType>
|
|
<xs:attribute name='serviceToken' type='xs:string' use='required'/>
|
|
<xs:attribute name='userToken' type='xs:string' use='required'/>
|
|
<xs:attribute name='privilegeId' type='PrivilegeId' use='required'/>
|
|
</xs:complexType>
|
|
</xs:element>
|
|
|
|
<xs:element name='hasPrivilegeResponse'>
|
|
<xs:complexType>
|
|
<xs:attribute name='result' type='xs:boolean' use='required'/>
|
|
</xs:complexType>
|
|
</xs:element>
|
|
|
|
<xs:element name='downloadPrivileges'>
|
|
<xs:complexType>
|
|
<xs:attribute name='serviceToken' type='xs:string' use='required'/>
|
|
<xs:attribute name='userToken' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
</xs:element>
|
|
|
|
<xs:element name='downloadPrivilegesResponse'>
|
|
<xs:complexType>
|
|
<xs:choice minOccurs='0' maxOccurs='unbounded'>
|
|
<xs:element name='include' type='Privilege'/>
|
|
<xs:element name='exclude' type='Privilege'/>
|
|
</xs:choice>
|
|
</xs:complexType>
|
|
</xs:element>
|
|
|
|
<xs:complexType name='GetToken'>
|
|
<xs:attribute name='deviceId' type='xs:string' use='optional'/>
|
|
<xs:attribute name='serviceId' type='xs:string' use='optional'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='GetTokenResponse'>
|
|
<xs:attribute name='token' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='Friend'>
|
|
<xs:attribute name='jid' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='FriendResponse'>
|
|
<xs:complexContent>
|
|
<xs:extension base='Friend'>
|
|
<xs:attribute name='result' type='xs:boolean' use='required'/>
|
|
<xs:attribute name='secondaryTrustAllowed' type='xs:boolean' use='optional' default='false'/>
|
|
</xs:extension>
|
|
</xs:complexContent>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='CanReadBase' abstract='true'>
|
|
<xs:choice minOccurs='0' maxOccurs='unbounded'>
|
|
<xs:element name='node'>
|
|
<xs:complexType>
|
|
<xs:attribute name='nodeId' type='xs:string' use='required'/>
|
|
<xs:attribute name='sourceId' type='xs:string' use='optional'/>
|
|
<xs:attribute name='cacheType' type='xs:string' use='optional'/>
|
|
</xs:complexType>
|
|
</xs:element>
|
|
<xs:element name='field'>
|
|
<xs:complexType>
|
|
<xs:attribute name='name' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
</xs:element>
|
|
</xs:choice>
|
|
<xs:attributeGroup ref='sn:fieldTypes'/>
|
|
<xs:attribute name='all' type='xs:boolean' use='optional' default='false'/>
|
|
<xs:attribute name='historical' type='xs:boolean' use='optional' default='false'/>
|
|
<xs:attribute name='jid' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='CanRead'>
|
|
<xs:complexContent>
|
|
<xs:extension base='CanReadBase'>
|
|
<xs:attribute name='token' type='xs:string' use='optional'/>
|
|
</xs:extension>
|
|
</xs:complexContent>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='CanReadResponse'>
|
|
<xs:complexContent>
|
|
<xs:extension base='CanReadBase'>
|
|
<xs:attribute name='result' type='xs:boolean' use='required'/>
|
|
</xs:extension>
|
|
</xs:complexContent>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='CanControlBase' abstract='true'>
|
|
<xs:choice minOccurs='0' maxOccurs='unbounded'>
|
|
<xs:element name='node'>
|
|
<xs:complexType>
|
|
<xs:attribute name='nodeId' type='xs:string' use='required'/>
|
|
<xs:attribute name='sourceId' type='xs:string' use='optional'/>
|
|
<xs:attribute name='cacheType' type='xs:string' use='optional'/>
|
|
</xs:complexType>
|
|
</xs:element>
|
|
<xs:element name='parameter'>
|
|
<xs:complexType>
|
|
<xs:attribute name='name' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
</xs:element>
|
|
</xs:choice>
|
|
<xs:attribute name='jid' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='CanControl'>
|
|
<xs:complexContent>
|
|
<xs:extension base='CanControlBase'>
|
|
<xs:attribute name='token' type='xs:string' use='optional'/>
|
|
</xs:extension>
|
|
</xs:complexContent>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='CanControlResponse'>
|
|
<xs:complexContent>
|
|
<xs:extension base='CanControlBase'>
|
|
<xs:attribute name='result' type='xs:boolean' use='required'/>
|
|
</xs:extension>
|
|
</xs:complexContent>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='JidCredential'>
|
|
<xs:attribute name='value' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='Ip4Credential'>
|
|
<xs:attribute name='value' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='Ip6Credential'>
|
|
<xs:attribute name='value' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='HostNameCredential'>
|
|
<xs:attribute name='value' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='X509CertificateCredential'>
|
|
<xs:attribute name='base64' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='X509CertificateThumbprintCredential'>
|
|
<xs:attribute name='base64' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='UserNameCredential'>
|
|
<xs:attribute name='value' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='LongitudeCredential'>
|
|
<xs:attribute name='value' type='xs:double' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='LatitudeCredential'>
|
|
<xs:attribute name='value' type='xs:double' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='AltitudeCredential'>
|
|
<xs:attribute name='value' type='xs:double' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='SsoCredential'>
|
|
<xs:attribute name='value' type='xs:string' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='ProtocolCredential'>
|
|
<xs:attribute name='value' type='ProtocolName' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:complexType name='Privilege'>
|
|
<xs:attribute name='id' type='PrivilegeId' use='required'/>
|
|
</xs:complexType>
|
|
|
|
<xs:simpleType name='ProtocolName'>
|
|
<xs:restriction base='xs:string'>
|
|
<xs:enumeration value='XMPP'/>
|
|
<xs:enumeration value='HTTP'/>
|
|
<xs:enumeration value='HTTPS'/>
|
|
<xs:enumeration value='TcpSocket'/>
|
|
<xs:enumeration value='UdpSocket'/>
|
|
<xs:enumeration value='Queue'/>
|
|
<xs:enumeration value='Internal'/>
|
|
<xs:enumeration value='Other'/>
|
|
</xs:restriction>
|
|
</xs:simpleType>
|
|
|
|
<xs:simpleType name='PrivilegeId'>
|
|
<xs:restriction base='xs:string'>
|
|
<xs:pattern value='^[^.]+([.][^.]+)*$'/>
|
|
</xs:restriction>
|
|
</xs:simpleType>
|
|
|
|
</xs:schema>]]>
|
|
</code>
|
|
</section1>
|
|
<section1 topic='For more information' anchor='moreinfo'>
|
|
<p>
|
|
For more information, please see the following resources:
|
|
</p>
|
|
<ul>
|
|
<li>
|
|
<p>
|
|
The <link url='http://wiki.xmpp.org/web/Tech_pages/SensorNetworks'>Sensor Network section of the XMPP Wiki</link> contains further information about the
|
|
use of the sensor network XEPs, links to implementations, discussions, etc.
|
|
</p>
|
|
</li>
|
|
<li>
|
|
<p>
|
|
The XEP's and related projects are also available on <link url='https://github.com/joachimlindborg/'>github</link>, thanks to Joachim Lindborg.
|
|
</p>
|
|
</li>
|
|
<li>
|
|
<p>
|
|
A presentation giving an overview of all extensions related to Internet of Things can be found here:
|
|
<link url='http://prezi.com/esosntqhewhs/iot-xmpp/'>http://prezi.com/esosntqhewhs/iot-xmpp/</link>.
|
|
</p>
|
|
</li>
|
|
</ul>
|
|
</section1>
|
|
<section1 topic='Acknowledgements' anchor='ack'>
|
|
<p>Thanks to Joachim Lindborg, Karin Forsell, Tina Beckman and Teemu Väisänen for all valuable feedback.</p>
|
|
</section1>
|
|
</xep>
|