@digital-alchemy/hass 24.9.3 → 24.9.5

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 (276) hide show
  1. package/README.md +3 -0
  2. package/dist/dynamic.d.ts +1 -1
  3. package/dist/dynamic.js +7 -2
  4. package/dist/dynamic.js.map +1 -1
  5. package/dist/extensions/area.extension.d.ts +2 -12
  6. package/dist/extensions/area.extension.js +20 -26
  7. package/dist/extensions/area.extension.js.map +1 -1
  8. package/dist/extensions/backup.extension.d.ts +2 -7
  9. package/dist/extensions/backup.extension.js +5 -8
  10. package/dist/extensions/backup.extension.js.map +1 -1
  11. package/dist/extensions/call-proxy.extension.d.ts +1 -1
  12. package/dist/extensions/call-proxy.extension.js +3 -22
  13. package/dist/extensions/call-proxy.extension.js.map +1 -1
  14. package/dist/extensions/config.extension.d.ts +2 -6
  15. package/dist/extensions/config.extension.js +21 -25
  16. package/dist/extensions/config.extension.js.map +1 -1
  17. package/dist/extensions/conversation.extension.d.ts +2 -6
  18. package/dist/extensions/conversation.extension.js +5 -8
  19. package/dist/extensions/conversation.extension.js.map +1 -1
  20. package/dist/extensions/device.extension.d.ts +2 -5
  21. package/dist/extensions/device.extension.js +16 -22
  22. package/dist/extensions/device.extension.js.map +1 -1
  23. package/dist/extensions/entity.extension.d.ts +2 -61
  24. package/dist/extensions/entity.extension.js +42 -83
  25. package/dist/extensions/entity.extension.js.map +1 -1
  26. package/dist/extensions/events.extension.d.ts +3 -11
  27. package/dist/extensions/events.extension.js +8 -11
  28. package/dist/extensions/events.extension.js.map +1 -1
  29. package/dist/extensions/fetch-api.extension.d.ts +12 -4
  30. package/dist/extensions/fetch-api.extension.js +23 -35
  31. package/dist/extensions/fetch-api.extension.js.map +1 -1
  32. package/dist/extensions/floor.extension.d.ts +2 -9
  33. package/dist/extensions/floor.extension.js +17 -23
  34. package/dist/extensions/floor.extension.js.map +1 -1
  35. package/dist/extensions/id-by.extension.js +15 -20
  36. package/dist/extensions/id-by.extension.js.map +1 -1
  37. package/dist/extensions/index.d.ts +1 -0
  38. package/dist/extensions/index.js +16 -18
  39. package/dist/extensions/index.js.map +1 -1
  40. package/dist/extensions/internal.extension.d.ts +18 -0
  41. package/dist/extensions/internal.extension.js +102 -0
  42. package/dist/extensions/internal.extension.js.map +1 -0
  43. package/dist/extensions/label.extension.d.ts +2 -9
  44. package/dist/extensions/label.extension.js +17 -23
  45. package/dist/extensions/label.extension.js.map +1 -1
  46. package/dist/extensions/reference.extension.d.ts +2 -12
  47. package/dist/extensions/reference.extension.js +19 -25
  48. package/dist/extensions/reference.extension.js.map +1 -1
  49. package/dist/extensions/registry.extension.d.ts +2 -7
  50. package/dist/extensions/registry.extension.js +1 -4
  51. package/dist/extensions/registry.extension.js.map +1 -1
  52. package/dist/extensions/websocket-api.extension.d.ts +3 -78
  53. package/dist/extensions/websocket-api.extension.js +82 -165
  54. package/dist/extensions/websocket-api.extension.js.map +1 -1
  55. package/dist/extensions/zone.extension.d.ts +2 -7
  56. package/dist/extensions/zone.extension.js +15 -21
  57. package/dist/extensions/zone.extension.js.map +1 -1
  58. package/dist/hass.module.d.ts +47 -36
  59. package/dist/hass.module.js +70 -70
  60. package/dist/hass.module.js.map +1 -1
  61. package/dist/helpers/backup.helper.js +1 -2
  62. package/dist/helpers/constants.helper.js +15 -18
  63. package/dist/helpers/constants.helper.js.map +1 -1
  64. package/dist/helpers/device.helper.js +2 -5
  65. package/dist/helpers/device.helper.js.map +1 -1
  66. package/dist/helpers/entity-state.helper.d.ts +3 -8
  67. package/dist/helpers/entity-state.helper.js +1 -8
  68. package/dist/helpers/entity-state.helper.js.map +1 -1
  69. package/dist/helpers/features.helper.js +79 -85
  70. package/dist/helpers/features.helper.js.map +1 -1
  71. package/dist/helpers/fetch/calendar.js +1 -2
  72. package/dist/helpers/fetch/configuration.js +2 -5
  73. package/dist/helpers/fetch/configuration.js.map +1 -1
  74. package/dist/helpers/fetch/index.js +5 -8
  75. package/dist/helpers/fetch/index.js.map +1 -1
  76. package/dist/helpers/fetch/server-log.js +1 -2
  77. package/dist/helpers/fetch/server-log.js.map +1 -1
  78. package/dist/helpers/fetch/service-list.js +1 -2
  79. package/dist/helpers/fetch/weather-forecasts.js +1 -2
  80. package/dist/helpers/fetch/weather-forecasts.js.map +1 -1
  81. package/dist/helpers/fetch.helper.d.ts +162 -0
  82. package/dist/helpers/fetch.helper.js +161 -0
  83. package/dist/helpers/fetch.helper.js.map +1 -0
  84. package/dist/helpers/id-by.helper.js +1 -2
  85. package/dist/helpers/index.d.ts +2 -1
  86. package/dist/helpers/index.js +13 -15
  87. package/dist/helpers/index.js.map +1 -1
  88. package/dist/helpers/interfaces.helper.d.ts +228 -0
  89. package/dist/helpers/interfaces.helper.js +10 -0
  90. package/dist/helpers/interfaces.helper.js.map +1 -0
  91. package/dist/helpers/manifest.helper.d.ts +0 -1
  92. package/dist/helpers/manifest.helper.js +0 -1
  93. package/dist/helpers/notify.helper.d.ts +13 -5
  94. package/dist/helpers/notify.helper.js +1 -2
  95. package/dist/helpers/registry.js +7 -10
  96. package/dist/helpers/registry.js.map +1 -1
  97. package/dist/helpers/utility.helper.d.ts +6 -1
  98. package/dist/helpers/utility.helper.js +9 -13
  99. package/dist/helpers/utility.helper.js.map +1 -1
  100. package/dist/helpers/websocket.helper.d.ts +1 -2
  101. package/dist/helpers/websocket.helper.js +1 -2
  102. package/dist/index.js +5 -8
  103. package/dist/index.js.map +1 -1
  104. package/dist/mock_assistant/extensions/area.extension.d.ts +8 -0
  105. package/dist/mock_assistant/extensions/area.extension.js +51 -0
  106. package/dist/mock_assistant/extensions/area.extension.js.map +1 -0
  107. package/dist/mock_assistant/extensions/config.extension.d.ts +14 -0
  108. package/dist/mock_assistant/extensions/config.extension.js +29 -0
  109. package/dist/mock_assistant/extensions/config.extension.js.map +1 -0
  110. package/dist/mock_assistant/extensions/device.extension.d.ts +8 -0
  111. package/dist/mock_assistant/extensions/device.extension.js +33 -0
  112. package/dist/mock_assistant/extensions/device.extension.js.map +1 -0
  113. package/dist/mock_assistant/extensions/entity-registry.extension.d.ts +13 -0
  114. package/dist/mock_assistant/extensions/entity-registry.extension.js +28 -0
  115. package/dist/mock_assistant/extensions/entity-registry.extension.js.map +1 -0
  116. package/dist/mock_assistant/extensions/entity.extension.d.ts +30 -0
  117. package/dist/mock_assistant/extensions/entity.extension.js +77 -0
  118. package/dist/mock_assistant/extensions/entity.extension.js.map +1 -0
  119. package/dist/mock_assistant/extensions/events.extension.d.ts +1 -1
  120. package/dist/mock_assistant/extensions/events.extension.js +3 -6
  121. package/dist/mock_assistant/extensions/events.extension.js.map +1 -1
  122. package/dist/mock_assistant/extensions/fetch.extension.d.ts +1 -0
  123. package/dist/mock_assistant/extensions/fetch.extension.js +4 -0
  124. package/dist/mock_assistant/extensions/fetch.extension.js.map +1 -0
  125. package/dist/mock_assistant/extensions/fixtures.extension.d.ts +1 -1
  126. package/dist/mock_assistant/extensions/fixtures.extension.js +29 -33
  127. package/dist/mock_assistant/extensions/fixtures.extension.js.map +1 -1
  128. package/dist/mock_assistant/extensions/floor.extension.d.ts +8 -0
  129. package/dist/mock_assistant/extensions/floor.extension.js +51 -0
  130. package/dist/mock_assistant/extensions/floor.extension.js.map +1 -0
  131. package/dist/mock_assistant/extensions/index.d.ts +10 -0
  132. package/dist/mock_assistant/extensions/index.js +12 -5
  133. package/dist/mock_assistant/extensions/index.js.map +1 -1
  134. package/dist/mock_assistant/extensions/label.extension.d.ts +8 -0
  135. package/dist/mock_assistant/extensions/label.extension.js +51 -0
  136. package/dist/mock_assistant/extensions/label.extension.js.map +1 -0
  137. package/dist/mock_assistant/extensions/services.extension.d.ts +12 -0
  138. package/dist/mock_assistant/extensions/services.extension.js +20 -0
  139. package/dist/mock_assistant/extensions/services.extension.js.map +1 -0
  140. package/dist/mock_assistant/extensions/websocket-api.extension.d.ts +15 -0
  141. package/dist/mock_assistant/extensions/websocket-api.extension.js +68 -0
  142. package/dist/mock_assistant/extensions/websocket-api.extension.js.map +1 -0
  143. package/dist/mock_assistant/extensions/zone.extension.d.ts +8 -0
  144. package/dist/mock_assistant/extensions/zone.extension.js +51 -0
  145. package/dist/mock_assistant/extensions/zone.extension.js.map +1 -0
  146. package/dist/mock_assistant/helpers/fixtures.js +1 -2
  147. package/dist/mock_assistant/helpers/index.d.ts +0 -1
  148. package/dist/mock_assistant/helpers/index.js +1 -5
  149. package/dist/mock_assistant/helpers/index.js.map +1 -1
  150. package/dist/mock_assistant/index.js +3 -6
  151. package/dist/mock_assistant/index.js.map +1 -1
  152. package/dist/mock_assistant/main.js +11 -15
  153. package/dist/mock_assistant/main.js.map +1 -1
  154. package/dist/mock_assistant/mock-assistant.module.d.ts +156 -3
  155. package/dist/mock_assistant/mock-assistant.module.js +56 -11
  156. package/dist/mock_assistant/mock-assistant.module.js.map +1 -1
  157. package/dist/quickboot.module.js +5 -8
  158. package/dist/quickboot.module.js.map +1 -1
  159. package/dist/testing/area.spec.js +106 -194
  160. package/dist/testing/area.spec.js.map +1 -1
  161. package/dist/testing/backup.spec.js +97 -139
  162. package/dist/testing/backup.spec.js.map +1 -1
  163. package/dist/testing/config.spec.js +79 -153
  164. package/dist/testing/config.spec.js.map +1 -1
  165. package/dist/testing/device.spec.js +35 -69
  166. package/dist/testing/device.spec.js.map +1 -1
  167. package/dist/testing/entity.spec.js +94 -149
  168. package/dist/testing/entity.spec.js.map +1 -1
  169. package/dist/testing/events.spec.js +33 -57
  170. package/dist/testing/events.spec.js.map +1 -1
  171. package/dist/testing/fetch-api.spec.js +242 -427
  172. package/dist/testing/fetch-api.spec.js.map +1 -1
  173. package/dist/testing/fixtures.spec.d.ts +1 -0
  174. package/dist/testing/fixtures.spec.js +150 -0
  175. package/dist/testing/fixtures.spec.js.map +1 -0
  176. package/dist/testing/floor.spec.js +106 -194
  177. package/dist/testing/floor.spec.js.map +1 -1
  178. package/dist/testing/id-by.spec.js +68 -107
  179. package/dist/testing/id-by.spec.js.map +1 -1
  180. package/dist/testing/label.spec.js +106 -194
  181. package/dist/testing/label.spec.js.map +1 -1
  182. package/dist/testing/ref-by.spec.js +155 -219
  183. package/dist/testing/ref-by.spec.js.map +1 -1
  184. package/dist/testing/websocket.spec.d.ts +1 -8
  185. package/dist/testing/websocket.spec.js +35 -50
  186. package/dist/testing/websocket.spec.js.map +1 -1
  187. package/dist/testing/workflow.spec.js +82 -81
  188. package/dist/testing/workflow.spec.js.map +1 -1
  189. package/dist/testing/zone.spec.js +61 -113
  190. package/dist/testing/zone.spec.js.map +1 -1
  191. package/package.json +62 -44
  192. package/scripts/mock-assistant.sh +5 -0
  193. package/scripts/run-e2e.sh +7 -0
  194. package/scripts/test.sh +2 -0
  195. package/src/dynamic.ts +4254 -0
  196. package/src/extensions/area.extension.ts +118 -0
  197. package/src/extensions/backup.extension.ts +63 -0
  198. package/src/extensions/call-proxy.extension.ts +113 -0
  199. package/src/extensions/config.extension.ts +119 -0
  200. package/src/extensions/conversation.extension.ts +46 -0
  201. package/src/extensions/device.extension.ts +56 -0
  202. package/src/extensions/entity.extension.ts +344 -0
  203. package/src/extensions/events.extension.ts +25 -0
  204. package/src/extensions/fetch-api.extension.ts +269 -0
  205. package/src/extensions/floor.extension.ts +76 -0
  206. package/src/extensions/id-by.extension.ts +157 -0
  207. package/src/extensions/index.ts +16 -0
  208. package/src/extensions/internal.extension.ts +145 -0
  209. package/src/extensions/label.extension.ts +83 -0
  210. package/src/extensions/reference.extension.ts +330 -0
  211. package/src/extensions/registry.extension.ts +44 -0
  212. package/src/extensions/websocket-api.extension.ts +554 -0
  213. package/src/extensions/zone.extension.ts +69 -0
  214. package/src/hass.module.ts +217 -0
  215. package/src/helpers/backup.helper.ts +11 -0
  216. package/src/helpers/constants.helper.ts +30 -0
  217. package/src/helpers/device.helper.ts +25 -0
  218. package/src/helpers/entity-state.helper.ts +171 -0
  219. package/src/helpers/features.helper.ts +580 -0
  220. package/src/helpers/fetch/calendar.ts +54 -0
  221. package/src/helpers/fetch/configuration.ts +75 -0
  222. package/src/helpers/fetch/index.ts +5 -0
  223. package/src/helpers/fetch/server-log.ts +28 -0
  224. package/src/helpers/fetch/service-list.ts +64 -0
  225. package/src/helpers/fetch/weather-forecasts.ts +86 -0
  226. package/src/helpers/fetch.helper.ts +328 -0
  227. package/src/helpers/id-by.helper.ts +53 -0
  228. package/src/helpers/index.ts +13 -0
  229. package/src/helpers/interfaces.helper.ts +340 -0
  230. package/src/helpers/manifest.helper.ts +0 -0
  231. package/src/helpers/notify.helper.ts +302 -0
  232. package/src/helpers/registry.ts +281 -0
  233. package/src/helpers/utility.helper.ts +147 -0
  234. package/src/helpers/websocket.helper.ts +117 -0
  235. package/src/index.ts +5 -0
  236. package/src/mock_assistant/extensions/area.extension.ts +62 -0
  237. package/src/mock_assistant/extensions/config.extension.ts +33 -0
  238. package/src/mock_assistant/extensions/device.extension.ts +44 -0
  239. package/src/mock_assistant/extensions/entity-registry.extension.ts +41 -0
  240. package/src/mock_assistant/extensions/entity.extension.ts +114 -0
  241. package/src/mock_assistant/extensions/events.extension.ts +37 -0
  242. package/src/mock_assistant/extensions/fetch.extension.ts +3 -0
  243. package/src/mock_assistant/extensions/fixtures.extension.ts +79 -0
  244. package/src/mock_assistant/extensions/floor.extension.ts +64 -0
  245. package/src/mock_assistant/extensions/index.ts +12 -0
  246. package/src/mock_assistant/extensions/label.extension.ts +64 -0
  247. package/src/mock_assistant/extensions/services.extension.ts +25 -0
  248. package/src/mock_assistant/extensions/websocket-api.extension.ts +84 -0
  249. package/src/mock_assistant/extensions/zone.extension.ts +65 -0
  250. package/src/mock_assistant/helpers/fixtures.ts +22 -0
  251. package/src/mock_assistant/helpers/index.ts +1 -0
  252. package/src/mock_assistant/index.ts +3 -0
  253. package/src/mock_assistant/main.ts +46 -0
  254. package/src/mock_assistant/mock-assistant.module.ts +90 -0
  255. package/src/quickboot.module.ts +23 -0
  256. package/src/testing/area.spec.ts +189 -0
  257. package/src/testing/backup.spec.ts +157 -0
  258. package/src/testing/config.spec.ts +188 -0
  259. package/src/testing/device.spec.ts +89 -0
  260. package/src/testing/entity.spec.ts +171 -0
  261. package/src/testing/events.spec.ts +78 -0
  262. package/src/testing/fetch-api.spec.ts +410 -0
  263. package/src/testing/fixtures.spec.ts +158 -0
  264. package/src/testing/floor.spec.ts +186 -0
  265. package/src/testing/id-by.spec.ts +140 -0
  266. package/src/testing/label.spec.ts +186 -0
  267. package/src/testing/ref-by.spec.ts +300 -0
  268. package/src/testing/websocket.spec.ts +63 -0
  269. package/src/testing/workflow.spec.ts +195 -0
  270. package/src/testing/zone.spec.ts +109 -0
  271. package/dist/helpers/metrics.helper.d.ts +0 -29
  272. package/dist/helpers/metrics.helper.js +0 -62
  273. package/dist/helpers/metrics.helper.js.map +0 -1
  274. package/dist/mock_assistant/helpers/utils.d.ts +0 -4
  275. package/dist/mock_assistant/helpers/utils.js +0 -57
  276. package/dist/mock_assistant/helpers/utils.js.map +0 -1
@@ -0,0 +1,186 @@
1
+ import { sleep } from "@digital-alchemy/core";
2
+
3
+ import { TFloorId } from "../dynamic";
4
+ import { FLOOR_REGISTRY_UPDATED, FloorDetails } from "../helpers";
5
+ import { hassTestRunner } from "../mock_assistant";
6
+
7
+ describe("Floor", () => {
8
+ const EXAMPLE_FLOOR = {
9
+ aliases: [],
10
+ floor_id: "upstairs",
11
+ icon: null,
12
+ level: 2,
13
+ name: "Upstairs",
14
+ } as unknown as FloorDetails;
15
+
16
+ afterEach(async () => {
17
+ await hassTestRunner.teardown();
18
+ jest.restoreAllMocks();
19
+ });
20
+
21
+ describe("Lifecycle", () => {
22
+ it("should force values to be available before ready", async () => {
23
+ expect.assertions(1);
24
+ await hassTestRunner.run(({ lifecycle, hass }) => {
25
+ const spy = jest
26
+ .spyOn(hass.socket, "sendMessage")
27
+ .mockImplementation(async () => [EXAMPLE_FLOOR]);
28
+ lifecycle.onReady(async () => {
29
+ await hass.floor.list();
30
+ expect(spy).toHaveBeenCalledWith(
31
+ expect.objectContaining({ type: "config/floor_registry/list" }),
32
+ );
33
+ });
34
+ });
35
+ });
36
+ });
37
+
38
+ describe("API", () => {
39
+ describe("Formatting", () => {
40
+ it("should call list properly", async () => {
41
+ expect.assertions(1);
42
+ await hassTestRunner.run(({ lifecycle, hass }) => {
43
+ const spy = jest.spyOn(hass.socket, "sendMessage").mockImplementation(async () => []);
44
+ lifecycle.onReady(async () => {
45
+ await hass.floor.list();
46
+ expect(spy).toHaveBeenCalledWith(
47
+ expect.objectContaining({ type: "config/floor_registry/list" }),
48
+ );
49
+ });
50
+ });
51
+ });
52
+
53
+ it("should call update properly", async () => {
54
+ expect.assertions(1);
55
+ await hassTestRunner.run(({ lifecycle, hass, event }) => {
56
+ const spy = jest
57
+ .spyOn(hass.socket, "sendMessage")
58
+ .mockImplementation(async () => undefined);
59
+ lifecycle.onReady(async () => {
60
+ setImmediate(() => event.emit(FLOOR_REGISTRY_UPDATED));
61
+ await hass.floor.update(EXAMPLE_FLOOR);
62
+
63
+ expect(spy).toHaveBeenCalledWith({
64
+ type: "config/floor_registry/update",
65
+ ...EXAMPLE_FLOOR,
66
+ });
67
+ });
68
+ });
69
+ });
70
+
71
+ it("should call delete properly", async () => {
72
+ expect.assertions(1);
73
+ await hassTestRunner.run(({ lifecycle, hass, event }) => {
74
+ const spy = jest
75
+ .spyOn(hass.socket, "sendMessage")
76
+ .mockImplementation(async () => undefined);
77
+ lifecycle.onReady(async () => {
78
+ setImmediate(() => event.emit(FLOOR_REGISTRY_UPDATED));
79
+ await hass.floor.delete(EXAMPLE_FLOOR.floor_id);
80
+
81
+ expect(spy).toHaveBeenCalledWith({
82
+ floor_id: "upstairs",
83
+ type: "config/floor_registry/delete",
84
+ });
85
+ });
86
+ });
87
+ });
88
+
89
+ it("should call create properly", async () => {
90
+ expect.assertions(1);
91
+ await hassTestRunner.run(({ lifecycle, hass, event }) => {
92
+ const spy = jest
93
+ .spyOn(hass.socket, "sendMessage")
94
+ .mockImplementation(async () => undefined);
95
+ lifecycle.onReady(async () => {
96
+ setImmediate(() => event.emit(FLOOR_REGISTRY_UPDATED));
97
+ await hass.floor.create(EXAMPLE_FLOOR);
98
+
99
+ expect(spy).toHaveBeenCalledWith({
100
+ type: "config/floor_registry/create",
101
+ ...EXAMPLE_FLOOR,
102
+ });
103
+ });
104
+ });
105
+ });
106
+ });
107
+
108
+ describe("Order of operations", () => {
109
+ it("should wait for an update before returning when updating", async () => {
110
+ expect.assertions(1);
111
+ await hassTestRunner.run(({ lifecycle, hass, event }) => {
112
+ jest.spyOn(hass.socket, "sendMessage").mockImplementation(async () => undefined);
113
+ lifecycle.onReady(async () => {
114
+ const response = hass.floor.update(EXAMPLE_FLOOR);
115
+ let order = "";
116
+ setTimeout(() => {
117
+ order += "a";
118
+ event.emit(FLOOR_REGISTRY_UPDATED);
119
+ }, 5);
120
+ await response;
121
+ order += "b";
122
+ expect(order).toEqual("ab");
123
+ });
124
+ });
125
+ });
126
+
127
+ it("should debounce updates properly", async () => {
128
+ expect.assertions(1);
129
+ await hassTestRunner.run(({ lifecycle, hass }) => {
130
+ jest.spyOn(hass.socket, "sendMessage").mockImplementation(async () => undefined);
131
+ let counter = 0;
132
+ hass.events.onFloorRegistryUpdate(() => counter++);
133
+ lifecycle.onReady(async () => {
134
+ setImmediate(async () => {
135
+ hass.socket.socketEvents.emit("floor_registry_updated");
136
+ await sleep(5);
137
+ hass.socket.socketEvents.emit("floor_registry_updated");
138
+ await sleep(5);
139
+ hass.socket.socketEvents.emit("floor_registry_updated");
140
+ await sleep(75);
141
+ hass.socket.socketEvents.emit("floor_registry_updated");
142
+ });
143
+ await sleep(200);
144
+ expect(counter).toBe(2);
145
+ });
146
+ });
147
+ });
148
+
149
+ it("should wait for an update before returning when deleting", async () => {
150
+ expect.assertions(1);
151
+ await hassTestRunner.run(({ lifecycle, hass, event }) => {
152
+ jest.spyOn(hass.socket, "sendMessage").mockImplementation(async () => undefined);
153
+ lifecycle.onReady(async () => {
154
+ const response = hass.floor.delete("example_floor" as TFloorId);
155
+ let order = "";
156
+ setTimeout(() => {
157
+ order += "a";
158
+ event.emit(FLOOR_REGISTRY_UPDATED);
159
+ }, 5);
160
+ await response;
161
+ order += "b";
162
+ expect(order).toEqual("ab");
163
+ });
164
+ });
165
+ });
166
+
167
+ it("should wait for an update before returning when creating", async () => {
168
+ expect.assertions(1);
169
+ await hassTestRunner.run(({ lifecycle, hass, event }) => {
170
+ jest.spyOn(hass.socket, "sendMessage").mockImplementation(async () => undefined);
171
+ lifecycle.onReady(async () => {
172
+ const response = hass.floor.create(EXAMPLE_FLOOR);
173
+ let order = "";
174
+ setTimeout(() => {
175
+ order += "a";
176
+ event.emit(FLOOR_REGISTRY_UPDATED);
177
+ }, 5);
178
+ await response;
179
+ order += "b";
180
+ expect(order).toEqual("ab");
181
+ });
182
+ });
183
+ });
184
+ });
185
+ });
186
+ });
@@ -0,0 +1,140 @@
1
+ import { PICK_ENTITY } from "../helpers";
2
+ import { hassTestRunner } from "../mock_assistant";
3
+
4
+ describe("ID By", () => {
5
+ afterEach(async () => {
6
+ await hassTestRunner.teardown();
7
+ jest.restoreAllMocks();
8
+ });
9
+
10
+ describe("area", () => {
11
+ it("find entities by area", async () => {
12
+ expect.assertions(2);
13
+ await hassTestRunner.run(({ lifecycle, hass }) => {
14
+ lifecycle.onReady(() => {
15
+ const bedroom = hass.idBy.area("bedroom");
16
+ const kitchen = hass.idBy.area("kitchen");
17
+ expect(bedroom.length).toBe(2);
18
+ expect(kitchen.length).toBe(1);
19
+ });
20
+ });
21
+ });
22
+
23
+ it("finds entities only related by device", async () => {
24
+ expect.assertions(1);
25
+ await hassTestRunner.run(({ lifecycle, hass }) => {
26
+ lifecycle.onReady(() => {
27
+ // merges 1 from direct area, 2 via device
28
+ // ignores 2 from the device assigned to another area
29
+ const list = hass.idBy.area("test") as PICK_ENTITY[];
30
+ const expected = [
31
+ "sensor.sun_next_dusk",
32
+ "climate.hallway_thermostat",
33
+ "binary_sensor.garage_door",
34
+ ] as PICK_ENTITY[];
35
+ expect(expected.every(expected => list.includes(expected))).toBe(true);
36
+ });
37
+ });
38
+ });
39
+
40
+ it("find entities by area limiting by domain", async () => {
41
+ expect.assertions(2);
42
+ await hassTestRunner.run(({ lifecycle, hass }) => {
43
+ lifecycle.onReady(() => {
44
+ const bedroom = hass.idBy.area("bedroom", "light");
45
+ const kitchen = hass.idBy.area("kitchen", "light");
46
+ expect(bedroom.length).toBe(1);
47
+ expect(kitchen.length).toBe(0);
48
+ });
49
+ });
50
+ });
51
+ });
52
+
53
+ describe("label", () => {
54
+ it("find entities by label", async () => {
55
+ expect.assertions(1);
56
+ await hassTestRunner.run(({ lifecycle, hass }) => {
57
+ lifecycle.onReady(() => {
58
+ const synapse = hass.idBy.label("synapse");
59
+ expect(synapse.length).toBe(7);
60
+ });
61
+ });
62
+ });
63
+
64
+ it("find entities by label limiting by domain", async () => {
65
+ expect.assertions(1);
66
+ await hassTestRunner.run(({ lifecycle, hass }) => {
67
+ lifecycle.onReady(() => {
68
+ const synapse = hass.idBy.label("synapse", "light");
69
+ expect(synapse.length).toBe(0);
70
+ });
71
+ });
72
+ });
73
+ });
74
+
75
+ describe("device", () => {
76
+ it("find entities by device", async () => {
77
+ expect.assertions(1);
78
+ await hassTestRunner.run(({ lifecycle, hass }) => {
79
+ lifecycle.onReady(() => {
80
+ const synapse = hass.idBy.device("308e39cf50a9fc6c30b4110724ed1f2e");
81
+ expect(synapse.length).toBe(9);
82
+ });
83
+ });
84
+ });
85
+
86
+ it("find entities by device limiting by domain", async () => {
87
+ expect.assertions(1);
88
+ await hassTestRunner.run(({ lifecycle, hass }) => {
89
+ lifecycle.onReady(() => {
90
+ const synapse = hass.idBy.device("308e39cf50a9fc6c30b4110724ed1f2e", "light");
91
+ expect(synapse.length).toBe(0);
92
+ });
93
+ });
94
+ });
95
+ });
96
+
97
+ describe("platform", () => {
98
+ it("find entities by platform", async () => {
99
+ expect.assertions(1);
100
+ await hassTestRunner.run(({ lifecycle, hass }) => {
101
+ lifecycle.onReady(() => {
102
+ const synapse = hass.idBy.platform("synapse");
103
+ expect(synapse.length).toBe(7);
104
+ });
105
+ });
106
+ });
107
+
108
+ it("find entities by platform limiting by domain", async () => {
109
+ expect.assertions(1);
110
+ await hassTestRunner.run(({ lifecycle, hass }) => {
111
+ lifecycle.onReady(() => {
112
+ const synapse = hass.idBy.platform("synapse", "light");
113
+ expect(synapse.length).toBe(0);
114
+ });
115
+ });
116
+ });
117
+ });
118
+
119
+ describe("floor", () => {
120
+ it("find entities by floor", async () => {
121
+ expect.assertions(1);
122
+ await hassTestRunner.run(({ lifecycle, hass }) => {
123
+ lifecycle.onReady(() => {
124
+ const synapse = hass.idBy.floor("downstairs");
125
+ expect(synapse.length).toBe(3);
126
+ });
127
+ });
128
+ });
129
+
130
+ it("find entities by floor limiting by domain", async () => {
131
+ expect.assertions(1);
132
+ await hassTestRunner.run(({ lifecycle, hass }) => {
133
+ lifecycle.onReady(() => {
134
+ const synapse = hass.idBy.floor("downstairs", "light");
135
+ expect(synapse.length).toBe(0);
136
+ });
137
+ });
138
+ });
139
+ });
140
+ });
@@ -0,0 +1,186 @@
1
+ import { sleep } from "@digital-alchemy/core";
2
+
3
+ import { TLabelId } from "../dynamic";
4
+ import { LABEL_REGISTRY_UPDATED, LabelDefinition } from "../helpers";
5
+ import { hassTestRunner } from "../mock_assistant";
6
+
7
+ describe("Label", () => {
8
+ const EXAMPLE_LABEL = {
9
+ color: "accent",
10
+ description: "test",
11
+ icon: "mdi:brain",
12
+ label_id: "synapse",
13
+ name: "synapse",
14
+ } as unknown as LabelDefinition;
15
+
16
+ afterEach(async () => {
17
+ await hassTestRunner.teardown();
18
+ jest.restoreAllMocks();
19
+ });
20
+
21
+ describe("Lifecycle", () => {
22
+ it("should force values to be available before ready", async () => {
23
+ expect.assertions(1);
24
+ await hassTestRunner.run(({ lifecycle, hass }) => {
25
+ const spy = jest
26
+ .spyOn(hass.socket, "sendMessage")
27
+ .mockImplementation(async () => [EXAMPLE_LABEL]);
28
+ lifecycle.onReady(async () => {
29
+ await hass.label.list();
30
+ expect(spy).toHaveBeenCalledWith(
31
+ expect.objectContaining({ type: "config/label_registry/list" }),
32
+ );
33
+ });
34
+ });
35
+ });
36
+ });
37
+
38
+ describe("API", () => {
39
+ describe("Formatting", () => {
40
+ it("should call list properly", async () => {
41
+ expect.assertions(1);
42
+ await hassTestRunner.run(({ lifecycle, hass }) => {
43
+ const spy = jest.spyOn(hass.socket, "sendMessage").mockImplementation(async () => []);
44
+ lifecycle.onReady(async () => {
45
+ await hass.label.list();
46
+ expect(spy).toHaveBeenCalledWith(
47
+ expect.objectContaining({ type: "config/label_registry/list" }),
48
+ );
49
+ });
50
+ });
51
+ });
52
+
53
+ it("should call update properly", async () => {
54
+ expect.assertions(1);
55
+ await hassTestRunner.run(({ lifecycle, hass, event }) => {
56
+ const spy = jest
57
+ .spyOn(hass.socket, "sendMessage")
58
+ .mockImplementation(async () => undefined);
59
+ lifecycle.onReady(async () => {
60
+ setImmediate(() => event.emit(LABEL_REGISTRY_UPDATED));
61
+ await hass.label.update(EXAMPLE_LABEL);
62
+
63
+ expect(spy).toHaveBeenCalledWith({
64
+ type: "config/label_registry/update",
65
+ ...EXAMPLE_LABEL,
66
+ });
67
+ });
68
+ });
69
+ });
70
+
71
+ it("should call delete properly", async () => {
72
+ expect.assertions(1);
73
+ await hassTestRunner.run(({ lifecycle, hass, event }) => {
74
+ const spy = jest
75
+ .spyOn(hass.socket, "sendMessage")
76
+ .mockImplementation(async () => undefined);
77
+ lifecycle.onReady(async () => {
78
+ setImmediate(() => event.emit(LABEL_REGISTRY_UPDATED));
79
+ await hass.label.delete(EXAMPLE_LABEL.label_id);
80
+
81
+ expect(spy).toHaveBeenCalledWith({
82
+ label_id: "synapse",
83
+ type: "config/label_registry/delete",
84
+ });
85
+ });
86
+ });
87
+ });
88
+
89
+ it("should call create properly", async () => {
90
+ expect.assertions(1);
91
+ await hassTestRunner.run(({ lifecycle, hass, event }) => {
92
+ const spy = jest
93
+ .spyOn(hass.socket, "sendMessage")
94
+ .mockImplementation(async () => undefined);
95
+ lifecycle.onReady(async () => {
96
+ setImmediate(() => event.emit(LABEL_REGISTRY_UPDATED));
97
+ await hass.label.create(EXAMPLE_LABEL);
98
+
99
+ expect(spy).toHaveBeenCalledWith({
100
+ type: "config/label_registry/create",
101
+ ...EXAMPLE_LABEL,
102
+ });
103
+ });
104
+ });
105
+ });
106
+ });
107
+
108
+ describe("Order of operations", () => {
109
+ it("should wait for an update before returning when updating", async () => {
110
+ expect.assertions(1);
111
+ await hassTestRunner.run(({ lifecycle, hass, event }) => {
112
+ jest.spyOn(hass.socket, "sendMessage").mockImplementation(async () => undefined);
113
+ lifecycle.onReady(async () => {
114
+ const response = hass.label.update(EXAMPLE_LABEL);
115
+ let order = "";
116
+ setTimeout(() => {
117
+ order += "a";
118
+ event.emit(LABEL_REGISTRY_UPDATED);
119
+ }, 5);
120
+ await response;
121
+ order += "b";
122
+ expect(order).toEqual("ab");
123
+ });
124
+ });
125
+ });
126
+
127
+ it("should debounce updates properly", async () => {
128
+ expect.assertions(1);
129
+ await hassTestRunner.run(({ lifecycle, hass }) => {
130
+ jest.spyOn(hass.socket, "sendMessage").mockImplementation(async () => undefined);
131
+ let counter = 0;
132
+ hass.events.onLabelRegistryUpdate(() => counter++);
133
+ lifecycle.onReady(async () => {
134
+ setImmediate(async () => {
135
+ hass.socket.socketEvents.emit("label_registry_updated");
136
+ await sleep(5);
137
+ hass.socket.socketEvents.emit("label_registry_updated");
138
+ await sleep(5);
139
+ hass.socket.socketEvents.emit("label_registry_updated");
140
+ await sleep(75);
141
+ hass.socket.socketEvents.emit("label_registry_updated");
142
+ });
143
+ await sleep(200);
144
+ expect(counter).toBe(2);
145
+ });
146
+ });
147
+ });
148
+
149
+ it("should wait for an update before returning when deleting", async () => {
150
+ expect.assertions(1);
151
+ await hassTestRunner.run(({ lifecycle, hass, event }) => {
152
+ jest.spyOn(hass.socket, "sendMessage").mockImplementation(async () => undefined);
153
+ lifecycle.onReady(async () => {
154
+ const response = hass.label.delete("example_label" as TLabelId);
155
+ let order = "";
156
+ setTimeout(() => {
157
+ order += "a";
158
+ event.emit(LABEL_REGISTRY_UPDATED);
159
+ }, 5);
160
+ await response;
161
+ order += "b";
162
+ expect(order).toEqual("ab");
163
+ });
164
+ });
165
+ });
166
+
167
+ it("should wait for an update before returning when creating", async () => {
168
+ expect.assertions(1);
169
+ await hassTestRunner.run(({ lifecycle, hass, event }) => {
170
+ jest.spyOn(hass.socket, "sendMessage").mockImplementation(async () => undefined);
171
+ lifecycle.onReady(async () => {
172
+ const response = hass.label.create(EXAMPLE_LABEL);
173
+ let order = "";
174
+ setTimeout(() => {
175
+ order += "a";
176
+ event.emit(LABEL_REGISTRY_UPDATED);
177
+ }, 5);
178
+ await response;
179
+ order += "b";
180
+ expect(order).toEqual("ab");
181
+ });
182
+ });
183
+ });
184
+ });
185
+ });
186
+ });