@willieee802/zigbee-herdsman 0.19.2 → 0.34.3

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 (530) hide show
  1. package/.babelrc.js +0 -4
  2. package/.release-please-manifest.json +1 -2
  3. package/CHANGELOG.md +376 -0
  4. package/README.md +1 -1
  5. package/dist/adapter/adapter.d.ts +61 -61
  6. package/dist/adapter/adapter.d.ts.map +1 -1
  7. package/dist/adapter/adapter.js +158 -146
  8. package/dist/adapter/adapter.js.map +1 -1
  9. package/dist/adapter/deconz/adapter/deconzAdapter.d.ts +68 -68
  10. package/dist/adapter/deconz/adapter/deconzAdapter.d.ts.map +1 -1
  11. package/dist/adapter/deconz/adapter/deconzAdapter.js +1081 -1060
  12. package/dist/adapter/deconz/adapter/deconzAdapter.js.map +1 -1
  13. package/dist/adapter/deconz/adapter/index.d.ts +2 -2
  14. package/dist/adapter/deconz/adapter/index.js +10 -10
  15. package/dist/adapter/deconz/driver/constants.d.ts +104 -104
  16. package/dist/adapter/deconz/driver/constants.js +55 -55
  17. package/dist/adapter/deconz/driver/driver.d.ts +81 -81
  18. package/dist/adapter/deconz/driver/driver.d.ts.map +1 -1
  19. package/dist/adapter/deconz/driver/driver.js +750 -732
  20. package/dist/adapter/deconz/driver/driver.js.map +1 -1
  21. package/dist/adapter/deconz/driver/frame.d.ts +6 -6
  22. package/dist/adapter/deconz/driver/frame.js +13 -13
  23. package/dist/adapter/deconz/driver/frameParser.d.ts +2 -2
  24. package/dist/adapter/deconz/driver/frameParser.js +443 -443
  25. package/dist/adapter/deconz/driver/frameParser.js.map +1 -1
  26. package/dist/adapter/deconz/driver/parser.d.ts +12 -12
  27. package/dist/adapter/deconz/driver/parser.js +63 -61
  28. package/dist/adapter/deconz/driver/parser.js.map +1 -1
  29. package/dist/adapter/deconz/driver/writer.d.ts +8 -8
  30. package/dist/adapter/deconz/driver/writer.js +44 -44
  31. package/dist/adapter/ember/adapter/emberAdapter.d.ts +817 -0
  32. package/dist/adapter/ember/adapter/emberAdapter.d.ts.map +1 -0
  33. package/dist/adapter/ember/adapter/emberAdapter.js +2981 -0
  34. package/dist/adapter/ember/adapter/emberAdapter.js.map +1 -0
  35. package/dist/adapter/ember/adapter/endpoints.d.ts +25 -0
  36. package/dist/adapter/ember/adapter/endpoints.d.ts.map +1 -0
  37. package/dist/adapter/ember/adapter/endpoints.js +66 -0
  38. package/dist/adapter/ember/adapter/endpoints.js.map +1 -0
  39. package/dist/adapter/ember/adapter/index.d.ts +3 -0
  40. package/dist/adapter/ember/adapter/index.d.ts.map +1 -0
  41. package/dist/adapter/ember/adapter/index.js +6 -0
  42. package/dist/adapter/ember/adapter/index.js.map +1 -0
  43. package/dist/adapter/ember/adapter/oneWaitress.d.ts +97 -0
  44. package/dist/adapter/ember/adapter/oneWaitress.d.ts.map +1 -0
  45. package/dist/adapter/ember/adapter/oneWaitress.js +226 -0
  46. package/dist/adapter/ember/adapter/oneWaitress.js.map +1 -0
  47. package/dist/adapter/ember/adapter/requestQueue.d.ts +59 -0
  48. package/dist/adapter/ember/adapter/requestQueue.d.ts.map +1 -0
  49. package/dist/adapter/ember/adapter/requestQueue.js +144 -0
  50. package/dist/adapter/ember/adapter/requestQueue.js.map +1 -0
  51. package/dist/adapter/ember/adapter/tokensManager.d.ts +69 -0
  52. package/dist/adapter/ember/adapter/tokensManager.d.ts.map +1 -0
  53. package/dist/adapter/ember/adapter/tokensManager.js +685 -0
  54. package/dist/adapter/ember/adapter/tokensManager.js.map +1 -0
  55. package/dist/adapter/ember/consts.d.ts +198 -0
  56. package/dist/adapter/ember/consts.d.ts.map +1 -0
  57. package/dist/adapter/ember/consts.js +253 -0
  58. package/dist/adapter/ember/consts.js.map +1 -0
  59. package/dist/adapter/ember/enums.d.ts +2184 -0
  60. package/dist/adapter/ember/enums.d.ts.map +1 -0
  61. package/dist/adapter/ember/enums.js +2391 -0
  62. package/dist/adapter/ember/enums.js.map +1 -0
  63. package/dist/adapter/ember/ezsp/buffalo.d.ts +156 -0
  64. package/dist/adapter/ember/ezsp/buffalo.d.ts.map +1 -0
  65. package/dist/adapter/ember/ezsp/buffalo.js +1033 -0
  66. package/dist/adapter/ember/ezsp/buffalo.js.map +1 -0
  67. package/dist/adapter/ember/ezsp/consts.d.ts +116 -0
  68. package/dist/adapter/ember/ezsp/consts.d.ts.map +1 -0
  69. package/dist/adapter/ember/ezsp/consts.js +128 -0
  70. package/dist/adapter/ember/ezsp/consts.js.map +1 -0
  71. package/dist/adapter/ember/ezsp/enums.d.ts +879 -0
  72. package/dist/adapter/ember/ezsp/enums.d.ts.map +1 -0
  73. package/dist/adapter/ember/ezsp/enums.js +948 -0
  74. package/dist/adapter/ember/ezsp/enums.js.map +1 -0
  75. package/dist/adapter/ember/ezsp/ezsp.d.ts +2664 -0
  76. package/dist/adapter/ember/ezsp/ezsp.d.ts.map +1 -0
  77. package/dist/adapter/ember/ezsp/ezsp.js +6438 -0
  78. package/dist/adapter/ember/ezsp/ezsp.js.map +1 -0
  79. package/dist/adapter/ember/types.d.ts +733 -0
  80. package/dist/adapter/ember/types.d.ts.map +1 -0
  81. package/dist/adapter/ember/types.js +3 -0
  82. package/dist/adapter/ember/types.js.map +1 -0
  83. package/dist/adapter/ember/uart/ash.d.ts +451 -0
  84. package/dist/adapter/ember/uart/ash.d.ts.map +1 -0
  85. package/dist/adapter/ember/uart/ash.js +1584 -0
  86. package/dist/adapter/ember/uart/ash.js.map +1 -0
  87. package/dist/adapter/ember/uart/consts.d.ts +91 -0
  88. package/dist/adapter/ember/uart/consts.d.ts.map +1 -0
  89. package/dist/adapter/ember/uart/consts.js +100 -0
  90. package/dist/adapter/ember/uart/consts.js.map +1 -0
  91. package/dist/adapter/ember/uart/enums.d.ts +191 -0
  92. package/dist/adapter/ember/uart/enums.d.ts.map +1 -0
  93. package/dist/adapter/ember/uart/enums.js +197 -0
  94. package/dist/adapter/ember/uart/enums.js.map +1 -0
  95. package/dist/adapter/ember/uart/parser.d.ts +10 -0
  96. package/dist/adapter/ember/uart/parser.d.ts.map +1 -0
  97. package/dist/adapter/ember/uart/parser.js +41 -0
  98. package/dist/adapter/ember/uart/parser.js.map +1 -0
  99. package/dist/adapter/ember/uart/queues.d.ts +85 -0
  100. package/dist/adapter/ember/uart/queues.d.ts.map +1 -0
  101. package/dist/adapter/ember/uart/queues.js +212 -0
  102. package/dist/adapter/ember/uart/queues.js.map +1 -0
  103. package/dist/adapter/ember/uart/writer.d.ts +15 -0
  104. package/dist/adapter/ember/uart/writer.d.ts.map +1 -0
  105. package/dist/adapter/ember/uart/writer.js +48 -0
  106. package/dist/adapter/ember/uart/writer.js.map +1 -0
  107. package/dist/adapter/ember/utils/initters.d.ts +20 -0
  108. package/dist/adapter/ember/utils/initters.d.ts.map +1 -0
  109. package/dist/adapter/ember/utils/initters.js +58 -0
  110. package/dist/adapter/ember/utils/initters.js.map +1 -0
  111. package/dist/adapter/ember/utils/math.d.ts +51 -0
  112. package/dist/adapter/ember/utils/math.d.ts.map +1 -0
  113. package/dist/adapter/ember/utils/math.js +102 -0
  114. package/dist/adapter/ember/utils/math.js.map +1 -0
  115. package/dist/adapter/ember/zdo.d.ts +921 -0
  116. package/dist/adapter/ember/zdo.d.ts.map +1 -0
  117. package/dist/adapter/ember/zdo.js +723 -0
  118. package/dist/adapter/ember/zdo.js.map +1 -0
  119. package/dist/adapter/events.d.ts +47 -47
  120. package/dist/adapter/events.js +13 -14
  121. package/dist/adapter/events.js.map +1 -1
  122. package/dist/adapter/ezsp/adapter/backup.d.ts +9 -9
  123. package/dist/adapter/ezsp/adapter/backup.d.ts.map +1 -1
  124. package/dist/adapter/ezsp/adapter/backup.js +72 -53
  125. package/dist/adapter/ezsp/adapter/backup.js.map +1 -1
  126. package/dist/adapter/ezsp/adapter/ezspAdapter.d.ts +61 -59
  127. package/dist/adapter/ezsp/adapter/ezspAdapter.d.ts.map +1 -1
  128. package/dist/adapter/ezsp/adapter/ezspAdapter.js +629 -603
  129. package/dist/adapter/ezsp/adapter/ezspAdapter.js.map +1 -1
  130. package/dist/adapter/ezsp/adapter/index.d.ts +2 -2
  131. package/dist/adapter/ezsp/adapter/index.js +10 -10
  132. package/dist/adapter/ezsp/driver/commands.d.ts +36 -36
  133. package/dist/adapter/ezsp/driver/commands.d.ts.map +1 -1
  134. package/dist/adapter/ezsp/driver/commands.js +2388 -2359
  135. package/dist/adapter/ezsp/driver/commands.js.map +1 -1
  136. package/dist/adapter/ezsp/driver/consts.d.ts +10 -10
  137. package/dist/adapter/ezsp/driver/consts.js +13 -13
  138. package/dist/adapter/ezsp/driver/driver.d.ts +106 -103
  139. package/dist/adapter/ezsp/driver/driver.d.ts.map +1 -1
  140. package/dist/adapter/ezsp/driver/driver.js +731 -639
  141. package/dist/adapter/ezsp/driver/driver.js.map +1 -1
  142. package/dist/adapter/ezsp/driver/ezsp.d.ts +105 -96
  143. package/dist/adapter/ezsp/driver/ezsp.d.ts.map +1 -1
  144. package/dist/adapter/ezsp/driver/ezsp.js +651 -586
  145. package/dist/adapter/ezsp/driver/ezsp.js.map +1 -1
  146. package/dist/adapter/ezsp/driver/frame.d.ts +40 -0
  147. package/dist/adapter/ezsp/driver/frame.d.ts.map +1 -0
  148. package/dist/adapter/ezsp/driver/frame.js +101 -0
  149. package/dist/adapter/ezsp/driver/frame.js.map +1 -0
  150. package/dist/adapter/ezsp/driver/index.d.ts +3 -3
  151. package/dist/adapter/ezsp/driver/index.js +8 -8
  152. package/dist/adapter/ezsp/driver/multicast.d.ts +12 -12
  153. package/dist/adapter/ezsp/driver/multicast.d.ts.map +1 -1
  154. package/dist/adapter/ezsp/driver/multicast.js +77 -74
  155. package/dist/adapter/ezsp/driver/multicast.js.map +1 -1
  156. package/dist/adapter/ezsp/driver/parser.d.ts +11 -12
  157. package/dist/adapter/ezsp/driver/parser.d.ts.map +1 -1
  158. package/dist/adapter/ezsp/driver/parser.js +104 -111
  159. package/dist/adapter/ezsp/driver/parser.js.map +1 -1
  160. package/dist/adapter/ezsp/driver/types/basic.d.ts +62 -62
  161. package/dist/adapter/ezsp/driver/types/basic.js +208 -208
  162. package/dist/adapter/ezsp/driver/types/basic.js.map +1 -1
  163. package/dist/adapter/ezsp/driver/types/index.d.ts +9 -9
  164. package/dist/adapter/ezsp/driver/types/index.d.ts.map +1 -1
  165. package/dist/adapter/ezsp/driver/types/index.js +138 -133
  166. package/dist/adapter/ezsp/driver/types/index.js.map +1 -1
  167. package/dist/adapter/ezsp/driver/types/named.d.ts +1287 -697
  168. package/dist/adapter/ezsp/driver/types/named.d.ts.map +1 -1
  169. package/dist/adapter/ezsp/driver/types/named.js +2329 -1726
  170. package/dist/adapter/ezsp/driver/types/named.js.map +1 -1
  171. package/dist/adapter/ezsp/driver/types/struct.d.ts +270 -251
  172. package/dist/adapter/ezsp/driver/types/struct.d.ts.map +1 -1
  173. package/dist/adapter/ezsp/driver/types/struct.js +803 -708
  174. package/dist/adapter/ezsp/driver/types/struct.js.map +1 -1
  175. package/dist/adapter/ezsp/driver/uart.d.ts +48 -44
  176. package/dist/adapter/ezsp/driver/uart.d.ts.map +1 -1
  177. package/dist/adapter/ezsp/driver/uart.js +382 -368
  178. package/dist/adapter/ezsp/driver/uart.js.map +1 -1
  179. package/dist/adapter/ezsp/driver/utils/crc16ccitt.d.ts +2 -2
  180. package/dist/adapter/ezsp/driver/utils/crc16ccitt.js +55 -55
  181. package/dist/adapter/ezsp/driver/utils/crc16ccitt.js.map +1 -1
  182. package/dist/adapter/ezsp/driver/utils/index.d.ts +18 -18
  183. package/dist/adapter/ezsp/driver/utils/index.js +72 -67
  184. package/dist/adapter/ezsp/driver/utils/index.js.map +1 -1
  185. package/dist/adapter/ezsp/driver/writer.d.ts +13 -13
  186. package/dist/adapter/ezsp/driver/writer.d.ts.map +1 -1
  187. package/dist/adapter/ezsp/driver/writer.js +85 -88
  188. package/dist/adapter/ezsp/driver/writer.js.map +1 -1
  189. package/dist/adapter/index.d.ts +4 -4
  190. package/dist/adapter/index.js +35 -35
  191. package/dist/adapter/serialPort.d.ts +10 -8
  192. package/dist/adapter/serialPort.d.ts.map +1 -1
  193. package/dist/adapter/serialPort.js +53 -22
  194. package/dist/adapter/serialPort.js.map +1 -1
  195. package/dist/adapter/serialPortUtils.d.ts +12 -12
  196. package/dist/adapter/serialPortUtils.js +18 -18
  197. package/dist/adapter/serialPortUtils.js.map +1 -1
  198. package/dist/adapter/socketPortUtils.d.ts +10 -10
  199. package/dist/adapter/socketPortUtils.js +16 -16
  200. package/dist/adapter/tstype.d.ts +85 -85
  201. package/dist/adapter/tstype.d.ts.map +1 -1
  202. package/dist/adapter/tstype.js +2 -2
  203. package/dist/adapter/z-stack/adapter/adapter-backup.d.ts +62 -62
  204. package/dist/adapter/z-stack/adapter/adapter-backup.d.ts.map +1 -1
  205. package/dist/adapter/z-stack/adapter/adapter-backup.js +462 -460
  206. package/dist/adapter/z-stack/adapter/adapter-backup.js.map +1 -1
  207. package/dist/adapter/z-stack/adapter/adapter-nv-memory.d.ts +150 -150
  208. package/dist/adapter/z-stack/adapter/adapter-nv-memory.js +258 -258
  209. package/dist/adapter/z-stack/adapter/adapter-nv-memory.js.map +1 -1
  210. package/dist/adapter/z-stack/adapter/endpoints.d.ts +11 -11
  211. package/dist/adapter/z-stack/adapter/endpoints.js +73 -73
  212. package/dist/adapter/z-stack/adapter/index.d.ts +2 -2
  213. package/dist/adapter/z-stack/adapter/index.js +8 -8
  214. package/dist/adapter/z-stack/adapter/manager.d.ts +86 -86
  215. package/dist/adapter/z-stack/adapter/manager.d.ts.map +1 -1
  216. package/dist/adapter/z-stack/adapter/manager.js +482 -476
  217. package/dist/adapter/z-stack/adapter/manager.js.map +1 -1
  218. package/dist/adapter/z-stack/adapter/tstype.d.ts +6 -6
  219. package/dist/adapter/z-stack/adapter/tstype.js +9 -10
  220. package/dist/adapter/z-stack/adapter/tstype.js.map +1 -1
  221. package/dist/adapter/z-stack/adapter/zStackAdapter.d.ts +81 -81
  222. package/dist/adapter/z-stack/adapter/zStackAdapter.d.ts.map +1 -1
  223. package/dist/adapter/z-stack/adapter/zStackAdapter.js +891 -868
  224. package/dist/adapter/z-stack/adapter/zStackAdapter.js.map +1 -1
  225. package/dist/adapter/z-stack/constants/af.d.ts +23 -23
  226. package/dist/adapter/z-stack/constants/af.js +27 -27
  227. package/dist/adapter/z-stack/constants/common.d.ts +278 -278
  228. package/dist/adapter/z-stack/constants/common.d.ts.map +1 -1
  229. package/dist/adapter/z-stack/constants/common.js +292 -289
  230. package/dist/adapter/z-stack/constants/common.js.map +1 -1
  231. package/dist/adapter/z-stack/constants/dbg.d.ts +22 -22
  232. package/dist/adapter/z-stack/constants/dbg.js +24 -24
  233. package/dist/adapter/z-stack/constants/index.d.ts +10 -10
  234. package/dist/adapter/z-stack/constants/index.js +47 -47
  235. package/dist/adapter/z-stack/constants/mac.d.ts +127 -127
  236. package/dist/adapter/z-stack/constants/mac.js +129 -129
  237. package/dist/adapter/z-stack/constants/sapi.d.ts +24 -24
  238. package/dist/adapter/z-stack/constants/sapi.js +26 -26
  239. package/dist/adapter/z-stack/constants/sys.d.ts +71 -71
  240. package/dist/adapter/z-stack/constants/sys.js +73 -73
  241. package/dist/adapter/z-stack/constants/util.d.ts +81 -81
  242. package/dist/adapter/z-stack/constants/util.js +83 -83
  243. package/dist/adapter/z-stack/constants/utils.d.ts +4 -4
  244. package/dist/adapter/z-stack/constants/utils.js +14 -14
  245. package/dist/adapter/z-stack/constants/zdo.d.ts +102 -102
  246. package/dist/adapter/z-stack/constants/zdo.js +104 -104
  247. package/dist/adapter/z-stack/models/index.d.ts +1 -1
  248. package/dist/adapter/z-stack/models/index.js +17 -17
  249. package/dist/adapter/z-stack/models/startup-options.d.ts +12 -12
  250. package/dist/adapter/z-stack/models/startup-options.js +2 -2
  251. package/dist/adapter/z-stack/structs/entries/address-manager-entry.d.ts +23 -23
  252. package/dist/adapter/z-stack/structs/entries/address-manager-entry.js +45 -45
  253. package/dist/adapter/z-stack/structs/entries/address-manager-entry.js.map +1 -1
  254. package/dist/adapter/z-stack/structs/entries/address-manager-table.d.ts +10 -10
  255. package/dist/adapter/z-stack/structs/entries/address-manager-table.js +22 -22
  256. package/dist/adapter/z-stack/structs/entries/aps-link-key-data-entry.d.ts +10 -10
  257. package/dist/adapter/z-stack/structs/entries/aps-link-key-data-entry.js +21 -21
  258. package/dist/adapter/z-stack/structs/entries/aps-link-key-data-table.d.ts +10 -10
  259. package/dist/adapter/z-stack/structs/entries/aps-link-key-data-table.js +23 -23
  260. package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-entry.d.ts +10 -10
  261. package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-entry.js +24 -24
  262. package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-table.d.ts +10 -10
  263. package/dist/adapter/z-stack/structs/entries/aps-tc-link-key-table.js +23 -23
  264. package/dist/adapter/z-stack/structs/entries/channel-list.d.ts +8 -8
  265. package/dist/adapter/z-stack/structs/entries/channel-list.js +15 -15
  266. package/dist/adapter/z-stack/structs/entries/has-configured.d.ts +8 -8
  267. package/dist/adapter/z-stack/structs/entries/has-configured.js +16 -16
  268. package/dist/adapter/z-stack/structs/entries/index.d.ts +16 -16
  269. package/dist/adapter/z-stack/structs/entries/index.js +32 -32
  270. package/dist/adapter/z-stack/structs/entries/nib.d.ts +10 -10
  271. package/dist/adapter/z-stack/structs/entries/nib.js +68 -68
  272. package/dist/adapter/z-stack/structs/entries/nwk-key-descriptor.d.ts +10 -10
  273. package/dist/adapter/z-stack/structs/entries/nwk-key-descriptor.js +18 -18
  274. package/dist/adapter/z-stack/structs/entries/nwk-key.d.ts +8 -8
  275. package/dist/adapter/z-stack/structs/entries/nwk-key.js +15 -15
  276. package/dist/adapter/z-stack/structs/entries/nwk-pan-id.d.ts +8 -8
  277. package/dist/adapter/z-stack/structs/entries/nwk-pan-id.js +15 -15
  278. package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-entry.d.ts +13 -13
  279. package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-entry.js +23 -23
  280. package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-table.d.ts +10 -10
  281. package/dist/adapter/z-stack/structs/entries/nwk-sec-material-descriptor-table.js +22 -22
  282. package/dist/adapter/z-stack/structs/entries/security-manager-entry.d.ts +20 -20
  283. package/dist/adapter/z-stack/structs/entries/security-manager-entry.js +36 -36
  284. package/dist/adapter/z-stack/structs/entries/security-manager-entry.js.map +1 -1
  285. package/dist/adapter/z-stack/structs/entries/security-manager-table.d.ts +10 -10
  286. package/dist/adapter/z-stack/structs/entries/security-manager-table.js +24 -24
  287. package/dist/adapter/z-stack/structs/index.d.ts +4 -4
  288. package/dist/adapter/z-stack/structs/index.js +20 -20
  289. package/dist/adapter/z-stack/structs/serializable-memory-object.d.ts +13 -13
  290. package/dist/adapter/z-stack/structs/serializable-memory-object.js +2 -2
  291. package/dist/adapter/z-stack/structs/struct.d.ts +99 -99
  292. package/dist/adapter/z-stack/structs/struct.js +296 -295
  293. package/dist/adapter/z-stack/structs/struct.js.map +1 -1
  294. package/dist/adapter/z-stack/structs/table.d.ts +94 -94
  295. package/dist/adapter/z-stack/structs/table.js +163 -161
  296. package/dist/adapter/z-stack/structs/table.js.map +1 -1
  297. package/dist/adapter/z-stack/unpi/constants.d.ts +28 -28
  298. package/dist/adapter/z-stack/unpi/constants.js +39 -41
  299. package/dist/adapter/z-stack/unpi/constants.js.map +1 -1
  300. package/dist/adapter/z-stack/unpi/frame.d.ts +16 -16
  301. package/dist/adapter/z-stack/unpi/frame.js +54 -48
  302. package/dist/adapter/z-stack/unpi/frame.js.map +1 -1
  303. package/dist/adapter/z-stack/unpi/index.d.ts +5 -5
  304. package/dist/adapter/z-stack/unpi/index.js +37 -37
  305. package/dist/adapter/z-stack/unpi/parser.d.ts +10 -10
  306. package/dist/adapter/z-stack/unpi/parser.js +75 -74
  307. package/dist/adapter/z-stack/unpi/parser.js.map +1 -1
  308. package/dist/adapter/z-stack/unpi/writer.d.ts +10 -10
  309. package/dist/adapter/z-stack/unpi/writer.js +44 -44
  310. package/dist/adapter/z-stack/utils/channel-list.d.ts +20 -20
  311. package/dist/adapter/z-stack/utils/channel-list.js +40 -40
  312. package/dist/adapter/z-stack/utils/channel-list.js.map +1 -1
  313. package/dist/adapter/z-stack/utils/index.d.ts +2 -2
  314. package/dist/adapter/z-stack/utils/index.js +18 -18
  315. package/dist/adapter/z-stack/utils/network-options.d.ts +8 -8
  316. package/dist/adapter/z-stack/utils/network-options.js +22 -22
  317. package/dist/adapter/z-stack/znp/buffaloZnp.d.ts +11 -11
  318. package/dist/adapter/z-stack/znp/buffaloZnp.js +113 -113
  319. package/dist/adapter/z-stack/znp/buffaloZnp.js.map +1 -1
  320. package/dist/adapter/z-stack/znp/definition.d.ts +5 -5
  321. package/dist/adapter/z-stack/znp/definition.js +3050 -3050
  322. package/dist/adapter/z-stack/znp/index.d.ts +3 -3
  323. package/dist/adapter/z-stack/znp/index.js +10 -10
  324. package/dist/adapter/z-stack/znp/parameterType.d.ts +22 -22
  325. package/dist/adapter/z-stack/znp/parameterType.js +25 -25
  326. package/dist/adapter/z-stack/znp/tstype.d.ts +21 -21
  327. package/dist/adapter/z-stack/znp/tstype.js +2 -2
  328. package/dist/adapter/z-stack/znp/znp.d.ts +44 -43
  329. package/dist/adapter/z-stack/znp/znp.d.ts.map +1 -1
  330. package/dist/adapter/z-stack/znp/znp.js +326 -325
  331. package/dist/adapter/z-stack/znp/znp.js.map +1 -1
  332. package/dist/adapter/z-stack/znp/zpiObject.d.ts +19 -19
  333. package/dist/adapter/z-stack/znp/zpiObject.js +102 -96
  334. package/dist/adapter/z-stack/znp/zpiObject.js.map +1 -1
  335. package/dist/adapter/zigate/adapter/index.d.ts +2 -2
  336. package/dist/adapter/zigate/adapter/index.js +10 -10
  337. package/dist/adapter/zigate/adapter/zigateAdapter.d.ts +70 -69
  338. package/dist/adapter/zigate/adapter/zigateAdapter.d.ts.map +1 -1
  339. package/dist/adapter/zigate/adapter/zigateAdapter.js +689 -678
  340. package/dist/adapter/zigate/adapter/zigateAdapter.js.map +1 -1
  341. package/dist/adapter/zigate/debug.d.ts +7 -7
  342. package/dist/adapter/zigate/debug.d.ts.map +1 -1
  343. package/dist/adapter/zigate/debug.js +19 -22
  344. package/dist/adapter/zigate/debug.js.map +1 -1
  345. package/dist/adapter/zigate/driver/buffaloZiGate.d.ts +18 -18
  346. package/dist/adapter/zigate/driver/buffaloZiGate.js +139 -139
  347. package/dist/adapter/zigate/driver/buffaloZiGate.js.map +1 -1
  348. package/dist/adapter/zigate/driver/commandType.d.ts +41 -41
  349. package/dist/adapter/zigate/driver/commandType.js +385 -385
  350. package/dist/adapter/zigate/driver/commandType.js.map +1 -1
  351. package/dist/adapter/zigate/driver/constants.d.ts +276 -276
  352. package/dist/adapter/zigate/driver/constants.d.ts.map +1 -1
  353. package/dist/adapter/zigate/driver/constants.js +371 -371
  354. package/dist/adapter/zigate/driver/constants.js.map +1 -1
  355. package/dist/adapter/zigate/driver/frame.d.ts +26 -26
  356. package/dist/adapter/zigate/driver/frame.js +172 -172
  357. package/dist/adapter/zigate/driver/frame.js.map +1 -1
  358. package/dist/adapter/zigate/driver/messageType.d.ts +11 -11
  359. package/dist/adapter/zigate/driver/messageType.js +278 -278
  360. package/dist/adapter/zigate/driver/messageType.js.map +1 -1
  361. package/dist/adapter/zigate/driver/parameterType.d.ts +20 -20
  362. package/dist/adapter/zigate/driver/parameterType.js +23 -23
  363. package/dist/adapter/zigate/driver/ziGateObject.d.ts +23 -23
  364. package/dist/adapter/zigate/driver/ziGateObject.js +110 -106
  365. package/dist/adapter/zigate/driver/ziGateObject.js.map +1 -1
  366. package/dist/adapter/zigate/driver/zigate.d.ts +49 -49
  367. package/dist/adapter/zigate/driver/zigate.d.ts.map +1 -1
  368. package/dist/adapter/zigate/driver/zigate.js +296 -303
  369. package/dist/adapter/zigate/driver/zigate.js.map +1 -1
  370. package/dist/buffalo/buffalo.d.ts +50 -50
  371. package/dist/buffalo/buffalo.js +324 -322
  372. package/dist/buffalo/buffalo.js.map +1 -1
  373. package/dist/buffalo/index.d.ts +3 -3
  374. package/dist/buffalo/index.js +33 -33
  375. package/dist/buffalo/tstype.d.ts +8 -8
  376. package/dist/buffalo/tstype.js +2 -2
  377. package/dist/controller/controller.d.ts +113 -113
  378. package/dist/controller/controller.d.ts.map +1 -1
  379. package/dist/controller/controller.js +641 -619
  380. package/dist/controller/controller.js.map +1 -1
  381. package/dist/controller/database.d.ts +18 -18
  382. package/dist/controller/database.js +96 -93
  383. package/dist/controller/database.js.map +1 -1
  384. package/dist/controller/events.d.ts +58 -58
  385. package/dist/controller/events.d.ts.map +1 -1
  386. package/dist/controller/events.js +108 -101
  387. package/dist/controller/events.js.map +1 -1
  388. package/dist/controller/greenPower.d.ts +12 -12
  389. package/dist/controller/greenPower.js +221 -220
  390. package/dist/controller/greenPower.js.map +1 -1
  391. package/dist/controller/helpers/index.d.ts +2 -2
  392. package/dist/controller/helpers/index.js +28 -28
  393. package/dist/controller/helpers/request.d.ts +21 -22
  394. package/dist/controller/helpers/request.d.ts.map +1 -1
  395. package/dist/controller/helpers/request.js +77 -71
  396. package/dist/controller/helpers/request.js.map +1 -1
  397. package/dist/controller/helpers/requestQueue.d.ts +13 -0
  398. package/dist/controller/helpers/requestQueue.d.ts.map +1 -0
  399. package/dist/controller/helpers/requestQueue.js +116 -0
  400. package/dist/controller/helpers/requestQueue.js.map +1 -0
  401. package/dist/controller/helpers/zclFrameConverter.d.ts +7 -7
  402. package/dist/controller/helpers/zclFrameConverter.d.ts.map +1 -1
  403. package/dist/controller/helpers/zclFrameConverter.js +50 -31
  404. package/dist/controller/helpers/zclFrameConverter.js.map +1 -1
  405. package/dist/controller/helpers/zclTransactionSequenceNumber.d.ts +5 -5
  406. package/dist/controller/helpers/zclTransactionSequenceNumber.js +13 -13
  407. package/dist/controller/helpers/zclTransactionSequenceNumber.js.map +1 -1
  408. package/dist/controller/index.d.ts +5 -5
  409. package/dist/controller/index.js +8 -8
  410. package/dist/controller/logger-stub.d.ts +6 -6
  411. package/dist/controller/logger-stub.js +2 -2
  412. package/dist/controller/model/device.d.ts +132 -133
  413. package/dist/controller/model/device.d.ts.map +1 -1
  414. package/dist/controller/model/device.js +724 -717
  415. package/dist/controller/model/device.js.map +1 -1
  416. package/dist/controller/model/endpoint.d.ts +128 -131
  417. package/dist/controller/model/endpoint.d.ts.map +1 -1
  418. package/dist/controller/model/endpoint.js +755 -816
  419. package/dist/controller/model/endpoint.js.map +1 -1
  420. package/dist/controller/model/entity.d.ts +14 -14
  421. package/dist/controller/model/entity.js +26 -26
  422. package/dist/controller/model/entity.js.map +1 -1
  423. package/dist/controller/model/group.d.ts +38 -38
  424. package/dist/controller/model/group.d.ts.map +1 -1
  425. package/dist/controller/model/group.js +225 -221
  426. package/dist/controller/model/group.js.map +1 -1
  427. package/dist/controller/model/index.d.ts +5 -5
  428. package/dist/controller/model/index.js +14 -14
  429. package/dist/controller/touchlink.d.ts +19 -19
  430. package/dist/controller/touchlink.js +159 -157
  431. package/dist/controller/touchlink.js.map +1 -1
  432. package/dist/controller/tstype.d.ts +20 -21
  433. package/dist/controller/tstype.d.ts.map +1 -1
  434. package/dist/controller/tstype.js +8 -9
  435. package/dist/controller/tstype.js.map +1 -1
  436. package/dist/index.d.ts +3 -3
  437. package/dist/index.js +33 -33
  438. package/dist/models/backup-storage-legacy.d.ts +26 -26
  439. package/dist/models/backup-storage-legacy.js +2 -2
  440. package/dist/models/backup-storage-unified.d.ts +49 -49
  441. package/dist/models/backup-storage-unified.js +2 -2
  442. package/dist/models/backup.d.ts +37 -37
  443. package/dist/models/backup.js +2 -2
  444. package/dist/models/index.d.ts +4 -4
  445. package/dist/models/index.js +20 -20
  446. package/dist/models/network-options.d.ts +12 -12
  447. package/dist/models/network-options.js +2 -2
  448. package/dist/utils/assertString.d.ts +2 -2
  449. package/dist/utils/assertString.js +8 -8
  450. package/dist/utils/assertString.js.map +1 -1
  451. package/dist/utils/backup.d.ts +20 -20
  452. package/dist/utils/backup.d.ts.map +1 -1
  453. package/dist/utils/backup.js +189 -187
  454. package/dist/utils/backup.js.map +1 -1
  455. package/dist/utils/equalsPartial.d.ts +2 -2
  456. package/dist/utils/equalsPartial.js +11 -11
  457. package/dist/utils/index.d.ts +9 -9
  458. package/dist/utils/index.js +45 -45
  459. package/dist/utils/isNumberArray.d.ts +2 -2
  460. package/dist/utils/isNumberArray.js +6 -6
  461. package/dist/utils/queue.d.ts +11 -11
  462. package/dist/utils/queue.d.ts.map +1 -1
  463. package/dist/utils/queue.js +61 -50
  464. package/dist/utils/queue.js.map +1 -1
  465. package/dist/utils/realpathSync.d.ts +2 -2
  466. package/dist/utils/realpathSync.js +12 -12
  467. package/dist/utils/wait.d.ts +2 -2
  468. package/dist/utils/wait.js +8 -8
  469. package/dist/utils/waitress.d.ts +21 -21
  470. package/dist/utils/waitress.d.ts.map +1 -1
  471. package/dist/utils/waitress.js +68 -61
  472. package/dist/utils/waitress.js.map +1 -1
  473. package/dist/zcl/buffaloZcl.d.ts +41 -41
  474. package/dist/zcl/buffaloZcl.d.ts.map +1 -1
  475. package/dist/zcl/buffaloZcl.js +594 -591
  476. package/dist/zcl/buffaloZcl.js.map +1 -1
  477. package/dist/zcl/definition/buffaloZclDataType.d.ts +17 -17
  478. package/dist/zcl/definition/buffaloZclDataType.js +20 -20
  479. package/dist/zcl/definition/cluster.d.ts +29 -29
  480. package/dist/zcl/definition/cluster.d.ts.map +1 -1
  481. package/dist/zcl/definition/cluster.js +5520 -5335
  482. package/dist/zcl/definition/cluster.js.map +1 -1
  483. package/dist/zcl/definition/dataType.d.ts +59 -59
  484. package/dist/zcl/definition/dataType.js +64 -64
  485. package/dist/zcl/definition/direction.d.ts +5 -5
  486. package/dist/zcl/definition/direction.js +8 -8
  487. package/dist/zcl/definition/endpointDeviceType.d.ts +4 -4
  488. package/dist/zcl/definition/endpointDeviceType.js +15 -15
  489. package/dist/zcl/definition/foundation.d.ts +11 -11
  490. package/dist/zcl/definition/foundation.js +167 -167
  491. package/dist/zcl/definition/frameControl.d.ts +10 -10
  492. package/dist/zcl/definition/frameControl.js +2 -2
  493. package/dist/zcl/definition/frameType.d.ts +5 -5
  494. package/dist/zcl/definition/frameType.js +8 -8
  495. package/dist/zcl/definition/index.d.ts +13 -13
  496. package/dist/zcl/definition/index.js +51 -51
  497. package/dist/zcl/definition/manufacturerCode.d.ts +1077 -1074
  498. package/dist/zcl/definition/manufacturerCode.d.ts.map +1 -1
  499. package/dist/zcl/definition/manufacturerCode.js +1082 -1079
  500. package/dist/zcl/definition/manufacturerCode.js.map +1 -1
  501. package/dist/zcl/definition/powerSource.d.ts +4 -4
  502. package/dist/zcl/definition/powerSource.js +12 -12
  503. package/dist/zcl/definition/status.d.ts +38 -38
  504. package/dist/zcl/definition/status.js +41 -41
  505. package/dist/zcl/definition/tstype.d.ts +16 -16
  506. package/dist/zcl/definition/tstype.js +2 -2
  507. package/dist/zcl/index.d.ts +16 -15
  508. package/dist/zcl/index.d.ts.map +1 -1
  509. package/dist/zcl/index.js +55 -55
  510. package/dist/zcl/index.js.map +1 -1
  511. package/dist/zcl/tstype.d.ts +56 -56
  512. package/dist/zcl/tstype.js +9 -10
  513. package/dist/zcl/tstype.js.map +1 -1
  514. package/dist/zcl/utils.d.ts +6 -6
  515. package/dist/zcl/utils.js +164 -165
  516. package/dist/zcl/utils.js.map +1 -1
  517. package/dist/zcl/zclFrame.d.ts +40 -45
  518. package/dist/zcl/zclFrame.d.ts.map +1 -1
  519. package/dist/zcl/zclFrame.js +351 -347
  520. package/dist/zcl/zclFrame.js.map +1 -1
  521. package/dist/zcl/zclHeader.d.ts +9 -0
  522. package/dist/zcl/zclHeader.d.ts.map +1 -0
  523. package/dist/zcl/zclHeader.js +3 -0
  524. package/dist/zcl/zclHeader.js.map +1 -0
  525. package/dist/zcl/zclStatusError.d.ts +5 -5
  526. package/dist/zcl/zclStatusError.js +14 -13
  527. package/dist/zcl/zclStatusError.js.map +1 -1
  528. package/package.json +11 -11
  529. package/release-please-config.json +1 -5
  530. package/tsconfig.json +4 -2
@@ -1,718 +1,725 @@
1
- "use strict";
2
- var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
- if (k2 === undefined) k2 = k;
4
- var desc = Object.getOwnPropertyDescriptor(m, k);
5
- if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
- desc = { enumerable: true, get: function() { return m[k]; } };
7
- }
8
- Object.defineProperty(o, k2, desc);
9
- }) : (function(o, m, k, k2) {
10
- if (k2 === undefined) k2 = k;
11
- o[k2] = m[k];
12
- }));
13
- var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
- Object.defineProperty(o, "default", { enumerable: true, value: v });
15
- }) : function(o, v) {
16
- o["default"] = v;
17
- });
18
- var __importStar = (this && this.__importStar) || function (mod) {
19
- if (mod && mod.__esModule) return mod;
20
- var result = {};
21
- if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
- __setModuleDefault(result, mod);
23
- return result;
24
- };
25
- var __importDefault = (this && this.__importDefault) || function (mod) {
26
- return (mod && mod.__esModule) ? mod : { "default": mod };
27
- };
28
- Object.defineProperty(exports, "__esModule", { value: true });
29
- const zclTransactionSequenceNumber_1 = __importDefault(require("../helpers/zclTransactionSequenceNumber"));
30
- const endpoint_1 = __importDefault(require("./endpoint"));
31
- const entity_1 = __importDefault(require("./entity"));
32
- const utils_1 = require("../../utils");
33
- const debug_1 = __importDefault(require("debug"));
34
- const Zcl = __importStar(require("../../zcl"));
35
- const assert_1 = __importDefault(require("assert"));
36
- const helpers_1 = require("../helpers");
37
- /**
38
- * @ignore
39
- */
40
- const OneJanuary2000 = new Date('January 01, 2000 00:00:00 UTC+00:00').getTime();
41
- const debug = {
42
- error: (0, debug_1.default)('zigbee-herdsman:controller:device:error'),
43
- log: (0, debug_1.default)('zigbee-herdsman:controller:device:log'),
44
- };
45
- class Device extends entity_1.default {
46
- // Getters/setters
47
- get ieeeAddr() { return this._ieeeAddr; }
48
- set ieeeAddr(ieeeAddr) { this._ieeeAddr = ieeeAddr; }
49
- get applicationVersion() { return this._applicationVersion; }
50
- set applicationVersion(applicationVersion) { this._applicationVersion = applicationVersion; }
51
- get endpoints() { return this._endpoints; }
52
- get interviewCompleted() { return this._interviewCompleted; }
53
- get interviewing() { return this._interviewing; }
54
- get lastSeen() { return this._lastSeen; }
55
- get manufacturerID() { return this._manufacturerID; }
56
- get isDeleted() { return this._deleted; }
57
- set type(type) { this._type = type; }
58
- get type() { return this._type; }
59
- get dateCode() { return this._dateCode; }
60
- set dateCode(dateCode) { this._dateCode = dateCode; }
61
- set hardwareVersion(hardwareVersion) { this._hardwareVersion = hardwareVersion; }
62
- get hardwareVersion() { return this._hardwareVersion; }
63
- get manufacturerName() { return this._manufacturerName; }
64
- set manufacturerName(manufacturerName) { this._manufacturerName = manufacturerName; }
65
- set modelID(modelID) { this._modelID = modelID; }
66
- get modelID() { return this._modelID; }
67
- get networkAddress() { return this._networkAddress; }
68
- set networkAddress(networkAddress) {
69
- this._networkAddress = networkAddress;
70
- for (const endpoint of this._endpoints) {
71
- endpoint.deviceNetworkAddress = networkAddress;
72
- }
73
- }
74
- get powerSource() { return this._powerSource; }
75
- set powerSource(powerSource) {
76
- this._powerSource = typeof powerSource === 'number' ? Zcl.PowerSource[powerSource & ~(1 << 7)] : powerSource;
77
- }
78
- get softwareBuildID() { return this._softwareBuildID; }
79
- set softwareBuildID(softwareBuildID) { this._softwareBuildID = softwareBuildID; }
80
- get stackVersion() { return this._stackVersion; }
81
- set stackVersion(stackVersion) { this._stackVersion = stackVersion; }
82
- get zclVersion() { return this._zclVersion; }
83
- set zclVersion(zclVersion) { this._zclVersion = zclVersion; }
84
- get linkquality() { return this._linkquality; }
85
- set linkquality(linkquality) { this._linkquality = linkquality; }
86
- get skipDefaultResponse() { return this._skipDefaultResponse; }
87
- set skipDefaultResponse(skipDefaultResponse) { this._skipDefaultResponse = skipDefaultResponse; }
88
- get skipTimeResponse() { return this._skipTimeResponse; }
89
- set skipTimeResponse(skipTimeResponse) { this._skipTimeResponse = skipTimeResponse; }
90
- get defaultSendRequestWhen() { return this._defaultSendRequestWhen; }
91
- set defaultSendRequestWhen(defaultSendRequestWhen) {
92
- this._defaultSendRequestWhen = defaultSendRequestWhen;
93
- }
94
- get checkinInterval() { return this._checkinInterval; }
95
- get pendingRequestTimeout() { return this._pendingRequestTimeout; }
96
- set pendingRequestTimeout(pendingRequestTimeout) { this._pendingRequestTimeout = pendingRequestTimeout; }
97
- constructor(databaseID, ID, type, ieeeAddr, networkAddress, manufacturerID, endpoints, manufacturerName, powerSource, modelID, applicationVersion, stackVersion, zclVersion, hardwareVersion, dateCode, softwareBuildID, interviewCompleted, meta, lastSeen, defaultSendRequestWhen, checkinInterval, pendingRequestTimeout) {
98
- super();
99
- this.databaseID = databaseID;
100
- this.ID = ID;
101
- this._type = type;
102
- this.ieeeAddr = ieeeAddr;
103
- this._networkAddress = networkAddress;
104
- this._manufacturerID = manufacturerID;
105
- this._endpoints = endpoints;
106
- this._manufacturerName = manufacturerName;
107
- this._powerSource = powerSource;
108
- this._modelID = modelID;
109
- this._applicationVersion = applicationVersion;
110
- this._stackVersion = stackVersion;
111
- this._zclVersion = zclVersion;
112
- this.hardwareVersion = hardwareVersion;
113
- this._dateCode = dateCode;
114
- this._softwareBuildID = softwareBuildID;
115
- this._interviewCompleted = interviewCompleted;
116
- this._interviewing = false;
117
- this._skipDefaultResponse = false;
118
- this._skipTimeResponse = false;
119
- this.meta = meta;
120
- this._lastSeen = lastSeen;
121
- this._defaultSendRequestWhen = defaultSendRequestWhen;
122
- this._checkinInterval = checkinInterval;
123
- this._pendingRequestTimeout = pendingRequestTimeout;
124
- }
125
- createEndpoint(ID) {
126
- if (this.getEndpoint(ID)) {
127
- throw new Error(`Device '${this.ieeeAddr}' already has an endpoint '${ID}'`);
128
- }
129
- const endpoint = endpoint_1.default.create(this.databaseID, ID, undefined, undefined, [], [], this.networkAddress, this.ieeeAddr);
130
- this.endpoints.push(endpoint);
131
- this.save();
132
- return endpoint;
133
- }
134
- changeIeeeAddress(ieeeAddr) {
135
- delete Device.devices.get(this.databaseID)[this.ieeeAddr];
136
- this.ieeeAddr = ieeeAddr;
137
- Device.devices.get(this.databaseID)[this.ieeeAddr] = this;
138
- this.endpoints.forEach((e) => e.deviceIeeeAddress = ieeeAddr);
139
- this.save();
140
- }
141
- getEndpoint(ID) {
142
- return this.endpoints.find((e) => e.ID === ID);
143
- }
144
- // There might be multiple endpoints with same DeviceId but it is not supported and first endpoint is returned
145
- getEndpointByDeviceType(deviceType) {
146
- const deviceID = Zcl.EndpointDeviceType[deviceType];
147
- return this.endpoints.find((d) => d.deviceID === deviceID);
148
- }
149
- implicitCheckin() {
150
- this.endpoints.forEach(async (e) => e.sendPendingRequests(false));
151
- }
152
- updateLastSeen() {
153
- this._lastSeen = Date.now();
154
- }
155
- hasPendingRequests() {
156
- return this.endpoints.find(e => e.hasPendingRequests()) !== undefined;
157
- }
158
- async onZclData(dataPayload, endpoint) {
159
- var _a;
160
- const frame = dataPayload.frame;
161
- // Update reportable properties
162
- if (frame.isCluster('genBasic') && (frame.isCommand('readRsp') || frame.isCommand('report'))) {
163
- for (const [key, value] of Object.entries(helpers_1.ZclFrameConverter.attributeKeyValue(frame))) {
164
- (_a = Device.ReportablePropertiesMapping[key]) === null || _a === void 0 ? void 0 : _a.set(value, this);
165
- }
166
- }
167
- // Respond to enroll requests
168
- if (frame.isSpecific() && frame.isCluster('ssIasZone') && frame.isCommand('enrollReq')) {
169
- debug.log(`IAS - '${this.ieeeAddr}' responding to enroll response`);
170
- const payload = { enrollrspcode: 0, zoneid: 23 };
171
- await endpoint.command('ssIasZone', 'enrollRsp', payload, { disableDefaultResponse: true });
172
- }
173
- // Reponse to read requests
174
- if (frame.isGlobal() && frame.isCommand('read')) {
175
- const time = Math.round(((new Date()).getTime() - OneJanuary2000) / 1000);
176
- const attributes = {
177
- ...endpoint.clusters,
178
- genTime: { attributes: {
179
- timeStatus: 3,
180
- time: time,
181
- timeZone: ((new Date()).getTimezoneOffset() * -1) * 60,
182
- localTime: time - (new Date()).getTimezoneOffset() * 60,
183
- lastSetTime: time,
184
- validUntilTime: time + (24 * 60 * 60), // valid for 24 hours
185
- } },
186
- };
187
- if (frame.Cluster.name in attributes && (frame.Cluster.name !== 'genTime' || !this._skipTimeResponse)) {
188
- const response = {};
189
- for (const entry of frame.Payload) {
190
- if (frame.Cluster.hasAttribute(entry.attrId)) {
191
- const name = frame.Cluster.getAttribute(entry.attrId).name;
192
- if (name in attributes[frame.Cluster.name].attributes) {
193
- response[name] = attributes[frame.Cluster.name].attributes[name];
194
- }
195
- }
196
- }
197
- try {
198
- await endpoint.readResponse(frame.Cluster.ID, frame.Header.transactionSequenceNumber, response, { srcEndpoint: dataPayload.destinationEndpoint });
199
- }
200
- catch (error) {
201
- debug.error(`Read response to ${this.ieeeAddr} failed`);
202
- }
203
- }
204
- }
205
- // Handle check-in from sleeping end devices
206
- if (frame.isSpecific() && frame.isCluster("genPollCtrl") && frame.isCommand("checkin")) {
207
- try {
208
- if (this.hasPendingRequests() || (this._checkinInterval === undefined)) {
209
- const payload = {
210
- startFastPolling: true,
211
- fastPollTimeout: 0,
212
- };
213
- debug.log(`check-in from ${this.ieeeAddr}: accepting fast-poll`);
214
- await endpoint.command(frame.Cluster.ID, 'checkinRsp', payload, { sendWhen: 'immediate' });
215
- // This is a good time to read the checkin interval if we haven't stored it previously
216
- if (this._checkinInterval === undefined) {
217
- const pollPeriod = await endpoint.read('genPollCtrl', ['checkinInterval'], { sendWhen: 'immediate' });
218
- this._checkinInterval = pollPeriod.checkinInterval / 4; // convert to seconds
219
- this.pendingRequestTimeout = this._checkinInterval * 1000; // milliseconds
220
- debug.log(`Request Queue (${this.ieeeAddr}): default expiration timeout set to ${this.pendingRequestTimeout}`);
221
- }
222
- await Promise.all(this.endpoints.map(async (e) => e.sendPendingRequests(true)));
223
- // We *must* end fast-poll when we're done sending things. Otherwise
224
- // we cause undue power-drain.
225
- debug.log(`check-in from ${this.ieeeAddr}: stopping fast-poll`);
226
- await endpoint.command(frame.Cluster.ID, 'fastPollStop', {}, { sendWhen: 'immediate' });
227
- }
228
- else {
229
- const payload = {
230
- startFastPolling: false,
231
- fastPollTimeout: 0,
232
- };
233
- debug.log(`check-in from ${this.ieeeAddr}: declining fast-poll`);
234
- await endpoint.command(frame.Cluster.ID, 'checkinRsp', payload, { sendWhen: 'immediate' });
235
- }
236
- }
237
- catch (error) {
238
- /* istanbul ignore next */
239
- debug.error(`Handling of poll check-in form ${this.ieeeAddr} failed`);
240
- }
241
- }
242
- // Send a default response if necessary.
243
- const isDefaultResponse = frame.isGlobal() && frame.getCommand().name === 'defaultRsp';
244
- const commandHasResponse = frame.getCommand().hasOwnProperty('response');
245
- const disableDefaultResponse = frame.Header.frameControl.disableDefaultResponse;
246
- // Sometimes messages are received twice, prevent responding twice
247
- const alreadyResponded = this._lastDefaultResponseSequenceNumber === frame.Header.transactionSequenceNumber;
248
- if (this.type !== 'GreenPower' && !dataPayload.wasBroadcast && !disableDefaultResponse && !isDefaultResponse &&
249
- !commandHasResponse && !this._skipDefaultResponse && !alreadyResponded) {
250
- try {
251
- this._lastDefaultResponseSequenceNumber = frame.Header.transactionSequenceNumber;
252
- // In the ZCL it is not documented what the direction of the default response should be
253
- // In https://github.com/Koenkk/zigbee2mqtt/issues/18096 a commandResponse (SERVER_TO_CLIENT)
254
- // is send and the device expects a CLIENT_TO_SERVER back.
255
- // Previously SERVER_TO_CLIENT was always used.
256
- // Therefore for non-global commands we inverse the direction.
257
- const direction = frame.isGlobal() ? Zcl.Direction.SERVER_TO_CLIENT : (frame.Header.frameControl.direction === Zcl.Direction.CLIENT_TO_SERVER
258
- ? Zcl.Direction.SERVER_TO_CLIENT : Zcl.Direction.CLIENT_TO_SERVER);
259
- await endpoint.defaultResponse(frame.getCommand().ID, 0, frame.Cluster.ID, frame.Header.transactionSequenceNumber, { direction });
260
- }
261
- catch (error) {
262
- debug.error(`Default response to ${this.ieeeAddr} failed`);
263
- }
264
- }
265
- }
266
- /*
267
- * CRUD
268
- */
269
- static fromDatabaseEntry(entry, databaseID) {
270
- const networkAddress = entry.nwkAddr;
271
- const ieeeAddr = entry.ieeeAddr;
272
- const endpoints = Object.values(entry.endpoints).map((e) => {
273
- return endpoint_1.default.fromDatabaseRecord(e, networkAddress, ieeeAddr, databaseID);
274
- });
275
- const meta = entry.meta ? entry.meta : {};
276
- if (entry.type === 'Group') {
277
- throw new Error('Cannot load device from group');
278
- }
279
- let defaultSendRequestWhen = entry.defaultSendRequestWhen;
280
- /* istanbul ignore next */
281
- if (defaultSendRequestWhen == null) {
282
- // Guess defaultSendRequestWhen based on old useImplicitCheckin/defaultSendWhenActive
283
- if (entry.hasOwnProperty('useImplicitCheckin') && !entry.useImplicitCheckin) {
284
- defaultSendRequestWhen = 'fastpoll';
285
- }
286
- else if (entry.hasOwnProperty('defaultSendWhenActive') && entry.defaultSendWhenActive) {
287
- defaultSendRequestWhen = 'active';
288
- }
289
- else {
290
- defaultSendRequestWhen = 'immediate';
291
- }
292
- }
293
- // default: no timeout (messages expire immediately after first send attempt)
294
- let pendingRequestTimeout = 0;
295
- if ((endpoints.filter((e) => e.supportsInputCluster('genPollCtrl'))).length > 0) {
296
- // default for devices that support genPollCtrl cluster (RX off when idle): 1 day
297
- pendingRequestTimeout = 86400000;
298
- /* istanbul ignore else */
299
- if (entry.hasOwnProperty('checkinInterval')) {
300
- // if the checkin interval is known, messages expire by default after one checkin interval
301
- pendingRequestTimeout = entry.checkinInterval * 1000; // milliseconds
302
- }
303
- }
304
- debug.log(`Request Queue (${ieeeAddr}): default expiration timeout set to ${pendingRequestTimeout}`);
305
- return new Device(databaseID, entry.id, entry.type, ieeeAddr, networkAddress, entry.manufId, endpoints, entry.manufName, entry.powerSource, entry.modelId, entry.appVersion, entry.stackVersion, entry.zclVersion, entry.hwVersion, entry.dateCode, entry.swBuildId, entry.interviewCompleted, meta, entry.lastSeen || null, defaultSendRequestWhen, entry.checkinInterval, pendingRequestTimeout);
306
- }
307
- toDatabaseEntry() {
308
- const epList = this.endpoints.map((e) => e.ID);
309
- const endpoints = {};
310
- for (const endpoint of this.endpoints) {
311
- endpoints[endpoint.ID] = endpoint.toDatabaseRecord();
312
- }
313
- return {
314
- id: this.ID, type: this.type, ieeeAddr: this.ieeeAddr, nwkAddr: this.networkAddress,
315
- manufId: this.manufacturerID, manufName: this.manufacturerName, powerSource: this.powerSource,
316
- modelId: this.modelID, epList, endpoints, appVersion: this.applicationVersion,
317
- stackVersion: this.stackVersion, hwVersion: this.hardwareVersion, dateCode: this.dateCode,
318
- swBuildId: this.softwareBuildID, zclVersion: this.zclVersion, interviewCompleted: this.interviewCompleted,
319
- meta: this.meta, lastSeen: this.lastSeen, defaultSendRequestWhen: this.defaultSendRequestWhen,
320
- checkinInterval: this.checkinInterval
321
- };
322
- }
323
- save(writeDatabase = true) {
324
- entity_1.default.getDatabaseByID(this.databaseID).update(this.toDatabaseEntry(), writeDatabase);
325
- }
326
- static loadFromDatabaseIfNecessary() {
327
- entity_1.default.databases.forEach(database => {
328
- if (!Device.devices.get(database.id)) {
329
- Device.devices.set(database.id, {});
330
- const entries = database.getEntries(['Coordinator', 'EndDevice', 'Router', 'GreenPower', 'Unknown']);
331
- for (const entry of entries) {
332
- const device = Device.fromDatabaseEntry(entry, database.id);
333
- Device.devices.get(database.id)[device.ieeeAddr] = device;
334
- }
335
- }
336
- });
337
- }
338
- static byIeeeAddr(databaseID, ieeeAddr, includeDeleted = false) {
339
- Device.loadFromDatabaseIfNecessary();
340
- const device = Device.devices.get(databaseID)[ieeeAddr];
341
- return (device === null || device === void 0 ? void 0 : device._deleted) && !includeDeleted ? undefined : device;
342
- }
343
- static byNetworkAddress(networkAddress, databaseID) {
344
- return Device.allByDatabaseID(databaseID).find(d => d.networkAddress === networkAddress);
345
- }
346
- static byType(type, databaseID) {
347
- return Device.allByDatabaseID(databaseID).filter(d => d.type === type);
348
- }
349
- static all() {
350
- Device.loadFromDatabaseIfNecessary();
351
- const devices = [];
352
- Device.devices.forEach(deviceDB => {
353
- devices.push(...Object.values(deviceDB).filter(d => !d._deleted));
354
- });
355
- return devices;
356
- }
357
- static allByDatabaseID(databaseID) {
358
- Device.loadFromDatabaseIfNecessary();
359
- return Object.values(Device.devices.get(databaseID)).filter(d => !d._deleted);
360
- }
361
- undelete(interviewCompleted) {
362
- (0, assert_1.default)(this._deleted, `Device '${this.ieeeAddr}' is not deleted`);
363
- this._deleted = false;
364
- this._interviewCompleted = interviewCompleted !== null && interviewCompleted !== void 0 ? interviewCompleted : this._interviewCompleted;
365
- entity_1.default.getDatabaseByID(this.databaseID).insert(this.toDatabaseEntry());
366
- }
367
- static create(type, ieeeAddr, networkAddress, manufacturerID, manufacturerName, powerSource, modelID, interviewCompleted, endpoints, databaseID) {
368
- Device.loadFromDatabaseIfNecessary();
369
- const database = entity_1.default.getDatabaseByID(databaseID);
370
- if (Device.devices.get(databaseID)[ieeeAddr] && !Device.devices.get(databaseID)[ieeeAddr]._deleted) {
371
- throw new Error(`Device with ieeeAddr '${ieeeAddr}' already exists`);
372
- }
373
- const endpointsMapped = endpoints.map((e) => {
374
- return endpoint_1.default.create(databaseID, e.ID, e.profileID, e.deviceID, e.inputClusters, e.outputClusters, networkAddress, ieeeAddr);
375
- });
376
- const ID = database.newID();
377
- const device = new Device(databaseID, ID, type, ieeeAddr, networkAddress, manufacturerID, endpointsMapped, manufacturerName, powerSource, modelID, undefined, undefined, undefined, undefined, undefined, undefined, interviewCompleted, {}, null, 'immediate', undefined, 0);
378
- database.insert(device.toDatabaseEntry());
379
- Device.devices.get(databaseID)[device.ieeeAddr] = device;
380
- return device;
381
- }
382
- /*
383
- * Zigbee functions
384
- */
385
- async interview() {
386
- if (this.interviewing) {
387
- const message = `Interview - interview already in progress for '${this.ieeeAddr}'`;
388
- debug.log(message);
389
- throw new Error(message);
390
- }
391
- let error;
392
- this._interviewing = true;
393
- debug.log(`Interview - start device '${this.ieeeAddr}'`);
394
- try {
395
- await this.interviewInternal();
396
- debug.log(`Interview - completed for device '${this.ieeeAddr}'`);
397
- this._interviewCompleted = true;
398
- }
399
- catch (e) {
400
- if (this.interviewQuirks()) {
401
- debug.log(`Interview - completed for device '${this.ieeeAddr}' because of quirks ('${e}')`);
402
- }
403
- else {
404
- debug.log(`Interview - failed for device '${this.ieeeAddr}' with error '${e.stack}'`);
405
- error = e;
406
- }
407
- }
408
- finally {
409
- this._interviewing = false;
410
- this.save();
411
- }
412
- if (error) {
413
- throw error;
414
- }
415
- }
416
- interviewQuirks() {
417
- var _a, _b, _c;
418
- debug.log(`Interview - quirks check for '${this.modelID}'-'${this.manufacturerName}'-'${this.type}'`);
419
- // TuYa devices are typically hard to interview. They also don't require a full interview to work correctly
420
- // e.g. no ias enrolling is required for the devices to work.
421
- // Assume that in case we got both the manufacturerName and modelID the device works correctly.
422
- // https://github.com/Koenkk/zigbee2mqtt/issues/7564:
423
- // Fails during ias enroll due to UNSUPPORTED_ATTRIBUTE
424
- // https://github.com/Koenkk/zigbee2mqtt/issues/4655
425
- // Device does not change zoneState after enroll (event with original gateway)
426
- // modelID is mostly in the form of e.g. TS0202 and manufacturerName like e.g. _TYZB01_xph99wvr
427
- if (((_a = this.modelID) === null || _a === void 0 ? void 0 : _a.match('^TS\\d*$')) &&
428
- (((_b = this.manufacturerName) === null || _b === void 0 ? void 0 : _b.match('^_TZ.*_.*$')) || ((_c = this.manufacturerName) === null || _c === void 0 ? void 0 : _c.match('^_TYZB01_.*$')))) {
429
- this._powerSource = this._powerSource || 'Battery';
430
- this._interviewing = false;
431
- this._interviewCompleted = true;
432
- debug.log(`Interview - quirks matched for TuYa end device`);
433
- return true;
434
- }
435
- // Some devices, e.g. Xiaomi end devices have a different interview procedure, after pairing they
436
- // report it's modelID trough a readResponse. The readResponse is received by the controller and set
437
- // on the device.
438
- const lookup = {
439
- '^3R.*?Z': {
440
- type: 'EndDevice', powerSource: 'Battery'
441
- },
442
- 'lumi\..*': {
443
- type: 'EndDevice', manufacturerID: 4151, manufacturerName: 'LUMI', powerSource: 'Battery'
444
- },
445
- 'TERNCY-PP01': {
446
- type: 'EndDevice', manufacturerID: 4648, manufacturerName: 'TERNCY', powerSource: 'Battery'
447
- },
448
- '3RWS18BZ': {},
449
- 'MULTI-MECI--EA01': {},
450
- 'MOT003': {}, // https://github.com/Koenkk/zigbee2mqtt/issues/12471
451
- };
452
- const match = Object.keys(lookup).find((key) => this.modelID && this.modelID.match(key));
453
- if (match) {
454
- const info = lookup[match];
455
- debug.log(`Interview procedure failed but got modelID matching '${match}', assuming interview succeeded`);
456
- this._type = this._type === 'Unknown' ? info.type : this._type;
457
- this._manufacturerID = this._manufacturerID || info.manufacturerID;
458
- this._manufacturerName = this._manufacturerName || info.manufacturerName;
459
- this._powerSource = this._powerSource || info.powerSource;
460
- this._interviewing = false;
461
- this._interviewCompleted = true;
462
- debug.log(`Interview - quirks matched on '${match}'`);
463
- return true;
464
- }
465
- else {
466
- debug.log('Interview - quirks did not match');
467
- return false;
468
- }
469
- }
470
- async interviewInternal() {
471
- const nodeDescriptorQuery = async () => {
472
- const nodeDescriptor = await entity_1.default.getAdapterByID(this.databaseID).nodeDescriptor(this.networkAddress);
473
- this._manufacturerID = nodeDescriptor.manufacturerCode;
474
- this._type = nodeDescriptor.type;
475
- debug.log(`Interview - got node descriptor for device '${this.ieeeAddr}'`);
476
- };
477
- const hasNodeDescriptor = () => this._manufacturerID != null && this._type != null;
478
- if (!hasNodeDescriptor()) {
479
- for (let attempt = 0; attempt < 6; attempt++) {
480
- try {
481
- await nodeDescriptorQuery();
482
- break;
483
- }
484
- catch (error) {
485
- if (this.interviewQuirks()) {
486
- debug.log(`Interview - completed for device '${this.ieeeAddr}' because of quirks ('${error}')`);
487
- return;
488
- }
489
- else {
490
- // Most of the times the first node descriptor query fails and the seconds one succeeds.
491
- debug.log(`Interview - node descriptor request failed for '${this.ieeeAddr}', attempt ${attempt + 1}`);
492
- }
493
- }
494
- }
495
- }
496
- else {
497
- debug.log(`Interview - skip node descriptor request for '${this.ieeeAddr}', already got it`);
498
- }
499
- if (!hasNodeDescriptor()) {
500
- throw new Error(`Interview failed because can not get node descriptor ('${this.ieeeAddr}')`);
501
- }
502
- if (this.manufacturerID === 4619 && this._type === 'EndDevice') {
503
- // Give TuYa end device some time to pair. Otherwise they leave immediately.
504
- // https://github.com/Koenkk/zigbee2mqtt/issues/5814
505
- debug.log("Interview - Detected TuYa end device, waiting 10 seconds...");
506
- await (0, utils_1.Wait)(10000);
507
- }
508
- else if ([0, 4098].includes(this.manufacturerID)) {
509
- // Potentially a TuYa device, some sleep fast so make sure to read the modelId and manufacturerName quickly.
510
- // In case the device responds, the endoint and modelID/manufacturerName are set
511
- // in controller.onZclOrRawData()
512
- // https://github.com/Koenkk/zigbee2mqtt/issues/7553
513
- debug.log("Interview - Detected potential TuYa end device, reading modelID and manufacturerName...");
514
- try {
515
- const endpoint = endpoint_1.default.create(this.databaseID, 1, undefined, undefined, [], [], this.networkAddress, this.ieeeAddr);
516
- const result = await endpoint.read('genBasic', ['modelId', 'manufacturerName'], { sendWhen: 'immediate' });
517
- Object.entries(result)
518
- .forEach((entry) => Device.ReportablePropertiesMapping[entry[0]].set(entry[1], this));
519
- }
520
- catch (error) {
521
- /* istanbul ignore next */
522
- debug.log(`Interview - TuYa read modelID and manufacturerName failed (${error})`);
523
- }
524
- }
525
- // e.g. Xiaomi Aqara Opple devices fail to respond to the first active endpoints request, therefore try 2 times
526
- // https://github.com/Koenkk/zigbee-herdsman/pull/103
527
- const adapter = entity_1.default.getAdapterByID(this.databaseID);
528
- let activeEndpoints;
529
- for (let attempt = 0; attempt < 2; attempt++) {
530
- try {
531
- activeEndpoints = await adapter.activeEndpoints(this.networkAddress);
532
- break;
533
- }
534
- catch (error) {
535
- debug.log(`Interview - active endpoints request failed for '${this.ieeeAddr}', attempt ${attempt + 1}`);
536
- }
537
- }
538
- if (!activeEndpoints) {
539
- throw new Error(`Interview failed because can not get active endpoints ('${this.ieeeAddr}')`);
540
- }
541
- // Make sure that the endpoint are sorted.
542
- activeEndpoints.endpoints.sort((a, b) => a - b);
543
- // Some devices, e.g. TERNCY return endpoint 0 in the active endpoints request.
544
- // This is not a valid endpoint number according to the ZCL, requesting a simple descriptor will result
545
- // into an error. Therefore we filter it, more info: https://github.com/Koenkk/zigbee-herdsman/issues/82
546
- activeEndpoints.endpoints.filter((e) => e !== 0 && !this.getEndpoint(e)).forEach((e) => this._endpoints.push(endpoint_1.default.create(this.databaseID, e, undefined, undefined, [], [], this.networkAddress, this.ieeeAddr)));
547
- debug.log(`Interview - got active endpoints for device '${this.ieeeAddr}'`);
548
- for (const endpointID of activeEndpoints.endpoints.filter((e) => e !== 0)) {
549
- const endpoint = this.getEndpoint(endpointID);
550
- const simpleDescriptor = await adapter.simpleDescriptor(this.networkAddress, endpoint.ID);
551
- endpoint.profileID = simpleDescriptor.profileID;
552
- endpoint.deviceID = simpleDescriptor.deviceID;
553
- endpoint.inputClusters = simpleDescriptor.inputClusters;
554
- endpoint.outputClusters = simpleDescriptor.outputClusters;
555
- debug.log(`Interview - got simple descriptor for endpoint '${endpoint.ID}' device '${this.ieeeAddr}'`);
556
- // Read attributes, nice to have but not required for succesfull pairing as most of the attributes
557
- // are not mandatory in ZCL specification.
558
- if (endpoint.supportsInputCluster('genBasic')) {
559
- for (const [key, item] of Object.entries(Device.ReportablePropertiesMapping)) {
560
- if (!this[item.key]) {
561
- try {
562
- let result;
563
- try {
564
- result = await endpoint.read('genBasic', [key], { sendWhen: 'immediate' });
565
- }
566
- catch (error) {
567
- // Reading attributes can fail for many reason, e.g. it could be that device rejoins
568
- // while joining like in:
569
- // https://github.com/Koenkk/zigbee-herdsman-converters/issues/2485.
570
- // The modelID and manufacturerName are crucial for device identification, so retry.
571
- if (item.key === 'modelID' || item.key === 'manufacturerName') {
572
- debug.log(`Interview - first ${item.key} retrieval attempt failed, ` +
573
- `retrying after 10 seconds...`);
574
- await (0, utils_1.Wait)(10000);
575
- result = await endpoint.read('genBasic', [key], { sendWhen: 'immediate' });
576
- }
577
- else {
578
- throw error;
579
- }
580
- }
581
- item.set(result[key], this);
582
- debug.log(`Interview - got '${item.key}' for device '${this.ieeeAddr}'`);
583
- }
584
- catch (error) {
585
- debug.log(`Interview - failed to read attribute '${item.key}' from ` +
586
- `endpoint '${endpoint.ID}' (${error})`);
587
- }
588
- }
589
- }
590
- }
591
- }
592
- const coordinator = Device.byType('Coordinator', this.databaseID)[0];
593
- // Enroll IAS device
594
- for (const endpoint of this.endpoints.filter((e) => e.supportsInputCluster('ssIasZone'))) {
595
- debug.log(`Interview - IAS - enrolling '${this.ieeeAddr}' endpoint '${endpoint.ID}'`);
596
- const stateBefore = await endpoint.read('ssIasZone', ['iasCieAddr', 'zoneState'], { sendWhen: 'immediate' });
597
- debug.log(`Interview - IAS - before enrolling state: '${JSON.stringify(stateBefore)}'`);
598
- // Do not enroll when device has already been enrolled
599
- if (stateBefore.zoneState !== 1 || stateBefore.iasCieAddr !== coordinator.ieeeAddr) {
600
- debug.log(`Interview - IAS - not enrolled, enrolling`);
601
- await endpoint.write('ssIasZone', { 'iasCieAddr': coordinator.ieeeAddr }, { sendWhen: 'immediate' });
602
- debug.log(`Interview - IAS - wrote iasCieAddr`);
603
- // There are 2 enrollment procedures:
604
- // - Auto enroll: coordinator has to send enrollResponse without receiving an enroll request
605
- // this case is handled below.
606
- // - Manual enroll: coordinator replies to enroll request with an enroll response.
607
- // this case in hanled in onZclData().
608
- // https://github.com/Koenkk/zigbee2mqtt/issues/4569#issuecomment-706075676
609
- await (0, utils_1.Wait)(500);
610
- debug.log(`IAS - '${this.ieeeAddr}' sending enroll response (auto enroll)`);
611
- const payload = { enrollrspcode: 0, zoneid: 23 };
612
- await endpoint.command('ssIasZone', 'enrollRsp', payload, { disableDefaultResponse: true, sendWhen: 'immediate' });
613
- let enrolled = false;
614
- for (let attempt = 0; attempt < 20; attempt++) {
615
- await (0, utils_1.Wait)(500);
616
- const stateAfter = await endpoint.read('ssIasZone', ['iasCieAddr', 'zoneState'], { sendWhen: 'immediate' });
617
- debug.log(`Interview - IAS - after enrolling state (${attempt}): '${JSON.stringify(stateAfter)}'`);
618
- if (stateAfter.zoneState === 1) {
619
- enrolled = true;
620
- break;
621
- }
622
- }
623
- if (enrolled) {
624
- debug.log(`Interview - IAS successfully enrolled '${this.ieeeAddr}' endpoint '${endpoint.ID}'`);
625
- }
626
- else {
627
- throw new Error(`Interview failed because of failed IAS enroll (zoneState didn't change ('${this.ieeeAddr}')`);
628
- }
629
- }
630
- else {
631
- debug.log(`Interview - IAS - already enrolled, skipping enroll`);
632
- }
633
- }
634
- // Bind poll control
635
- try {
636
- for (const endpoint of this.endpoints.filter((e) => e.supportsInputCluster('genPollCtrl'))) {
637
- debug.log(`Interview - Poll control - binding '${this.ieeeAddr}' endpoint '${endpoint.ID}'`);
638
- await endpoint.bind('genPollCtrl', coordinator.endpoints[0]);
639
- const pollPeriod = await endpoint.read('genPollCtrl', ['checkinInterval'], { sendWhen: 'immediate' });
640
- this._checkinInterval = pollPeriod.checkinInterval / 4; // convert to seconds
641
- this.pendingRequestTimeout = this._checkinInterval * 1000; // milliseconds
642
- if (pollPeriod.checkinInterval <= 2400) { // 10 minutes
643
- this.defaultSendRequestWhen = 'fastpoll';
644
- }
645
- else {
646
- this.defaultSendRequestWhen = 'active';
647
- }
648
- }
649
- }
650
- catch (error) {
651
- /* istanbul ignore next */
652
- debug.log(`Interview - failed to bind genPollCtrl (${error})`);
653
- }
654
- }
655
- async removeFromNetwork() {
656
- if (this._type === 'GreenPower') {
657
- const payload = {
658
- options: 0x002550,
659
- srcID: Number(this.ieeeAddr),
660
- };
661
- const frame = Zcl.ZclFrame.create(Zcl.FrameType.SPECIFIC, Zcl.Direction.SERVER_TO_CLIENT, true, null, zclTransactionSequenceNumber_1.default.next(), 'pairing', 33, payload);
662
- await entity_1.default.getAdapterByID(this.databaseID).sendZclFrameToAll(242, frame, 242);
663
- }
664
- else
665
- await entity_1.default.getAdapterByID(this.databaseID).removeDevice(this.networkAddress, this.ieeeAddr);
666
- await this.removeFromDatabase();
667
- }
668
- async removeFromDatabase() {
669
- Device.loadFromDatabaseIfNecessary();
670
- const database = entity_1.default.getDatabaseByID(this.databaseID);
671
- for (const endpoint of this.endpoints) {
672
- endpoint.removeFromAllGroupsDatabase();
673
- }
674
- if (database.has(this.ID)) {
675
- database.remove(this.ID);
676
- }
677
- this._deleted = true;
678
- // Clear all data in case device joins again
679
- this._interviewCompleted = false;
680
- this._interviewing = false;
681
- this.meta = {};
682
- const newEndpoints = [];
683
- for (const endpoint of this.endpoints) {
684
- newEndpoints.push(endpoint_1.default.create(database.id, endpoint.ID, endpoint.profileID, endpoint.deviceID, endpoint.inputClusters, endpoint.outputClusters, this.networkAddress, this.ieeeAddr));
685
- }
686
- this._endpoints = newEndpoints;
687
- }
688
- async lqi() {
689
- return entity_1.default.getAdapterByID(this.databaseID).lqi(this.networkAddress);
690
- }
691
- async routingTable() {
692
- return entity_1.default.getAdapterByID(this.databaseID).routingTable(this.networkAddress);
693
- }
694
- async ping(disableRecovery = true) {
695
- var _a;
696
- // Zigbee does not have an official pining mechamism. Use a read request
697
- // of a mandatory basic cluster attribute to keep it as lightweight as
698
- // possible.
699
- const endpoint = (_a = this.endpoints.find((ep) => ep.inputClusters.includes(0))) !== null && _a !== void 0 ? _a : this.endpoints[0];
700
- await endpoint.read('genBasic', ['zclVersion'], { disableRecovery });
701
- }
702
- }
703
- // This lookup contains all devices that are queried from the database, this is to ensure that always
704
- // the same instance is returned.
705
- Device.devices = new Map();
706
- Device.ReportablePropertiesMapping = {
707
- modelId: { key: 'modelID', set: (v, d) => { d.modelID = v; } },
708
- manufacturerName: { key: 'manufacturerName', set: (v, d) => { d.manufacturerName = v; } },
709
- powerSource: { key: 'powerSource', set: (v, d) => { d.powerSource = v; } },
710
- zclVersion: { key: 'zclVersion', set: (v, d) => { d.zclVersion = v; } },
711
- appVersion: { key: 'applicationVersion', set: (v, d) => { d.applicationVersion = v; } },
712
- stackVersion: { key: 'stackVersion', set: (v, d) => { d.stackVersion = v; } },
713
- hwVersion: { key: 'hardwareVersion', set: (v, d) => { d.hardwareVersion = v; } },
714
- dateCode: { key: 'dateCode', set: (v, d) => { d.dateCode = v; } },
715
- swBuildId: { key: 'softwareBuildID', set: (v, d) => { d.softwareBuildID = v; } },
716
- };
717
- exports.default = Device;
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || function (mod) {
19
+ if (mod && mod.__esModule) return mod;
20
+ var result = {};
21
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
+ __setModuleDefault(result, mod);
23
+ return result;
24
+ };
25
+ var __importDefault = (this && this.__importDefault) || function (mod) {
26
+ return (mod && mod.__esModule) ? mod : { "default": mod };
27
+ };
28
+ Object.defineProperty(exports, "__esModule", { value: true });
29
+ const zclTransactionSequenceNumber_1 = __importDefault(require("../helpers/zclTransactionSequenceNumber"));
30
+ const endpoint_1 = __importDefault(require("./endpoint"));
31
+ const entity_1 = __importDefault(require("./entity"));
32
+ const utils_1 = require("../../utils");
33
+ const debug_1 = __importDefault(require("debug"));
34
+ const Zcl = __importStar(require("../../zcl"));
35
+ const assert_1 = __importDefault(require("assert"));
36
+ const helpers_1 = require("../helpers");
37
+ /**
38
+ * @ignore
39
+ */
40
+ const OneJanuary2000 = new Date('January 01, 2000 00:00:00 UTC+00:00').getTime();
41
+ const debug = {
42
+ error: (0, debug_1.default)('zigbee-herdsman:controller:device:error'),
43
+ log: (0, debug_1.default)('zigbee-herdsman:controller:device:log'),
44
+ };
45
+ class Device extends entity_1.default {
46
+ databaseID;
47
+ ID;
48
+ _applicationVersion;
49
+ _dateCode;
50
+ _endpoints;
51
+ _hardwareVersion;
52
+ _ieeeAddr;
53
+ _interviewCompleted;
54
+ _interviewing;
55
+ _lastSeen;
56
+ _manufacturerID;
57
+ _manufacturerName;
58
+ _modelID;
59
+ _networkAddress;
60
+ _powerSource;
61
+ _softwareBuildID;
62
+ _stackVersion;
63
+ _type;
64
+ _zclVersion;
65
+ _linkquality;
66
+ _skipDefaultResponse;
67
+ _skipTimeResponse;
68
+ _deleted;
69
+ _lastDefaultResponseSequenceNumber;
70
+ _checkinInterval;
71
+ _pendingRequestTimeout;
72
+ // Getters/setters
73
+ get ieeeAddr() { return this._ieeeAddr; }
74
+ set ieeeAddr(ieeeAddr) { this._ieeeAddr = ieeeAddr; }
75
+ get applicationVersion() { return this._applicationVersion; }
76
+ set applicationVersion(applicationVersion) { this._applicationVersion = applicationVersion; }
77
+ get endpoints() { return this._endpoints; }
78
+ get interviewCompleted() { return this._interviewCompleted; }
79
+ get interviewing() { return this._interviewing; }
80
+ get lastSeen() { return this._lastSeen; }
81
+ get manufacturerID() { return this._manufacturerID; }
82
+ get isDeleted() { return this._deleted; }
83
+ set type(type) { this._type = type; }
84
+ get type() { return this._type; }
85
+ get dateCode() { return this._dateCode; }
86
+ set dateCode(dateCode) { this._dateCode = dateCode; }
87
+ set hardwareVersion(hardwareVersion) { this._hardwareVersion = hardwareVersion; }
88
+ get hardwareVersion() { return this._hardwareVersion; }
89
+ get manufacturerName() { return this._manufacturerName; }
90
+ set manufacturerName(manufacturerName) { this._manufacturerName = manufacturerName; }
91
+ set modelID(modelID) { this._modelID = modelID; }
92
+ get modelID() { return this._modelID; }
93
+ get networkAddress() { return this._networkAddress; }
94
+ set networkAddress(networkAddress) {
95
+ this._networkAddress = networkAddress;
96
+ for (const endpoint of this._endpoints) {
97
+ endpoint.deviceNetworkAddress = networkAddress;
98
+ }
99
+ }
100
+ get powerSource() { return this._powerSource; }
101
+ set powerSource(powerSource) {
102
+ this._powerSource = typeof powerSource === 'number' ? Zcl.PowerSource[powerSource & ~(1 << 7)] : powerSource;
103
+ }
104
+ get softwareBuildID() { return this._softwareBuildID; }
105
+ set softwareBuildID(softwareBuildID) { this._softwareBuildID = softwareBuildID; }
106
+ get stackVersion() { return this._stackVersion; }
107
+ set stackVersion(stackVersion) { this._stackVersion = stackVersion; }
108
+ get zclVersion() { return this._zclVersion; }
109
+ set zclVersion(zclVersion) { this._zclVersion = zclVersion; }
110
+ get linkquality() { return this._linkquality; }
111
+ set linkquality(linkquality) { this._linkquality = linkquality; }
112
+ get skipDefaultResponse() { return this._skipDefaultResponse; }
113
+ set skipDefaultResponse(skipDefaultResponse) { this._skipDefaultResponse = skipDefaultResponse; }
114
+ get skipTimeResponse() { return this._skipTimeResponse; }
115
+ set skipTimeResponse(skipTimeResponse) { this._skipTimeResponse = skipTimeResponse; }
116
+ get checkinInterval() { return this._checkinInterval; }
117
+ set checkinInterval(checkinInterval) {
118
+ this._checkinInterval = checkinInterval;
119
+ this.resetPendingRequestTimeout();
120
+ }
121
+ get pendingRequestTimeout() { return this._pendingRequestTimeout; }
122
+ set pendingRequestTimeout(pendingRequestTimeout) { this._pendingRequestTimeout = pendingRequestTimeout; }
123
+ meta;
124
+ // This lookup contains all devices that are queried from the database, this is to ensure that always
125
+ // the same instance is returned.
126
+ static devices = new Map();
127
+ static ReportablePropertiesMapping = {
128
+ modelId: { key: 'modelID', set: (v, d) => { d.modelID = v; } },
129
+ manufacturerName: { key: 'manufacturerName', set: (v, d) => { d.manufacturerName = v; } },
130
+ powerSource: { key: 'powerSource', set: (v, d) => { d.powerSource = v; } },
131
+ zclVersion: { key: 'zclVersion', set: (v, d) => { d.zclVersion = v; } },
132
+ appVersion: { key: 'applicationVersion', set: (v, d) => { d.applicationVersion = v; } },
133
+ stackVersion: { key: 'stackVersion', set: (v, d) => { d.stackVersion = v; } },
134
+ hwVersion: { key: 'hardwareVersion', set: (v, d) => { d.hardwareVersion = v; } },
135
+ dateCode: { key: 'dateCode', set: (v, d) => { d.dateCode = v; } },
136
+ swBuildId: { key: 'softwareBuildID', set: (v, d) => { d.softwareBuildID = v; } },
137
+ };
138
+ constructor(databaseID, ID, type, ieeeAddr, networkAddress, manufacturerID, endpoints, manufacturerName, powerSource, modelID, applicationVersion, stackVersion, zclVersion, hardwareVersion, dateCode, softwareBuildID, interviewCompleted, meta, lastSeen, checkinInterval, pendingRequestTimeout) {
139
+ super();
140
+ this.databaseID = databaseID;
141
+ this.ID = ID;
142
+ this._type = type;
143
+ this.ieeeAddr = ieeeAddr;
144
+ this._networkAddress = networkAddress;
145
+ this._manufacturerID = manufacturerID;
146
+ this._endpoints = endpoints;
147
+ this._manufacturerName = manufacturerName;
148
+ this._powerSource = powerSource;
149
+ this._modelID = modelID;
150
+ this._applicationVersion = applicationVersion;
151
+ this._stackVersion = stackVersion;
152
+ this._zclVersion = zclVersion;
153
+ this.hardwareVersion = hardwareVersion;
154
+ this._dateCode = dateCode;
155
+ this._softwareBuildID = softwareBuildID;
156
+ this._interviewCompleted = interviewCompleted;
157
+ this._interviewing = false;
158
+ this._skipDefaultResponse = false;
159
+ this._skipTimeResponse = false;
160
+ this.meta = meta;
161
+ this._lastSeen = lastSeen;
162
+ this._checkinInterval = checkinInterval;
163
+ this._pendingRequestTimeout = pendingRequestTimeout;
164
+ }
165
+ createEndpoint(ID) {
166
+ if (this.getEndpoint(ID)) {
167
+ throw new Error(`Device '${this.ieeeAddr}' already has an endpoint '${ID}'`);
168
+ }
169
+ const endpoint = endpoint_1.default.create(this.databaseID, ID, undefined, undefined, [], [], this.networkAddress, this.ieeeAddr);
170
+ this.endpoints.push(endpoint);
171
+ this.save();
172
+ return endpoint;
173
+ }
174
+ changeIeeeAddress(ieeeAddr) {
175
+ delete Device.devices.get(this.databaseID)[this.ieeeAddr];
176
+ this.ieeeAddr = ieeeAddr;
177
+ Device.devices.get(this.databaseID)[this.ieeeAddr] = this;
178
+ this.endpoints.forEach((e) => e.deviceIeeeAddress = ieeeAddr);
179
+ this.save();
180
+ }
181
+ getEndpoint(ID) {
182
+ return this.endpoints.find((e) => e.ID === ID);
183
+ }
184
+ // There might be multiple endpoints with same DeviceId but it is not supported and first endpoint is returned
185
+ getEndpointByDeviceType(deviceType) {
186
+ const deviceID = Zcl.EndpointDeviceType[deviceType];
187
+ return this.endpoints.find((d) => d.deviceID === deviceID);
188
+ }
189
+ implicitCheckin() {
190
+ this.endpoints.forEach(async (e) => e.sendPendingRequests(false));
191
+ }
192
+ updateLastSeen() {
193
+ this._lastSeen = Date.now();
194
+ }
195
+ resetPendingRequestTimeout() {
196
+ // pendingRequestTimeout can be changed dynamically at runtime, and it is not persisted.
197
+ // Default timeout is one checkin interval in milliseconds.
198
+ this._pendingRequestTimeout = this._checkinInterval * 1000;
199
+ }
200
+ hasPendingRequests() {
201
+ return this.endpoints.find(e => e.hasPendingRequests()) !== undefined;
202
+ }
203
+ async onZclData(dataPayload, endpoint) {
204
+ const frame = dataPayload.frame;
205
+ // Update reportable properties
206
+ if (frame.isCluster('genBasic') && (frame.isCommand('readRsp') || frame.isCommand('report'))) {
207
+ for (const [key, val] of Object.entries(helpers_1.ZclFrameConverter.attributeKeyValue(frame, this.manufacturerID))) {
208
+ Device.ReportablePropertiesMapping[key]?.set(val, this);
209
+ }
210
+ }
211
+ // Respond to enroll requests
212
+ if (frame.isSpecific() && frame.isCluster('ssIasZone') && frame.isCommand('enrollReq')) {
213
+ debug.log(`IAS - '${this.ieeeAddr}' responding to enroll response`);
214
+ const payload = { enrollrspcode: 0, zoneid: 23 };
215
+ await endpoint.command('ssIasZone', 'enrollRsp', payload, { disableDefaultResponse: true });
216
+ }
217
+ // Reponse to read requests
218
+ if (frame.isGlobal() && frame.isCommand('read')) {
219
+ const time = Math.round(((new Date()).getTime() - OneJanuary2000) / 1000);
220
+ const attributes = {
221
+ ...endpoint.clusters,
222
+ genTime: { attributes: {
223
+ timeStatus: 3, // Time-master + synchronised
224
+ time: time,
225
+ timeZone: ((new Date()).getTimezoneOffset() * -1) * 60,
226
+ localTime: time - (new Date()).getTimezoneOffset() * 60,
227
+ lastSetTime: time,
228
+ validUntilTime: time + (24 * 60 * 60), // valid for 24 hours
229
+ } },
230
+ };
231
+ if (frame.Cluster.name in attributes && (frame.Cluster.name !== 'genTime' || !this._skipTimeResponse)) {
232
+ const response = {};
233
+ for (const entry of frame.Payload) {
234
+ if (frame.Cluster.hasAttribute(entry.attrId)) {
235
+ const name = frame.Cluster.getAttribute(entry.attrId).name;
236
+ if (name in attributes[frame.Cluster.name].attributes) {
237
+ response[name] = attributes[frame.Cluster.name].attributes[name];
238
+ }
239
+ }
240
+ }
241
+ try {
242
+ await endpoint.readResponse(frame.Cluster.ID, frame.Header.transactionSequenceNumber, response, { srcEndpoint: dataPayload.destinationEndpoint });
243
+ }
244
+ catch (error) {
245
+ debug.error(`Read response to ${this.ieeeAddr} failed`);
246
+ }
247
+ }
248
+ }
249
+ // Handle check-in from sleeping end devices
250
+ if (frame.isSpecific() && frame.isCluster("genPollCtrl") && frame.isCommand("checkin")) {
251
+ try {
252
+ if (this.hasPendingRequests() || (this._checkinInterval === undefined)) {
253
+ const payload = {
254
+ startFastPolling: true,
255
+ fastPollTimeout: 0,
256
+ };
257
+ debug.log(`check-in from ${this.ieeeAddr}: accepting fast-poll`);
258
+ await endpoint.command(frame.Cluster.ID, 'checkinRsp', payload, { sendPolicy: 'immediate' });
259
+ // This is a good time to read the checkin interval if we haven't stored it previously
260
+ if (this._checkinInterval === undefined) {
261
+ const pollPeriod = await endpoint.read('genPollCtrl', ['checkinInterval'], { sendPolicy: 'immediate' });
262
+ this._checkinInterval = pollPeriod.checkinInterval / 4; // convert to seconds
263
+ this.resetPendingRequestTimeout();
264
+ debug.log(`Request Queue (${this.ieeeAddr}): default expiration timeout set to ${this.pendingRequestTimeout}`);
265
+ }
266
+ await Promise.all(this.endpoints.map(async (e) => e.sendPendingRequests(true)));
267
+ // We *must* end fast-poll when we're done sending things. Otherwise
268
+ // we cause undue power-drain.
269
+ debug.log(`check-in from ${this.ieeeAddr}: stopping fast-poll`);
270
+ await endpoint.command(frame.Cluster.ID, 'fastPollStop', {}, { sendPolicy: 'immediate' });
271
+ }
272
+ else {
273
+ const payload = {
274
+ startFastPolling: false,
275
+ fastPollTimeout: 0,
276
+ };
277
+ debug.log(`check-in from ${this.ieeeAddr}: declining fast-poll`);
278
+ await endpoint.command(frame.Cluster.ID, 'checkinRsp', payload, { sendPolicy: 'immediate' });
279
+ }
280
+ }
281
+ catch (error) {
282
+ /* istanbul ignore next */
283
+ debug.error(`Handling of poll check-in form ${this.ieeeAddr} failed`);
284
+ }
285
+ }
286
+ // Send a default response if necessary.
287
+ const isDefaultResponse = frame.isGlobal() && frame.getCommand().name === 'defaultRsp';
288
+ const commandHasResponse = frame.getCommand().hasOwnProperty('response');
289
+ const disableDefaultResponse = frame.Header.frameControl.disableDefaultResponse;
290
+ // Sometimes messages are received twice, prevent responding twice
291
+ const alreadyResponded = this._lastDefaultResponseSequenceNumber === frame.Header.transactionSequenceNumber;
292
+ if (this.type !== 'GreenPower' && !dataPayload.wasBroadcast && !disableDefaultResponse && !isDefaultResponse &&
293
+ !commandHasResponse && !this._skipDefaultResponse && !alreadyResponded) {
294
+ try {
295
+ this._lastDefaultResponseSequenceNumber = frame.Header.transactionSequenceNumber;
296
+ // In the ZCL it is not documented what the direction of the default response should be
297
+ // In https://github.com/Koenkk/zigbee2mqtt/issues/18096 a commandResponse (SERVER_TO_CLIENT)
298
+ // is send and the device expects a CLIENT_TO_SERVER back.
299
+ // Previously SERVER_TO_CLIENT was always used.
300
+ // Therefore for non-global commands we inverse the direction.
301
+ const direction = frame.isGlobal() ? Zcl.Direction.SERVER_TO_CLIENT : (frame.Header.frameControl.direction === Zcl.Direction.CLIENT_TO_SERVER
302
+ ? Zcl.Direction.SERVER_TO_CLIENT : Zcl.Direction.CLIENT_TO_SERVER);
303
+ await endpoint.defaultResponse(frame.getCommand().ID, 0, frame.Cluster.ID, frame.Header.transactionSequenceNumber, { direction });
304
+ }
305
+ catch (error) {
306
+ debug.error(`Default response to ${this.ieeeAddr} failed`);
307
+ }
308
+ }
309
+ }
310
+ /*
311
+ * CRUD
312
+ */
313
+ static fromDatabaseEntry(entry, databaseID) {
314
+ const networkAddress = entry.nwkAddr;
315
+ const ieeeAddr = entry.ieeeAddr;
316
+ const endpoints = Object.values(entry.endpoints).map((e) => {
317
+ return endpoint_1.default.fromDatabaseRecord(e, networkAddress, ieeeAddr, databaseID);
318
+ });
319
+ const meta = entry.meta ? entry.meta : {};
320
+ if (entry.type === 'Group') {
321
+ throw new Error('Cannot load device from group');
322
+ }
323
+ // default: no timeout (messages expire immediately after first send attempt)
324
+ let pendingRequestTimeout = 0;
325
+ if ((endpoints.filter((e) => e.supportsInputCluster('genPollCtrl'))).length > 0) {
326
+ // default for devices that support genPollCtrl cluster (RX off when idle): 1 day
327
+ pendingRequestTimeout = 86400000;
328
+ /* istanbul ignore else */
329
+ if (entry.hasOwnProperty('checkinInterval')) {
330
+ // if the checkin interval is known, messages expire by default after one checkin interval
331
+ pendingRequestTimeout = entry.checkinInterval * 1000; // milliseconds
332
+ }
333
+ }
334
+ debug.log(`Request Queue (${ieeeAddr}): default expiration timeout set to ${pendingRequestTimeout}`);
335
+ return new Device(databaseID, entry.id, entry.type, ieeeAddr, networkAddress, entry.manufId, endpoints, entry.manufName, entry.powerSource, entry.modelId, entry.appVersion, entry.stackVersion, entry.zclVersion, entry.hwVersion, entry.dateCode, entry.swBuildId, entry.interviewCompleted, meta, entry.lastSeen || null, entry.checkinInterval, pendingRequestTimeout);
336
+ }
337
+ toDatabaseEntry() {
338
+ const epList = this.endpoints.map((e) => e.ID);
339
+ const endpoints = {};
340
+ for (const endpoint of this.endpoints) {
341
+ endpoints[endpoint.ID] = endpoint.toDatabaseRecord();
342
+ }
343
+ return {
344
+ id: this.ID, type: this.type, ieeeAddr: this.ieeeAddr, nwkAddr: this.networkAddress,
345
+ manufId: this.manufacturerID, manufName: this.manufacturerName, powerSource: this.powerSource,
346
+ modelId: this.modelID, epList, endpoints, appVersion: this.applicationVersion,
347
+ stackVersion: this.stackVersion, hwVersion: this.hardwareVersion, dateCode: this.dateCode,
348
+ swBuildId: this.softwareBuildID, zclVersion: this.zclVersion, interviewCompleted: this.interviewCompleted,
349
+ meta: this.meta, lastSeen: this.lastSeen, checkinInterval: this.checkinInterval
350
+ };
351
+ }
352
+ save(writeDatabase = true) {
353
+ entity_1.default.getDatabaseByID(this.databaseID).update(this.toDatabaseEntry(), writeDatabase);
354
+ }
355
+ static loadFromDatabaseIfNecessary() {
356
+ entity_1.default.databases.forEach(database => {
357
+ if (!Device.devices.get(database.id)) {
358
+ Device.devices.set(database.id, {});
359
+ const entries = database.getEntries(['Coordinator', 'EndDevice', 'Router', 'GreenPower', 'Unknown']);
360
+ for (const entry of entries) {
361
+ const device = Device.fromDatabaseEntry(entry, database.id);
362
+ Device.devices.get(database.id)[device.ieeeAddr] = device;
363
+ }
364
+ }
365
+ });
366
+ }
367
+ static byIeeeAddr(databaseID, ieeeAddr, includeDeleted = false) {
368
+ Device.loadFromDatabaseIfNecessary();
369
+ const device = Device.devices.get(databaseID)[ieeeAddr];
370
+ return device?._deleted && !includeDeleted ? undefined : device;
371
+ }
372
+ static byNetworkAddress(networkAddress, databaseID) {
373
+ return Device.allByDatabaseID(databaseID).find(d => d.networkAddress === networkAddress);
374
+ }
375
+ static byType(type, databaseID) {
376
+ return Device.allByDatabaseID(databaseID).filter(d => d.type === type);
377
+ }
378
+ static all() {
379
+ Device.loadFromDatabaseIfNecessary();
380
+ const devices = [];
381
+ Device.devices.forEach(deviceDB => {
382
+ devices.push(...Object.values(deviceDB).filter(d => !d._deleted));
383
+ });
384
+ return devices;
385
+ }
386
+ static allByDatabaseID(databaseID) {
387
+ Device.loadFromDatabaseIfNecessary();
388
+ return Object.values(Device.devices.get(databaseID)).filter(d => !d._deleted);
389
+ }
390
+ undelete(interviewCompleted) {
391
+ (0, assert_1.default)(this._deleted, `Device '${this.ieeeAddr}' is not deleted`);
392
+ this._deleted = false;
393
+ this._interviewCompleted = interviewCompleted ?? this._interviewCompleted;
394
+ entity_1.default.getDatabaseByID(this.databaseID).insert(this.toDatabaseEntry());
395
+ }
396
+ static create(type, ieeeAddr, networkAddress, manufacturerID, manufacturerName, powerSource, modelID, interviewCompleted, endpoints, databaseID) {
397
+ Device.loadFromDatabaseIfNecessary();
398
+ const database = entity_1.default.getDatabaseByID(databaseID);
399
+ if (Device.devices.get(databaseID)[ieeeAddr] && !Device.devices.get(databaseID)[ieeeAddr]._deleted) {
400
+ throw new Error(`Device with ieeeAddr '${ieeeAddr}' already exists`);
401
+ }
402
+ const endpointsMapped = endpoints.map((e) => {
403
+ return endpoint_1.default.create(databaseID, e.ID, e.profileID, e.deviceID, e.inputClusters, e.outputClusters, networkAddress, ieeeAddr);
404
+ });
405
+ const ID = database.newID();
406
+ const device = new Device(databaseID, ID, type, ieeeAddr, networkAddress, manufacturerID, endpointsMapped, manufacturerName, powerSource, modelID, undefined, undefined, undefined, undefined, undefined, undefined, interviewCompleted, {}, null, undefined, 0);
407
+ database.insert(device.toDatabaseEntry());
408
+ Device.devices.get(databaseID)[device.ieeeAddr] = device;
409
+ return device;
410
+ }
411
+ /*
412
+ * Zigbee functions
413
+ */
414
+ async interview() {
415
+ if (this.interviewing) {
416
+ const message = `Interview - interview already in progress for '${this.ieeeAddr}'`;
417
+ debug.log(message);
418
+ throw new Error(message);
419
+ }
420
+ let error;
421
+ this._interviewing = true;
422
+ debug.log(`Interview - start device '${this.ieeeAddr}'`);
423
+ try {
424
+ await this.interviewInternal();
425
+ debug.log(`Interview - completed for device '${this.ieeeAddr}'`);
426
+ this._interviewCompleted = true;
427
+ }
428
+ catch (e) {
429
+ if (this.interviewQuirks()) {
430
+ debug.log(`Interview - completed for device '${this.ieeeAddr}' because of quirks ('${e}')`);
431
+ }
432
+ else {
433
+ debug.log(`Interview - failed for device '${this.ieeeAddr}' with error '${e.stack}'`);
434
+ error = e;
435
+ }
436
+ }
437
+ finally {
438
+ this._interviewing = false;
439
+ this.save();
440
+ }
441
+ if (error) {
442
+ throw error;
443
+ }
444
+ }
445
+ interviewQuirks() {
446
+ debug.log(`Interview - quirks check for '${this.modelID}'-'${this.manufacturerName}'-'${this.type}'`);
447
+ // TuYa devices are typically hard to interview. They also don't require a full interview to work correctly
448
+ // e.g. no ias enrolling is required for the devices to work.
449
+ // Assume that in case we got both the manufacturerName and modelID the device works correctly.
450
+ // https://github.com/Koenkk/zigbee2mqtt/issues/7564:
451
+ // Fails during ias enroll due to UNSUPPORTED_ATTRIBUTE
452
+ // https://github.com/Koenkk/zigbee2mqtt/issues/4655
453
+ // Device does not change zoneState after enroll (event with original gateway)
454
+ // modelID is mostly in the form of e.g. TS0202 and manufacturerName like e.g. _TYZB01_xph99wvr
455
+ if (this.modelID?.match('^TS\\d*$') &&
456
+ (this.manufacturerName?.match('^_TZ.*_.*$') || this.manufacturerName?.match('^_TYZB01_.*$'))) {
457
+ this._powerSource = this._powerSource || 'Battery';
458
+ this._interviewing = false;
459
+ this._interviewCompleted = true;
460
+ debug.log(`Interview - quirks matched for TuYa end device`);
461
+ return true;
462
+ }
463
+ // Some devices, e.g. Xiaomi end devices have a different interview procedure, after pairing they
464
+ // report it's modelID trough a readResponse. The readResponse is received by the controller and set
465
+ // on the device.
466
+ const lookup = {
467
+ '^3R.*?Z': {
468
+ type: 'EndDevice', powerSource: 'Battery'
469
+ },
470
+ 'lumi\..*': {
471
+ type: 'EndDevice', manufacturerID: 4151, manufacturerName: 'LUMI', powerSource: 'Battery'
472
+ },
473
+ 'TERNCY-PP01': {
474
+ type: 'EndDevice', manufacturerID: 4648, manufacturerName: 'TERNCY', powerSource: 'Battery'
475
+ },
476
+ '3RWS18BZ': {}, // https://github.com/Koenkk/zigbee-herdsman-converters/pull/2710
477
+ 'MULTI-MECI--EA01': {},
478
+ 'MOT003': {}, // https://github.com/Koenkk/zigbee2mqtt/issues/12471
479
+ };
480
+ const match = Object.keys(lookup).find((key) => this.modelID && this.modelID.match(key));
481
+ if (match) {
482
+ const info = lookup[match];
483
+ debug.log(`Interview procedure failed but got modelID matching '${match}', assuming interview succeeded`);
484
+ this._type = this._type === 'Unknown' ? info.type : this._type;
485
+ this._manufacturerID = this._manufacturerID || info.manufacturerID;
486
+ this._manufacturerName = this._manufacturerName || info.manufacturerName;
487
+ this._powerSource = this._powerSource || info.powerSource;
488
+ this._interviewing = false;
489
+ this._interviewCompleted = true;
490
+ debug.log(`Interview - quirks matched on '${match}'`);
491
+ return true;
492
+ }
493
+ else {
494
+ debug.log('Interview - quirks did not match');
495
+ return false;
496
+ }
497
+ }
498
+ async interviewInternal() {
499
+ const nodeDescriptorQuery = async () => {
500
+ const nodeDescriptor = await entity_1.default.getAdapterByID(this.databaseID).nodeDescriptor(this.networkAddress);
501
+ this._manufacturerID = nodeDescriptor.manufacturerCode;
502
+ this._type = nodeDescriptor.type;
503
+ debug.log(`Interview - got node descriptor for device '${this.ieeeAddr}'`);
504
+ };
505
+ const hasNodeDescriptor = () => this._manufacturerID != null && this._type != null;
506
+ if (!hasNodeDescriptor()) {
507
+ for (let attempt = 0; attempt < 6; attempt++) {
508
+ try {
509
+ await nodeDescriptorQuery();
510
+ break;
511
+ }
512
+ catch (error) {
513
+ if (this.interviewQuirks()) {
514
+ debug.log(`Interview - completed for device '${this.ieeeAddr}' because of quirks ('${error}')`);
515
+ return;
516
+ }
517
+ else {
518
+ // Most of the times the first node descriptor query fails and the seconds one succeeds.
519
+ debug.log(`Interview - node descriptor request failed for '${this.ieeeAddr}', attempt ${attempt + 1}`);
520
+ }
521
+ }
522
+ }
523
+ }
524
+ else {
525
+ debug.log(`Interview - skip node descriptor request for '${this.ieeeAddr}', already got it`);
526
+ }
527
+ if (!hasNodeDescriptor()) {
528
+ throw new Error(`Interview failed because can not get node descriptor ('${this.ieeeAddr}')`);
529
+ }
530
+ if (this.manufacturerID === 4619 && this._type === 'EndDevice') {
531
+ // Give TuYa end device some time to pair. Otherwise they leave immediately.
532
+ // https://github.com/Koenkk/zigbee2mqtt/issues/5814
533
+ debug.log("Interview - Detected TuYa end device, waiting 10 seconds...");
534
+ await (0, utils_1.Wait)(10000);
535
+ }
536
+ else if ([0, 4098].includes(this.manufacturerID)) {
537
+ // Potentially a TuYa device, some sleep fast so make sure to read the modelId and manufacturerName quickly.
538
+ // In case the device responds, the endoint and modelID/manufacturerName are set
539
+ // in controller.onZclOrRawData()
540
+ // https://github.com/Koenkk/zigbee2mqtt/issues/7553
541
+ debug.log("Interview - Detected potential TuYa end device, reading modelID and manufacturerName...");
542
+ try {
543
+ const endpoint = endpoint_1.default.create(this.databaseID, 1, undefined, undefined, [], [], this.networkAddress, this.ieeeAddr);
544
+ const result = await endpoint.read('genBasic', ['modelId', 'manufacturerName'], { sendPolicy: 'immediate' });
545
+ Object.entries(result)
546
+ .forEach((entry) => Device.ReportablePropertiesMapping[entry[0]].set(entry[1], this));
547
+ }
548
+ catch (error) {
549
+ /* istanbul ignore next */
550
+ debug.log(`Interview - TuYa read modelID and manufacturerName failed (${error})`);
551
+ }
552
+ }
553
+ // e.g. Xiaomi Aqara Opple devices fail to respond to the first active endpoints request, therefore try 2 times
554
+ // https://github.com/Koenkk/zigbee-herdsman/pull/103
555
+ const adapter = entity_1.default.getAdapterByID(this.databaseID);
556
+ let activeEndpoints;
557
+ for (let attempt = 0; attempt < 2; attempt++) {
558
+ try {
559
+ activeEndpoints = await adapter.activeEndpoints(this.networkAddress);
560
+ break;
561
+ }
562
+ catch (error) {
563
+ debug.log(`Interview - active endpoints request failed for '${this.ieeeAddr}', attempt ${attempt + 1}`);
564
+ }
565
+ }
566
+ if (!activeEndpoints) {
567
+ throw new Error(`Interview failed because can not get active endpoints ('${this.ieeeAddr}')`);
568
+ }
569
+ // Make sure that the endpoint are sorted.
570
+ activeEndpoints.endpoints.sort((a, b) => a - b);
571
+ // Some devices, e.g. TERNCY return endpoint 0 in the active endpoints request.
572
+ // This is not a valid endpoint number according to the ZCL, requesting a simple descriptor will result
573
+ // into an error. Therefore we filter it, more info: https://github.com/Koenkk/zigbee-herdsman/issues/82
574
+ activeEndpoints.endpoints.filter((e) => e !== 0 && !this.getEndpoint(e)).forEach((e) => this._endpoints.push(endpoint_1.default.create(this.databaseID, e, undefined, undefined, [], [], this.networkAddress, this.ieeeAddr)));
575
+ debug.log(`Interview - got active endpoints for device '${this.ieeeAddr}'`);
576
+ for (const endpointID of activeEndpoints.endpoints.filter((e) => e !== 0)) {
577
+ const endpoint = this.getEndpoint(endpointID);
578
+ const simpleDescriptor = await adapter.simpleDescriptor(this.networkAddress, endpoint.ID);
579
+ endpoint.profileID = simpleDescriptor.profileID;
580
+ endpoint.deviceID = simpleDescriptor.deviceID;
581
+ endpoint.inputClusters = simpleDescriptor.inputClusters;
582
+ endpoint.outputClusters = simpleDescriptor.outputClusters;
583
+ debug.log(`Interview - got simple descriptor for endpoint '${endpoint.ID}' device '${this.ieeeAddr}'`);
584
+ // Read attributes, nice to have but not required for succesfull pairing as most of the attributes
585
+ // are not mandatory in ZCL specification.
586
+ if (endpoint.supportsInputCluster('genBasic')) {
587
+ for (const [key, item] of Object.entries(Device.ReportablePropertiesMapping)) {
588
+ if (!this[item.key]) {
589
+ try {
590
+ let result;
591
+ try {
592
+ result = await endpoint.read('genBasic', [key], { sendPolicy: 'immediate' });
593
+ }
594
+ catch (error) {
595
+ // Reading attributes can fail for many reason, e.g. it could be that device rejoins
596
+ // while joining like in:
597
+ // https://github.com/Koenkk/zigbee-herdsman-converters/issues/2485.
598
+ // The modelID and manufacturerName are crucial for device identification, so retry.
599
+ if (item.key === 'modelID' || item.key === 'manufacturerName') {
600
+ debug.log(`Interview - first ${item.key} retrieval attempt failed, ` +
601
+ `retrying after 10 seconds...`);
602
+ await (0, utils_1.Wait)(10000);
603
+ result = await endpoint.read('genBasic', [key], { sendPolicy: 'immediate' });
604
+ }
605
+ else {
606
+ throw error;
607
+ }
608
+ }
609
+ item.set(result[key], this);
610
+ debug.log(`Interview - got '${item.key}' for device '${this.ieeeAddr}'`);
611
+ }
612
+ catch (error) {
613
+ debug.log(`Interview - failed to read attribute '${item.key}' from ` +
614
+ `endpoint '${endpoint.ID}' (${error})`);
615
+ }
616
+ }
617
+ }
618
+ }
619
+ }
620
+ const coordinator = Device.byType('Coordinator', this.databaseID)[0];
621
+ // Enroll IAS device
622
+ for (const endpoint of this.endpoints.filter((e) => e.supportsInputCluster('ssIasZone'))) {
623
+ debug.log(`Interview - IAS - enrolling '${this.ieeeAddr}' endpoint '${endpoint.ID}'`);
624
+ const stateBefore = await endpoint.read('ssIasZone', ['iasCieAddr', 'zoneState'], { sendPolicy: 'immediate' });
625
+ debug.log(`Interview - IAS - before enrolling state: '${JSON.stringify(stateBefore)}'`);
626
+ // Do not enroll when device has already been enrolled
627
+ if (stateBefore.zoneState !== 1 || stateBefore.iasCieAddr !== coordinator.ieeeAddr) {
628
+ debug.log(`Interview - IAS - not enrolled, enrolling`);
629
+ await endpoint.write('ssIasZone', { 'iasCieAddr': coordinator.ieeeAddr }, { sendPolicy: 'immediate' });
630
+ debug.log(`Interview - IAS - wrote iasCieAddr`);
631
+ // There are 2 enrollment procedures:
632
+ // - Auto enroll: coordinator has to send enrollResponse without receiving an enroll request
633
+ // this case is handled below.
634
+ // - Manual enroll: coordinator replies to enroll request with an enroll response.
635
+ // this case in hanled in onZclData().
636
+ // https://github.com/Koenkk/zigbee2mqtt/issues/4569#issuecomment-706075676
637
+ await (0, utils_1.Wait)(500);
638
+ debug.log(`IAS - '${this.ieeeAddr}' sending enroll response (auto enroll)`);
639
+ const payload = { enrollrspcode: 0, zoneid: 23 };
640
+ await endpoint.command('ssIasZone', 'enrollRsp', payload, { disableDefaultResponse: true, sendPolicy: 'immediate' });
641
+ let enrolled = false;
642
+ for (let attempt = 0; attempt < 20; attempt++) {
643
+ await (0, utils_1.Wait)(500);
644
+ const stateAfter = await endpoint.read('ssIasZone', ['iasCieAddr', 'zoneState'], { sendPolicy: 'immediate' });
645
+ debug.log(`Interview - IAS - after enrolling state (${attempt}): '${JSON.stringify(stateAfter)}'`);
646
+ if (stateAfter.zoneState === 1) {
647
+ enrolled = true;
648
+ break;
649
+ }
650
+ }
651
+ if (enrolled) {
652
+ debug.log(`Interview - IAS successfully enrolled '${this.ieeeAddr}' endpoint '${endpoint.ID}'`);
653
+ }
654
+ else {
655
+ throw new Error(`Interview failed because of failed IAS enroll (zoneState didn't change ('${this.ieeeAddr}')`);
656
+ }
657
+ }
658
+ else {
659
+ debug.log(`Interview - IAS - already enrolled, skipping enroll`);
660
+ }
661
+ }
662
+ // Bind poll control
663
+ try {
664
+ for (const endpoint of this.endpoints.filter((e) => e.supportsInputCluster('genPollCtrl'))) {
665
+ debug.log(`Interview - Poll control - binding '${this.ieeeAddr}' endpoint '${endpoint.ID}'`);
666
+ await endpoint.bind('genPollCtrl', coordinator.endpoints[0]);
667
+ const pollPeriod = await endpoint.read('genPollCtrl', ['checkinInterval'], { sendPolicy: 'immediate' });
668
+ this._checkinInterval = pollPeriod.checkinInterval / 4; // convert to seconds
669
+ this.resetPendingRequestTimeout();
670
+ }
671
+ }
672
+ catch (error) {
673
+ /* istanbul ignore next */
674
+ debug.log(`Interview - failed to bind genPollCtrl (${error})`);
675
+ }
676
+ }
677
+ async removeFromNetwork() {
678
+ if (this._type === 'GreenPower') {
679
+ const payload = {
680
+ options: 0x002550,
681
+ srcID: Number(this.ieeeAddr),
682
+ };
683
+ const frame = Zcl.ZclFrame.create(Zcl.FrameType.SPECIFIC, Zcl.Direction.SERVER_TO_CLIENT, true, null, zclTransactionSequenceNumber_1.default.next(), 'pairing', 33, payload);
684
+ await entity_1.default.getAdapterByID(this.databaseID).sendZclFrameToAll(242, frame, 242);
685
+ }
686
+ else
687
+ await entity_1.default.getAdapterByID(this.databaseID).removeDevice(this.networkAddress, this.ieeeAddr);
688
+ this.removeFromDatabase();
689
+ }
690
+ removeFromDatabase() {
691
+ Device.loadFromDatabaseIfNecessary();
692
+ const database = entity_1.default.getDatabaseByID(this.databaseID);
693
+ for (const endpoint of this.endpoints) {
694
+ endpoint.removeFromAllGroupsDatabase();
695
+ }
696
+ if (database.has(this.ID)) {
697
+ database.remove(this.ID);
698
+ }
699
+ this._deleted = true;
700
+ // Clear all data in case device joins again
701
+ this._interviewCompleted = false;
702
+ this._interviewing = false;
703
+ this.meta = {};
704
+ const newEndpoints = [];
705
+ for (const endpoint of this.endpoints) {
706
+ newEndpoints.push(endpoint_1.default.create(database.id, endpoint.ID, endpoint.profileID, endpoint.deviceID, endpoint.inputClusters, endpoint.outputClusters, this.networkAddress, this.ieeeAddr));
707
+ }
708
+ this._endpoints = newEndpoints;
709
+ }
710
+ async lqi() {
711
+ return entity_1.default.getAdapterByID(this.databaseID).lqi(this.networkAddress);
712
+ }
713
+ async routingTable() {
714
+ return entity_1.default.getAdapterByID(this.databaseID).routingTable(this.networkAddress);
715
+ }
716
+ async ping(disableRecovery = true) {
717
+ // Zigbee does not have an official pining mechamism. Use a read request
718
+ // of a mandatory basic cluster attribute to keep it as lightweight as
719
+ // possible.
720
+ const endpoint = this.endpoints.find((ep) => ep.inputClusters.includes(0)) ?? this.endpoints[0];
721
+ await endpoint.read('genBasic', ['zclVersion'], { disableRecovery });
722
+ }
723
+ }
724
+ exports.default = Device;
718
725
  //# sourceMappingURL=device.js.map