All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.microsoft.azure.iothub.DeviceClient Maven / Gradle / Ivy

There is a newer version: 1.0.16
Show newest version
// Copyright (c) Microsoft. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for full license information.

package com.microsoft.azure.iothub;

import com.microsoft.azure.iothub.transport.amqps.AmqpsTransport;
import com.microsoft.azure.iothub.transport.https.HttpsTransport;
import com.microsoft.azure.iothub.transport.IotHubReceiveTask;
import com.microsoft.azure.iothub.transport.IotHubSendTask;
import com.microsoft.azure.iothub.transport.IotHubTransport;
import com.microsoft.azure.iothub.transport.mqtt.MqttTransport;
import java.io.Closeable;

import java.io.IOError;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.net.URISyntaxException;
import java.net.URLDecoder;
import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

/**
 * 

* The public-facing API. Allows a single logical or physical device to connect * to an IoT Hub. The IoT Hub client supports sending events to and receiving * messages from an IoT Hub. *

*

* To support these workflows, the client library will provide the following * abstractions: a message, with its payload and associated properties; and a * client, which sends and receives messages. *

*

* The client buffers messages while the network is down, and re-sends them when * the network comes back online. It also batches messages to improve * communication efficiency (HTTPS only). *

* The client supports HTTPS 1.1 and AMQPS 1.0 transports. */ public final class DeviceClient implements Closeable { /** The state of the IoT Hub client's connection with the IoT Hub. */ protected enum IotHubClientState { OPEN, CLOSED } protected final static String SET_MINIMUM_POLLING_INTERVAL = "SetMinimumPollingInterval"; protected final static String SET_CERTIFICATE_PATH = "SetCertificatePath"; protected final static String SET_SAS_TOKEN_EXPIRY_TIME = "SetSASTokenExpiryTime"; /** * The number of milliseconds the transport will wait between * sending out messages. */ public static long SEND_PERIOD_MILLIS = 10l; /** * The number of milliseconds the transport will wait between * polling for messages. */ public static long RECEIVE_PERIOD_MILLIS_AMQPS = 10l; public static long RECEIVE_PERIOD_MILLIS_MQTT = 10l; public static long RECEIVE_PERIOD_MILLIS_HTTPS = 25*60*1000; /*25 minutes*/ /** The hostname attribute name in a connection string. */ public static final String HOSTNAME_ATTRIBUTE = "HostName="; /** The device ID attribute name in a connection string. */ public static final String DEVICE_ID_ATTRIBUTE = "DeviceId="; /** The shared access key attribute name in a connection string. */ public static final String SHARED_ACCESS_KEY_ATTRIBUTE = "SharedAccessKey="; /** The shared access signature attribute name in a connection string. */ public static final String SHARED_ACCESS_TOKEN_ATTRIBUTE = "SharedAccessSignature="; /** * The charset used for URL-encoding the device ID in the connection * string. */ public static final Charset CONNECTION_STRING_CHARSET = StandardCharsets.UTF_8; protected DeviceClientConfig config; protected IotHubTransport transport; protected ScheduledExecutorService taskScheduler; protected IotHubClientState state; protected long RECEIVE_PERIOD_MILLIS; /** * Constructor that takes a connection string as an argument. * * @param connString the connection string. The connection string is a set * of key-value pairs that are separated by ';', with the keys and values * separated by '='. It should contain values for the following keys: * {@code HostName}, {@code DeviceId}, and {@code SharedAccessKey}. * @param protocol the communication protocol used (i.e. HTTPS). * * @throws IllegalArgumentException if any of {@code connString} or * {@code protocol} are {@code null}; or if {@code connString} is missing * one of the following attributes:{@code HostName}, {@code DeviceId}, or * {@code SharedAccessKey}. * @throws URISyntaxException if the IoT hub hostname does not conform to * RFC 3986. */ public DeviceClient(String connString, IotHubClientProtocol protocol) throws URISyntaxException { // Codes_SRS_DEVICECLIENT_11_047: [If the connection string is null, the function shall throw an IllegalArgumentException.] if (connString == null) { throw new IllegalArgumentException( "IoT Hub connection string cannot be null."); } // Codes_SRS_DEVICECLIENT_11_042: [The constructor shall interpret the connection string as a set of key-value pairs delimited by ';', with keys and values separated by '='.] String[] connStringAttrs = connString.split(";"); String hostname = null; String deviceId = null; String sharedAccessKey = null; String sharedAccessToken = null; for (String attr : connStringAttrs) { // Codes_SRS_DEVICECLIENT_11_043: [The constructor shall save the IoT Hub hostname as the value of 'HostName' in the connection string.] if (attr.startsWith(HOSTNAME_ATTRIBUTE)) { hostname = attr.substring(HOSTNAME_ATTRIBUTE.length()); } // Codes_SRS_DEVICECLIENT_11_044: [The constructor shall save the device ID as the UTF-8 URL-decoded value of 'DeviceId' in the connection string.] else if (attr.startsWith(DEVICE_ID_ATTRIBUTE)) { String urlEncodedDeviceId = attr.substring(DEVICE_ID_ATTRIBUTE.length()); try { deviceId = URLDecoder.decode(urlEncodedDeviceId,CONNECTION_STRING_CHARSET.name()); } catch (UnsupportedEncodingException e) { // should never happen, since the encoding is hard-coded. throw new IllegalStateException(e); } } // Codes_SRS_DEVICECLIENT_11_045: [The constructor shall save the device key as the value of 'SharedAccessKey' in the connection string.] else if (attr.startsWith(SHARED_ACCESS_KEY_ATTRIBUTE)) { sharedAccessKey = attr.substring(SHARED_ACCESS_KEY_ATTRIBUTE.length()); } // Codes_SRS_DEVICECLIENT_25_052: [The constructor shall save the shared access token as the value of 'sharedAccessToken' in the connection string.] else if (attr.startsWith(SHARED_ACCESS_TOKEN_ATTRIBUTE)) { sharedAccessToken = attr.substring(SHARED_ACCESS_TOKEN_ATTRIBUTE.length()); } } initIotHubClient(hostname, deviceId, sharedAccessKey, sharedAccessToken, protocol); } /** * Starts asynchronously sending and receiving messages from an IoT Hub. If * the client is already open, the function shall do nothing. * * @throws IOException if a connection to an IoT Hub is cannot be * established. */ public void open() throws IOException { // Codes_SRS_DEVICECLIENT_11_028: [If the client is already open, the function shall do nothing.] if (this.state == IotHubClientState.OPEN) { return; } // Codes_SRS_DEVICECLIENT_11_035: [The function shall open the transport to communicate with an IoT Hub.] // Codes_SRS_DEVICECLIENT_11_036: [If an error occurs in opening the transport, the function shall throw an IOException.] this.transport.open(); IotHubSendTask sendTask = new IotHubSendTask(this.transport); IotHubReceiveTask receiveTask = new IotHubReceiveTask(this.transport); this.taskScheduler = Executors.newScheduledThreadPool(2); // the scheduler waits until each execution is finished before // scheduling the next one, so executions of a given task // will never overlap. // Codes_SRS_DEVICECLIENT_11_023: [The function shall schedule send tasks to run every SEND_PERIOD_MILLIS milliseconds.] this.taskScheduler.scheduleAtFixedRate(sendTask, 0, SEND_PERIOD_MILLIS, TimeUnit.MILLISECONDS); // Codes_SRS_DEVICECLIENT_11_024: [The function shall schedule receive tasks to run every RECEIVE_PERIOD_MILLIS milliseconds.] this.taskScheduler.scheduleAtFixedRate(receiveTask, 0, RECEIVE_PERIOD_MILLIS, TimeUnit.MILLISECONDS); this.state = IotHubClientState.OPEN; } /** * Completes all current outstanding requests and closes the IoT Hub client. * Must be called to terminate the background thread that is sending data to * IoT Hub. After {@code close()} is called, the IoT Hub client is no longer * usable. If the client is already closed, the function shall do nothing. * * @throws IOException if the connection to an IoT Hub cannot be closed. */ public void close() throws IOException { // Codes_SRS_DEVICECLIENT_11_031: [If the client is already closed, the function shall do nothing.] if (this.state == IotHubClientState.CLOSED) { return; } // Codes_SRS_DEVICECLIENT_11_010: [The function shall finish all ongoing tasks.] // Codes_SRS_DEVICECLIENT_11_011: [The function shall cancel all recurring tasks.] while (!this.transport.isEmpty()) { } this.taskScheduler.shutdown(); // Codes_SRS_DEVICECLIENT_11_037: [The function shall close the transport.] this.transport.close(); this.state = IotHubClientState.CLOSED; } /** * Asynchronously sends an event message to the IoT Hub. * * @param message the message to be sent. * @param callback the callback to be invoked when a response is received. * Can be {@code null}. * @param callbackContext a context to be passed to the callback. Can be * {@code null} if no callback is provided. * * @throws IllegalArgumentException if the message provided is {@code null}. * @throws IllegalStateException if the client has not been opened yet or is * already closed. */ public void sendEventAsync(Message message, IotHubEventCallback callback, Object callbackContext) { // Codes_SRS_DEVICECLIENT_11_039: [If the client is closed, the function shall throw an IllegalStateException.] if (this.state == IotHubClientState.CLOSED) { throw new IllegalStateException( "Cannot send event from " + "an IoT Hub client that is closed."); } // Codes_SRS_DEVICECLIENT_11_033: [If the message given is null, the function shall throw an IllegalArgumentException.] if (message == null) { throw new IllegalArgumentException("Cannot send message 'null'."); } // Codes_SRS_DEVICECLIENT_11_006: [The function shall add the message, with its associated callback and callback context, to the transport.] transport.addMessage(message, callback, callbackContext); } /** * Sets the message callback. * * @param callback the message callback. Can be {@code null}. * @param context the context to be passed to the callback. Can be {@code null}. * * @return itself, for fluent setting. * * @throws IllegalArgumentException if the callback is {@code null} but a context is * passed in. * @throws IllegalStateException if the callback is set after the client is * closed. */ public DeviceClient setMessageCallback( MessageCallback callback, Object context) { // Codes_SRS_DEVICECLIENT_11_032: [If the callback is null but the context is non-null, the function shall throw an IllegalArgumentException.] if (callback == null && context != null) { throw new IllegalArgumentException( "Cannot give non-null context for a null callback."); } // Codes_SRS_DEVICECLIENT_11_012: [The function shall set the message callback, with its associated context.] this.config.setMessageCallback(callback, context); return this; } /** * Initializes an IoT Hub device client with the given parameters. * * @param iotHubHostname the IoT Hub hostname. * @param deviceId the device ID. * @param deviceKey the device key. * @param protocol the communication protocol used (i.e. HTTPS). * * @throws IllegalArgumentException if any of {@code iotHubHostname}, {@code deviceId}, * {@code deviceKey}, or {@code protocol} are {@code null}; or if {@code iotHubHostname} * does not contain a valid IoT Hub name as its prefix. * @throws URISyntaxException if the IoT Hub hostname does not conform to * RFC 3986. */ protected void initIotHubClient(String iotHubHostname, String deviceId, String deviceKey, String sharedAccessToken, IotHubClientProtocol protocol) throws URISyntaxException { // Codes_SRS_DEVICECLIENT_11_048: [If no value for 'HostName' is found in the connection string, the function shall throw an IllegalArgumentException.] // Codes_SRS_DEVICECLIENT_11_025: [If iotHubHostname is null, the constructor shall throw an IllegalArgumentException.] if (iotHubHostname == null) { throw new IllegalArgumentException( "IoT Hub hostname cannot be null."); } // Codes_SRS_DEVICECLIENT_11_049: [If no value for 'DeviceId' is found in the connection string, the function shall throw an IllegalArgumentException.] // Codes_SRS_DEVICECLIENT_11_026: [If deviceId is null, the constructor shall throw an IllegalArgumentException.] if (deviceId == null) { throw new IllegalArgumentException("Device ID cannot be null."); } // Codes_SRS_DEVICECLIENT_25_053: [**If no argument for 'sharedAccessToken' and 'SharedAccessKey' is found in the connection string, the function shall throw an IllegalArgumentException.**]** // Codes_SRS_DEVICECLIENT_11_027: [If deviceKey is null, the constructor shall throw an IllegalArgumentException.] if (deviceKey == null && sharedAccessToken == null) { throw new IllegalArgumentException("Device key and Shared Access Signature both cannot be null."); } // Codes_SRS_DEVICECLIENT_25_054: [**The constructor shall only accept either 'sharedAccessToken' or 'SharedAccessKey' from the connection string and throw an IllegalArgumentException if both are found**]** if (sharedAccessToken != null && deviceKey != null) { throw new IllegalArgumentException("Either of device key or Shared Access Signature should be provided."); } // Codes_SRS_DEVICECLIENT_11_051: [If protocol is null, the function shall throw an IllegalArgumentException.] // Codes_SRS_DEVICECLIENT_11_034: [If protocol is null, the constructor shall throw an IllegalArgumentException.] if (protocol == null) { throw new IllegalArgumentException("Protocol cannot be null."); } // Codes_SRS_DEVICECLIENT_11_052: [The constructor shall save the IoT Hub hostname, device ID, and device key as configuration parameters.] this.config = new DeviceClientConfig(iotHubHostname, deviceId, deviceKey, sharedAccessToken); // Codes_SRS_DEVICECLIENT_11_046: [The constructor shall initialize the IoT Hub transport that uses the protocol specified.] // Codes_SRS_DEVICECLIENT_11_004: [The constructor shall initialize the IoT Hub transport that uses the protocol specified.] switch (protocol) { case HTTPS: this.transport = new HttpsTransport(this.config); RECEIVE_PERIOD_MILLIS = RECEIVE_PERIOD_MILLIS_HTTPS; break; case AMQPS: this.transport = new AmqpsTransport(this.config, false); RECEIVE_PERIOD_MILLIS = RECEIVE_PERIOD_MILLIS_AMQPS; break; case AMQPS_WS: this.transport = new AmqpsTransport(this.config, true); RECEIVE_PERIOD_MILLIS = RECEIVE_PERIOD_MILLIS_AMQPS; break; case MQTT: this.transport = new MqttTransport(this.config); RECEIVE_PERIOD_MILLIS = RECEIVE_PERIOD_MILLIS_MQTT; break; default: // should never happen. throw new IllegalStateException( "Invalid client protocol specified."); } this.state = IotHubClientState.CLOSED; } protected DeviceClient() { this.config = null; this.transport = null; this.taskScheduler = null; } private void setOption_SetMinimumPollingInterval(Object value) { if (!value.equals(null)) { if (this.state != IotHubClientState.CLOSED) { throw new IllegalStateException("setOption " + SET_MINIMUM_POLLING_INTERVAL + " only works when the transport is closed"); } else { // Codes_SRS_DEVICECLIENT_02_004: [Value needs to have type long]. if (value instanceof Long) { this.RECEIVE_PERIOD_MILLIS = (long) value; } else { throw new IllegalArgumentException("value is not long = " + value); } } } else throw new IllegalArgumentException("value cannot be null"); } private void setOption_SetCertificatePath(Object value) { if (this.state != IotHubClientState.CLOSED) { throw new IllegalStateException("setOption " + SET_CERTIFICATE_PATH + "only works when the transport is closed"); } else { if (!value.equals(null)) this.config.setPathToCert((String) value); else throw new IllegalArgumentException("value cannot be null"); } } private void setOption_SetSASTokenExpiryTime(Object value) { if (!value.equals(null)) { //**Codes_SRS_DEVICECLIENT_25_009: [**"SetSASTokenExpiryTime" should have value type long**.]** long validTimeInSeconds; if (value instanceof Long) { validTimeInSeconds = (long) value; } else { throw new IllegalArgumentException("value is not long = " + value); } boolean restart = false; if (this.state != IotHubClientState.CLOSED) { try { /* Codes_SRS_DEVICECLIENT_25_010: [**"SetSASTokenExpiryTime" shall restart the transport * 1. If the device currently uses device key and * 2. If transport is already open * after updating expiry time */ if (this.config.getDeviceKey() != null) { restart = true; this.close(); } } catch (IOException e) { throw new IOError(e); } } this.config.setTokenValidSecs(validTimeInSeconds); if (restart) { if (this.state == IotHubClientState.CLOSED) { try { restart = false; this.open(); } catch (IOException e) { throw new IOError(e); } } } } else { throw new IllegalArgumentException("value should be in secs"); } } /** * Sets a runtime option identified by parameter {@code optionName} * to {@code value}. * * The options that can be set via this API are: * - SetMinimumPollingInterval - this option is applicable only * when the transport configured with this client is HTTP. This * option specifies the interval in milliseconds between calls to * the service checking for availability of new messages. The value * is expected to be of type {@code long}. * - SetCertificatePath - this option is applicable only * when the transport configured with this client is AMQP. This * option specifies the path to the certificate used to verify peer. * The value is expected to be of type {@code String}. * - SetSASTokenExpiryTime - this option is applicable for HTTP/ * AMQP/MQTT. This option specifies the interval in seconds after which * SASToken expires. If the transport is already open then setting this * option will restart the transport with the updated expiry time. The * value is expected to be of type {@code long}. * * @param optionName the option name to modify * @param value an object of the appropriate type for the option's value */ public void setOption(String optionName, Object value) { // Codes_SRS_DEVICECLIENT_02_001: [If optionName is null or not an option handled by the client, then // it shall throw IllegalArgumentException.] if (optionName == null) { throw new IllegalArgumentException("optionName is null"); } else { switch (optionName) { //**Codes_SRS_DEVICECLIENT_02_002: [**"SetMinimumPollingInterval" - time in miliseconds // between 2 consecutive polls.**]** case SET_MINIMUM_POLLING_INTERVAL: { // Codes_SRS_DEVICECLIENT_02_003: [Available only for HTTP.] if (this.transport.getClass() == HttpsTransport.class) { setOption_SetMinimumPollingInterval(value); } else { // Codes_SRS_DEVICECLIENT_02_001: [If optionName is null or not an option // handled by the client, then it shall throw IllegalArgumentException.] throw new IllegalArgumentException("optionName is unknown = " + optionName + " for " + this.transport.getClass()); } break; } //**Codes_SRS_DEVICECLIENT_25_005: [**"SetCertificatePath" - path to the certificate to verify peer.**]** case SET_CERTIFICATE_PATH: { //**Codes_SRS_DEVICECLIENT_25_006: [**"SetCertificatePath" is available only for AMQP.**]** if (this.transport.getClass() == AmqpsTransport.class) { setOption_SetCertificatePath(value); } else { // Codes_SRS_DEVICECLIENT_02_001: [If optionName is null or not an option handled by the // client, then it shall throw IllegalArgumentException.] throw new IllegalArgumentException("optionName is unknown = " + optionName + " for " + this.transport.getClass()); } break; } case SET_SAS_TOKEN_EXPIRY_TIME: { //**Codes__SRS_DEVICECLIENT_25_008: ["SetSASTokenExpiryTime" is available for HTTPS/AMQP/MQTT.] if (this.transport.getClass() == AmqpsTransport.class || this.transport.getClass() == HttpsTransport.class || this.transport.getClass() == MqttTransport.class) { setOption_SetSASTokenExpiryTime(value); } else { // Codes_SRS_DEVICECLIENT_02_001: [If optionName is null or not an option handled by the // client, then it shall throw IllegalArgumentException.] throw new IllegalArgumentException("optionName is unknown = " + optionName + " for " + this.transport.getClass()); } break; } default: throw new IllegalArgumentException("optionName is unknown = " + optionName); } } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy