我正在探索Azure IoT Hub Device Streams功能。
我有一个C#客户端连接到IoT中心(位于美国中部)并处于活动状态。
客户端使用以下算法等待传入连接(使用SDK版本1.29.0-preview-004):
var buffer = new byte[1024];
using var cancellationTokenSource = new CancellationTokenSource(TimeSpan.FromMinutes(5));
DeviceStreamRequest streamRequest = await deviceClient.WaitForDeviceStreamRequestAsync(cancellationTokenSource.Token);
if (streamRequest is null)
return;
在服务器端,我有一个在特定终结点上调用的Azure应用服务。它检索IoT中心的连接字符串
ServiceClient serviceClient = ServiceClient.CreateFromConnectionString(Constants.IoTHub.ConnectionString, TransportType.Amqp);
DeviceStreamRequest deviceStreamRequest = new DeviceStreamRequest("portal");
DeviceStreamResponse result = await serviceClient.CreateStreamAsync(serialNumber, deviceStreamRequest);
我无法建立连接。 CreateStreamAsync
呼叫在1分钟后超时,客户端似乎从未退出WaitForDeviceStreamRequestAsync
呼叫。
我一直遵循troubleshooting guide来了解发生了什么,但我什么也没得到:
问题
为什么我错过了阻止IoT设备流连接的原因?
答案 0 :(得分:1)
请注意,Azure IoT中心设备流功能仍在公开预览中(超过一年)。 您在设备端使用SDK 1.29.0-preview-004 ,在服务端使用 1.27.0-preview-004 ,并采样 DeviceStreamingSample 从 azure-iot-samples-csharp-master 软件包中。
设备流功能用于服务和设备之间的握手过程,与调用设备方法相同的通信概念。可以通过REST POST请求轻松测试此握手阶段。
出于演示目的,我将使用IoT Hub Tester(已实现了设备流功能),请参阅附录A2部分中的更多详细信息。
步骤1.为您的设备运行DeviceClientStreamingSample程序。您应该使用 Transport.Amqp 。请注意,其他人在我的测试中未通过。
步骤2。使用REST客户端工具生成POST请求。以下网址供我测试:
https://xxxxxxxxxxx.azure-devices.net/twins/device1/streams/teststream?api-version=2018-08-30-preview
标题:
accept: application/json
iothub-streaming-response-timeout-in-seconds: 15
iothub-streaming-connect-timeout-in-seconds: 5
Authorization: sas-token
有效载荷: 任何或空的
以下屏幕片段演示了POST已发送到设备1的时间:
和以下代码片段显示了在 WaitForDeviceStreamRequestAsync 方法之后的断点处停止程序:
以下屏幕截图显示了设备与服务(在本例中为测试仪)之间的完整握手和流式传输:
正如我提到的那样,Azure IoT中心测试器已实现设备流功能,以下屏幕片段显示了流缓冲区:
请注意,为 TransportType.Mqtt 运行模拟设备无法正常工作,症状与您的相同,超时。看起来(基于REST POST响应),存在一个用于为设备预订主题的错误,例如 $ iothub / streams / POST /#
但是,当您的设备对IoT中心使用直接MQTT协议时,一切运行良好,请参阅我的测试仪的屏幕片段,其中 device1 已连接到IoT中心:
一旦在屏幕上显示了Azure IoT中心测试器,就可以将其用于测试服务SDK以进行流传输,例如SDK示例中的以下行:
DeviceStreamResponse result = await _serviceClient.CreateStreamAsync(_deviceId, deviceStreamRequest).ConfigureAwait(false);
最后,根据我的上述测试,当将模拟设备配置为 TransportType.Amqp 协议时,您应该可以成功使用SDK进行流传输。
更新:
在使用REST客户端工具的情况下,您可以在标头中看到来自设备握手的响应:
此响应由以下行生成:
await _deviceClient.AcceptDeviceStreamRequestAsync(streamRequest, cancellationTokenSource.Token).ConfigureAwait(false);
基于响应头,例如
iothub-streaming-is-accepted
iothub-streaming-url
iothub-streaming-auth-token
可以通过IoT中心在设备和服务之间建立WebSocket流通信。
请注意,此后,如果使用REST客户端工具,则模拟设备将失败。
答案 1 :(得分:0)
我使用的DeviceClient
实例存在问题,该问题是根据来自预配置服务的信息构建的:
ProvisioningDeviceClient provClient = ProvisioningDeviceClient.Create(Constants.IoTProvisioningService.GlobalDeviceEndpoint, Constants.IoTProvisioningService.IdScope, security, new ProvisioningTransportHandlerHttp());
var result = await provClient.RegisterAsync();
if (result.Status != ProvisioningRegistrationStatusType.Assigned)
return;
var auth = new DeviceAuthenticationWithTpm(result.DeviceId, security);
--> DeviceClient deviceClient = DeviceClient.Create(result.AssignedHub, auth, TransportType.Amqp);
然后我正在使用DeviceClient
实例来等待请求:
var buffer = new byte[1024];
using var cancellationTokenSource = new CancellationTokenSource(TimeSpan.FromMinutes(5));
--> DeviceStreamRequest streamRequest = await deviceClient.WaitForDeviceStreamRequestAsync(cancellationTokenSource.Token);
但是它不起作用。
我意识到in the samples(DeviceClientStreamingSample),连接字符串中必须包含我没有的设备ID:
// String containing Hostname, Device Id & Device Key in one of the following formats:
// "HostName=<iothub_host_name>;DeviceId=<device_id>;SharedAccessKey=<device_key>"
// "HostName=<iothub_host_name>;CredentialType=SharedAccessSignature;DeviceId=<device_id>;SharedAccessSignature=SharedAccessSignature sr=<iot_host>/devices/<device_id>&sig=<token>&se=<expiry_time>";
// For this sample either
// - pass this value as a command-prompt argument
// - set the IOTHUB_DEVICE_CONN_STRING environment variable
// - create a launchSettings.json (see launchSettings.json.template) containing the variable
private static string s_deviceConnectionString = Environment.GetEnvironmentVariable("IOTHUB_DEVICE_CONN_STRING");
因此,我不再重复使用从预配服务获得的DeviceClient
实例,并构建了一个新实例(该代码尚未投入生产-这是一个原型):
var buffer = new byte[1024];
using var cancellationTokenSource = new CancellationTokenSource(TimeSpan.FromMinutes(5));
var deviceClient = DeviceClient.CreateFromConnectionString(Constants.IoTHub.ConnectionString + $";DeviceId={Constants.Unit.SerialNumber}", TransportType.Amqp);
DeviceStreamRequest streamRequest = await deviceClient.WaitForDeviceStreamRequestAsync(cancellationTokenSource.Token).ConfigureAwait(false);
if (streamRequest is null)
return;
await deviceClient.AcceptDeviceStreamRequestAsync(streamRequest, cancellationTokenSource.Token);