@switchbot/homebridge-switchbot 5.0.0-beta.40 → 5.0.0-beta.42

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.
@@ -30,10 +30,12 @@ class PluginUiServer extends HomebridgePluginUiServer {
30
30
  });
31
31
  }
32
32
  // Return the array
33
+ console.warn(`[Homebridge UI] getCachedAccessories returning ${devicesToReturn.length} device(s)`);
33
34
  return devicesToReturn;
34
35
  }
35
- catch {
36
+ catch (e) {
36
37
  // Just return an empty accessory list in case of any errors
38
+ console.error(`[Homebridge UI] getCachedAccessories error: ${e?.message ?? e}`);
37
39
  return [];
38
40
  }
39
41
  };
@@ -47,31 +49,52 @@ class PluginUiServer extends HomebridgePluginUiServer {
47
49
  const devicesToReturn = [];
48
50
  const accFile = `${this.homebridgeStoragePath}/accessories/cachedAccessories`;
49
51
  const matterFile = `${this.homebridgeStoragePath}/accessories/cachedMatterAccessories`;
52
+ // Log all files in the accessories directory for debugging
53
+ try {
54
+ const accessoriesDir = `${this.homebridgeStoragePath}/accessories`;
55
+ if (fs.existsSync(accessoriesDir)) {
56
+ const files = fs.readdirSync(accessoriesDir);
57
+ console.warn(`[Homebridge UI] Files in accessories directory: ${files.join(', ')}`);
58
+ }
59
+ }
60
+ catch (e) {
61
+ console.error(`[Homebridge UI] Error listing accessories directory: ${e?.message ?? e}`);
62
+ }
63
+ console.warn(`[Homebridge UI] Checking for cached files:`);
64
+ console.warn(`[Homebridge UI] - cachedAccessories: ${fs.existsSync(accFile)}`);
65
+ console.warn(`[Homebridge UI] - cachedMatterAccessories: ${fs.existsSync(matterFile)}`);
50
66
  const readAndCollect = (filePath) => {
51
67
  if (!fs.existsSync(filePath)) {
52
68
  return;
53
69
  }
54
70
  try {
55
71
  const parsed = JSON.parse(fs.readFileSync(filePath, 'utf8'));
72
+ console.warn(`[Homebridge UI] - ${filePath}: found ${parsed.length} total entries`);
73
+ let matchCount = 0;
56
74
  parsed.forEach((entry) => {
57
75
  // Entry shape varies between Homebridge versions; try common locations
58
76
  const pluginName = entry.plugin || entry?.accessory?.plugin || entry?.accessory?.pluginName;
59
77
  const acc = entry.accessory ?? entry;
60
78
  if (pluginNames.includes(pluginName)) {
61
79
  devicesToReturn.push(acc);
80
+ matchCount++;
62
81
  }
63
82
  });
83
+ console.warn(`[Homebridge UI] - ${filePath}: matched ${matchCount} SwitchBot entries`);
64
84
  }
65
- catch {
85
+ catch (e) {
66
86
  // ignore parse errors for a single file
87
+ console.error(`[Homebridge UI] - ${filePath}: parse error - ${e?.message ?? e}`);
67
88
  }
68
89
  };
69
90
  // Read both canonical files (some Homebridge versions use one or the other)
70
91
  readAndCollect(accFile);
71
92
  readAndCollect(matterFile);
93
+ console.warn(`[Homebridge UI] getCachedMatterAccessories returning ${devicesToReturn.length} device(s)`);
72
94
  return devicesToReturn;
73
95
  }
74
- catch {
96
+ catch (e) {
97
+ console.error(`[Homebridge UI] getCachedMatterAccessories error: ${e?.message ?? e}`);
75
98
  return [];
76
99
  }
77
100
  };
@@ -1 +1 @@
1
- {"version":3,"file":"server.js","sourceRoot":"","sources":["../../src/homebridge-ui/server.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../../src/homebridge-ui/server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=apiRequestTracker.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"apiRequestTracker.test.d.ts","sourceRoot":"","sources":["../../src/test/apiRequestTracker.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,392 @@
1
+ import { existsSync, mkdirSync, readdirSync, readFileSync, rmdirSync, unlinkSync } from 'node:fs';
2
+ import { tmpdir } from 'node:os';
3
+ import { join } from 'node:path';
4
+ import { describe, expect, it, vi } from 'vitest';
5
+ import { ApiRequestTracker } from '../utils.js';
6
+ // Helper to create isolated test environment for each test
7
+ function createTestEnvironment(pluginName = 'SwitchBotTest') {
8
+ const testId = Math.random().toString(36).substring(7);
9
+ const testDir = join(tmpdir(), `switchbot-test-${testId}`);
10
+ // Create test directory
11
+ if (!existsSync(testDir)) {
12
+ mkdirSync(testDir, { recursive: true });
13
+ }
14
+ const testStatsFile = join(testDir, `${pluginName.toLowerCase()}-api-stats.json`);
15
+ // Mock API with a unique storage path per test
16
+ const mockApi = {
17
+ user: {
18
+ storagePath: () => testDir,
19
+ },
20
+ };
21
+ // Mock logger
22
+ const mockLog = {
23
+ info: vi.fn(),
24
+ warn: vi.fn(),
25
+ error: vi.fn(),
26
+ debug: vi.fn(),
27
+ };
28
+ return { mockApi, mockLog, testStatsFile, testDir };
29
+ }
30
+ // Cleanup helper
31
+ function cleanup(testDir) {
32
+ try {
33
+ if (existsSync(testDir)) {
34
+ const files = readdirSync(testDir);
35
+ for (const file of files) {
36
+ try {
37
+ unlinkSync(join(testDir, file));
38
+ }
39
+ catch {
40
+ // ignore
41
+ }
42
+ }
43
+ rmdirSync(testDir);
44
+ }
45
+ }
46
+ catch {
47
+ // ignore
48
+ }
49
+ }
50
+ describe('apiRequestTracker', () => {
51
+ describe('initialization', () => {
52
+ it('should create a new tracker with default limits', () => {
53
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
54
+ try {
55
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest');
56
+ expect(tracker).toBeDefined();
57
+ expect(tracker.getCount()).toBe(0);
58
+ expect(tracker.getDate()).toBe(new Date().toISOString().split('T')[0]);
59
+ }
60
+ finally {
61
+ cleanup(testDir);
62
+ }
63
+ });
64
+ it('should respect custom daily limit', () => {
65
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
66
+ try {
67
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
68
+ dailyLimit: 5000,
69
+ reserveForCommands: 500,
70
+ });
71
+ expect(tracker).toBeDefined();
72
+ }
73
+ finally {
74
+ cleanup(testDir);
75
+ }
76
+ });
77
+ it('should load existing stats from file', () => {
78
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
79
+ try {
80
+ // Create a tracker, increment, and verify persistence
81
+ const tracker1 = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest');
82
+ tracker1.track();
83
+ tracker1.track();
84
+ expect(tracker1.getCount()).toBe(2);
85
+ // Create a new tracker instance and verify it loads the count
86
+ const tracker2 = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest');
87
+ expect(tracker2.getCount()).toBe(2);
88
+ }
89
+ finally {
90
+ cleanup(testDir);
91
+ }
92
+ });
93
+ });
94
+ describe('track() - legacy method', () => {
95
+ it('should increment the counter', () => {
96
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
97
+ try {
98
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest');
99
+ expect(tracker.getCount()).toBe(0);
100
+ tracker.track();
101
+ expect(tracker.getCount()).toBe(1);
102
+ tracker.track();
103
+ expect(tracker.getCount()).toBe(2);
104
+ }
105
+ finally {
106
+ cleanup(testDir);
107
+ }
108
+ });
109
+ it('should persist count to file', () => {
110
+ const { mockApi, mockLog, testStatsFile, testDir } = createTestEnvironment();
111
+ try {
112
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest');
113
+ tracker.track();
114
+ tracker.track();
115
+ tracker.track();
116
+ // Read the stats file directly
117
+ const statsContent = readFileSync(testStatsFile, 'utf8');
118
+ const stats = JSON.parse(statsContent);
119
+ expect(stats.count).toBe(3);
120
+ expect(stats.date).toBe(new Date().toISOString().split('T')[0]);
121
+ }
122
+ finally {
123
+ cleanup(testDir);
124
+ }
125
+ });
126
+ });
127
+ describe('trySpend() - budget enforcement', () => {
128
+ it('should allow commands when under soft cap', () => {
129
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
130
+ try {
131
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
132
+ dailyLimit: 100,
133
+ reserveForCommands: 20,
134
+ });
135
+ // Use 50 requests (well under soft cap of 80)
136
+ for (let i = 0; i < 50; i++) {
137
+ expect(tracker.trySpend('command')).toBe(true);
138
+ }
139
+ expect(tracker.getCount()).toBe(50);
140
+ }
141
+ finally {
142
+ cleanup(testDir);
143
+ }
144
+ });
145
+ it('should allow polling when under soft cap', () => {
146
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
147
+ try {
148
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
149
+ dailyLimit: 100,
150
+ reserveForCommands: 20,
151
+ });
152
+ // Use 50 requests
153
+ for (let i = 0; i < 50; i++) {
154
+ expect(tracker.trySpend('poll')).toBe(true);
155
+ }
156
+ expect(tracker.getCount()).toBe(50);
157
+ }
158
+ finally {
159
+ cleanup(testDir);
160
+ }
161
+ });
162
+ it('should block polling at soft cap when pausePollingAtReserve is true', () => {
163
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
164
+ try {
165
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
166
+ dailyLimit: 100,
167
+ reserveForCommands: 20,
168
+ pausePollingAtReserve: true, // Enable soft cap blocking
169
+ });
170
+ // Use up to soft cap (80 requests)
171
+ for (let i = 0; i < 80; i++) {
172
+ tracker.track();
173
+ }
174
+ expect(tracker.getCount()).toBe(80);
175
+ // Polling should be blocked at soft cap
176
+ expect(tracker.trySpend('poll')).toBe(false);
177
+ expect(tracker.trySpend('discovery')).toBe(false);
178
+ // Commands should still work
179
+ expect(tracker.trySpend('command')).toBe(true);
180
+ expect(tracker.getCount()).toBe(81);
181
+ }
182
+ finally {
183
+ cleanup(testDir);
184
+ }
185
+ });
186
+ it('should block all requests at hard cap', () => {
187
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
188
+ try {
189
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
190
+ dailyLimit: 100,
191
+ reserveForCommands: 20,
192
+ });
193
+ // Use up to hard cap
194
+ for (let i = 0; i < 100; i++) {
195
+ tracker.track();
196
+ }
197
+ expect(tracker.getCount()).toBe(100);
198
+ // All request types should be blocked
199
+ expect(tracker.trySpend('poll')).toBe(false);
200
+ expect(tracker.trySpend('discovery')).toBe(false);
201
+ expect(tracker.trySpend('command')).toBe(false);
202
+ expect(tracker.getCount()).toBe(100);
203
+ }
204
+ finally {
205
+ cleanup(testDir);
206
+ }
207
+ });
208
+ it('should support batch spending', () => {
209
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
210
+ try {
211
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
212
+ dailyLimit: 100,
213
+ reserveForCommands: 20,
214
+ });
215
+ expect(tracker.trySpend('poll', 10)).toBe(true);
216
+ expect(tracker.getCount()).toBe(10);
217
+ expect(tracker.trySpend('command', 5)).toBe(true);
218
+ expect(tracker.getCount()).toBe(15);
219
+ }
220
+ finally {
221
+ cleanup(testDir);
222
+ }
223
+ });
224
+ });
225
+ describe('webhookOnlyOnReserve mode', () => {
226
+ it('should continue polling beyond soft cap when pausePollingAtReserve is false', () => {
227
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
228
+ try {
229
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
230
+ dailyLimit: 100,
231
+ reserveForCommands: 20,
232
+ pausePollingAtReserve: false,
233
+ });
234
+ // Use 85 requests (past soft cap)
235
+ for (let i = 0; i < 85; i++) {
236
+ tracker.track();
237
+ }
238
+ expect(tracker.getCount()).toBe(85);
239
+ // Polling should still work (not paused at reserve)
240
+ expect(tracker.trySpend('poll')).toBe(true);
241
+ expect(tracker.getCount()).toBe(86);
242
+ }
243
+ finally {
244
+ cleanup(testDir);
245
+ }
246
+ });
247
+ it('should stop polling at soft cap when pausePollingAtReserve is true', () => {
248
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
249
+ try {
250
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
251
+ dailyLimit: 100,
252
+ reserveForCommands: 20,
253
+ pausePollingAtReserve: true,
254
+ });
255
+ // Use up to soft cap
256
+ for (let i = 0; i < 80; i++) {
257
+ tracker.track();
258
+ }
259
+ expect(tracker.getCount()).toBe(80);
260
+ // Polling should be blocked
261
+ expect(tracker.trySpend('poll')).toBe(false);
262
+ expect(tracker.getCount()).toBe(80);
263
+ // Commands should still work
264
+ expect(tracker.trySpend('command')).toBe(true);
265
+ expect(tracker.getCount()).toBe(81);
266
+ }
267
+ finally {
268
+ cleanup(testDir);
269
+ }
270
+ });
271
+ });
272
+ describe('warning logs', () => {
273
+ it('should log warning when reaching soft cap with pausePollingAtReserve enabled', () => {
274
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
275
+ try {
276
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
277
+ dailyLimit: 100,
278
+ reserveForCommands: 20,
279
+ pausePollingAtReserve: true, // Enable soft cap warning
280
+ });
281
+ // Use up to soft cap
282
+ for (let i = 0; i < 80; i++) {
283
+ tracker.track();
284
+ }
285
+ // Trigger soft cap warning by attempting poll (will be blocked)
286
+ tracker.trySpend('poll');
287
+ expect(mockLog.warn).toHaveBeenCalledWith(expect.stringContaining('Near daily limit'));
288
+ }
289
+ finally {
290
+ cleanup(testDir);
291
+ }
292
+ });
293
+ it('should log error when reaching hard cap', () => {
294
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
295
+ try {
296
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
297
+ dailyLimit: 100,
298
+ reserveForCommands: 20,
299
+ });
300
+ // Use up to hard cap
301
+ for (let i = 0; i < 100; i++) {
302
+ tracker.track();
303
+ }
304
+ // Trigger hard cap error
305
+ tracker.trySpend('command');
306
+ expect(mockLog.error).toHaveBeenCalledWith(expect.stringContaining('Daily limit'));
307
+ }
308
+ finally {
309
+ cleanup(testDir);
310
+ }
311
+ });
312
+ });
313
+ describe('hourly logging', () => {
314
+ it('should log immediately on startup', () => {
315
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
316
+ try {
317
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest');
318
+ tracker.startHourlyLogging();
319
+ tracker.stopHourlyLogging();
320
+ expect(mockLog.info).toHaveBeenCalledWith(expect.stringContaining('[API Stats] Today'));
321
+ }
322
+ finally {
323
+ cleanup(testDir);
324
+ }
325
+ });
326
+ it('should stop logging when stopHourlyLogging is called', () => {
327
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
328
+ try {
329
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest');
330
+ tracker.startHourlyLogging();
331
+ tracker.stopHourlyLogging();
332
+ // Should not throw
333
+ expect(true).toBe(true);
334
+ }
335
+ finally {
336
+ cleanup(testDir);
337
+ }
338
+ });
339
+ });
340
+ describe('edge cases', () => {
341
+ it('should handle zero daily limit', () => {
342
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
343
+ try {
344
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
345
+ dailyLimit: 0,
346
+ reserveForCommands: 0,
347
+ });
348
+ // All requests should be blocked immediately
349
+ expect(tracker.trySpend('poll')).toBe(false);
350
+ expect(tracker.trySpend('command')).toBe(false);
351
+ expect(tracker.getCount()).toBe(0);
352
+ }
353
+ finally {
354
+ cleanup(testDir);
355
+ }
356
+ });
357
+ it('should handle reserve larger than limit', () => {
358
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
359
+ try {
360
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
361
+ dailyLimit: 100,
362
+ reserveForCommands: 150,
363
+ pausePollingAtReserve: true, // Enable soft cap blocking
364
+ });
365
+ // Soft cap would be negative (100 - 150 = -50), clamped to 0
366
+ // With pausePollingAtReserve=true, polling should be blocked immediately
367
+ expect(tracker.trySpend('poll')).toBe(false);
368
+ // Commands up to hard cap should work
369
+ expect(tracker.trySpend('command')).toBe(true);
370
+ }
371
+ finally {
372
+ cleanup(testDir);
373
+ }
374
+ });
375
+ it('should handle negative values in config', () => {
376
+ const { mockApi, mockLog, testDir } = createTestEnvironment();
377
+ try {
378
+ const tracker = new ApiRequestTracker(mockApi, mockLog, 'SwitchBotTest', {
379
+ dailyLimit: -100,
380
+ reserveForCommands: -50,
381
+ });
382
+ // Should be treated as 0
383
+ expect(tracker.trySpend('poll')).toBe(false);
384
+ expect(tracker.trySpend('command')).toBe(false);
385
+ }
386
+ finally {
387
+ cleanup(testDir);
388
+ }
389
+ });
390
+ });
391
+ });
392
+ //# sourceMappingURL=apiRequestTracker.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"apiRequestTracker.test.js","sourceRoot":"","sources":["../../src/test/apiRequestTracker.test.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,UAAU,EAAE,SAAS,EAAE,WAAW,EAAE,YAAY,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,SAAS,CAAA;AACjG,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAA;AAChC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAA;AAEhC,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAA;AAEjD,OAAO,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAA;AAE/C,2DAA2D;AAC3D,SAAS,qBAAqB,CAAC,UAAU,GAAG,eAAe;IACzD,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,CAAA;IACtD,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,EAAE,EAAE,kBAAkB,MAAM,EAAE,CAAC,CAAA;IAE1D,wBAAwB;IACxB,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;QACzB,SAAS,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAA;IACzC,CAAC;IAED,MAAM,aAAa,GAAG,IAAI,CAAC,OAAO,EAAE,GAAG,UAAU,CAAC,WAAW,EAAE,iBAAiB,CAAC,CAAA;IAEjF,+CAA+C;IAC/C,MAAM,OAAO,GAAG;QACd,IAAI,EAAE;YACJ,WAAW,EAAE,GAAG,EAAE,CAAC,OAAO;SAC3B;KACgB,CAAA;IAEnB,cAAc;IACd,MAAM,OAAO,GAAG;QACd,IAAI,EAAE,EAAE,CAAC,EAAE,EAAE;QACb,IAAI,EAAE,EAAE,CAAC,EAAE,EAAE;QACb,KAAK,EAAE,EAAE,CAAC,EAAE,EAAE;QACd,KAAK,EAAE,EAAE,CAAC,EAAE,EAAE;KACO,CAAA;IAEvB,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,aAAa,EAAE,OAAO,EAAE,CAAA;AACrD,CAAC;AAED,iBAAiB;AACjB,SAAS,OAAO,CAAC,OAAe;IAC9B,IAAI,CAAC;QACH,IAAI,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;YACxB,MAAM,KAAK,GAAG,WAAW,CAAC,OAAO,CAAC,CAAA;YAClC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,IAAI,CAAC;oBACH,UAAU,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC,CAAA;gBACjC,CAAC;gBAAC,MAAM,CAAC;oBACP,SAAS;gBACX,CAAC;YACH,CAAC;YACD,SAAS,CAAC,OAAO,CAAC,CAAA;QACpB,CAAC;IACH,CAAC;IAAC,MAAM,CAAC;QACP,SAAS;IACX,CAAC;AACH,CAAC;AAED,QAAQ,CAAC,mBAAmB,EAAE,GAAG,EAAE;IACjC,QAAQ,CAAC,gBAAgB,EAAE,GAAG,EAAE;QAC9B,EAAE,CAAC,iDAAiD,EAAE,GAAG,EAAE;YACzD,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,CAAC,CAAA;gBACxE,MAAM,CAAC,OAAO,CAAC,CAAC,WAAW,EAAE,CAAA;gBAC7B,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;gBAClC,MAAM,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAA;YACxE,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,mCAAmC,EAAE,GAAG,EAAE;YAC3C,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,IAAI;oBAChB,kBAAkB,EAAE,GAAG;iBACxB,CAAC,CAAA;gBACF,MAAM,CAAC,OAAO,CAAC,CAAC,WAAW,EAAE,CAAA;YAC/B,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,sCAAsC,EAAE,GAAG,EAAE;YAC9C,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,sDAAsD;gBACtD,MAAM,QAAQ,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,CAAC,CAAA;gBACzE,QAAQ,CAAC,KAAK,EAAE,CAAA;gBAChB,QAAQ,CAAC,KAAK,EAAE,CAAA;gBAChB,MAAM,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;gBAEnC,8DAA8D;gBAC9D,MAAM,QAAQ,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,CAAC,CAAA;gBACzE,MAAM,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;YACrC,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;IAEF,QAAQ,CAAC,yBAAyB,EAAE,GAAG,EAAE;QACvC,EAAE,CAAC,8BAA8B,EAAE,GAAG,EAAE;YACtC,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,CAAC,CAAA;gBACxE,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;gBAClC,OAAO,CAAC,KAAK,EAAE,CAAA;gBACf,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;gBAClC,OAAO,CAAC,KAAK,EAAE,CAAA;gBACf,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;YACpC,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,8BAA8B,EAAE,GAAG,EAAE;YACtC,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,aAAa,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC5E,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,CAAC,CAAA;gBACxE,OAAO,CAAC,KAAK,EAAE,CAAA;gBACf,OAAO,CAAC,KAAK,EAAE,CAAA;gBACf,OAAO,CAAC,KAAK,EAAE,CAAA;gBAEf,+BAA+B;gBAC/B,MAAM,YAAY,GAAG,YAAY,CAAC,aAAa,EAAE,MAAM,CAAC,CAAA;gBACxD,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,CAAA;gBACtC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;gBAC3B,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAA;YACjE,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;IAEF,QAAQ,CAAC,iCAAiC,EAAE,GAAG,EAAE;QAC/C,EAAE,CAAC,2CAA2C,EAAE,GAAG,EAAE;YACnD,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,EAAE;iBACvB,CAAC,CAAA;gBACF,8CAA8C;gBAC9C,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC;oBAC5B,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;gBAChD,CAAC;gBACD,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;YACrC,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;YAClD,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,EAAE;iBACvB,CAAC,CAAA;gBACF,kBAAkB;gBAClB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC;oBAC5B,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;gBAC7C,CAAC;gBACD,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;YACrC,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,qEAAqE,EAAE,GAAG,EAAE;YAC7E,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,EAAE;oBACtB,qBAAqB,EAAE,IAAI,EAAE,2BAA2B;iBACzD,CAAC,CAAA;gBACF,mCAAmC;gBACnC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC;oBAC5B,OAAO,CAAC,KAAK,EAAE,CAAA;gBACjB,CAAC;gBACD,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;gBAEnC,wCAAwC;gBACxC,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;gBAC5C,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;gBAEjD,6BAA6B;gBAC7B,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;gBAC9C,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;YACrC,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,uCAAuC,EAAE,GAAG,EAAE;YAC/C,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,EAAE;iBACvB,CAAC,CAAA;gBACF,qBAAqB;gBACrB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;oBAC7B,OAAO,CAAC,KAAK,EAAE,CAAA;gBACjB,CAAC;gBACD,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;gBAEpC,sCAAsC;gBACtC,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;gBAC5C,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;gBACjD,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;gBAC/C,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;YACtC,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,+BAA+B,EAAE,GAAG,EAAE;YACvC,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,EAAE;iBACvB,CAAC,CAAA;gBACF,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;gBAC/C,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;gBAEnC,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;gBACjD,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;YACrC,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;IAEF,QAAQ,CAAC,2BAA2B,EAAE,GAAG,EAAE;QACzC,EAAE,CAAC,6EAA6E,EAAE,GAAG,EAAE;YACrF,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,EAAE;oBACtB,qBAAqB,EAAE,KAAK;iBAC7B,CAAC,CAAA;gBACF,kCAAkC;gBAClC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC;oBAC5B,OAAO,CAAC,KAAK,EAAE,CAAA;gBACjB,CAAC;gBACD,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;gBAEnC,oDAAoD;gBACpD,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;gBAC3C,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;YACrC,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,oEAAoE,EAAE,GAAG,EAAE;YAC5E,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,EAAE;oBACtB,qBAAqB,EAAE,IAAI;iBAC5B,CAAC,CAAA;gBACF,qBAAqB;gBACrB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC;oBAC5B,OAAO,CAAC,KAAK,EAAE,CAAA;gBACjB,CAAC;gBACD,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;gBAEnC,4BAA4B;gBAC5B,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;gBAC5C,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;gBAEnC,6BAA6B;gBAC7B,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;gBAC9C,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;YACrC,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;IAEF,QAAQ,CAAC,cAAc,EAAE,GAAG,EAAE;QAC5B,EAAE,CAAC,8EAA8E,EAAE,GAAG,EAAE;YACtF,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,EAAE;oBACtB,qBAAqB,EAAE,IAAI,EAAE,0BAA0B;iBACxD,CAAC,CAAA;gBACF,qBAAqB;gBACrB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC;oBAC5B,OAAO,CAAC,KAAK,EAAE,CAAA;gBACjB,CAAC;gBAED,gEAAgE;gBAChE,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAA;gBACxB,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,oBAAoB,CACvC,MAAM,CAAC,gBAAgB,CAAC,kBAAkB,CAAC,CAC5C,CAAA;YACH,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,yCAAyC,EAAE,GAAG,EAAE;YACjD,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,EAAE;iBACvB,CAAC,CAAA;gBACF,qBAAqB;gBACrB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC;oBAC7B,OAAO,CAAC,KAAK,EAAE,CAAA;gBACjB,CAAC;gBAED,yBAAyB;gBACzB,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAA;gBAC3B,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,oBAAoB,CACxC,MAAM,CAAC,gBAAgB,CAAC,aAAa,CAAC,CACvC,CAAA;YACH,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;IAEF,QAAQ,CAAC,gBAAgB,EAAE,GAAG,EAAE;QAC9B,EAAE,CAAC,mCAAmC,EAAE,GAAG,EAAE;YAC3C,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,CAAC,CAAA;gBACxE,OAAO,CAAC,kBAAkB,EAAE,CAAA;gBAC5B,OAAO,CAAC,iBAAiB,EAAE,CAAA;gBAC3B,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,oBAAoB,CACvC,MAAM,CAAC,gBAAgB,CAAC,mBAAmB,CAAC,CAC7C,CAAA;YACH,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,sDAAsD,EAAE,GAAG,EAAE;YAC9D,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,CAAC,CAAA;gBACxE,OAAO,CAAC,kBAAkB,EAAE,CAAA;gBAC5B,OAAO,CAAC,iBAAiB,EAAE,CAAA;gBAC3B,mBAAmB;gBACnB,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YACzB,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;IAEF,QAAQ,CAAC,YAAY,EAAE,GAAG,EAAE;QAC1B,EAAE,CAAC,gCAAgC,EAAE,GAAG,EAAE;YACxC,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,CAAC;oBACb,kBAAkB,EAAE,CAAC;iBACtB,CAAC,CAAA;gBACF,6CAA6C;gBAC7C,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;gBAC5C,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;gBAC/C,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAA;YACpC,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,yCAAyC,EAAE,GAAG,EAAE;YACjD,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,GAAG;oBACvB,qBAAqB,EAAE,IAAI,EAAE,2BAA2B;iBACzD,CAAC,CAAA;gBACF,6DAA6D;gBAC7D,yEAAyE;gBACzE,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;gBAC5C,sCAAsC;gBACtC,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YAChD,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;QAEF,EAAE,CAAC,yCAAyC,EAAE,GAAG,EAAE;YACjD,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,GAAG,qBAAqB,EAAE,CAAA;YAC7D,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,IAAI,iBAAiB,CAAC,OAAO,EAAE,OAAO,EAAE,eAAe,EAAE;oBACvE,UAAU,EAAE,CAAC,GAAG;oBAChB,kBAAkB,EAAE,CAAC,EAAE;iBACxB,CAAC,CAAA;gBACF,yBAAyB;gBACzB,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;gBAC5C,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAA;YACjD,CAAC;oBAAS,CAAC;gBACT,OAAO,CAAC,OAAO,CAAC,CAAA;YAClB,CAAC;QACH,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAA;AACJ,CAAC,CAAC,CAAA"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=server.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.test.d.ts","sourceRoot":"","sources":["../../../src/test/homebridge-ui/server.test.ts"],"names":[],"mappings":""}