資料介紹
描述
早在 2020 年 11 月,我就發布了我的 The Things Network V2 Azure IoT Hubs & IoT Central Gateway。該項目是關于構建物聯網 (TTN)HTTP 應用程序集成,它啟用Azure IoT 中心和Azure IoT Central與Azure IoT 中心設備供應服務 (DPS)供應支持的連接。
該項目使用了“符合流行語”的Microsoft Azure服務選擇,但它不支持云到設備 (C2D) 消息,存在消息排序問題,并且部署和設置復雜。(還有其他問題,但不值得在這里重新討論)
然后在 2021 年 3 月,我再次嘗試使用我的 The Things Industries(TTI) V3 Azure IoT 連接器,它是一個TTI(消息隊列遙測傳輸)MQTT 集成,也使用了 TTI應用程序和終端設備 API 。
此版本使用MQTTNet (這是一個很棒的庫)和在應用程序啟動時連接到Azure IoT Hub或Azure IoT Central的 TTI 設備。即使我使用多個線程并對 Application 和 EndDevice 請求進行分頁,這個過程也很慢。
該應用程序更容易調試,因為我可以在桌面上運行它,并且更容易配置,因為我將配置轉移回appsettings.json文件(我可能會重新考慮放棄Azure Key Vault支持的決定)。
此版本具有基本的Azure 數字孿生定義語言 (DTDL)支持,因此可以在Azure IoT Central中“自動”預配設備。
我還添加了對Azure IoT Hubs和Azure IoT Central的C2D 支持,并基于下行鏈路消息有效負載確認標志跟蹤消息傳遞。我發現TTI 交付進度更新的順序可能有問題。
在生產環境中使用基于MQTT的集成后,我發現它過于“有狀態”并且無法從意外事件中很好地恢復。(還有其他問題,但不值得在這里重新討論)
然后在 2021 年 10 月,我決定我的“學習之旅”還沒有結束,我將構建另一個TTI 連接器,該連接器將 Azure 存儲隊列用于 C2D 和 D2C 消息。
試用該應用程序后,我意識到消息排序和部署復雜性可能是一個問題(我忘記了我的 TTN V2 網關學習),所以我暫停了項目。(雖然我確實認為這個項目可能對一些集成項目有用)
此時,我回顧了從多個 TTI 集成項目中學到的知識,并決定再次嘗試使用The Things Stack(TTS) 網絡掛鉤集成。
我的“The Things Industries(TTI) V3 connector revisited”項目是一個身份轉換云網關,它將LoRaWAN EndDevices映射到Azure IoT Hub Devices 。
連接器為每個LoRaWAN設備創建一個DeviceClient ,并且可以使用Azure 設備連接字符串或Azure 設備預配服務 (DPS) 。
在我所有的集成中,TTI 一直是設備配置的單一事實來源 (SSOT) ,因為LoRaWAN配置設置的數量和復雜性會使從其他應用程序管理它成為一個難題。(我還考慮過使用TTSEndDevice 模板來創建我可能會回來的設備)
當前版本的一個限制是,EndDevice 將連接到Azure IoT Hub (提供應用程序配置連接字符串或Azure IoT Hub DPS ),并且只有在收到 TTI 上行鏈路或Azure IoT Hub D2C 消息后才會處理 C2D 消息。一體化。
這可能是一個問題(尤其是在重新啟動集成后)或配置了新設備。我考慮過添加幾個Azure HTTP 觸發器函數,應用程序調用這些函數可以檢查設備的連接狀態并可選擇啟動連接。(短期內從 TTI EndDevice 用戶界面或 API 模擬上行鏈路應該可以工作)
我從D2C 消息傳遞開始,然后添加了C2D 消息傳遞,然后添加了支持DTDLV2 的 Device Provisioning(DPS) ,然后擴展了C2D 消息傳遞,最后實現了Azure IoT Central D2C和C2D (使用少參數、單值和JavaScript 對象表示法(JSON )有效載荷命令)
該應用程序的核心是五個Azure HTTP 觸發函數(已發送函數當前未使用)和一個為 C2D 調用的方法(與SetReceiveMessageHandlerAsync方法連接)消息。
Azure IoT 中心可以調用方法(同步)或將消息(異步)發送到設備進行處理。Azure IoT 中心 DeviceClient有兩個方法SetMethodDefaultHandlerAsync和SetReceiveMessageHandlerAsync ,它們可以處理直接方法和消息。
在對以前的 TTI 連接器進行了一些實驗之后,我發現DirectMethods的同步特性不適用于LoRAWAN通常“不規則”的上行鏈路,因此目前不支持它們。
該集成廣泛使用了Microsoft.Extensions.Logging功能和Azure Application Insights ,因此調試、監控和故障查找更省時。
我已將有用的“元數據”添加到各個日志項中,因此更容易跟蹤為處理事件而執行的所有步驟,例如 ReceiveMessageCallback 、AbandonAsync 、CompleteAsync和RejectAsync C2D 消息處理中使用的 LockToken。
應用程序配置概述
可以使用appsettings.json文件配置應用程序(對桌面開發和調試很有用)
{
"Values": {
"AzureWebJobsStorage": "DefaultEndpointsProtocol=https;AccountName=...",
"FUNCTIONS_WORKER_RUNTIME": "dotnet-isolated",
"APPINSIGHTS_INSTRUMENTATIONKEY": "..."
},
"TheThingsIndustries": {
"WebhookBaseURL": "https://....eu1.cloud.thethings.industries/api/v3/as/applicat ions",
"Applications": {
"seeeduinolorawan": {
"webhookId": "azure-iot-hub-connector",
"APIKey": "..."
},
"Wisnode Devices": {
"webhookId": "azure-iot-hub-connector",
"APIKey": "..."
},
"dragino-lht65": {
"webhookId": "azure-iot-hub-connector",
"APIKey": "..."
},
"SeeeduinoLoRaWAN100": {
"webhookId": "azure-iot-hub-connector",
"APIKey": "..."
},
"rak3172": {
"webhookId": "azure-iot-hub-connector",
"APIKey": "..."
},
"application1": {
"webhookId": "azure-iot-hub-connector",
"APIKey": "..."
}
}
},
"AzureIoT": {
"DeviceClientCacheSlidingExpiration": "P2H30M",
"IoTHub": {
"IoTHubConnectionString": "HostName=...",
"Applications": {
"SeeeduinoLoRaWAN": {
"DtdlModelId": "dtmi:ttnv3connectorclient:SeeeduinoLoRaWAN4cz;1"
},
"Wisnode Devices": {
},
"Dragino LHT65": {
}
}
},
"DeviceProvisioningService": {
"IdScope": "0ne..",
"Applications": {
"seeeduinolorawan": {
"DtdlModelId": "dtmi:ttnv3connectorclient:SeeeduinoLoRaWAN4cz;1",
"GroupEnrollmentKey": "...",
},
"Wisnode Devices": {
"GroupEnrollmentKey": "..."
},
"dragino-lht65": {
"GroupEnrollmentKey": "..."
},
"rak3172": {
"GroupEnrollmentKey": "..."
},
"application1": {
"DtdlModelId": "dtmi:ttnv3connectorclient:FezduinoWisnodeV14x8;4",
"GroupEnrollmentKey": "..."
}
}
},
"IoTCentral": {
"methods": {
"LightsGoOn": {
"Port": 10,
"Payload": "{"value_1": 1}"
},
"LightsGoOff": {
"Port": 10,
"Payload": "{"value_1": 0}"
},
"value_0": {
"Port": 20
},
"value_1": {
"Port": 21
},
"value_2": {
"Port": 22
},
"TemperatureOOBAlertMinimumAndMaximum": {
"Port": 23
},
}
}
暫存和生產部署的首選方法)是使用Azure 門戶Azure 功能配置刀片
Things Industries Webhook 集成配置
要發送下行鏈路和接收上行鏈路消息,必須配置 TTI 應用程序和TTI 連接器并配置 API 密鑰。
注意 – TTN URL 和 Azure IoT 中心設備標識符區分大小寫
TTI 連接器需要webhookbaseURL ,然后是每個 TTI 應用程序和一個API 密鑰,以及 WebhookId
當調用 Azure Functions 時,Azure Function Host Key會在名為“x-functions-key”的HTTP 標頭中傳遞
當調用TTI webhook 下行鏈路端點時, TTI 應用程序密鑰在標準HTTP 授權標頭中傳遞。
Azure IoT 中心連接配置
TTI 連接器需要共享訪問簽名 (SAS) 設備策略連接字符串才能連接到Azure IoT 中心。
Azure IoT Hub設備必須手動或通過Azure IoT Hub REST API進行預配。我已經試用了一個Azure 邏輯應用程序,它管理設備配置并且可以在操作失敗時穩健地處理所需的補償事務。
如果同時配置了Azure IoT 中心/Azure IoT 中心設備預配 (DPS) 支持,則 TTI 連接器應用程序將不會啟動。
Azure IoT 中心設備預配服務 (DPS) 配置
TTI 連接器支持用于獨立Azure IoT 中心應用程序的Azure IoT 中心設備預配服務 (DPS) 。TTI 連接器實現還支持用于設備配置的Azure IoT Central 數字孿生定義語言( DTDL V2 )。
Azure IoT 中心設備預配服務支持使用X.509證書、可信平臺模塊 (TPM)和使用共享訪問簽名(SAS) 安全令牌的對稱密鑰進行設備證明。
Things Industries(TTI) V3 Azure IoT 連接器僅支持對稱密鑰設備證明。
如果Azure IoT 中心/ Azure IoT 中心設備預配 (DPS)支持兩者/兩者均未配置,則 TTI 連接器應用程序將不會啟動。
Azure IoT 中心設備預配服務 (DPS)具有確定設備分配方式的服務級別設置。有四種支持的分配策略:
- 均勻加權分布:鏈接的物聯網中心同樣可能有設備配置給它們。默認設置。如果您僅將設備預配到一個 IoT 中心,則可以保留此設置。
- 最低延遲:將設備預配到設備延遲最低的 IoT 中心。如果多個鏈接的 IoT 集線器將提供相同的最低延遲,則供應服務會在這些集線器之間散列設備
- 通過注冊列表進行靜態配置:注冊列表中所需 IoT 中心的規范優先于服務級別分配策略。
- 自定義(使用 Azure 函數):自定義分配策略使您可以更好地控制如何將設備分配給 IoT 中心。這是通過使用 Azure 函數中的自定義代碼將設備分配給 IoT 中心來實現的。設備預配服務調用您的 Azure 函數代碼,提供有關設備和代碼注冊的所有相關信息。您的函數代碼將執行并返回用于預配設備的 IoT 中心信息。
在我的測試環境中,我使用均勻加權分布,當我預置 1000 臺設備時,它們分布在我的五個Azure IoT 中心。
Azure IoT Central DPS 配置
TTI 連接器支持Azure IoT Central應用程序所需的Azure IoT 中心設備預配服務 (DPS) (有一種預配單個設備的方法) 。TTI 連接器實現還支持用于“自動”設備預配的Azure IoT Central 數字孿生定義語言( DTDL V2 )。
如果同時配置了Azure IoT 中心/Azure IoT 中心設備預配 (DPS) 支持,則 TTI 連接器應用程序將不會啟動。
第一步是配置Azure IoT Central 注冊組(確保“自動連接該組中的設備”為“零接觸”配置)并將IDScope和組注冊密鑰復制到 TTI 連接器配置
然后,我為我的RAK3172 分線板基于 .Net Core 供電的測試設備創建了一個 Azure IoT Central 模板。
還可以使用在 TTI 連接器配置中指定的可選 dtdlmodelid 為 TTI 應用程序設置設備模板 @Id。
Azure IoT Hub 設備到云 (D2C)
LoRaWAN設備使用共享訪問簽名 (SAS) 設備策略連接字符串連接到Azure IoT 中心。我正在使用Device Twin Explorer顯示遙測數據并向我的傳感器節點發送消息。
如果有效負載已被有效負載格式化程序解碼,則將對其進行后處理,然后包含在消息有效負載中。
try
{
JObject telemetryEvent = new JObject
{
{ "ApplicationID", applicationId },
{ "DeviceEUI" , payload.EndDeviceIds.DeviceEui},
{ "DeviceID", deviceId },
{ "Port", port },
{ "Simulated", payload.Simulated },
{ "ReceivedAtUtc", payload.UplinkMessage.ReceivedAtUtc.ToString("s", CultureInfo.InvariantCulture) },
{ "PayloadRaw", payload.UplinkMessage.PayloadRaw }
};
// If the payload has been decoded by payload formatter, put it in the message body.
if (payload.UplinkMessage.PayloadDecoded != null)
{
EnumerateChildren(telemetryEvent, payload.UplinkMessage.PayloadDecoded);
}
// Send the message to Azure IoT Hub
using (Message ioTHubmessage = new Message(Encoding.ASCII.GetBytes(JsonConvert.SerializeObject(telemetryEvent))))
{
// Ensure the displayed time is the acquired time rather than the uploaded time.
ioTHubmessage.Properties.Add("iothub-creation-time-utc", payload.UplinkMessage.ReceivedAtUtc.ToString("s", CultureInfo.InvariantCulture));
ioTHubmessage.Properties.Add("ApplicationId", applicationId);
ioTHubmessage.Properties.Add("DeviceEUI", payload.EndDeviceIds.DeviceEui);
ioTHubmessage.Properties.Add("DeviceId", deviceId);
ioTHubmessage.Properties.Add("port", port.ToString());
ioTHubmessage.Properties.Add("Simulated", payload.Simulated.ToString());
await deviceClient.SendEventAsync(ioTHubmessage);
logger.LogInformation("Uplink-DeviceID:{deviceId} SendEventAsync success", deviceId);
}
}
catch( Exception ex)
{
logger.LogError(ex, "Uplink-DeviceID:{deviceId} SendEventAsync failure", deviceId);
// If retries etc fail remove from the cache and it will get tried again on the next message
_DeviceClients.Remove(deviceId);
}
Azure IoT Hub 云到設備 (C2D)
基本 Azure IoT 中心 C2D 消息傳遞僅需要端口號、TTI 確認、隊列和優先級(如果未提供)使用默認值。
- 確認 - 真/假
- 隊列 - 推送/替換
- 優先級 - 最低/低/低于正常/正常/高于正常/高/最高
這些選項在消息屬性中指定。為了測試此功能,我使用了Azure Device Explorer Twin應用程序,該應用程序還顯示消息傳遞進度。
如果負載無效,則假定JSON是Base64編碼的(需要額外驗證)并復制到下行鏈路消息的 payload_raw 字段中。
如果有效載荷是有效的JSON ,它被“嫁接”(想不出更好的詞)到TTI 下行鏈路消息 decoded_pa??yload 字段中。
Azure IoT Central 設備到云 (D2C)
連接器“轉換”了The Things Industries(TTI) MyDevices Cayenne 低功耗協議 (LPP) 有效負載格式化程序的輸出(它還支持自定義編碼器/解碼器,但尚未經過廣泛測試),以便它可以被Azure IoT Central攝取.
用于處理TTI 上行鏈路消息的Azure 函數首先反序列化JSON負載,丟棄任何LoRaWAN 控制消息和具有空負載的消息。
為了測試更復雜的場景,我創建了一個Azure IoT Central 設備模板,該模板具有“功能類型”的位置。
如果消息已由有效負載格式化程序成功解碼,則 PayloadDecoded 內容將被“嫁接”到Azure IoT Central 遙測消息中。
Azure IoT Central 位置遙測消息的格式與 TTI Cayenne LPP Payload格式化程序的輸出格式略有不同,因此必須對有效負載進行“后處理”(使用新的Azure IoT Central 地圖遙測入口功能,這可能不是必需的) .
我可能必須擴展后處理以支持其他Cayenne LPP 或第三方有效負載格式化程序。
Azure IoT Central 云到設備 (C2D)
要發送下行鏈路消息,TTI 需要一個無法通過 Azure IoT Central 命令設置提供的LoRaWAN 端口號(加上可選隊列、確認和優先級值),因此這些值在集成配置中進行配置。
我的集成僅使用離線排隊命令,因為消息通常不會立即傳遞到傳感器節點,特別是如果傳感器節點僅每半小時/小時/天發送一條消息。
每個 TTI 應用程序都有零個或多個Azure IoT Central 命令配置,這些配置指定 LoRaWAN 端口號,以及可選的有效負載、已確認的 TTI 下行鏈路消息、優先級和隊列設置。
無參數命令
即使該命令沒有參數,也必須配置下行鏈路消息負載(當前只有JSON編碼的負載,考慮到原始Base64負載支持)
此示例說明如何使用內置的Cayenne LPP 有效負載格式化程序配置打開和關閉燈的命令。
枚舉參數
此示例顯示如何通過從選項列表中選擇所需狀態來配置打開和關閉風扇的命令。
單值參數
此示例顯示如何配置用于設置警報的最低溫度的命令。
JSON 參數
此示例說明如何配置命令以設置警報的最低和最高溫度。
交貨確認
為了處理消息傳遞確認,將包含消息LockToken 的相關標識符添加到下行鏈路有效負載中的相關 ID 。
唯一需要的消息屬性是 LoRaWAN 端口號,確認、隊列、優先級和有效負載字段是可選的。
如果端口號屬性或任何其他屬性不正確,則調用DeviceClient.RejectAsync ,這會從設備隊列中刪除消息并向服務器指示無法處理該消息。
使用存儲在 TTI CorrelationID 中的 Azure 令牌跟蹤消息傳遞確認過程。
未確認的消息
TTI 連接器調用CompleteAsync方法(使用 TTI CorrelationIDs 列表中的 LockToken),該方法在調用“排隊”Azure 函數時從Azure IoT 中心設備隊列中刪除消息。
確認消息
如果消息傳遞成功(調用 Ack 函數),則會調用CompleteAsync方法(使用 TTI CorrelationIDs 列表中的 LockToken)從 Azure IoT 中心設備隊列中刪除消息。
如果消息傳遞失敗(調用失敗的函數),則調用AbandonAsync方法(使用 TTI CorrelationIDs 列表中的 LockToken)將下行鏈路消息放回 Azure IoT 中心設備隊列。
如果消息傳遞不成功(調用 Nack 函數),則會調用RejectAsync方法(使用 CorrelationIDs 列表中的 LockToken),該方法從設備隊列中刪除消息并向服務器指示無法處理該消息。
消息 Failed( AbandonAsync )、Ack( CompleteAsync ) 和 Nack( RejectAsync ) 的處理方式需要進行更多測試,以確認我對 TTI 確認消息傳遞順序的理解。
謹防
當Azure IoT 中心下行鏈路消息超時并重新發送時,將確認消息與不定期發送上行鏈路消息的設備一起使用可能會導致奇怪的問題。
執行摘要
這個項目已經付出了一年多的努力。我學到了很多關于LoRaWAN以及The Things Industries如何運作的知識。
有時是一些愚蠢的事情,比如拖慢進度的錯字
在我確信它已準備好投入生產之前,我對該軟件進行了一個月的浸泡測試,但有幾次我達到了我的Azure 支出限制,這禁用了我的所有服務,因此我不得不重新運行浸泡測試。
如果您有任何問題或反饋給我留言,我在Twitter上,我的博客上有更多關于我的“學習之旅”的詳細信息。
- FET430UIF V3固件降級程序 0次下載
- The Things Network V2 Azure物聯網中心和物聯網中心網關
- 如何將手機連接到Azure IoT Central
- 3M卡連接器microSD數據手冊.pdf 6次下載
- NODEMCU V3燒寫 AT固件
- 正點原子ESP8266獲取網絡天氣適配戰艦V3 25次下載
- 戰艦V3 STM32103封裝庫下載 79次下載
- GAKATO光速達中控解決方案V3 0次下載
- V3硬件設計指南V1.0 68次下載
- FY2005K編程器軟件V3安裝程序 40次下載
- ITIL V3白皮書 0次下載
- PDP4218三星V3屏電源工作原理及時序
- S60 V3(opda)手機權限破解+去除簽名限制
- 摩托羅拉V3手機原理圖紙
- 摩托羅拉v3電路圖
- 連接器的類型及應用 801次閱讀
- 什么是連接器?連接器有哪些種類? 2917次閱讀
- 3D打印技術制造連接器有什么優勢 1126次閱讀
- 連接器到底是什么?連接器有哪些類型 9260次閱讀
- 連接器的基礎知識入門介紹 1.1w次閱讀
- FFC連接器與FPC連接器的差別及常用種類有哪些 8721次閱讀
- 貼片連接器是什么它有什么作用 1w次閱讀
- 射頻連接器的選用 1918次閱讀
- fpc連接器結構_fpc連接器原理 1w次閱讀
- 電子連接器是什么_電子連接器的材料 5704次閱讀
- 關于射頻同軸連接器的詳細介紹 1.2w次閱讀
- 什么是射頻連接器_射頻連接器有什么用 3.7w次閱讀
- 電子連接器結構 2206次閱讀
- 射頻連接器是什么_射頻連接器分類與規格介紹 2.5w次閱讀
- 連接器的應用與連接器的市場發展 2414次閱讀
下載排行
本周
- 1ADI高性能電源管理解決方案
- 2.43 MB | 449次下載 | 免費
- 2免費開源CC3D飛控資料(電路圖&PCB源文件、BOM、
- 5.67 MB | 136次下載 | 1 積分
- 3基于STM32單片機智能手環心率計步器體溫顯示設計
- 0.10 MB | 123次下載 | 免費
- 4550W充電機原理圖
- 0.13 MB | 2次下載 | 6 積分
- 5USB的PD快充協議電壓誘騙控制器FS312A中文手冊
- 1.51 MB | 2次下載 | 免費
- 6USB的PD和OC快充協議電壓誘騙控制器FS312B中文手冊
- 1.35 MB | 2次下載 | 免費
- 7USB Type_C PD快充協議智能觸發芯片FS8025B應用手冊
- 1.48 MB | 1次下載 | 免費
- 8ADI公司串行端口開發和故障排除指南
- 343.09KB | 1次下載 | 免費
本月
- 1ADI高性能電源管理解決方案
- 2.43 MB | 449次下載 | 免費
- 2免費開源CC3D飛控資料(電路圖&PCB源文件、BOM、
- 5.67 MB | 136次下載 | 1 積分
- 3基于STM32單片機智能手環心率計步器體溫顯示設計
- 0.10 MB | 123次下載 | 免費
- 4使用單片機實現七人表決器的程序和仿真資料免費下載
- 2.96 MB | 44次下載 | 免費
- 53314A函數發生器維修手冊
- 16.30 MB | 31次下載 | 免費
- 6美的電磁爐維修手冊大全
- 1.56 MB | 22次下載 | 5 積分
- 7使用TL431設計電源
- 0.67 MB | 10次下載 | 免費
- 8感應筆電路圖
- 0.06 MB | 10次下載 | 免費
總榜
- 1matlab軟件下載入口
- 未知 | 935119次下載 | 10 積分
- 2開源硬件-PMP21529.1-4 開關降壓/升壓雙向直流/直流轉換器 PCB layout 設計
- 1.48MB | 420062次下載 | 10 積分
- 3Altium DXP2002下載入口
- 未知 | 233084次下載 | 10 積分
- 4電路仿真軟件multisim 10.0免費下載
- 340992 | 191367次下載 | 10 積分
- 5十天學會AVR單片機與C語言視頻教程 下載
- 158M | 183335次下載 | 10 積分
- 6labview8.5下載
- 未知 | 81581次下載 | 10 積分
- 7Keil工具MDK-Arm免費下載
- 0.02 MB | 73807次下載 | 10 積分
- 8LabVIEW 8.6下載
- 未知 | 65987次下載 | 10 積分
評論
查看更多