zigbee-herdsman 3.5.1 → 4.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (726) hide show
  1. package/.github/dependabot.yml +9 -3
  2. package/.github/workflows/ci.yml +2 -4
  3. package/.github/workflows/stale.yml +1 -1
  4. package/.release-please-manifest.json +1 -1
  5. package/.vscode/extensions.json +3 -0
  6. package/.vscode/settings.json +11 -0
  7. package/CHANGELOG.md +25 -0
  8. package/biome.json +89 -0
  9. package/dist/adapter/adapter.d.ts +11 -11
  10. package/dist/adapter/adapter.d.ts.map +1 -1
  11. package/dist/adapter/adapter.js +14 -14
  12. package/dist/adapter/adapterDiscovery.d.ts +6 -6
  13. package/dist/adapter/adapterDiscovery.d.ts.map +1 -1
  14. package/dist/adapter/adapterDiscovery.js +139 -141
  15. package/dist/adapter/adapterDiscovery.js.map +1 -1
  16. package/dist/adapter/const.d.ts +1 -1
  17. package/dist/adapter/const.js +2 -2
  18. package/dist/adapter/deconz/adapter/deconzAdapter.d.ts +16 -16
  19. package/dist/adapter/deconz/adapter/deconzAdapter.d.ts.map +1 -1
  20. package/dist/adapter/deconz/adapter/deconzAdapter.js +95 -100
  21. package/dist/adapter/deconz/adapter/deconzAdapter.js.map +1 -1
  22. package/dist/adapter/deconz/driver/constants.d.ts +5 -5
  23. package/dist/adapter/deconz/driver/constants.d.ts.map +1 -1
  24. package/dist/adapter/deconz/driver/driver.d.ts +9 -9
  25. package/dist/adapter/deconz/driver/driver.d.ts.map +1 -1
  26. package/dist/adapter/deconz/driver/driver.js +112 -106
  27. package/dist/adapter/deconz/driver/driver.js.map +1 -1
  28. package/dist/adapter/deconz/driver/frame.d.ts +1 -1
  29. package/dist/adapter/deconz/driver/frame.d.ts.map +1 -1
  30. package/dist/adapter/deconz/driver/frame.js +1 -2
  31. package/dist/adapter/deconz/driver/frame.js.map +1 -1
  32. package/dist/adapter/deconz/driver/frameParser.d.ts +1 -1
  33. package/dist/adapter/deconz/driver/frameParser.d.ts.map +1 -1
  34. package/dist/adapter/deconz/driver/frameParser.js +66 -70
  35. package/dist/adapter/deconz/driver/frameParser.js.map +1 -1
  36. package/dist/adapter/deconz/driver/parser.d.ts +1 -1
  37. package/dist/adapter/deconz/driver/parser.d.ts.map +1 -1
  38. package/dist/adapter/deconz/driver/parser.js +2 -2
  39. package/dist/adapter/deconz/driver/parser.js.map +1 -1
  40. package/dist/adapter/deconz/driver/writer.d.ts +2 -2
  41. package/dist/adapter/deconz/driver/writer.d.ts.map +1 -1
  42. package/dist/adapter/deconz/driver/writer.js +1 -1
  43. package/dist/adapter/ember/adapter/emberAdapter.d.ts +19 -19
  44. package/dist/adapter/ember/adapter/emberAdapter.d.ts.map +1 -1
  45. package/dist/adapter/ember/adapter/emberAdapter.js +117 -97
  46. package/dist/adapter/ember/adapter/emberAdapter.js.map +1 -1
  47. package/dist/adapter/ember/adapter/endpoints.d.ts +1 -1
  48. package/dist/adapter/ember/adapter/endpoints.d.ts.map +1 -1
  49. package/dist/adapter/ember/adapter/oneWaitress.d.ts +5 -5
  50. package/dist/adapter/ember/adapter/oneWaitress.d.ts.map +1 -1
  51. package/dist/adapter/ember/adapter/oneWaitress.js +2 -2
  52. package/dist/adapter/ember/adapter/oneWaitress.js.map +1 -1
  53. package/dist/adapter/ember/adapter/tokensManager.d.ts +3 -3
  54. package/dist/adapter/ember/adapter/tokensManager.d.ts.map +1 -1
  55. package/dist/adapter/ember/adapter/tokensManager.js +20 -23
  56. package/dist/adapter/ember/adapter/tokensManager.js.map +1 -1
  57. package/dist/adapter/ember/enums.d.ts +4 -4
  58. package/dist/adapter/ember/enums.d.ts.map +1 -1
  59. package/dist/adapter/ember/enums.js +4 -6
  60. package/dist/adapter/ember/enums.js.map +1 -1
  61. package/dist/adapter/ember/ezsp/buffalo.d.ts +4 -4
  62. package/dist/adapter/ember/ezsp/buffalo.d.ts.map +1 -1
  63. package/dist/adapter/ember/ezsp/buffalo.js +30 -32
  64. package/dist/adapter/ember/ezsp/buffalo.js.map +1 -1
  65. package/dist/adapter/ember/ezsp/ezsp.d.ts +46 -46
  66. package/dist/adapter/ember/ezsp/ezsp.d.ts.map +1 -1
  67. package/dist/adapter/ember/ezsp/ezsp.js +85 -111
  68. package/dist/adapter/ember/ezsp/ezsp.js.map +1 -1
  69. package/dist/adapter/ember/ezspError.d.ts +1 -1
  70. package/dist/adapter/ember/types.d.ts +14 -14
  71. package/dist/adapter/ember/types.d.ts.map +1 -1
  72. package/dist/adapter/ember/uart/ash.d.ts +5 -5
  73. package/dist/adapter/ember/uart/ash.d.ts.map +1 -1
  74. package/dist/adapter/ember/uart/ash.js +80 -71
  75. package/dist/adapter/ember/uart/ash.js.map +1 -1
  76. package/dist/adapter/ember/uart/parser.d.ts +2 -2
  77. package/dist/adapter/ember/uart/parser.d.ts.map +1 -1
  78. package/dist/adapter/ember/uart/parser.js +2 -1
  79. package/dist/adapter/ember/uart/parser.js.map +1 -1
  80. package/dist/adapter/ember/uart/queues.js +27 -29
  81. package/dist/adapter/ember/uart/queues.js.map +1 -1
  82. package/dist/adapter/ember/uart/writer.d.ts +1 -1
  83. package/dist/adapter/ember/uart/writer.d.ts.map +1 -1
  84. package/dist/adapter/ember/uart/writer.js +3 -5
  85. package/dist/adapter/ember/uart/writer.js.map +1 -1
  86. package/dist/adapter/ember/utils/initters.d.ts +2 -2
  87. package/dist/adapter/ember/utils/initters.d.ts.map +1 -1
  88. package/dist/adapter/ember/utils/initters.js +1 -1
  89. package/dist/adapter/ember/utils/math.js +1 -1
  90. package/dist/adapter/events.d.ts +1 -1
  91. package/dist/adapter/events.d.ts.map +1 -1
  92. package/dist/adapter/ezsp/adapter/backup.d.ts +3 -3
  93. package/dist/adapter/ezsp/adapter/backup.d.ts.map +1 -1
  94. package/dist/adapter/ezsp/adapter/backup.js +8 -10
  95. package/dist/adapter/ezsp/adapter/backup.js.map +1 -1
  96. package/dist/adapter/ezsp/adapter/ezspAdapter.d.ts +12 -12
  97. package/dist/adapter/ezsp/adapter/ezspAdapter.d.ts.map +1 -1
  98. package/dist/adapter/ezsp/adapter/ezspAdapter.js +45 -44
  99. package/dist/adapter/ezsp/adapter/ezspAdapter.js.map +1 -1
  100. package/dist/adapter/ezsp/driver/driver.d.ts +12 -12
  101. package/dist/adapter/ezsp/driver/driver.d.ts.map +1 -1
  102. package/dist/adapter/ezsp/driver/driver.js +148 -141
  103. package/dist/adapter/ezsp/driver/driver.js.map +1 -1
  104. package/dist/adapter/ezsp/driver/ezsp.d.ts +7 -7
  105. package/dist/adapter/ezsp/driver/ezsp.d.ts.map +1 -1
  106. package/dist/adapter/ezsp/driver/ezsp.js +61 -59
  107. package/dist/adapter/ezsp/driver/ezsp.js.map +1 -1
  108. package/dist/adapter/ezsp/driver/frame.js +2 -2
  109. package/dist/adapter/ezsp/driver/index.d.ts +2 -2
  110. package/dist/adapter/ezsp/driver/multicast.d.ts +8 -5
  111. package/dist/adapter/ezsp/driver/multicast.d.ts.map +1 -1
  112. package/dist/adapter/ezsp/driver/multicast.js +9 -9
  113. package/dist/adapter/ezsp/driver/multicast.js.map +1 -1
  114. package/dist/adapter/ezsp/driver/parser.d.ts +1 -1
  115. package/dist/adapter/ezsp/driver/parser.js +3 -3
  116. package/dist/adapter/ezsp/driver/types/basic.d.ts +5 -5
  117. package/dist/adapter/ezsp/driver/types/basic.d.ts.map +1 -1
  118. package/dist/adapter/ezsp/driver/types/basic.js +12 -33
  119. package/dist/adapter/ezsp/driver/types/basic.js.map +1 -1
  120. package/dist/adapter/ezsp/driver/types/index.d.ts +3 -3
  121. package/dist/adapter/ezsp/driver/types/index.d.ts.map +1 -1
  122. package/dist/adapter/ezsp/driver/types/index.js +8 -9
  123. package/dist/adapter/ezsp/driver/types/index.js.map +1 -1
  124. package/dist/adapter/ezsp/driver/types/named.d.ts +2 -2
  125. package/dist/adapter/ezsp/driver/types/named.d.ts.map +1 -1
  126. package/dist/adapter/ezsp/driver/types/named.js +8 -11
  127. package/dist/adapter/ezsp/driver/types/named.js.map +1 -1
  128. package/dist/adapter/ezsp/driver/types/struct.d.ts +2 -2
  129. package/dist/adapter/ezsp/driver/types/struct.d.ts.map +1 -1
  130. package/dist/adapter/ezsp/driver/types/struct.js +207 -214
  131. package/dist/adapter/ezsp/driver/types/struct.js.map +1 -1
  132. package/dist/adapter/ezsp/driver/uart.d.ts +2 -2
  133. package/dist/adapter/ezsp/driver/uart.d.ts.map +1 -1
  134. package/dist/adapter/ezsp/driver/uart.js +48 -37
  135. package/dist/adapter/ezsp/driver/uart.js.map +1 -1
  136. package/dist/adapter/ezsp/driver/utils/crc16ccitt.d.ts +3 -3
  137. package/dist/adapter/ezsp/driver/utils/crc16ccitt.d.ts.map +1 -1
  138. package/dist/adapter/ezsp/driver/utils/crc16ccitt.js +16 -34
  139. package/dist/adapter/ezsp/driver/utils/crc16ccitt.js.map +1 -1
  140. package/dist/adapter/ezsp/driver/utils/index.d.ts +4 -4
  141. package/dist/adapter/ezsp/driver/utils/index.d.ts.map +1 -1
  142. package/dist/adapter/ezsp/driver/utils/index.js +4 -4
  143. package/dist/adapter/ezsp/driver/utils/index.js.map +1 -1
  144. package/dist/adapter/ezsp/driver/writer.d.ts +1 -1
  145. package/dist/adapter/ezsp/driver/writer.js +2 -2
  146. package/dist/adapter/index.d.ts +3 -3
  147. package/dist/adapter/serialPort.d.ts +3 -3
  148. package/dist/adapter/serialPort.d.ts.map +1 -1
  149. package/dist/adapter/serialPort.js +1 -0
  150. package/dist/adapter/serialPort.js.map +1 -1
  151. package/dist/adapter/socketPortUtils.js +3 -3
  152. package/dist/adapter/tstype.d.ts +6 -6
  153. package/dist/adapter/z-stack/adapter/adapter-backup.d.ts +4 -4
  154. package/dist/adapter/z-stack/adapter/adapter-backup.d.ts.map +1 -1
  155. package/dist/adapter/z-stack/adapter/adapter-backup.js +106 -119
  156. package/dist/adapter/z-stack/adapter/adapter-backup.js.map +1 -1
  157. package/dist/adapter/z-stack/adapter/adapter-nv-memory.d.ts +10 -10
  158. package/dist/adapter/z-stack/adapter/adapter-nv-memory.d.ts.map +1 -1
  159. package/dist/adapter/z-stack/adapter/adapter-nv-memory.js +23 -24
  160. package/dist/adapter/z-stack/adapter/adapter-nv-memory.js.map +1 -1
  161. package/dist/adapter/z-stack/adapter/manager.d.ts +6 -6
  162. package/dist/adapter/z-stack/adapter/manager.d.ts.map +1 -1
  163. package/dist/adapter/z-stack/adapter/manager.js +115 -127
  164. package/dist/adapter/z-stack/adapter/manager.js.map +1 -1
  165. package/dist/adapter/z-stack/adapter/tstype.d.ts +3 -3
  166. package/dist/adapter/z-stack/adapter/tstype.js +3 -3
  167. package/dist/adapter/z-stack/adapter/zStackAdapter.d.ts +10 -10
  168. package/dist/adapter/z-stack/adapter/zStackAdapter.d.ts.map +1 -1
  169. package/dist/adapter/z-stack/adapter/zStackAdapter.js +125 -137
  170. package/dist/adapter/z-stack/adapter/zStackAdapter.js.map +1 -1
  171. package/dist/adapter/z-stack/constants/common.d.ts.map +1 -1
  172. package/dist/adapter/z-stack/constants/common.js +0 -3
  173. package/dist/adapter/z-stack/constants/common.js.map +1 -1
  174. package/dist/adapter/z-stack/constants/index.d.ts +10 -10
  175. package/dist/adapter/z-stack/constants/index.d.ts.map +1 -1
  176. package/dist/adapter/z-stack/constants/index.js +2 -2
  177. package/dist/adapter/z-stack/constants/index.js.map +1 -1
  178. package/dist/adapter/z-stack/constants/utils.d.ts +1 -1
  179. package/dist/adapter/z-stack/constants/utils.js +2 -2
  180. package/dist/adapter/z-stack/constants/utils.js.map +1 -1
  181. package/dist/adapter/z-stack/models/startup-options.d.ts +2 -2
  182. package/dist/adapter/z-stack/models/startup-options.d.ts.map +1 -1
  183. package/dist/adapter/z-stack/structs/entries/address-manager-entry.d.ts +1 -1
  184. package/dist/adapter/z-stack/structs/entries/address-manager-entry.d.ts.map +1 -1
  185. package/dist/adapter/z-stack/structs/entries/address-manager-entry.js +4 -5
  186. package/dist/adapter/z-stack/structs/entries/address-manager-entry.js.map +1 -1
  187. package/dist/adapter/z-stack/structs/entries/address-manager-table.d.ts +2 -2
  188. package/dist/adapter/z-stack/structs/entries/address-manager-table.d.ts.map +1 -1
  189. package/dist/adapter/z-stack/structs/entries/address-manager-table.js +3 -4
  190. package/dist/adapter/z-stack/structs/entries/address-manager-table.js.map +1 -1
  191. package/dist/adapter/z-stack/structs/entries/aps-link-key-data-entry.d.ts +1 -1
  192. package/dist/adapter/z-stack/structs/entries/aps-link-key-data-entry.d.ts.map +1 -1
  193. package/dist/adapter/z-stack/structs/entries/aps-link-key-data-entry.js +1 -2
  194. package/dist/adapter/z-stack/structs/entries/aps-link-key-data-entry.js.map +1 -1
  195. package/dist/adapter/z-stack/structs/entries/aps-link-key-data-table.d.ts +2 -2
  196. package/dist/adapter/z-stack/structs/entries/aps-link-key-data-table.d.ts.map +1 -1
  197. package/dist/adapter/z-stack/structs/entries/aps-link-key-data-table.js +3 -4
  198. package/dist/adapter/z-stack/structs/entries/aps-link-key-data-table.js.map +1 -1
  199. package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-entry.d.ts +1 -1
  200. package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-entry.d.ts.map +1 -1
  201. package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-entry.js +6 -7
  202. package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-entry.js.map +1 -1
  203. package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-table.d.ts +2 -2
  204. package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-table.d.ts.map +1 -1
  205. package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-table.js +3 -4
  206. package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-table.js.map +1 -1
  207. package/dist/adapter/z-stack/structs/entries/channel-list.d.ts +1 -1
  208. package/dist/adapter/z-stack/structs/entries/channel-list.d.ts.map +1 -1
  209. package/dist/adapter/z-stack/structs/entries/channel-list.js +1 -2
  210. package/dist/adapter/z-stack/structs/entries/channel-list.js.map +1 -1
  211. package/dist/adapter/z-stack/structs/entries/has-configured.d.ts +1 -1
  212. package/dist/adapter/z-stack/structs/entries/has-configured.d.ts.map +1 -1
  213. package/dist/adapter/z-stack/structs/entries/has-configured.js +2 -3
  214. package/dist/adapter/z-stack/structs/entries/has-configured.js.map +1 -1
  215. package/dist/adapter/z-stack/structs/entries/index.d.ts +16 -16
  216. package/dist/adapter/z-stack/structs/entries/nib.d.ts +1 -1
  217. package/dist/adapter/z-stack/structs/entries/nib.d.ts.map +1 -1
  218. package/dist/adapter/z-stack/structs/entries/nib.js +49 -50
  219. package/dist/adapter/z-stack/structs/entries/nib.js.map +1 -1
  220. package/dist/adapter/z-stack/structs/entries/nwk-key-descriptor.d.ts +1 -1
  221. package/dist/adapter/z-stack/structs/entries/nwk-key-descriptor.d.ts.map +1 -1
  222. package/dist/adapter/z-stack/structs/entries/nwk-key-descriptor.js +1 -2
  223. package/dist/adapter/z-stack/structs/entries/nwk-key-descriptor.js.map +1 -1
  224. package/dist/adapter/z-stack/structs/entries/nwk-key.d.ts +1 -1
  225. package/dist/adapter/z-stack/structs/entries/nwk-key.d.ts.map +1 -1
  226. package/dist/adapter/z-stack/structs/entries/nwk-key.js +1 -2
  227. package/dist/adapter/z-stack/structs/entries/nwk-key.js.map +1 -1
  228. package/dist/adapter/z-stack/structs/entries/nwk-pan-id.d.ts +1 -1
  229. package/dist/adapter/z-stack/structs/entries/nwk-pan-id.d.ts.map +1 -1
  230. package/dist/adapter/z-stack/structs/entries/nwk-pan-id.js +1 -2
  231. package/dist/adapter/z-stack/structs/entries/nwk-pan-id.js.map +1 -1
  232. package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-entry.d.ts +1 -1
  233. package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-entry.d.ts.map +1 -1
  234. package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-entry.js +3 -4
  235. package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-entry.js.map +1 -1
  236. package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-table.d.ts +2 -2
  237. package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-table.d.ts.map +1 -1
  238. package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-table.js +3 -4
  239. package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-table.js.map +1 -1
  240. package/dist/adapter/z-stack/structs/entries/security-manager-entry.d.ts +1 -1
  241. package/dist/adapter/z-stack/structs/entries/security-manager-entry.d.ts.map +1 -1
  242. package/dist/adapter/z-stack/structs/entries/security-manager-entry.js +4 -5
  243. package/dist/adapter/z-stack/structs/entries/security-manager-entry.js.map +1 -1
  244. package/dist/adapter/z-stack/structs/entries/security-manager-table.d.ts +2 -2
  245. package/dist/adapter/z-stack/structs/entries/security-manager-table.d.ts.map +1 -1
  246. package/dist/adapter/z-stack/structs/entries/security-manager-table.js +3 -4
  247. package/dist/adapter/z-stack/structs/entries/security-manager-table.js.map +1 -1
  248. package/dist/adapter/z-stack/structs/index.d.ts +4 -4
  249. package/dist/adapter/z-stack/structs/serializable-memory-object.d.ts +2 -2
  250. package/dist/adapter/z-stack/structs/serializable-memory-object.d.ts.map +1 -1
  251. package/dist/adapter/z-stack/structs/struct.d.ts +8 -8
  252. package/dist/adapter/z-stack/structs/struct.d.ts.map +1 -1
  253. package/dist/adapter/z-stack/structs/struct.js +42 -43
  254. package/dist/adapter/z-stack/structs/struct.js.map +1 -1
  255. package/dist/adapter/z-stack/structs/table.d.ts +3 -3
  256. package/dist/adapter/z-stack/structs/table.d.ts.map +1 -1
  257. package/dist/adapter/z-stack/structs/table.js +10 -10
  258. package/dist/adapter/z-stack/structs/table.js.map +1 -1
  259. package/dist/adapter/z-stack/unpi/frame.d.ts +1 -1
  260. package/dist/adapter/z-stack/unpi/frame.d.ts.map +1 -1
  261. package/dist/adapter/z-stack/unpi/frame.js +2 -4
  262. package/dist/adapter/z-stack/unpi/frame.js.map +1 -1
  263. package/dist/adapter/z-stack/unpi/index.d.ts +4 -4
  264. package/dist/adapter/z-stack/unpi/parser.d.ts +1 -1
  265. package/dist/adapter/z-stack/unpi/parser.js +3 -3
  266. package/dist/adapter/z-stack/unpi/parser.js.map +1 -1
  267. package/dist/adapter/z-stack/unpi/writer.d.ts +2 -2
  268. package/dist/adapter/z-stack/unpi/writer.d.ts.map +1 -1
  269. package/dist/adapter/z-stack/unpi/writer.js +1 -1
  270. package/dist/adapter/z-stack/utils/index.d.ts +2 -2
  271. package/dist/adapter/z-stack/utils/network-options.d.ts +1 -1
  272. package/dist/adapter/z-stack/utils/network-options.d.ts.map +1 -1
  273. package/dist/adapter/z-stack/znp/buffaloZnp.d.ts +3 -3
  274. package/dist/adapter/z-stack/znp/buffaloZnp.d.ts.map +1 -1
  275. package/dist/adapter/z-stack/znp/buffaloZnp.js +34 -20
  276. package/dist/adapter/z-stack/znp/buffaloZnp.js.map +1 -1
  277. package/dist/adapter/z-stack/znp/definition.d.ts +2 -2
  278. package/dist/adapter/z-stack/znp/definition.d.ts.map +1 -1
  279. package/dist/adapter/z-stack/znp/definition.js +1111 -1111
  280. package/dist/adapter/z-stack/znp/index.d.ts +2 -2
  281. package/dist/adapter/z-stack/znp/tstype.d.ts +7 -7
  282. package/dist/adapter/z-stack/znp/tstype.d.ts.map +1 -1
  283. package/dist/adapter/z-stack/znp/utils.d.ts +1 -1
  284. package/dist/adapter/z-stack/znp/utils.d.ts.map +1 -1
  285. package/dist/adapter/z-stack/znp/utils.js +2 -2
  286. package/dist/adapter/z-stack/znp/znp.d.ts +7 -7
  287. package/dist/adapter/z-stack/znp/znp.d.ts.map +1 -1
  288. package/dist/adapter/z-stack/znp/znp.js +46 -45
  289. package/dist/adapter/z-stack/znp/znp.js.map +1 -1
  290. package/dist/adapter/z-stack/znp/zpiObject.d.ts +8 -8
  291. package/dist/adapter/z-stack/znp/zpiObject.d.ts.map +1 -1
  292. package/dist/adapter/z-stack/znp/zpiObject.js +6 -6
  293. package/dist/adapter/z-stack/znp/zpiObject.js.map +1 -1
  294. package/dist/adapter/zboss/adapter/zbossAdapter.d.ts +11 -11
  295. package/dist/adapter/zboss/adapter/zbossAdapter.d.ts.map +1 -1
  296. package/dist/adapter/zboss/adapter/zbossAdapter.js +37 -36
  297. package/dist/adapter/zboss/adapter/zbossAdapter.js.map +1 -1
  298. package/dist/adapter/zboss/commands.d.ts +4 -6
  299. package/dist/adapter/zboss/commands.d.ts.map +1 -1
  300. package/dist/adapter/zboss/commands.js +321 -287
  301. package/dist/adapter/zboss/commands.js.map +1 -1
  302. package/dist/adapter/zboss/driver.d.ts +9 -9
  303. package/dist/adapter/zboss/driver.d.ts.map +1 -1
  304. package/dist/adapter/zboss/driver.js +25 -25
  305. package/dist/adapter/zboss/driver.js.map +1 -1
  306. package/dist/adapter/zboss/frame.d.ts +9 -9
  307. package/dist/adapter/zboss/frame.d.ts.map +1 -1
  308. package/dist/adapter/zboss/frame.js +15 -15
  309. package/dist/adapter/zboss/frame.js.map +1 -1
  310. package/dist/adapter/zboss/reader.d.ts +2 -2
  311. package/dist/adapter/zboss/reader.d.ts.map +1 -1
  312. package/dist/adapter/zboss/reader.js +6 -5
  313. package/dist/adapter/zboss/reader.js.map +1 -1
  314. package/dist/adapter/zboss/uart.d.ts +3 -3
  315. package/dist/adapter/zboss/uart.d.ts.map +1 -1
  316. package/dist/adapter/zboss/uart.js +36 -36
  317. package/dist/adapter/zboss/uart.js.map +1 -1
  318. package/dist/adapter/zboss/writer.d.ts +1 -1
  319. package/dist/adapter/zboss/writer.d.ts.map +1 -1
  320. package/dist/adapter/zboss/writer.js +3 -5
  321. package/dist/adapter/zboss/writer.js.map +1 -1
  322. package/dist/adapter/zigate/adapter/patchZdoBuffaloBE.js +2 -2
  323. package/dist/adapter/zigate/adapter/patchZdoBuffaloBE.js.map +1 -1
  324. package/dist/adapter/zigate/adapter/zigateAdapter.d.ts +14 -14
  325. package/dist/adapter/zigate/adapter/zigateAdapter.d.ts.map +1 -1
  326. package/dist/adapter/zigate/adapter/zigateAdapter.js +52 -53
  327. package/dist/adapter/zigate/adapter/zigateAdapter.js.map +1 -1
  328. package/dist/adapter/zigate/driver/buffaloZiGate.d.ts +6 -6
  329. package/dist/adapter/zigate/driver/buffaloZiGate.d.ts.map +1 -1
  330. package/dist/adapter/zigate/driver/buffaloZiGate.js +41 -23
  331. package/dist/adapter/zigate/driver/buffaloZiGate.js.map +1 -1
  332. package/dist/adapter/zigate/driver/commandType.d.ts +2 -2
  333. package/dist/adapter/zigate/driver/commandType.d.ts.map +1 -1
  334. package/dist/adapter/zigate/driver/commandType.js +37 -37
  335. package/dist/adapter/zigate/driver/commandType.js.map +1 -1
  336. package/dist/adapter/zigate/driver/constants.d.ts +27 -200
  337. package/dist/adapter/zigate/driver/constants.d.ts.map +1 -1
  338. package/dist/adapter/zigate/driver/constants.js +40 -236
  339. package/dist/adapter/zigate/driver/constants.js.map +1 -1
  340. package/dist/adapter/zigate/driver/frame.d.ts.map +1 -1
  341. package/dist/adapter/zigate/driver/frame.js +4 -3
  342. package/dist/adapter/zigate/driver/frame.js.map +1 -1
  343. package/dist/adapter/zigate/driver/messageType.d.ts +1 -1
  344. package/dist/adapter/zigate/driver/messageType.js +76 -76
  345. package/dist/adapter/zigate/driver/ziGateObject.d.ts +3 -3
  346. package/dist/adapter/zigate/driver/ziGateObject.d.ts.map +1 -1
  347. package/dist/adapter/zigate/driver/ziGateObject.js +7 -6
  348. package/dist/adapter/zigate/driver/ziGateObject.js.map +1 -1
  349. package/dist/adapter/zigate/driver/zigate.d.ts +8 -8
  350. package/dist/adapter/zigate/driver/zigate.d.ts.map +1 -1
  351. package/dist/adapter/zigate/driver/zigate.js +59 -54
  352. package/dist/adapter/zigate/driver/zigate.js.map +1 -1
  353. package/dist/adapter/zoh/adapter/utils.js +1 -1
  354. package/dist/adapter/zoh/adapter/zohAdapter.d.ts +13 -13
  355. package/dist/adapter/zoh/adapter/zohAdapter.d.ts.map +1 -1
  356. package/dist/adapter/zoh/adapter/zohAdapter.js +101 -67
  357. package/dist/adapter/zoh/adapter/zohAdapter.js.map +1 -1
  358. package/dist/buffalo/buffalo.d.ts +2 -2
  359. package/dist/buffalo/buffalo.js +4 -4
  360. package/dist/buffalo/buffalo.js.map +1 -1
  361. package/dist/buffalo/index.d.ts +1 -1
  362. package/dist/controller/controller.d.ts +7 -7
  363. package/dist/controller/controller.d.ts.map +1 -1
  364. package/dist/controller/controller.js +94 -98
  365. package/dist/controller/controller.js.map +1 -1
  366. package/dist/controller/database.d.ts +1 -1
  367. package/dist/controller/database.d.ts.map +1 -1
  368. package/dist/controller/database.js +9 -9
  369. package/dist/controller/database.js.map +1 -1
  370. package/dist/controller/events.d.ts +6 -6
  371. package/dist/controller/events.d.ts.map +1 -1
  372. package/dist/controller/greenPower.d.ts +21 -21
  373. package/dist/controller/greenPower.d.ts.map +1 -1
  374. package/dist/controller/greenPower.js +41 -41
  375. package/dist/controller/greenPower.js.map +1 -1
  376. package/dist/controller/helpers/index.d.ts +1 -1
  377. package/dist/controller/helpers/request.d.ts +2 -2
  378. package/dist/controller/helpers/request.d.ts.map +1 -1
  379. package/dist/controller/helpers/request.js +31 -27
  380. package/dist/controller/helpers/request.js.map +1 -1
  381. package/dist/controller/helpers/requestQueue.d.ts +3 -3
  382. package/dist/controller/helpers/requestQueue.d.ts.map +1 -1
  383. package/dist/controller/helpers/requestQueue.js +20 -20
  384. package/dist/controller/helpers/requestQueue.js.map +1 -1
  385. package/dist/controller/helpers/zclFrameConverter.d.ts +2 -2
  386. package/dist/controller/helpers/zclFrameConverter.d.ts.map +1 -1
  387. package/dist/controller/helpers/zclFrameConverter.js +1 -1
  388. package/dist/controller/helpers/zclFrameConverter.js.map +1 -1
  389. package/dist/controller/helpers/zclTransactionSequenceNumber.d.ts +6 -4
  390. package/dist/controller/helpers/zclTransactionSequenceNumber.d.ts.map +1 -1
  391. package/dist/controller/helpers/zclTransactionSequenceNumber.js +6 -5
  392. package/dist/controller/helpers/zclTransactionSequenceNumber.js.map +1 -1
  393. package/dist/controller/index.d.ts +1 -1
  394. package/dist/controller/model/device.d.ts +23 -19
  395. package/dist/controller/model/device.d.ts.map +1 -1
  396. package/dist/controller/model/device.js +153 -128
  397. package/dist/controller/model/device.js.map +1 -1
  398. package/dist/controller/model/endpoint.d.ts +11 -11
  399. package/dist/controller/model/endpoint.d.ts.map +1 -1
  400. package/dist/controller/model/endpoint.js +84 -67
  401. package/dist/controller/model/endpoint.js.map +1 -1
  402. package/dist/controller/model/entity.d.ts +3 -3
  403. package/dist/controller/model/entity.d.ts.map +1 -1
  404. package/dist/controller/model/group.d.ts +4 -4
  405. package/dist/controller/model/group.d.ts.map +1 -1
  406. package/dist/controller/model/group.js +21 -9
  407. package/dist/controller/model/group.js.map +1 -1
  408. package/dist/controller/model/index.d.ts +4 -4
  409. package/dist/controller/touchlink.d.ts +1 -1
  410. package/dist/controller/touchlink.d.ts.map +1 -1
  411. package/dist/controller/touchlink.js +11 -11
  412. package/dist/controller/tstype.d.ts +3 -3
  413. package/dist/index.d.ts +10 -10
  414. package/dist/models/backup-storage-legacy.d.ts +14 -16
  415. package/dist/models/backup-storage-legacy.d.ts.map +1 -1
  416. package/dist/models/backup-storage-unified.d.ts +1 -1
  417. package/dist/models/backup.d.ts +2 -2
  418. package/dist/models/backup.d.ts.map +1 -1
  419. package/dist/models/index.d.ts +4 -4
  420. package/dist/utils/backup.d.ts +1 -1
  421. package/dist/utils/backup.d.ts.map +1 -1
  422. package/dist/utils/backup.js +26 -26
  423. package/dist/utils/backup.js.map +1 -1
  424. package/dist/utils/index.d.ts +5 -5
  425. package/dist/utils/logger.js +3 -3
  426. package/dist/utils/patchBigIntSerialization.js +1 -1
  427. package/dist/utils/patchBigIntSerialization.js.map +1 -1
  428. package/dist/utils/queue.d.ts.map +1 -1
  429. package/dist/utils/queue.js +1 -0
  430. package/dist/utils/queue.js.map +1 -1
  431. package/dist/utils/utils.d.ts.map +1 -1
  432. package/dist/utils/utils.js +5 -4
  433. package/dist/utils/utils.js.map +1 -1
  434. package/dist/utils/waitress.d.ts +1 -1
  435. package/dist/utils/waitress.js +3 -3
  436. package/dist/zspec/consts.d.ts +1 -1
  437. package/dist/zspec/consts.d.ts.map +1 -1
  438. package/dist/zspec/consts.js +1 -1
  439. package/dist/zspec/index.d.ts +3 -3
  440. package/dist/zspec/tstypes.d.ts +1 -1
  441. package/dist/zspec/utils.d.ts +4 -4
  442. package/dist/zspec/utils.d.ts.map +1 -1
  443. package/dist/zspec/utils.js +9 -13
  444. package/dist/zspec/utils.js.map +1 -1
  445. package/dist/zspec/zcl/buffaloZcl.d.ts +3 -3
  446. package/dist/zspec/zcl/buffaloZcl.d.ts.map +1 -1
  447. package/dist/zspec/zcl/buffaloZcl.js +127 -84
  448. package/dist/zspec/zcl/buffaloZcl.js.map +1 -1
  449. package/dist/zspec/zcl/definition/cluster.d.ts +1 -1
  450. package/dist/zspec/zcl/definition/cluster.d.ts.map +1 -1
  451. package/dist/zspec/zcl/definition/cluster.js +894 -893
  452. package/dist/zspec/zcl/definition/cluster.js.map +1 -1
  453. package/dist/zspec/zcl/definition/consts.js +7 -7
  454. package/dist/zspec/zcl/definition/foundation.d.ts +3 -3
  455. package/dist/zspec/zcl/definition/foundation.d.ts.map +1 -1
  456. package/dist/zspec/zcl/definition/foundation.js +88 -89
  457. package/dist/zspec/zcl/definition/foundation.js.map +1 -1
  458. package/dist/zspec/zcl/definition/manufacturerCode.d.ts +1 -1
  459. package/dist/zspec/zcl/definition/manufacturerCode.js +1 -1
  460. package/dist/zspec/zcl/definition/manufacturerCode.js.map +1 -1
  461. package/dist/zspec/zcl/definition/status.d.ts +3 -3
  462. package/dist/zspec/zcl/definition/status.js +3 -3
  463. package/dist/zspec/zcl/definition/tstype.d.ts +3 -3
  464. package/dist/zspec/zcl/definition/tstype.d.ts.map +1 -1
  465. package/dist/zspec/zcl/index.d.ts +10 -10
  466. package/dist/zspec/zcl/utils.d.ts +3 -3
  467. package/dist/zspec/zcl/utils.d.ts.map +1 -1
  468. package/dist/zspec/zcl/utils.js +18 -21
  469. package/dist/zspec/zcl/utils.js.map +1 -1
  470. package/dist/zspec/zcl/zclFrame.d.ts +7 -7
  471. package/dist/zspec/zcl/zclFrame.d.ts.map +1 -1
  472. package/dist/zspec/zcl/zclFrame.js +38 -41
  473. package/dist/zspec/zcl/zclFrame.js.map +1 -1
  474. package/dist/zspec/zcl/zclHeader.d.ts +2 -2
  475. package/dist/zspec/zcl/zclHeader.d.ts.map +1 -1
  476. package/dist/zspec/zcl/zclHeader.js +3 -3
  477. package/dist/zspec/zcl/zclStatusError.d.ts +1 -1
  478. package/dist/zspec/zdo/buffaloZdo.d.ts +7 -7
  479. package/dist/zspec/zdo/buffaloZdo.d.ts.map +1 -1
  480. package/dist/zspec/zdo/buffaloZdo.js +4 -4
  481. package/dist/zspec/zdo/buffaloZdo.js.map +1 -1
  482. package/dist/zspec/zdo/definition/status.d.ts +1 -1
  483. package/dist/zspec/zdo/definition/status.js +1 -1
  484. package/dist/zspec/zdo/definition/tstypes.d.ts +48 -48
  485. package/dist/zspec/zdo/definition/tstypes.d.ts.map +1 -1
  486. package/dist/zspec/zdo/index.d.ts +7 -7
  487. package/dist/zspec/zdo/utils.d.ts +2 -2
  488. package/dist/zspec/zdo/utils.d.ts.map +1 -1
  489. package/dist/zspec/zdo/utils.js +1 -1
  490. package/dist/zspec/zdo/utils.js.map +1 -1
  491. package/dist/zspec/zdo/zdoStatusError.d.ts +1 -1
  492. package/examples/join-and-log.js +5 -5
  493. package/package.json +6 -13
  494. package/src/adapter/adapter.ts +189 -0
  495. package/src/adapter/adapterDiscovery.ts +653 -0
  496. package/src/adapter/const.ts +12 -0
  497. package/src/adapter/deconz/adapter/deconzAdapter.ts +768 -0
  498. package/src/adapter/deconz/driver/constants.ts +180 -0
  499. package/src/adapter/deconz/driver/driver.ts +900 -0
  500. package/src/adapter/deconz/driver/frame.ts +11 -0
  501. package/src/adapter/deconz/driver/frameParser.ts +557 -0
  502. package/src/adapter/deconz/driver/parser.ts +45 -0
  503. package/src/adapter/deconz/driver/writer.ts +22 -0
  504. package/src/adapter/deconz/types.d.ts +13 -0
  505. package/src/adapter/ember/adapter/emberAdapter.ts +2266 -0
  506. package/src/adapter/ember/adapter/endpoints.ts +86 -0
  507. package/src/adapter/ember/adapter/oneWaitress.ts +322 -0
  508. package/src/adapter/ember/adapter/tokensManager.ts +782 -0
  509. package/src/adapter/ember/consts.ts +178 -0
  510. package/src/adapter/ember/enums.ts +2123 -0
  511. package/src/adapter/ember/ezsp/buffalo.ts +1397 -0
  512. package/src/adapter/ember/ezsp/consts.ts +148 -0
  513. package/src/adapter/ember/ezsp/enums.ts +1087 -0
  514. package/src/adapter/ember/ezsp/ezsp.ts +8985 -0
  515. package/src/adapter/ember/ezspError.ts +10 -0
  516. package/src/adapter/ember/types.ts +866 -0
  517. package/src/adapter/ember/uart/ash.ts +1969 -0
  518. package/src/adapter/ember/uart/consts.ts +109 -0
  519. package/src/adapter/ember/uart/enums.ts +192 -0
  520. package/src/adapter/ember/uart/parser.ts +48 -0
  521. package/src/adapter/ember/uart/queues.ts +247 -0
  522. package/src/adapter/ember/uart/writer.ts +53 -0
  523. package/src/adapter/ember/utils/initters.ts +58 -0
  524. package/src/adapter/ember/utils/math.ts +73 -0
  525. package/src/adapter/events.ts +21 -0
  526. package/src/adapter/ezsp/adapter/backup.ts +109 -0
  527. package/src/adapter/ezsp/adapter/ezspAdapter.ts +614 -0
  528. package/src/adapter/ezsp/driver/commands.ts +2497 -0
  529. package/src/adapter/ezsp/driver/consts.ts +11 -0
  530. package/src/adapter/ezsp/driver/driver.ts +1002 -0
  531. package/src/adapter/ezsp/driver/ezsp.ts +802 -0
  532. package/src/adapter/ezsp/driver/frame.ts +101 -0
  533. package/src/adapter/ezsp/driver/index.ts +4 -0
  534. package/src/adapter/ezsp/driver/multicast.ts +78 -0
  535. package/src/adapter/ezsp/driver/parser.ts +81 -0
  536. package/src/adapter/ezsp/driver/types/basic.ts +201 -0
  537. package/src/adapter/ezsp/driver/types/index.ts +239 -0
  538. package/src/adapter/ezsp/driver/types/named.ts +2330 -0
  539. package/src/adapter/ezsp/driver/types/struct.ts +844 -0
  540. package/src/adapter/ezsp/driver/uart.ts +460 -0
  541. package/src/adapter/ezsp/driver/utils/crc16ccitt.ts +44 -0
  542. package/src/adapter/ezsp/driver/utils/index.ts +32 -0
  543. package/src/adapter/ezsp/driver/writer.ts +64 -0
  544. package/src/adapter/index.ts +3 -0
  545. package/src/adapter/serialPort.ts +58 -0
  546. package/src/adapter/socketPortUtils.ts +16 -0
  547. package/src/adapter/tstype.ts +78 -0
  548. package/src/adapter/z-stack/adapter/adapter-backup.ts +519 -0
  549. package/src/adapter/z-stack/adapter/adapter-nv-memory.ts +457 -0
  550. package/src/adapter/z-stack/adapter/endpoints.ts +57 -0
  551. package/src/adapter/z-stack/adapter/manager.ts +543 -0
  552. package/src/adapter/z-stack/adapter/tstype.ts +6 -0
  553. package/src/adapter/z-stack/adapter/zStackAdapter.ts +1190 -0
  554. package/src/adapter/z-stack/constants/af.ts +27 -0
  555. package/src/adapter/z-stack/constants/common.ts +285 -0
  556. package/src/adapter/z-stack/constants/dbg.ts +23 -0
  557. package/src/adapter/z-stack/constants/index.ts +11 -0
  558. package/src/adapter/z-stack/constants/mac.ts +128 -0
  559. package/src/adapter/z-stack/constants/sapi.ts +25 -0
  560. package/src/adapter/z-stack/constants/sys.ts +72 -0
  561. package/src/adapter/z-stack/constants/util.ts +82 -0
  562. package/src/adapter/z-stack/constants/utils.ts +14 -0
  563. package/src/adapter/z-stack/constants/zdo.ts +103 -0
  564. package/src/adapter/z-stack/models/startup-options.ts +13 -0
  565. package/src/adapter/z-stack/structs/entries/address-manager-entry.ts +44 -0
  566. package/src/adapter/z-stack/structs/entries/address-manager-table.ts +19 -0
  567. package/src/adapter/z-stack/structs/entries/aps-link-key-data-entry.ts +12 -0
  568. package/src/adapter/z-stack/structs/entries/aps-link-key-data-table.ts +21 -0
  569. package/src/adapter/z-stack/structs/entries/aps-tc-link-key-entry.ts +19 -0
  570. package/src/adapter/z-stack/structs/entries/aps-tc-link-key-table.ts +21 -0
  571. package/src/adapter/z-stack/structs/entries/channel-list.ts +8 -0
  572. package/src/adapter/z-stack/structs/entries/has-configured.ts +16 -0
  573. package/src/adapter/z-stack/structs/entries/index.ts +16 -0
  574. package/src/adapter/z-stack/structs/entries/nib.ts +66 -0
  575. package/src/adapter/z-stack/structs/entries/nwk-key-descriptor.ts +15 -0
  576. package/src/adapter/z-stack/structs/entries/nwk-key.ts +13 -0
  577. package/src/adapter/z-stack/structs/entries/nwk-pan-id.ts +8 -0
  578. package/src/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-entry.ts +20 -0
  579. package/src/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-table.ts +19 -0
  580. package/src/adapter/z-stack/structs/entries/security-manager-entry.ts +33 -0
  581. package/src/adapter/z-stack/structs/entries/security-manager-table.ts +22 -0
  582. package/src/adapter/z-stack/structs/index.ts +4 -0
  583. package/src/adapter/z-stack/structs/serializable-memory-object.ts +14 -0
  584. package/src/adapter/z-stack/structs/struct.ts +367 -0
  585. package/src/adapter/z-stack/structs/table.ts +198 -0
  586. package/src/adapter/z-stack/unpi/constants.ts +33 -0
  587. package/src/adapter/z-stack/unpi/frame.ts +62 -0
  588. package/src/adapter/z-stack/unpi/index.ts +4 -0
  589. package/src/adapter/z-stack/unpi/parser.ts +56 -0
  590. package/src/adapter/z-stack/unpi/writer.ts +21 -0
  591. package/src/adapter/z-stack/utils/channel-list.ts +40 -0
  592. package/src/adapter/z-stack/utils/index.ts +2 -0
  593. package/src/adapter/z-stack/utils/network-options.ts +26 -0
  594. package/src/adapter/z-stack/znp/buffaloZnp.ts +175 -0
  595. package/src/adapter/z-stack/znp/definition.ts +2713 -0
  596. package/src/adapter/z-stack/znp/index.ts +2 -0
  597. package/src/adapter/z-stack/znp/parameterType.ts +22 -0
  598. package/src/adapter/z-stack/znp/tstype.ts +44 -0
  599. package/src/adapter/z-stack/znp/utils.ts +10 -0
  600. package/src/adapter/z-stack/znp/znp.ts +342 -0
  601. package/src/adapter/z-stack/znp/zpiObject.ts +148 -0
  602. package/src/adapter/zboss/adapter/zbossAdapter.ts +527 -0
  603. package/src/adapter/zboss/commands.ts +1184 -0
  604. package/src/adapter/zboss/consts.ts +9 -0
  605. package/src/adapter/zboss/driver.ts +423 -0
  606. package/src/adapter/zboss/enums.ts +360 -0
  607. package/src/adapter/zboss/frame.ts +227 -0
  608. package/src/adapter/zboss/reader.ts +65 -0
  609. package/src/adapter/zboss/types.ts +0 -0
  610. package/src/adapter/zboss/uart.ts +428 -0
  611. package/src/adapter/zboss/utils.ts +58 -0
  612. package/src/adapter/zboss/writer.ts +49 -0
  613. package/src/adapter/zigate/adapter/patchZdoBuffaloBE.ts +27 -0
  614. package/src/adapter/zigate/adapter/zigateAdapter.ts +618 -0
  615. package/src/adapter/zigate/driver/LICENSE +17 -0
  616. package/src/adapter/zigate/driver/buffaloZiGate.ts +212 -0
  617. package/src/adapter/zigate/driver/commandType.ts +418 -0
  618. package/src/adapter/zigate/driver/constants.ts +150 -0
  619. package/src/adapter/zigate/driver/frame.ts +197 -0
  620. package/src/adapter/zigate/driver/messageType.ts +287 -0
  621. package/src/adapter/zigate/driver/parameterType.ts +32 -0
  622. package/src/adapter/zigate/driver/ziGateObject.ts +146 -0
  623. package/src/adapter/zigate/driver/zigate.ts +426 -0
  624. package/src/adapter/zoh/adapter/utils.ts +27 -0
  625. package/src/adapter/zoh/adapter/zohAdapter.ts +837 -0
  626. package/src/buffalo/buffalo.ts +335 -0
  627. package/src/buffalo/index.ts +1 -0
  628. package/src/controller/controller.ts +1048 -0
  629. package/src/controller/database.ts +124 -0
  630. package/src/controller/events.ts +51 -0
  631. package/src/controller/greenPower.ts +603 -0
  632. package/src/controller/helpers/index.ts +1 -0
  633. package/src/controller/helpers/request.ts +94 -0
  634. package/src/controller/helpers/requestQueue.ts +125 -0
  635. package/src/controller/helpers/zclFrameConverter.ts +51 -0
  636. package/src/controller/helpers/zclTransactionSequenceNumber.ts +19 -0
  637. package/src/controller/index.ts +6 -0
  638. package/src/controller/model/device.ts +1320 -0
  639. package/src/controller/model/endpoint.ts +1040 -0
  640. package/src/controller/model/entity.ts +23 -0
  641. package/src/controller/model/group.ts +333 -0
  642. package/src/controller/model/index.ts +4 -0
  643. package/src/controller/touchlink.ts +189 -0
  644. package/src/controller/tstype.ts +39 -0
  645. package/src/index.ts +12 -0
  646. package/src/models/backup-storage-legacy.ts +48 -0
  647. package/src/models/backup-storage-unified.ts +47 -0
  648. package/src/models/backup.ts +37 -0
  649. package/src/models/index.ts +5 -0
  650. package/src/models/network-options.ts +11 -0
  651. package/src/utils/backup.ts +152 -0
  652. package/src/utils/index.ts +5 -0
  653. package/src/utils/logger.ts +20 -0
  654. package/src/utils/patchBigIntSerialization.ts +8 -0
  655. package/src/utils/queue.ts +76 -0
  656. package/src/utils/types.d.ts +3 -0
  657. package/src/utils/utils.ts +20 -0
  658. package/src/utils/wait.ts +5 -0
  659. package/src/utils/waitress.ts +95 -0
  660. package/src/zspec/consts.ts +84 -0
  661. package/src/zspec/enums.ts +22 -0
  662. package/src/zspec/index.ts +3 -0
  663. package/src/zspec/tstypes.ts +18 -0
  664. package/src/zspec/utils.ts +308 -0
  665. package/src/zspec/zcl/buffaloZcl.ts +1133 -0
  666. package/src/zspec/zcl/definition/cluster.ts +5398 -0
  667. package/src/zspec/zcl/definition/consts.ts +24 -0
  668. package/src/zspec/zcl/definition/enums.ts +194 -0
  669. package/src/zspec/zcl/definition/foundation.ts +301 -0
  670. package/src/zspec/zcl/definition/manufacturerCode.ts +729 -0
  671. package/src/zspec/zcl/definition/status.ts +69 -0
  672. package/src/zspec/zcl/definition/tstype.ts +236 -0
  673. package/src/zspec/zcl/index.ts +10 -0
  674. package/src/zspec/zcl/utils.ts +342 -0
  675. package/src/zspec/zcl/zclFrame.ts +336 -0
  676. package/src/zspec/zcl/zclHeader.ts +102 -0
  677. package/src/zspec/zcl/zclStatusError.ts +10 -0
  678. package/src/zspec/zdo/buffaloZdo.ts +2345 -0
  679. package/src/zspec/zdo/definition/clusters.ts +722 -0
  680. package/src/zspec/zdo/definition/consts.ts +16 -0
  681. package/src/zspec/zdo/definition/enums.ts +99 -0
  682. package/src/zspec/zdo/definition/status.ts +105 -0
  683. package/src/zspec/zdo/definition/tstypes.ts +1062 -0
  684. package/src/zspec/zdo/index.ts +7 -0
  685. package/src/zspec/zdo/utils.ts +76 -0
  686. package/src/zspec/zdo/zdoStatusError.ts +10 -0
  687. package/test/adapter/adapter.test.ts +1062 -0
  688. package/test/adapter/ember/ash.test.ts +337 -0
  689. package/test/adapter/ember/consts.ts +131 -0
  690. package/test/adapter/ember/emberAdapter.test.ts +3449 -0
  691. package/test/adapter/ember/ezsp.test.ts +386 -0
  692. package/test/adapter/ember/ezspBuffalo.test.ts +92 -0
  693. package/test/adapter/ember/ezspError.test.ts +11 -0
  694. package/test/adapter/ember/math.test.ts +205 -0
  695. package/test/adapter/ezsp/frame.test.ts +29 -0
  696. package/test/adapter/ezsp/uart.test.ts +180 -0
  697. package/test/adapter/z-stack/adapter.test.ts +3985 -0
  698. package/test/adapter/z-stack/constants.test.ts +32 -0
  699. package/test/adapter/z-stack/structs.test.ts +114 -0
  700. package/test/adapter/z-stack/unpi.test.ts +212 -0
  701. package/test/adapter/z-stack/znp.test.ts +1284 -0
  702. package/test/adapter/zboss/fixZdoResponse.test.ts +178 -0
  703. package/test/adapter/zigate/patchZdoBuffaloBE.test.ts +80 -0
  704. package/test/adapter/zigate/zdo.test.ts +187 -0
  705. package/test/adapter/zoh/utils.test.ts +35 -0
  706. package/test/adapter/zoh/zohAdapter.test.ts +1306 -0
  707. package/test/buffalo.test.ts +430 -0
  708. package/test/controller.test.ts +10005 -0
  709. package/test/greenpower.test.ts +1421 -0
  710. package/test/mockAdapters.ts +65 -0
  711. package/test/mockDevices.ts +472 -0
  712. package/test/testUtils.ts +20 -0
  713. package/test/tsconfig.json +10 -0
  714. package/test/utils/math.ts +19 -0
  715. package/test/utils.test.ts +227 -0
  716. package/test/vitest.config.mts +25 -0
  717. package/test/zcl.test.ts +2832 -0
  718. package/test/zspec/utils.test.ts +131 -0
  719. package/test/zspec/zcl/buffalo.test.ts +1231 -0
  720. package/test/zspec/zcl/frame.test.ts +925 -0
  721. package/test/zspec/zcl/utils.test.ts +280 -0
  722. package/test/zspec/zdo/buffalo.test.ts +1849 -0
  723. package/test/zspec/zdo/utils.test.ts +240 -0
  724. package/.prettierignore +0 -2
  725. package/.prettierrc +0 -26
  726. package/eslint.config.mjs +0 -32
@@ -0,0 +1,1421 @@
1
+ import type {ZclPayload} from "../src/adapter/events";
2
+
3
+ import type {MockInstance} from "vitest";
4
+
5
+ import {GreenPower} from "../src/controller/greenPower";
6
+ import type {GreenPowerDeviceJoinedPayload} from "../src/controller/tstype";
7
+ import {logger} from "../src/utils/logger";
8
+ import {GP_ENDPOINT, GP_GROUP_ID} from "../src/zspec/consts";
9
+ import * as Zcl from "../src/zspec/zcl";
10
+
11
+ describe("GreenPower", () => {
12
+ let gp: GreenPower;
13
+ let logDebugSpy: MockInstance;
14
+ let logInfoSpy: MockInstance;
15
+ let logWarningSpy: MockInstance;
16
+ let logErrorSpy: MockInstance;
17
+
18
+ const clearLogMocks = (): void => {
19
+ logDebugSpy.mockClear();
20
+ logInfoSpy.mockClear();
21
+ logWarningSpy.mockClear();
22
+ logErrorSpy.mockClear();
23
+ };
24
+
25
+ const makeNotificationOptions = (
26
+ applicationId: number,
27
+ gpdfSecurityLevel: number,
28
+ gpdfSecurityKeyType: number,
29
+ bidirectionalInfo: number,
30
+ ): number => {
31
+ return (applicationId & 0x7) | ((gpdfSecurityLevel & 0x3) << 6) | ((gpdfSecurityKeyType & 0x7) << 8) | ((bidirectionalInfo & 0x3) << 11);
32
+ };
33
+
34
+ const makeHeader = (
35
+ sequenceNumber: number,
36
+ commandIdentifier: number,
37
+ applicationId: number,
38
+ gpdfSecurityLevel: number,
39
+ gpdfSecurityKeyType: number,
40
+ bidirectionalInfo: number,
41
+ sourceId: number,
42
+ gpdSecurityFrameCounter: number,
43
+ gpdCommandId: number,
44
+ payloadLength: number,
45
+ options?: number,
46
+ ): Buffer => {
47
+ const gpdHeader = Buffer.alloc(15);
48
+ gpdHeader.writeUInt8(0b00000001, 0); // frameControl: FrameType.SPECIFIC + Direction.CLIENT_TO_SERVER + disableDefaultResponse=false
49
+ gpdHeader.writeUInt8(sequenceNumber, 1);
50
+ gpdHeader.writeUInt8(commandIdentifier, 2);
51
+ gpdHeader.writeUInt16LE(options ?? makeNotificationOptions(applicationId, gpdfSecurityLevel, gpdfSecurityKeyType, bidirectionalInfo), 3);
52
+ gpdHeader.writeUInt32LE(sourceId, 5);
53
+ gpdHeader.writeUInt32LE(gpdSecurityFrameCounter, 9);
54
+ gpdHeader.writeUInt8(gpdCommandId, 13);
55
+ gpdHeader.writeUInt8(payloadLength, 14);
56
+
57
+ return gpdHeader;
58
+ };
59
+
60
+ const makeFooter = (options: number, gppNwkAddr?: number, gppGpdLink?: number, mic?: number): Buffer => {
61
+ const hasGppData = options & 0x800;
62
+ const hasMic = options & 0x200;
63
+ const gpdFooter = Buffer.alloc((hasGppData ? 3 : 0) + (hasMic ? 4 : 0));
64
+
65
+ if (hasGppData) {
66
+ gpdFooter.writeUInt16LE(gppNwkAddr!, 0);
67
+ gpdFooter.writeUInt8(gppGpdLink!, 2);
68
+ }
69
+
70
+ if (hasMic) {
71
+ gpdFooter.writeUInt32LE(mic!, hasGppData ? 3 : 0);
72
+ }
73
+
74
+ return gpdFooter;
75
+ };
76
+
77
+ const makePayload = (sourceId: number, buffer: Buffer, linkQuality: number): ZclPayload => {
78
+ return {
79
+ clusterID: Zcl.Clusters.greenPower.ID,
80
+ header: Zcl.Header.fromBuffer(buffer),
81
+ address: sourceId & 0xffff,
82
+ data: buffer,
83
+ endpoint: GP_ENDPOINT,
84
+ linkquality: linkQuality,
85
+ groupID: GP_GROUP_ID,
86
+ wasBroadcast: true,
87
+ destinationEndpoint: GP_ENDPOINT,
88
+ };
89
+ };
90
+
91
+ beforeAll(() => {
92
+ vi.useFakeTimers();
93
+
94
+ logDebugSpy = vi.spyOn(logger, "debug");
95
+ logInfoSpy = vi.spyOn(logger, "info");
96
+ logWarningSpy = vi.spyOn(logger, "warning");
97
+ logErrorSpy = vi.spyOn(logger, "error");
98
+ });
99
+
100
+ beforeEach(() => {
101
+ clearLogMocks();
102
+
103
+ gp = new GreenPower(
104
+ // @ts-expect-error minimal mock
105
+ {
106
+ getCoordinatorIEEE: vi.fn(),
107
+ sendZclFrameToAll: vi.fn(),
108
+ sendZclFrameToEndpoint: vi.fn(),
109
+ getNetworkParameters: vi.fn(),
110
+ },
111
+ );
112
+ });
113
+
114
+ afterAll(() => {
115
+ vi.useRealTimers();
116
+ });
117
+
118
+ it("encodes & decodes pairing options", () => {
119
+ let rawByte = 0b000000000110101000;
120
+ let rawOptions = {
121
+ appId: 0,
122
+ addSink: true,
123
+ removeGpd: false,
124
+ communicationMode: 0b01,
125
+ gpdFixed: true,
126
+ gpdMacSeqNumCapabilities: true,
127
+ securityLevel: 0,
128
+ securityKeyType: 0,
129
+ gpdSecurityFrameCounterPresent: false,
130
+ gpdSecurityKeyPresent: false,
131
+ assignedAliasPresent: false,
132
+ groupcastRadiusPresent: false,
133
+ };
134
+ let options = GreenPower.decodePairingOptions(rawByte);
135
+ let byte = GreenPower.encodePairingOptions(rawOptions);
136
+
137
+ expect(options).toStrictEqual(rawOptions);
138
+ expect(rawByte).toStrictEqual(byte);
139
+
140
+ rawByte = 0b001110010101001000;
141
+ rawOptions = {
142
+ appId: 0,
143
+ addSink: true,
144
+ removeGpd: false,
145
+ communicationMode: 0b10,
146
+ gpdFixed: false,
147
+ gpdMacSeqNumCapabilities: true,
148
+ securityLevel: 0b10,
149
+ securityKeyType: 0b100,
150
+ gpdSecurityFrameCounterPresent: true,
151
+ gpdSecurityKeyPresent: true,
152
+ assignedAliasPresent: false,
153
+ groupcastRadiusPresent: false,
154
+ };
155
+ options = GreenPower.decodePairingOptions(rawByte);
156
+ byte = GreenPower.encodePairingOptions(rawOptions);
157
+
158
+ expect(options).toStrictEqual(rawOptions);
159
+ expect(rawByte).toStrictEqual(byte);
160
+
161
+ rawByte = 0b001110010101101000;
162
+ rawOptions = {
163
+ appId: 0,
164
+ addSink: true,
165
+ removeGpd: false,
166
+ communicationMode: 0b11,
167
+ gpdFixed: false,
168
+ gpdMacSeqNumCapabilities: true,
169
+ securityLevel: 0b10,
170
+ securityKeyType: 0b100,
171
+ gpdSecurityFrameCounterPresent: true,
172
+ gpdSecurityKeyPresent: true,
173
+ assignedAliasPresent: false,
174
+ groupcastRadiusPresent: false,
175
+ };
176
+ options = GreenPower.decodePairingOptions(rawByte);
177
+ byte = GreenPower.encodePairingOptions(rawOptions);
178
+
179
+ expect(options).toStrictEqual(rawOptions);
180
+ expect(rawByte).toStrictEqual(byte);
181
+
182
+ rawByte = 0b000000000110110000;
183
+ rawOptions = {
184
+ appId: 0,
185
+ addSink: false,
186
+ removeGpd: true,
187
+ communicationMode: 0b01,
188
+ gpdFixed: true,
189
+ gpdMacSeqNumCapabilities: true,
190
+ securityLevel: 0b00,
191
+ securityKeyType: 0b000,
192
+ gpdSecurityFrameCounterPresent: false,
193
+ gpdSecurityKeyPresent: false,
194
+ assignedAliasPresent: false,
195
+ groupcastRadiusPresent: false,
196
+ };
197
+ options = GreenPower.decodePairingOptions(rawByte);
198
+ byte = GreenPower.encodePairingOptions(rawOptions);
199
+
200
+ expect(options).toStrictEqual(rawOptions);
201
+ expect(rawByte).toStrictEqual(byte);
202
+
203
+ // coverage
204
+ rawByte = 0b110000000010110000;
205
+ rawOptions = {
206
+ appId: 0,
207
+ addSink: false,
208
+ removeGpd: true,
209
+ communicationMode: 0b01,
210
+ gpdFixed: true,
211
+ gpdMacSeqNumCapabilities: false,
212
+ securityLevel: 0b00,
213
+ securityKeyType: 0b000,
214
+ gpdSecurityFrameCounterPresent: false,
215
+ gpdSecurityKeyPresent: false,
216
+ assignedAliasPresent: true,
217
+ groupcastRadiusPresent: true,
218
+ };
219
+ options = GreenPower.decodePairingOptions(rawByte);
220
+ byte = GreenPower.encodePairingOptions(rawOptions);
221
+
222
+ expect(options).toStrictEqual(rawOptions);
223
+ expect(rawByte).toStrictEqual(byte);
224
+ });
225
+
226
+ it("encodes & decodes commissioning mode options", () => {
227
+ let rawByte = 0x0b;
228
+ let rawOptions = {action: 1, commissioningWindowPresent: true, exitMode: 0b10, channelPresent: false, unicastCommunication: false};
229
+ let options = GreenPower.decodeCommissioningModeOptions(rawByte);
230
+ let byte = GreenPower.encodeCommissioningModeOptions(rawOptions);
231
+
232
+ expect(options).toStrictEqual(rawOptions);
233
+ expect(rawByte).toStrictEqual(byte);
234
+ rawByte = 0x2b;
235
+ rawOptions = {action: 1, commissioningWindowPresent: true, exitMode: 0b10, channelPresent: false, unicastCommunication: true};
236
+ options = GreenPower.decodeCommissioningModeOptions(rawByte);
237
+ byte = GreenPower.encodeCommissioningModeOptions(rawOptions);
238
+
239
+ expect(options).toStrictEqual(rawOptions);
240
+ expect(rawByte).toStrictEqual(byte);
241
+ rawByte = 0x0a;
242
+ rawOptions = {action: 0, commissioningWindowPresent: true, exitMode: 0b10, channelPresent: false, unicastCommunication: false};
243
+ options = GreenPower.decodeCommissioningModeOptions(rawByte);
244
+ byte = GreenPower.encodeCommissioningModeOptions(rawOptions);
245
+
246
+ expect(options).toStrictEqual(rawOptions);
247
+ expect(rawByte).toStrictEqual(byte);
248
+ expect(options).toStrictEqual(rawOptions);
249
+ expect(rawByte).toStrictEqual(byte);
250
+
251
+ // coverage
252
+ rawByte = 0b111100;
253
+ rawOptions = {action: 0, commissioningWindowPresent: false, exitMode: 0b11, channelPresent: true, unicastCommunication: true};
254
+ options = GreenPower.decodeCommissioningModeOptions(rawByte);
255
+ byte = GreenPower.encodeCommissioningModeOptions(rawOptions);
256
+
257
+ expect(options).toStrictEqual(rawOptions);
258
+ expect(rawByte).toStrictEqual(byte);
259
+ });
260
+
261
+ it("omits GPP data from raw payload", async () => {
262
+ const addr = {applicationId: 0, sourceId: 2777252112, endpoint: 0};
263
+ const options = 0x800;
264
+ const sequenceNumber = 18;
265
+ const gpdSecurityFrameCounter = 17326;
266
+ const gpdCommandId = 38;
267
+ const gpdCommandPayload = Buffer.from([0x3e]);
268
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.commissioningNotification.ID;
269
+ const gppNwkAddr = 24404;
270
+ const gppGpdLink = 207;
271
+
272
+ const gpdHeader = makeHeader(
273
+ sequenceNumber,
274
+ commandIdentifier,
275
+ 0,
276
+ 0,
277
+ 0,
278
+ 0,
279
+ addr.sourceId,
280
+ gpdSecurityFrameCounter,
281
+ gpdCommandId,
282
+ gpdCommandPayload.length,
283
+ options,
284
+ );
285
+ const gpdFooter = makeFooter(options, gppNwkAddr, gppGpdLink);
286
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload, gpdFooter]), 138);
287
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
288
+ const retFrame = await gp.processCommand(payload, frame, undefined);
289
+
290
+ expect(frame.payload.gppNwkAddr).toStrictEqual(gppNwkAddr);
291
+ expect(frame.payload.gppGpdLink).toStrictEqual(gppGpdLink);
292
+ expect(retFrame.payload.commandFrame).toStrictEqual({raw: gpdCommandPayload});
293
+ expect(retFrame.payload.gppNwkAddr).toStrictEqual(gppNwkAddr);
294
+ expect(retFrame.payload.gppGpdLink).toStrictEqual(gppGpdLink);
295
+ });
296
+
297
+ it("omits MIC from raw payload", async () => {
298
+ const securityKey = Buffer.from([227, 227, 225, 134, 235, 104, 141, 250, 162, 211, 104, 147, 201, 146, 67, 175]);
299
+ const addr = {applicationId: 0, sourceId: 2777252112, endpoint: 0};
300
+ const options = 0x30 | 0x200;
301
+ const sequenceNumber = 18;
302
+ const gpdSecurityFrameCounter = 17326;
303
+ const gpdCommandId = 38;
304
+ const gpdCommandPayload = Buffer.from([0x3e]);
305
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.commissioningNotification.ID;
306
+ const mic = 1441399364;
307
+
308
+ const gpdHeader = makeHeader(
309
+ sequenceNumber,
310
+ commandIdentifier,
311
+ 0,
312
+ 0,
313
+ 0,
314
+ 0,
315
+ addr.sourceId,
316
+ gpdSecurityFrameCounter,
317
+ gpdCommandId,
318
+ gpdCommandPayload.length,
319
+ options,
320
+ );
321
+ const gpdFooter = makeFooter(options, undefined, undefined, mic);
322
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload, gpdFooter]), 138);
323
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
324
+ const retFrame = await gp.processCommand(payload, frame, securityKey);
325
+
326
+ expect(frame.payload.mic).toBeDefined(); // garbage
327
+ expect(retFrame.payload.commandID).toStrictEqual(0x21); // just to be sure it decrypted properly
328
+ expect(retFrame.payload.commandFrame).toStrictEqual({raw: Buffer.from([207 /* decrypted, bogus data */])});
329
+ expect(retFrame.payload.mic).toStrictEqual(undefined); // removed once decrypted
330
+ });
331
+
332
+ it("omits GPP data and MIC from raw payload", async () => {
333
+ const securityKey = Buffer.from([227, 227, 225, 134, 235, 104, 141, 250, 162, 211, 104, 147, 201, 146, 67, 175]);
334
+ const addr = {applicationId: 0, sourceId: 2777252112, endpoint: 0};
335
+ const options = 0x30 | 0x200 | 0x800;
336
+ const sequenceNumber = 18;
337
+ const gpdSecurityFrameCounter = 17326;
338
+ const gpdCommandId = 38;
339
+ const gpdCommandPayload = Buffer.from([0x3e]);
340
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.commissioningNotification.ID;
341
+ const gppNwkAddr = 24404;
342
+ const gppGpdLink = 207;
343
+ const mic = 1441399364;
344
+
345
+ const gpdHeader = makeHeader(
346
+ sequenceNumber,
347
+ commandIdentifier,
348
+ 0,
349
+ 0,
350
+ 0,
351
+ 0,
352
+ addr.sourceId,
353
+ gpdSecurityFrameCounter,
354
+ gpdCommandId,
355
+ gpdCommandPayload.length,
356
+ options,
357
+ );
358
+ const gpdFooter = makeFooter(options, gppNwkAddr, gppGpdLink, mic);
359
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload, gpdFooter]), 138);
360
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
361
+ const retFrame = await gp.processCommand(payload, frame, securityKey);
362
+
363
+ expect(frame.payload.gppNwkAddr).toBeDefined(); // garbage
364
+ expect(frame.payload.gppGpdLink).toBeDefined(); // garbage
365
+ expect(frame.payload.mic).toBeDefined(); // garbage
366
+ expect(retFrame.payload.commandID).toStrictEqual(0x21); // just to be sure it decrypted properly
367
+ expect(retFrame.payload.commandFrame).toStrictEqual({raw: Buffer.from([207 /* decrypted, bogus data */])});
368
+ expect(retFrame.payload.gppNwkAddr).toStrictEqual(gppNwkAddr); // removed once decrypted
369
+ expect(retFrame.payload.gppGpdLink).toStrictEqual(gppGpdLink); // removed once decrypted
370
+ expect(retFrame.payload.mic).toStrictEqual(undefined); // removed once decrypted
371
+ });
372
+
373
+ it("does not parse command frame when FULLENCR security level - SINK", async () => {
374
+ const addr = {applicationId: 0, sourceId: 2888399791, endpoint: 0};
375
+ const securityLevelFullEncr = 3;
376
+ const securityKeyTypeNWK = 1;
377
+ const gpdLink = 207;
378
+ const sequenceNumber = 143;
379
+ const bidirectionalInfo = 0;
380
+ const gpdSecurityFrameCounter = 3727;
381
+ const gpdCommandId = 227; // this would otherwise be CHANNEL_REQUEST and result in bad parsing
382
+ const gpdCommandPayload = Buffer.from("", "hex");
383
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.notification.ID;
384
+
385
+ const gpdHeader = makeHeader(
386
+ sequenceNumber,
387
+ commandIdentifier,
388
+ addr.applicationId,
389
+ securityLevelFullEncr,
390
+ securityKeyTypeNWK,
391
+ bidirectionalInfo,
392
+ addr.sourceId,
393
+ gpdSecurityFrameCounter,
394
+ gpdCommandId,
395
+ gpdCommandPayload.length,
396
+ );
397
+
398
+ {
399
+ // mock bad frame, allows to bypass options check and validate that this errors out (trying to parse as CHANNEL_REQUEST)
400
+ const alteredSecurityGpdHeader = Buffer.from(gpdHeader);
401
+ alteredSecurityGpdHeader[3] = 0;
402
+ alteredSecurityGpdHeader[4] = 0;
403
+ const payload = makePayload(addr.sourceId, Buffer.concat([alteredSecurityGpdHeader, gpdCommandPayload]), gpdLink);
404
+
405
+ expect(() => {
406
+ Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
407
+ }).toThrow('The value of "offset" is out of range. It must be >= 0 and <= 14. Received 15');
408
+ }
409
+
410
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
411
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
412
+
413
+ expect(frame.payload.commandFrame).toBeUndefined(); // as opposed to `{}` when parsing (payloadSize=0)
414
+
415
+ const retFrame = await gp.processCommand(payload, frame, Buffer.alloc(16) /* just for the codepath, decrypting not important */);
416
+
417
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
418
+ 1,
419
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x9d srcID=2888399791 gpp=NO",
420
+ "zh:controller:greenpower",
421
+ );
422
+
423
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
424
+ clonedFrame.payload.commandID = 0x9d;
425
+ clonedFrame.payload.options = 256;
426
+ clonedFrame.payload.commandFrame = {};
427
+
428
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
429
+ });
430
+
431
+ it("does not parse command frame when FULLENCR security level - GPP", async () => {
432
+ const addr = {applicationId: 0, sourceId: 2888399791, endpoint: 0};
433
+ const gpdLink = 207;
434
+ const sequenceNumber = 143;
435
+ const gpdSecurityFrameCounter = 3727;
436
+ const gpdCommandId = 227; // this would otherwise be CHANNEL_REQUEST and result in bad parsing
437
+ const gpdCommandPayload = Buffer.from("", "hex");
438
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.commissioningNotification.ID;
439
+ const gppNwkAddr = 24404;
440
+ const gppGpdLink = 123;
441
+ const mic = 456;
442
+ const options = 2864;
443
+
444
+ const gpdHeader = makeHeader(
445
+ sequenceNumber,
446
+ commandIdentifier,
447
+ 0,
448
+ 0,
449
+ 0,
450
+ 0,
451
+ addr.sourceId,
452
+ gpdSecurityFrameCounter,
453
+ gpdCommandId,
454
+ gpdCommandPayload.length,
455
+ options,
456
+ );
457
+ const gpdFooter = makeFooter(options, gppNwkAddr, gppGpdLink, mic);
458
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload, gpdFooter]), gpdLink);
459
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
460
+
461
+ expect(frame.payload.commandFrame).toBeUndefined(); // as opposed to `{}` when parsing (payloadSize=0)
462
+
463
+ const retFrame = await gp.processCommand(payload, frame, Buffer.alloc(16) /* just for the codepath, decrypting not important */);
464
+
465
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
466
+ 1,
467
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x9d srcID=2888399791 gpp=24404 rssi=59 linkQuality=Moderate",
468
+ "zh:controller:greenpower",
469
+ );
470
+
471
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
472
+ clonedFrame.payload.commandID = 0x9d;
473
+ clonedFrame.payload.options = 2304;
474
+ clonedFrame.payload.commandFrame = {};
475
+ clonedFrame.payload.gppNwkAddr = gppNwkAddr;
476
+ clonedFrame.payload.gppGpdLink = gppGpdLink;
477
+ delete clonedFrame.payload.mic;
478
+
479
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
480
+ });
481
+
482
+ // @see https://github.com/Koenkk/zigbee2mqtt/issues/19405#issuecomment-2727338024
483
+ it("FULLENCR ZT-LP-ZEU2S-WH-MS MOES 2-gang vectors from ember", async () => {
484
+ let joinData: GreenPowerDeviceJoinedPayload | undefined;
485
+
486
+ gp.on("deviceJoined", (payload) => {
487
+ joinData = payload;
488
+ });
489
+
490
+ const addr = {applicationId: 0, sourceId: 1496140231, endpoint: 0};
491
+
492
+ {
493
+ const gpdLink = 214;
494
+ const sequenceNumber = 19;
495
+ const gpdfSecurityLevel = 0; // NONE
496
+ const gpdfSecurityKeyType = 0; // NONE
497
+ const bidirectionalInfo = 0;
498
+ const gpdSecurityFrameCounter = 4294967295;
499
+ const gpdCommandId = 224;
500
+ const gpdCommandPayload = Buffer.from("0289f31adb70a88d71196ee50c03580537767de27ad5331309000037647a62697061304047503030303157", "hex");
501
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.commissioningNotification.ID;
502
+
503
+ const gpdHeader = makeHeader(
504
+ sequenceNumber,
505
+ commandIdentifier,
506
+ addr.applicationId,
507
+ gpdfSecurityLevel,
508
+ gpdfSecurityKeyType,
509
+ bidirectionalInfo,
510
+ addr.sourceId,
511
+ gpdSecurityFrameCounter,
512
+ gpdCommandId,
513
+ gpdCommandPayload.length,
514
+ );
515
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
516
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
517
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey); // always undefined since not yet joined
518
+
519
+ await vi.waitUntil(() => joinData !== undefined);
520
+
521
+ expect(joinData).toStrictEqual({
522
+ sourceID: addr.sourceId,
523
+ deviceID: frame.payload.commandFrame.deviceID,
524
+ networkAddress: addr.sourceId & 0xffff,
525
+ securityKey: frame.payload.commandFrame.securityKey,
526
+ });
527
+ expect(logInfoSpy).toHaveBeenNthCalledWith(1, "[COMMISSIONING] srcID=1496140231 gpp=NO", "zh:controller:greenpower");
528
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
529
+ 1,
530
+ "[PAIRING] srcID=1496140231 gpp=NO options=58696 (addSink=true commMode=2)",
531
+ "zh:controller:greenpower",
532
+ );
533
+
534
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
535
+ clonedFrame.payload.commandID = 0xe0;
536
+ clonedFrame.payload.options = 0;
537
+ clonedFrame.payload.commandFrame = {
538
+ deviceID: 2,
539
+ options: 137,
540
+ extendedOptions: 243,
541
+ securityKey: joinData?.securityKey,
542
+ keyMic: 869628642,
543
+ outgoingCounter: 2323,
544
+ applicationInfo: addr.applicationId,
545
+ manufacturerID: 0,
546
+ modelID: 0,
547
+ numGpdCommands: 0,
548
+ gpdCommandIdList: Buffer.from([]),
549
+ numServerClusters: 0,
550
+ numClientClusters: 0,
551
+ gpdServerClusters: Buffer.from([]),
552
+ gpdClientClusters: Buffer.from([]),
553
+ genericSwitchConfig: 0,
554
+ currentContactStatus: 0,
555
+ };
556
+
557
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
558
+ expect(retFrame.payload.commandFrame.securityKey).toStrictEqual(joinData?.securityKey);
559
+ }
560
+
561
+ clearLogMocks();
562
+
563
+ const securityLevelFullEncr = 3;
564
+ const securityKeyTypeNWK = 1;
565
+
566
+ // left
567
+ {
568
+ const gpdLink = 220;
569
+ const sequenceNumber = 28;
570
+ const bidirectionalInfo = 0;
571
+ const gpdSecurityFrameCounter = 2332;
572
+ const gpdCommandId = 136;
573
+ const gpdCommandPayload = Buffer.from("", "hex");
574
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.notification.ID;
575
+
576
+ const gpdHeader = makeHeader(
577
+ sequenceNumber,
578
+ commandIdentifier,
579
+ addr.applicationId,
580
+ securityLevelFullEncr,
581
+ securityKeyTypeNWK,
582
+ bidirectionalInfo,
583
+ addr.sourceId,
584
+ gpdSecurityFrameCounter,
585
+ gpdCommandId,
586
+ gpdCommandPayload.length,
587
+ );
588
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
589
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
590
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
591
+
592
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
593
+ 1,
594
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x20 srcID=1496140231 gpp=NO",
595
+ "zh:controller:greenpower",
596
+ );
597
+
598
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
599
+ clonedFrame.payload.commandID = 0x20;
600
+ clonedFrame.payload.options = 256;
601
+ clonedFrame.payload.commandFrame = {};
602
+
603
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
604
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
605
+ }
606
+
607
+ clearLogMocks();
608
+
609
+ // left
610
+ {
611
+ const gpdLink = 220;
612
+ const sequenceNumber = 46;
613
+ const bidirectionalInfo = 0;
614
+ const gpdSecurityFrameCounter = 2350;
615
+ const gpdCommandId = 152;
616
+ const gpdCommandPayload = Buffer.from("", "hex");
617
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.notification.ID;
618
+
619
+ const gpdHeader = makeHeader(
620
+ sequenceNumber,
621
+ commandIdentifier,
622
+ addr.applicationId,
623
+ securityLevelFullEncr,
624
+ securityKeyTypeNWK,
625
+ bidirectionalInfo,
626
+ addr.sourceId,
627
+ gpdSecurityFrameCounter,
628
+ gpdCommandId,
629
+ gpdCommandPayload.length,
630
+ );
631
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
632
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
633
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
634
+
635
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
636
+ 1,
637
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x20 srcID=1496140231 gpp=NO",
638
+ "zh:controller:greenpower",
639
+ );
640
+
641
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
642
+ clonedFrame.payload.commandID = 0x20;
643
+ clonedFrame.payload.options = 256;
644
+ clonedFrame.payload.commandFrame = {};
645
+
646
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
647
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
648
+ }
649
+
650
+ clearLogMocks();
651
+
652
+ // left
653
+ {
654
+ const gpdLink = 223;
655
+ const sequenceNumber = 55;
656
+ const bidirectionalInfo = 0;
657
+ const gpdSecurityFrameCounter = 2359;
658
+ const gpdCommandId = 189;
659
+ const gpdCommandPayload = Buffer.from("", "hex");
660
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.notification.ID;
661
+
662
+ const gpdHeader = makeHeader(
663
+ sequenceNumber,
664
+ commandIdentifier,
665
+ addr.applicationId,
666
+ securityLevelFullEncr,
667
+ securityKeyTypeNWK,
668
+ bidirectionalInfo,
669
+ addr.sourceId,
670
+ gpdSecurityFrameCounter,
671
+ gpdCommandId,
672
+ gpdCommandPayload.length,
673
+ );
674
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
675
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
676
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
677
+
678
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
679
+ 1,
680
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x20 srcID=1496140231 gpp=NO",
681
+ "zh:controller:greenpower",
682
+ );
683
+
684
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
685
+ clonedFrame.payload.commandID = 0x20;
686
+ clonedFrame.payload.options = 256;
687
+ clonedFrame.payload.commandFrame = {};
688
+
689
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
690
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
691
+ }
692
+
693
+ clearLogMocks();
694
+
695
+ // right
696
+ {
697
+ const gpdLink = 218;
698
+ const sequenceNumber = 37;
699
+ const bidirectionalInfo = 0;
700
+ const gpdSecurityFrameCounter = 2341;
701
+ const gpdCommandId = 172;
702
+ const gpdCommandPayload = Buffer.from("", "hex");
703
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.notification.ID;
704
+
705
+ const gpdHeader = makeHeader(
706
+ sequenceNumber,
707
+ commandIdentifier,
708
+ addr.applicationId,
709
+ securityLevelFullEncr,
710
+ securityKeyTypeNWK,
711
+ bidirectionalInfo,
712
+ addr.sourceId,
713
+ gpdSecurityFrameCounter,
714
+ gpdCommandId,
715
+ gpdCommandPayload.length,
716
+ );
717
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
718
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
719
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
720
+
721
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
722
+ 1,
723
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x21 srcID=1496140231 gpp=NO",
724
+ "zh:controller:greenpower",
725
+ );
726
+
727
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
728
+ clonedFrame.payload.commandID = 0x21;
729
+ clonedFrame.payload.options = 256;
730
+ clonedFrame.payload.commandFrame = {};
731
+
732
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
733
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
734
+ }
735
+
736
+ clearLogMocks();
737
+
738
+ // right
739
+ {
740
+ const gpdLink = 222;
741
+ const sequenceNumber = 64;
742
+ const bidirectionalInfo = 0;
743
+ const gpdSecurityFrameCounter = 2368;
744
+ const gpdCommandId = 159;
745
+ const gpdCommandPayload = Buffer.from("", "hex");
746
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.notification.ID;
747
+
748
+ const gpdHeader = makeHeader(
749
+ sequenceNumber,
750
+ commandIdentifier,
751
+ addr.applicationId,
752
+ securityLevelFullEncr,
753
+ securityKeyTypeNWK,
754
+ bidirectionalInfo,
755
+ addr.sourceId,
756
+ gpdSecurityFrameCounter,
757
+ gpdCommandId,
758
+ gpdCommandPayload.length,
759
+ );
760
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
761
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
762
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
763
+
764
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
765
+ 1,
766
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x21 srcID=1496140231 gpp=NO",
767
+ "zh:controller:greenpower",
768
+ );
769
+
770
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
771
+ clonedFrame.payload.commandID = 0x21;
772
+ clonedFrame.payload.options = 256;
773
+ clonedFrame.payload.commandFrame = {};
774
+
775
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
776
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
777
+ }
778
+
779
+ clearLogMocks();
780
+
781
+ // right
782
+ {
783
+ const gpdLink = 222;
784
+ const sequenceNumber = 73;
785
+ const bidirectionalInfo = 0;
786
+ const gpdSecurityFrameCounter = 2377;
787
+ const gpdCommandId = 11;
788
+ const gpdCommandPayload = Buffer.from("", "hex");
789
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.notification.ID;
790
+
791
+ const gpdHeader = makeHeader(
792
+ sequenceNumber,
793
+ commandIdentifier,
794
+ addr.applicationId,
795
+ securityLevelFullEncr,
796
+ securityKeyTypeNWK,
797
+ bidirectionalInfo,
798
+ addr.sourceId,
799
+ gpdSecurityFrameCounter,
800
+ gpdCommandId,
801
+ gpdCommandPayload.length,
802
+ );
803
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
804
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
805
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
806
+
807
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
808
+ 1,
809
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x21 srcID=1496140231 gpp=NO",
810
+ "zh:controller:greenpower",
811
+ );
812
+
813
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
814
+ clonedFrame.payload.commandID = 0x21;
815
+ clonedFrame.payload.options = 256;
816
+ clonedFrame.payload.commandFrame = {};
817
+
818
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
819
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
820
+ }
821
+
822
+ clearLogMocks();
823
+
824
+ // mock FULLENCR with unknown security key
825
+ {
826
+ const gpdLink = 222;
827
+ const sequenceNumber = 73;
828
+ const bidirectionalInfo = 0;
829
+ const gpdSecurityFrameCounter = 2377;
830
+ const gpdCommandId = 11;
831
+ const gpdCommandPayload = Buffer.from("", "hex");
832
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.notification.ID;
833
+
834
+ const gpdHeader = makeHeader(
835
+ sequenceNumber,
836
+ commandIdentifier,
837
+ addr.applicationId,
838
+ securityLevelFullEncr,
839
+ securityKeyTypeNWK,
840
+ bidirectionalInfo,
841
+ addr.sourceId,
842
+ gpdSecurityFrameCounter,
843
+ gpdCommandId,
844
+ gpdCommandPayload.length,
845
+ );
846
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
847
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
848
+ const retFrame = await gp.processCommand(payload, frame, undefined);
849
+
850
+ expect(logErrorSpy).toHaveBeenNthCalledWith(
851
+ 1,
852
+ "[FULLENCR] srcID=1496140231 gpp=NO commandIdentifier=0 Unknown security key",
853
+ "zh:controller:greenpower",
854
+ );
855
+
856
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(frame)));
857
+ }
858
+
859
+ clearLogMocks();
860
+
861
+ // mock FULLENCR with gpp data
862
+ {
863
+ const gpdLink = 222;
864
+ const sequenceNumber = 73;
865
+ const gpdSecurityFrameCounter = 2377;
866
+ const gpdCommandId = 11;
867
+ const gpdCommandPayload = Buffer.from("", "hex");
868
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.notification.ID;
869
+ const gppNwkAddr = 24404;
870
+ const gppGpdLink = 207;
871
+ const options = ((0b11 & 0x3) << 6) | 0x4000;
872
+
873
+ const gpdHeader = makeHeader(
874
+ sequenceNumber,
875
+ commandIdentifier,
876
+ 0,
877
+ 0,
878
+ 0,
879
+ 0,
880
+ addr.sourceId,
881
+ gpdSecurityFrameCounter,
882
+ gpdCommandId,
883
+ gpdCommandPayload.length,
884
+ options,
885
+ );
886
+ const gpdFooter = Buffer.alloc(3);
887
+ gpdFooter.writeUInt16LE(gppNwkAddr, 0);
888
+ gpdFooter.writeUInt8(gppGpdLink, 2);
889
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload, gpdFooter]), gpdLink);
890
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
891
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
892
+
893
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
894
+ 1,
895
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x21 srcID=1496140231 gpp=24404 rssi=15 linkQuality=Excellent",
896
+ "zh:controller:greenpower",
897
+ );
898
+
899
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
900
+ clonedFrame.payload.commandID = 0x21;
901
+ clonedFrame.payload.options = 16384;
902
+ clonedFrame.payload.commandFrame = {};
903
+ clonedFrame.payload.gppNwkAddr = gppNwkAddr;
904
+ clonedFrame.payload.gppGpdLink = gppGpdLink;
905
+
906
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
907
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
908
+ }
909
+ });
910
+
911
+ // @see https://github.com/Koenkk/zigbee2mqtt/issues/19405#issuecomment-2732204071
912
+ it("FULLENCR ZT-LP-ZEU2S-WH-MS MOES 3-gang vectors from ember", async () => {
913
+ let joinData: GreenPowerDeviceJoinedPayload | undefined;
914
+
915
+ gp.on("deviceJoined", (payload) => {
916
+ joinData = payload;
917
+ });
918
+
919
+ const addr = {applicationId: 0, sourceId: 344902069, endpoint: 0};
920
+
921
+ {
922
+ const gpdLink = 219;
923
+ const sequenceNumber = 139;
924
+ const gpdfSecurityLevel = 0; // NONE
925
+ const gpdfSecurityKeyType = 0; // NONE
926
+ const bidirectionalInfo = 0;
927
+ const gpdSecurityFrameCounter = 4294967295;
928
+ const gpdCommandId = 224;
929
+ const gpdCommandPayload = Buffer.from("0289f35690230a93ea5f1951926f200236c7820891812a8b0400007165726837706f7840475030303031be", "hex");
930
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.commissioningNotification.ID;
931
+
932
+ const gpdHeader = makeHeader(
933
+ sequenceNumber,
934
+ commandIdentifier,
935
+ addr.applicationId,
936
+ gpdfSecurityLevel,
937
+ gpdfSecurityKeyType,
938
+ bidirectionalInfo,
939
+ addr.sourceId,
940
+ gpdSecurityFrameCounter,
941
+ gpdCommandId,
942
+ gpdCommandPayload.length,
943
+ );
944
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
945
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
946
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey); // always undefined since not yet joined
947
+
948
+ await vi.waitUntil(() => joinData !== undefined);
949
+
950
+ expect(joinData).toStrictEqual({
951
+ sourceID: addr.sourceId,
952
+ deviceID: frame.payload.commandFrame.deviceID,
953
+ networkAddress: addr.sourceId & 0xffff,
954
+ securityKey: frame.payload.commandFrame.securityKey,
955
+ });
956
+ expect(logInfoSpy).toHaveBeenNthCalledWith(1, "[COMMISSIONING] srcID=344902069 gpp=NO", "zh:controller:greenpower");
957
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
958
+ 1,
959
+ "[PAIRING] srcID=344902069 gpp=NO options=58696 (addSink=true commMode=2)",
960
+ "zh:controller:greenpower",
961
+ );
962
+
963
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
964
+ clonedFrame.payload.commandID = 0xe0;
965
+ clonedFrame.payload.options = 0;
966
+ clonedFrame.payload.commandFrame = {
967
+ deviceID: 2,
968
+ options: 137,
969
+ extendedOptions: 243,
970
+ securityKey: joinData?.securityKey,
971
+ keyMic: 713134344,
972
+ outgoingCounter: 1163,
973
+ applicationInfo: addr.applicationId,
974
+ manufacturerID: 0,
975
+ modelID: 0,
976
+ numGpdCommands: 0,
977
+ gpdCommandIdList: Buffer.from([]),
978
+ numServerClusters: 0,
979
+ numClientClusters: 0,
980
+ gpdServerClusters: Buffer.from([]),
981
+ gpdClientClusters: Buffer.from([]),
982
+ genericSwitchConfig: 0,
983
+ currentContactStatus: 0,
984
+ };
985
+
986
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
987
+ expect(retFrame.payload.commandFrame.securityKey).toStrictEqual(joinData?.securityKey);
988
+ }
989
+
990
+ clearLogMocks();
991
+
992
+ const securityLevelFullEncr = 3;
993
+ const securityKeyTypeNWK = 1;
994
+
995
+ // left
996
+ {
997
+ const gpdLink = 224;
998
+ const sequenceNumber = 175;
999
+ const bidirectionalInfo = 0;
1000
+ const gpdSecurityFrameCounter = 1199;
1001
+ const gpdCommandId = 92;
1002
+ const gpdCommandPayload = Buffer.from("", "hex");
1003
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.notification.ID;
1004
+
1005
+ const gpdHeader = makeHeader(
1006
+ sequenceNumber,
1007
+ commandIdentifier,
1008
+ addr.applicationId,
1009
+ securityLevelFullEncr,
1010
+ securityKeyTypeNWK,
1011
+ bidirectionalInfo,
1012
+ addr.sourceId,
1013
+ gpdSecurityFrameCounter,
1014
+ gpdCommandId,
1015
+ gpdCommandPayload.length,
1016
+ );
1017
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
1018
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1019
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
1020
+
1021
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
1022
+ 1,
1023
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x20 srcID=344902069 gpp=NO",
1024
+ "zh:controller:greenpower",
1025
+ );
1026
+
1027
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1028
+ clonedFrame.payload.commandID = 0x20;
1029
+ clonedFrame.payload.options = 256;
1030
+ clonedFrame.payload.commandFrame = {};
1031
+
1032
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
1033
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
1034
+ }
1035
+
1036
+ clearLogMocks();
1037
+
1038
+ // middle
1039
+ {
1040
+ const gpdLink = 225;
1041
+ const sequenceNumber = 184;
1042
+ const bidirectionalInfo = 0;
1043
+ const gpdSecurityFrameCounter = 1208;
1044
+ const gpdCommandId = 109;
1045
+ const gpdCommandPayload = Buffer.from("", "hex");
1046
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.notification.ID;
1047
+
1048
+ const gpdHeader = makeHeader(
1049
+ sequenceNumber,
1050
+ commandIdentifier,
1051
+ addr.applicationId,
1052
+ securityLevelFullEncr,
1053
+ securityKeyTypeNWK,
1054
+ bidirectionalInfo,
1055
+ addr.sourceId,
1056
+ gpdSecurityFrameCounter,
1057
+ gpdCommandId,
1058
+ gpdCommandPayload.length,
1059
+ );
1060
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
1061
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1062
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
1063
+
1064
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
1065
+ 1,
1066
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x21 srcID=344902069 gpp=NO",
1067
+ "zh:controller:greenpower",
1068
+ );
1069
+
1070
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1071
+ clonedFrame.payload.commandID = 0x21;
1072
+ clonedFrame.payload.options = 256;
1073
+ clonedFrame.payload.commandFrame = {};
1074
+
1075
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
1076
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
1077
+ }
1078
+
1079
+ clearLogMocks();
1080
+
1081
+ // right
1082
+ {
1083
+ const gpdLink = 225;
1084
+ const sequenceNumber = 193;
1085
+ const bidirectionalInfo = 0;
1086
+ const gpdSecurityFrameCounter = 1217;
1087
+ const gpdCommandId = 219;
1088
+ const gpdCommandPayload = Buffer.from("", "hex");
1089
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.notification.ID;
1090
+
1091
+ const gpdHeader = makeHeader(
1092
+ sequenceNumber,
1093
+ commandIdentifier,
1094
+ addr.applicationId,
1095
+ securityLevelFullEncr,
1096
+ securityKeyTypeNWK,
1097
+ bidirectionalInfo,
1098
+ addr.sourceId,
1099
+ gpdSecurityFrameCounter,
1100
+ gpdCommandId,
1101
+ gpdCommandPayload.length,
1102
+ );
1103
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload]), gpdLink);
1104
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1105
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
1106
+
1107
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
1108
+ 1,
1109
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x11 srcID=344902069 gpp=NO",
1110
+ "zh:controller:greenpower",
1111
+ );
1112
+
1113
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1114
+ clonedFrame.payload.commandID = 0x11;
1115
+ clonedFrame.payload.options = 256;
1116
+ clonedFrame.payload.commandFrame = {};
1117
+
1118
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
1119
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
1120
+ }
1121
+ });
1122
+
1123
+ // @see https://github.com/Koenkk/zigbee2mqtt/issues/19405#issuecomment-2744667458
1124
+ it("FULLENCR ZT-LP-ZEU2S-WH-MS MOES 2-gang vectors from zstack through GPP", async () => {
1125
+ const joinData: GreenPowerDeviceJoinedPayload = {
1126
+ sourceID: 2777252112,
1127
+ deviceID: 2,
1128
+ networkAddress: 2777252112 & 0xffff,
1129
+ securityKey: Buffer.from([227, 227, 225, 134, 235, 104, 141, 250, 162, 211, 104, 147, 201, 146, 67, 175]),
1130
+ };
1131
+ const addr = {applicationId: 0, sourceId: 2777252112, endpoint: 0};
1132
+ const gppNwkAddr = 24404;
1133
+ const options = 2864;
1134
+
1135
+ // right
1136
+ {
1137
+ const sequenceNumber = 18;
1138
+ const gpdSecurityFrameCounter = 17326;
1139
+ const gpdCommandId = 38;
1140
+ const gpdCommandPayload = Buffer.from("", "hex");
1141
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.commissioningNotification.ID;
1142
+ const gppGpdLink = 207;
1143
+ const mic = 1441399364;
1144
+
1145
+ const gpdHeader = makeHeader(
1146
+ sequenceNumber,
1147
+ commandIdentifier,
1148
+ 0,
1149
+ 0,
1150
+ 0,
1151
+ 0,
1152
+ addr.sourceId,
1153
+ gpdSecurityFrameCounter,
1154
+ gpdCommandId,
1155
+ gpdCommandPayload.length,
1156
+ options,
1157
+ );
1158
+ const gpdFooter = makeFooter(options, gppNwkAddr, gppGpdLink, mic);
1159
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload, gpdFooter]), 138);
1160
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1161
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
1162
+
1163
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
1164
+ 1,
1165
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x21 srcID=2777252112 gpp=24404 rssi=15 linkQuality=Excellent",
1166
+ "zh:controller:greenpower",
1167
+ );
1168
+
1169
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1170
+ clonedFrame.payload.commandID = 0x21;
1171
+ clonedFrame.payload.options = 2304;
1172
+ clonedFrame.payload.commandFrame = {};
1173
+ clonedFrame.payload.gppNwkAddr = gppNwkAddr;
1174
+ clonedFrame.payload.gppGpdLink = gppGpdLink;
1175
+ delete clonedFrame.payload.mic;
1176
+
1177
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
1178
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
1179
+ }
1180
+
1181
+ clearLogMocks();
1182
+
1183
+ // right
1184
+ {
1185
+ const sequenceNumber = 19;
1186
+ const gpdSecurityFrameCounter = 17335;
1187
+ const gpdCommandId = 17;
1188
+ const gpdCommandPayload = Buffer.from("", "hex");
1189
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.commissioningNotification.ID;
1190
+ const gppGpdLink = 207;
1191
+ const mic = 3064327344;
1192
+
1193
+ const gpdHeader = makeHeader(
1194
+ sequenceNumber,
1195
+ commandIdentifier,
1196
+ 0,
1197
+ 0,
1198
+ 0,
1199
+ 0,
1200
+ addr.sourceId,
1201
+ gpdSecurityFrameCounter,
1202
+ gpdCommandId,
1203
+ gpdCommandPayload.length,
1204
+ options,
1205
+ );
1206
+ const gpdFooter = makeFooter(options, gppNwkAddr, gppGpdLink, mic);
1207
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload, gpdFooter]), 127);
1208
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1209
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
1210
+
1211
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
1212
+ 1,
1213
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x21 srcID=2777252112 gpp=24404 rssi=15 linkQuality=Excellent",
1214
+ "zh:controller:greenpower",
1215
+ );
1216
+
1217
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1218
+ clonedFrame.payload.commandID = 0x21;
1219
+ clonedFrame.payload.options = 2304;
1220
+ clonedFrame.payload.commandFrame = {};
1221
+ clonedFrame.payload.gppNwkAddr = gppNwkAddr;
1222
+ clonedFrame.payload.gppGpdLink = gppGpdLink;
1223
+ delete clonedFrame.payload.mic;
1224
+
1225
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
1226
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
1227
+ }
1228
+
1229
+ clearLogMocks();
1230
+
1231
+ // right
1232
+ {
1233
+ const sequenceNumber = 20;
1234
+ const gpdSecurityFrameCounter = 17344;
1235
+ const gpdCommandId = 211;
1236
+ const gpdCommandPayload = Buffer.from("", "hex");
1237
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.commissioningNotification.ID;
1238
+ const gppGpdLink = 207;
1239
+ const mic = 3315864057;
1240
+
1241
+ const gpdHeader = makeHeader(
1242
+ sequenceNumber,
1243
+ commandIdentifier,
1244
+ 0,
1245
+ 0,
1246
+ 0,
1247
+ 0,
1248
+ addr.sourceId,
1249
+ gpdSecurityFrameCounter,
1250
+ gpdCommandId,
1251
+ gpdCommandPayload.length,
1252
+ options,
1253
+ );
1254
+ const gpdFooter = makeFooter(options, gppNwkAddr, gppGpdLink, mic);
1255
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload, gpdFooter]), 138);
1256
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1257
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
1258
+
1259
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
1260
+ 1,
1261
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x21 srcID=2777252112 gpp=24404 rssi=15 linkQuality=Excellent",
1262
+ "zh:controller:greenpower",
1263
+ );
1264
+
1265
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1266
+ clonedFrame.payload.commandID = 0x21;
1267
+ clonedFrame.payload.options = 2304;
1268
+ clonedFrame.payload.commandFrame = {};
1269
+ clonedFrame.payload.gppNwkAddr = gppNwkAddr;
1270
+ clonedFrame.payload.gppGpdLink = gppGpdLink;
1271
+ delete clonedFrame.payload.mic;
1272
+
1273
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
1274
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
1275
+ }
1276
+
1277
+ clearLogMocks();
1278
+
1279
+ // left
1280
+ {
1281
+ const sequenceNumber = 21;
1282
+ const gpdSecurityFrameCounter = 17353;
1283
+ const gpdCommandId = 174;
1284
+ const gpdCommandPayload = Buffer.from("", "hex");
1285
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.commissioningNotification.ID;
1286
+ const gppGpdLink = 142;
1287
+ const mic = 827946906;
1288
+
1289
+ const gpdHeader = makeHeader(
1290
+ sequenceNumber,
1291
+ commandIdentifier,
1292
+ 0,
1293
+ 0,
1294
+ 0,
1295
+ 0,
1296
+ addr.sourceId,
1297
+ gpdSecurityFrameCounter,
1298
+ gpdCommandId,
1299
+ gpdCommandPayload.length,
1300
+ options,
1301
+ );
1302
+ const gpdFooter = makeFooter(options, gppNwkAddr, gppGpdLink, mic);
1303
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload, gpdFooter]), 138);
1304
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1305
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
1306
+
1307
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
1308
+ 1,
1309
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x20 srcID=2777252112 gpp=24404 rssi=14 linkQuality=High",
1310
+ "zh:controller:greenpower",
1311
+ );
1312
+
1313
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1314
+ clonedFrame.payload.commandID = 0x20;
1315
+ clonedFrame.payload.options = 2304;
1316
+ clonedFrame.payload.commandFrame = {};
1317
+ clonedFrame.payload.gppNwkAddr = gppNwkAddr;
1318
+ clonedFrame.payload.gppGpdLink = gppGpdLink;
1319
+ delete clonedFrame.payload.mic;
1320
+
1321
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
1322
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
1323
+ }
1324
+
1325
+ clearLogMocks();
1326
+
1327
+ // left
1328
+ {
1329
+ const sequenceNumber = 22;
1330
+ const gpdSecurityFrameCounter = 17362;
1331
+ const gpdCommandId = 230;
1332
+ const gpdCommandPayload = Buffer.from("", "hex");
1333
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.commissioningNotification.ID;
1334
+ const gppGpdLink = 209;
1335
+ const mic = 2941277720;
1336
+
1337
+ const gpdHeader = makeHeader(
1338
+ sequenceNumber,
1339
+ commandIdentifier,
1340
+ 0,
1341
+ 0,
1342
+ 0,
1343
+ 0,
1344
+ addr.sourceId,
1345
+ gpdSecurityFrameCounter,
1346
+ gpdCommandId,
1347
+ gpdCommandPayload.length,
1348
+ options,
1349
+ );
1350
+ const gpdFooter = makeFooter(options, gppNwkAddr, gppGpdLink, mic);
1351
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload, gpdFooter]), 142);
1352
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1353
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
1354
+
1355
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
1356
+ 1,
1357
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x20 srcID=2777252112 gpp=24404 rssi=17 linkQuality=Excellent",
1358
+ "zh:controller:greenpower",
1359
+ );
1360
+
1361
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1362
+ clonedFrame.payload.commandID = 0x20;
1363
+ clonedFrame.payload.options = 2304;
1364
+ clonedFrame.payload.commandFrame = {};
1365
+ clonedFrame.payload.gppNwkAddr = gppNwkAddr;
1366
+ clonedFrame.payload.gppGpdLink = gppGpdLink;
1367
+ delete clonedFrame.payload.mic;
1368
+
1369
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
1370
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
1371
+ }
1372
+
1373
+ clearLogMocks();
1374
+
1375
+ // left
1376
+ {
1377
+ const sequenceNumber = 23;
1378
+ const gpdSecurityFrameCounter = 17371;
1379
+ const gpdCommandId = 59;
1380
+ const gpdCommandPayload = Buffer.from("", "hex");
1381
+ const commandIdentifier = Zcl.Clusters.greenPower.commands.commissioningNotification.ID;
1382
+ const gppGpdLink = 209;
1383
+ const mic = 3231351307;
1384
+
1385
+ const gpdHeader = makeHeader(
1386
+ sequenceNumber,
1387
+ commandIdentifier,
1388
+ 0,
1389
+ 0,
1390
+ 0,
1391
+ 0,
1392
+ addr.sourceId,
1393
+ gpdSecurityFrameCounter,
1394
+ gpdCommandId,
1395
+ gpdCommandPayload.length,
1396
+ options,
1397
+ );
1398
+ const gpdFooter = makeFooter(options, gppNwkAddr, gppGpdLink, mic);
1399
+ const payload = makePayload(addr.sourceId, Buffer.concat([gpdHeader, gpdCommandPayload, gpdFooter]), 138);
1400
+ const frame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1401
+ const retFrame = await gp.processCommand(payload, frame, joinData?.securityKey);
1402
+
1403
+ expect(logDebugSpy).toHaveBeenNthCalledWith(
1404
+ 1,
1405
+ "[UNHANDLED_CMD/PASSTHROUGH] command=0x20 srcID=2777252112 gpp=24404 rssi=17 linkQuality=Excellent",
1406
+ "zh:controller:greenpower",
1407
+ );
1408
+
1409
+ const clonedFrame = Zcl.Frame.fromBuffer(payload.clusterID, payload.header, payload.data, {});
1410
+ clonedFrame.payload.commandID = 0x20;
1411
+ clonedFrame.payload.options = 2304;
1412
+ clonedFrame.payload.commandFrame = {};
1413
+ clonedFrame.payload.gppNwkAddr = gppNwkAddr;
1414
+ clonedFrame.payload.gppGpdLink = gppGpdLink;
1415
+ delete clonedFrame.payload.mic;
1416
+
1417
+ expect(JSON.parse(JSON.stringify(retFrame))).toStrictEqual(JSON.parse(JSON.stringify(clonedFrame)));
1418
+ expect(retFrame.payload.commandFrame).toStrictEqual({});
1419
+ }
1420
+ });
1421
+ });