sonamu 0.8.13 → 0.8.14

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 (96) hide show
  1. package/dist/api/sonamu.d.ts.map +1 -1
  2. package/dist/api/sonamu.js +2 -3
  3. package/dist/auth/auth-generator.d.ts +8 -0
  4. package/dist/auth/auth-generator.d.ts.map +1 -1
  5. package/dist/auth/auth-generator.js +33 -1
  6. package/dist/auth/better-auth-entities.d.ts.map +1 -1
  7. package/dist/auth/better-auth-entities.js +12 -2
  8. package/dist/bin/cli.js +18 -3
  9. package/dist/cone/cone-generator.js +10 -4
  10. package/dist/database/knex.d.ts.map +1 -1
  11. package/dist/database/knex.js +64 -2
  12. package/dist/database/puri.d.ts +9 -1
  13. package/dist/database/puri.d.ts.map +1 -1
  14. package/dist/database/puri.js +42 -1
  15. package/dist/database/puri.types.d.ts +2 -0
  16. package/dist/database/puri.types.d.ts.map +1 -1
  17. package/dist/database/puri.types.js +6 -2
  18. package/dist/entity/entity-manager.d.ts +149 -1
  19. package/dist/entity/entity-manager.d.ts.map +1 -1
  20. package/dist/entity/entity-manager.js +68 -4
  21. package/dist/migration/__tests__/code-generation.search-text.test.js +435 -0
  22. package/dist/migration/code-generation.d.ts.map +1 -1
  23. package/dist/migration/code-generation.js +696 -32
  24. package/dist/migration/migration-set.js +3 -1
  25. package/dist/migration/postgresql-schema-reader.d.ts +16 -2
  26. package/dist/migration/postgresql-schema-reader.d.ts.map +1 -1
  27. package/dist/migration/postgresql-schema-reader.js +281 -7
  28. package/dist/stream/sse.js +5 -3
  29. package/dist/template/__tests__/generated.template.search-text.test.js +99 -0
  30. package/dist/template/generated.template.test-d.js +24 -0
  31. package/dist/template/implementations/generated.template.d.ts.map +1 -1
  32. package/dist/template/implementations/generated.template.js +2 -2
  33. package/dist/template/implementations/init_types.template.d.ts.map +1 -1
  34. package/dist/template/implementations/init_types.template.js +11 -3
  35. package/dist/template/zod-converter.d.ts.map +1 -1
  36. package/dist/template/zod-converter.js +6 -2
  37. package/dist/testing/dev-test-routes.d.ts.map +1 -1
  38. package/dist/testing/dev-test-routes.js +5 -3
  39. package/dist/testing/fixture-generator.d.ts +13 -0
  40. package/dist/testing/fixture-generator.d.ts.map +1 -1
  41. package/dist/testing/fixture-generator.js +105 -8
  42. package/dist/testing/fixture-manager.d.ts.map +1 -1
  43. package/dist/testing/fixture-manager.js +19 -2
  44. package/dist/types/__tests__/entity-json-schema-search-text.test.js +256 -0
  45. package/dist/types/types.d.ts +494 -1
  46. package/dist/types/types.d.ts.map +1 -1
  47. package/dist/types/types.js +117 -13
  48. package/dist/ui/api.d.ts.map +1 -1
  49. package/dist/ui/api.js +14 -2
  50. package/dist/ui/cdd-service.d.ts +16 -14
  51. package/dist/ui/cdd-service.d.ts.map +1 -1
  52. package/dist/ui/cdd-service.js +145 -37
  53. package/dist/ui/cdd-types.d.ts +60 -0
  54. package/dist/ui/cdd-types.d.ts.map +1 -0
  55. package/dist/ui/cdd-types.js +3 -0
  56. package/dist/ui-web/assets/index-D4XFBV-f.css +1 -0
  57. package/dist/ui-web/assets/{index-CQ_S40bD.js → index-D_19-Pi4.js} +87 -87
  58. package/dist/ui-web/index.html +2 -2
  59. package/package.json +7 -3
  60. package/src/api/sonamu.ts +1 -2
  61. package/src/auth/auth-generator.ts +38 -0
  62. package/src/auth/better-auth-entities.ts +18 -1
  63. package/src/bin/cli.ts +15 -1
  64. package/src/cone/cone-generator.ts +9 -3
  65. package/src/database/knex.ts +62 -4
  66. package/src/database/puri.ts +71 -0
  67. package/src/database/puri.types.ts +2 -0
  68. package/src/entity/entity-manager.ts +95 -3
  69. package/src/migration/__tests__/code-generation.search-text.test.ts +390 -0
  70. package/src/migration/code-generation.ts +848 -34
  71. package/src/migration/migration-set.ts +2 -0
  72. package/src/migration/postgresql-schema-reader.ts +366 -9
  73. package/src/skills/sonamu/auth-migration.md +80 -0
  74. package/src/skills/sonamu/cdd.md +148 -28
  75. package/src/skills/sonamu/cone.md +16 -0
  76. package/src/skills/sonamu/entity-relations.md +1 -1
  77. package/src/skills/sonamu/fixture-cli.md +4 -0
  78. package/src/skills/sonamu/frontend.md +65 -0
  79. package/src/skills/sonamu/migration.md +3 -1
  80. package/src/skills/sonamu/model.md +28 -0
  81. package/src/skills/sonamu/workflow.md +12 -5
  82. package/src/stream/sse.ts +4 -2
  83. package/src/template/__tests__/generated.template.search-text.test.ts +89 -0
  84. package/src/template/generated.template.test-d.ts +46 -0
  85. package/src/template/implementations/generated.template.ts +4 -1
  86. package/src/template/implementations/init_types.template.ts +20 -5
  87. package/src/template/zod-converter.ts +5 -0
  88. package/src/testing/dev-test-routes.ts +4 -2
  89. package/src/testing/fixture-generator.ts +157 -9
  90. package/src/testing/fixture-manager.ts +15 -1
  91. package/src/types/__tests__/entity-json-schema-search-text.test.ts +179 -0
  92. package/src/types/types.ts +168 -12
  93. package/src/ui/api.ts +24 -1
  94. package/src/ui/cdd-service.ts +195 -55
  95. package/src/ui/cdd-types.ts +73 -0
  96. package/dist/ui-web/assets/index-egkMxKos.css +0 -1
@@ -0,0 +1,435 @@
1
+ import { describe, expect, test } from "vitest";
2
+ import { Sonamu } from "../../api/index.js";
3
+ import { Entity } from "../../entity/entity.js";
4
+ import { EntityManager } from "../../entity/entity-manager.js";
5
+ import { setupBiome } from "../../utils/formatter.js";
6
+ import { generateAlterCode, generateCreateCode, setMigrationIndexDefaults } from "../code-generation.js";
7
+ import { getMigrationSetFromEntity } from "../migration-set.js";
8
+ const TEST_API_ROOT = "/Users/Nebuleto/Workspace/sonamu/modules/sonamu";
9
+ setupBiome(TEST_API_ROOT);
10
+ Sonamu.apiRootPath = TEST_API_ROOT;
11
+ let entitySeq = 0;
12
+ async function registerEntity(definition) {
13
+ entitySeq += 1;
14
+ const entity = {
15
+ ...definition,
16
+ id: `${definition.id}${entitySeq}`,
17
+ title: `${definition.id}${entitySeq}`,
18
+ table: `${definition.table}_${entitySeq}`,
19
+ subsets: {
20
+ A: definition.props.map((prop)=>prop.name)
21
+ },
22
+ enums: {}
23
+ };
24
+ await EntityManager.register(entity, {
25
+ deferSearchTextJsonSourceValidation: true
26
+ });
27
+ return EntityManager.get(entity.id);
28
+ }
29
+ function buildDbSetWithGeneratedSearchText(entitySet, expression) {
30
+ return {
31
+ ...entitySet,
32
+ columns: entitySet.columns.map((column)=>column.name === "search_text" ? {
33
+ ...column,
34
+ generated: {
35
+ type: "STORED",
36
+ expression
37
+ }
38
+ } : column)
39
+ };
40
+ }
41
+ describe("code-generation searchText/opclass DDL", ()=>{
42
+ test("searchText가 없으면 generated/helper SQL을 추가하지 않고 vector index를 유지해야 한다", async ()=>{
43
+ const entity = await registerEntity({
44
+ id: "CodeGenerationNoSearchText",
45
+ table: "code_generation_no_search_text",
46
+ props: [
47
+ {
48
+ name: "id",
49
+ type: "integer"
50
+ },
51
+ {
52
+ name: "title",
53
+ type: "string"
54
+ },
55
+ {
56
+ name: "embedding",
57
+ type: "vector",
58
+ dimensions: 1536
59
+ }
60
+ ],
61
+ indexes: [
62
+ {
63
+ type: "hnsw",
64
+ name: "code_generation_no_search_text_embedding_hnsw",
65
+ columns: [
66
+ {
67
+ name: "embedding",
68
+ vectorOps: "vector_cosine_ops"
69
+ }
70
+ ]
71
+ }
72
+ ]
73
+ });
74
+ const [migration] = await generateCreateCode(getMigrationSetFromEntity(entity));
75
+ expect(migration.formatted).not.toContain("GENERATED ALWAYS AS");
76
+ expect(migration.formatted).not.toContain("sonamu_text_array_agg");
77
+ expect(migration.formatted).not.toContain("sonamu_jsonb_array_agg");
78
+ expect(migration.formatted).toContain("USING hnsw (embedding vector_cosine_ops) WITH (m = 16, ef_construction = 64)");
79
+ });
80
+ test("searchText generated DDL, helper 함수, generic opclass를 올바르게 출력해야 한다", async ()=>{
81
+ const entity = await registerEntity({
82
+ id: "CodeGenerationSearchText",
83
+ table: "code_generation_search_text",
84
+ props: [
85
+ {
86
+ name: "id",
87
+ type: "integer"
88
+ },
89
+ {
90
+ name: "title_ci",
91
+ type: "string"
92
+ },
93
+ {
94
+ name: "code_cs",
95
+ type: "string"
96
+ },
97
+ {
98
+ name: "tags_ci",
99
+ type: "string[]"
100
+ },
101
+ {
102
+ name: "tags_cs",
103
+ type: "string[]"
104
+ },
105
+ {
106
+ name: "aliases_ci",
107
+ type: "json",
108
+ id: "StringArrayCi"
109
+ },
110
+ {
111
+ name: "aliases_cs",
112
+ type: "json",
113
+ id: "StringArrayCs"
114
+ },
115
+ {
116
+ name: "search_text",
117
+ type: "searchText",
118
+ sourceColumns: [
119
+ {
120
+ name: "title_ci",
121
+ caseInsensitive: true
122
+ },
123
+ {
124
+ name: "code_cs",
125
+ caseInsensitive: false
126
+ },
127
+ {
128
+ name: "tags_ci",
129
+ caseInsensitive: true
130
+ },
131
+ {
132
+ name: "tags_cs",
133
+ caseInsensitive: false
134
+ },
135
+ {
136
+ name: "aliases_ci",
137
+ caseInsensitive: true
138
+ },
139
+ {
140
+ name: "aliases_cs",
141
+ caseInsensitive: false
142
+ }
143
+ ]
144
+ }
145
+ ],
146
+ indexes: [
147
+ {
148
+ type: "index",
149
+ name: "code_generation_search_text_search_text_index",
150
+ using: "gin",
151
+ columns: [
152
+ {
153
+ name: "search_text",
154
+ opclass: "gin_trgm_ops"
155
+ }
156
+ ]
157
+ }
158
+ ]
159
+ });
160
+ const [migration] = await generateCreateCode(getMigrationSetFromEntity(entity));
161
+ expect(migration.formatted).toContain("CREATE OR REPLACE FUNCTION sonamu_text_array_agg(arr text[], ci boolean DEFAULT true)");
162
+ expect(migration.formatted).toContain("CREATE OR REPLACE FUNCTION sonamu_jsonb_array_agg(arr jsonb, ci boolean DEFAULT true)");
163
+ expect(migration.formatted).toContain("ADD COLUMN \"search_text\" text GENERATED ALWAYS AS (trim(lower(COALESCE(title_ci, '')) || ' ' || COALESCE(code_cs, '')");
164
+ expect(migration.formatted).toContain("COALESCE(sonamu_text_array_agg(tags_ci), '')");
165
+ expect(migration.formatted).toContain("COALESCE(sonamu_text_array_agg(tags_cs, false), '')");
166
+ expect(migration.formatted).toContain("COALESCE(sonamu_jsonb_array_agg(aliases_ci), '')");
167
+ expect(migration.formatted).toContain("COALESCE(sonamu_jsonb_array_agg(aliases_cs, false), '')");
168
+ expect(migration.formatted).toContain("CREATE INDEX code_generation_search_text_search_text_index ON");
169
+ expect(migration.formatted).toContain("USING gin(search_text gin_trgm_ops);");
170
+ });
171
+ test("searchText generated expression 변경 시 alter path에서 helper와 index를 함께 재생성해야 한다", async ()=>{
172
+ const entity = await registerEntity({
173
+ id: "CodeGenerationSearchTextAlter",
174
+ table: "code_generation_search_text_alter",
175
+ props: [
176
+ {
177
+ name: "id",
178
+ type: "integer"
179
+ },
180
+ {
181
+ name: "title_ci",
182
+ type: "string"
183
+ },
184
+ {
185
+ name: "tags_ci",
186
+ type: "string[]"
187
+ },
188
+ {
189
+ name: "search_text",
190
+ type: "searchText",
191
+ sourceColumns: [
192
+ {
193
+ name: "title_ci",
194
+ caseInsensitive: false
195
+ },
196
+ {
197
+ name: "tags_ci",
198
+ caseInsensitive: true
199
+ }
200
+ ]
201
+ }
202
+ ],
203
+ indexes: [
204
+ {
205
+ type: "index",
206
+ name: "code_generation_search_text_alter_search_text_index",
207
+ using: "gin",
208
+ columns: [
209
+ {
210
+ name: "search_text",
211
+ opclass: "gin_trgm_ops"
212
+ }
213
+ ]
214
+ }
215
+ ]
216
+ });
217
+ const entitySet = getMigrationSetFromEntity(entity);
218
+ const dbSet = buildDbSetWithGeneratedSearchText(entitySet, `trim(lower(COALESCE(title_ci, '')))`);
219
+ const [migration] = await generateAlterCode(entitySet, dbSet);
220
+ expect(migration.formatted).toContain('table.dropIndex(["search_text"], "code_generation_search_text_alter_search_text_index")');
221
+ expect(migration.formatted).toContain('table.dropColumns("search_text")');
222
+ expect(migration.formatted).toContain("CREATE OR REPLACE FUNCTION sonamu_text_array_agg(arr text[], ci boolean DEFAULT true)");
223
+ expect(migration.formatted).toContain(`ADD COLUMN "search_text" text GENERATED ALWAYS AS (trim(COALESCE(title_ci, '') || ' ' || COALESCE(sonamu_text_array_agg(tags_ci), ''))) STORED NOT NULL`);
224
+ expect(migration.formatted).toContain("CREATE INDEX code_generation_search_text_alter_search_text_index ON");
225
+ expect(migration.formatted).toContain(`ADD COLUMN "search_text" text GENERATED ALWAYS AS (trim(lower(COALESCE(title_ci, '')))) STORED NOT NULL`);
226
+ });
227
+ test("searchText generated expression이 Postgres canonical form과 semantically equivalent면 no-op이어야 한다", async ()=>{
228
+ const entity = await registerEntity({
229
+ id: "CodeGenerationSearchTextCanonicalNoop",
230
+ table: "code_generation_search_text_canonical_noop",
231
+ props: [
232
+ {
233
+ name: "id",
234
+ type: "integer"
235
+ },
236
+ {
237
+ name: "username",
238
+ type: "string"
239
+ },
240
+ {
241
+ name: "tags",
242
+ type: "string[]"
243
+ },
244
+ {
245
+ name: "search_text",
246
+ type: "searchText",
247
+ sourceColumns: [
248
+ {
249
+ name: "username",
250
+ caseInsensitive: false
251
+ },
252
+ {
253
+ name: "tags",
254
+ caseInsensitive: true
255
+ }
256
+ ]
257
+ }
258
+ ],
259
+ indexes: [
260
+ {
261
+ type: "index",
262
+ name: "code_generation_search_text_canonical_noop_search_text_index",
263
+ using: "gin",
264
+ columns: [
265
+ {
266
+ name: "search_text",
267
+ opclass: "gin_trgm_ops"
268
+ }
269
+ ]
270
+ }
271
+ ]
272
+ });
273
+ const entitySet = getMigrationSetFromEntity(entity);
274
+ const dbSet = buildDbSetWithGeneratedSearchText({
275
+ ...entitySet,
276
+ indexes: entitySet.indexes.map(setMigrationIndexDefaults)
277
+ }, `TRIM(BOTH FROM ((COALESCE(username, ''::text)) || ' '::text) || COALESCE(sonamu_text_array_agg(tags, true), ''::text))`);
278
+ const migrations = await generateAlterCode(entitySet, dbSet);
279
+ expect(migrations).toHaveLength(0);
280
+ });
281
+ test("searchText caseInsensitive 변경만 있어도 alter path에서 재생성해야 한다", async ()=>{
282
+ const entity = await registerEntity({
283
+ id: "CodeGenerationSearchTextCaseInsensitiveAlter",
284
+ table: "code_generation_search_text_case_insensitive_alter",
285
+ props: [
286
+ {
287
+ name: "id",
288
+ type: "integer"
289
+ },
290
+ {
291
+ name: "title_ci",
292
+ type: "string"
293
+ },
294
+ {
295
+ name: "search_text",
296
+ type: "searchText",
297
+ sourceColumns: [
298
+ {
299
+ name: "title_ci",
300
+ caseInsensitive: false
301
+ }
302
+ ]
303
+ }
304
+ ],
305
+ indexes: [
306
+ {
307
+ type: "index",
308
+ name: "code_generation_search_text_case_insensitive_alter_search_text_index",
309
+ using: "gin",
310
+ columns: [
311
+ {
312
+ name: "search_text",
313
+ opclass: "gin_trgm_ops"
314
+ }
315
+ ]
316
+ }
317
+ ]
318
+ });
319
+ const entitySet = getMigrationSetFromEntity(entity);
320
+ const dbSet = buildDbSetWithGeneratedSearchText(entitySet, `TRIM(BOTH FROM lower(COALESCE(title_ci, ''::text)))`);
321
+ const [migration] = await generateAlterCode(entitySet, dbSet);
322
+ expect(migration.formatted).toContain("table.dropIndex(");
323
+ expect(migration.formatted).toContain('"code_generation_search_text_case_insensitive_alter_search_text_index"');
324
+ expect(migration.formatted).toContain('table.dropColumns("search_text")');
325
+ expect(migration.formatted).toContain(`ADD COLUMN "search_text" text GENERATED ALWAYS AS (trim(COALESCE(title_ci, ''))) STORED NOT NULL`);
326
+ expect(migration.formatted).toContain("lower(COALESCE(title_ci");
327
+ });
328
+ test("searchText helper kind가 바뀌어도 down path는 복원 expression 기준 helper를 재생성해야 한다", async ()=>{
329
+ const entity = await registerEntity({
330
+ id: "CodeGenerationSearchTextRollbackHelperMismatch",
331
+ table: "code_generation_search_text_rollback_helper_mismatch",
332
+ props: [
333
+ {
334
+ name: "id",
335
+ type: "integer"
336
+ },
337
+ {
338
+ name: "tags",
339
+ type: "string[]"
340
+ },
341
+ {
342
+ name: "aliases",
343
+ type: "json",
344
+ id: "RollbackHelperMismatchJson"
345
+ },
346
+ {
347
+ name: "search_text",
348
+ type: "searchText",
349
+ sourceColumns: [
350
+ {
351
+ name: "tags",
352
+ caseInsensitive: true
353
+ }
354
+ ]
355
+ }
356
+ ],
357
+ indexes: [
358
+ {
359
+ type: "index",
360
+ name: "code_generation_search_text_rollback_helper_mismatch_search_text_index",
361
+ using: "gin",
362
+ columns: [
363
+ {
364
+ name: "search_text",
365
+ opclass: "gin_trgm_ops"
366
+ }
367
+ ]
368
+ }
369
+ ]
370
+ });
371
+ const entitySet = getMigrationSetFromEntity(entity);
372
+ const dbSet = buildDbSetWithGeneratedSearchText(entitySet, `trim(COALESCE(sonamu_jsonb_array_agg(aliases), ''))`);
373
+ const [migration] = await generateAlterCode(entitySet, dbSet);
374
+ expect(migration.formatted).toContain("CREATE OR REPLACE FUNCTION sonamu_text_array_agg(arr text[], ci boolean DEFAULT true)");
375
+ expect(migration.formatted).toContain(`ADD COLUMN "search_text" text GENERATED ALWAYS AS (trim(COALESCE(sonamu_text_array_agg(tags), ''))) STORED NOT NULL`);
376
+ expect(migration.formatted).toContain("CREATE OR REPLACE FUNCTION sonamu_jsonb_array_agg(arr jsonb, ci boolean DEFAULT true)");
377
+ expect(migration.formatted).toContain(`ADD COLUMN "search_text" text GENERATED ALWAYS AS (trim(COALESCE(sonamu_jsonb_array_agg(aliases), ''))) STORED NOT NULL`);
378
+ });
379
+ test("searchText가 제거되어도 down path는 이전 expression helper를 복원해야 한다", async ()=>{
380
+ const entity = await registerEntity({
381
+ id: "CodeGenerationSearchTextRollbackDropped",
382
+ table: "code_generation_search_text_rollback_dropped",
383
+ props: [
384
+ {
385
+ name: "id",
386
+ type: "integer"
387
+ },
388
+ {
389
+ name: "tags",
390
+ type: "string[]"
391
+ }
392
+ ],
393
+ indexes: []
394
+ });
395
+ const previousEntity = new Entity({
396
+ ...entity.toJson(),
397
+ props: [
398
+ ...entity.toJson().props,
399
+ {
400
+ name: "search_text",
401
+ type: "searchText",
402
+ sourceColumns: [
403
+ {
404
+ name: "tags",
405
+ caseInsensitive: true
406
+ }
407
+ ]
408
+ }
409
+ ],
410
+ indexes: [
411
+ {
412
+ type: "index",
413
+ name: "code_generation_search_text_rollback_dropped_search_text_index",
414
+ using: "gin",
415
+ columns: [
416
+ {
417
+ name: "search_text",
418
+ opclass: "gin_trgm_ops"
419
+ }
420
+ ]
421
+ }
422
+ ]
423
+ });
424
+ const entitySet = getMigrationSetFromEntity(entity);
425
+ const dbSet = buildDbSetWithGeneratedSearchText(getMigrationSetFromEntity(previousEntity), `trim(COALESCE(sonamu_text_array_agg(tags), ''))`);
426
+ const [migration] = await generateAlterCode(entitySet, dbSet);
427
+ expect(migration.formatted).toContain('table.dropColumns("search_text")');
428
+ expect(migration.formatted).toContain("CREATE OR REPLACE FUNCTION sonamu_text_array_agg(arr text[], ci boolean DEFAULT true)");
429
+ expect(migration.formatted).toContain(`ADD COLUMN "search_text" text GENERATED ALWAYS AS (trim(COALESCE(sonamu_text_array_agg(tags), ''))) STORED NOT NULL`);
430
+ expect(migration.formatted).toContain("CREATE INDEX code_generation_search_text_rollback_dropped_search_text_index ON");
431
+ expect(migration.formatted).toContain("USING gin(search_text gin_trgm_ops);");
432
+ });
433
+ });
434
+
435
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9taWdyYXRpb24vX190ZXN0c19fL2NvZGUtZ2VuZXJhdGlvbi5zZWFyY2gtdGV4dC50ZXN0LnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGRlc2NyaWJlLCBleHBlY3QsIHRlc3QgfSBmcm9tIFwidml0ZXN0XCI7XG5pbXBvcnQgeyBTb25hbXUgfSBmcm9tIFwiLi4vLi4vYXBpXCI7XG5pbXBvcnQgeyBFbnRpdHkgfSBmcm9tIFwiLi4vLi4vZW50aXR5L2VudGl0eVwiO1xuaW1wb3J0IHsgRW50aXR5TWFuYWdlciB9IGZyb20gXCIuLi8uLi9lbnRpdHkvZW50aXR5LW1hbmFnZXJcIjtcbmltcG9ydCB0eXBlIHsgTWlncmF0aW9uU2V0IH0gZnJvbSBcIi4uLy4uL3R5cGVzL3R5cGVzXCI7XG5pbXBvcnQgeyBzZXR1cEJpb21lIH0gZnJvbSBcIi4uLy4uL3V0aWxzL2Zvcm1hdHRlclwiO1xuaW1wb3J0IHtcbiAgZ2VuZXJhdGVBbHRlckNvZGUsXG4gIGdlbmVyYXRlQ3JlYXRlQ29kZSxcbiAgc2V0TWlncmF0aW9uSW5kZXhEZWZhdWx0cyxcbn0gZnJvbSBcIi4uL2NvZGUtZ2VuZXJhdGlvblwiO1xuaW1wb3J0IHsgZ2V0TWlncmF0aW9uU2V0RnJvbUVudGl0eSB9IGZyb20gXCIuLi9taWdyYXRpb24tc2V0XCI7XG5cbmNvbnN0IFRFU1RfQVBJX1JPT1QgPSBcIi9Vc2Vycy9OZWJ1bGV0by9Xb3Jrc3BhY2Uvc29uYW11L21vZHVsZXMvc29uYW11XCI7XG5cbnNldHVwQmlvbWUoVEVTVF9BUElfUk9PVCk7XG5Tb25hbXUuYXBpUm9vdFBhdGggPSBURVNUX0FQSV9ST09UO1xuXG5sZXQgZW50aXR5U2VxID0gMDtcblxuYXN5bmMgZnVuY3Rpb24gcmVnaXN0ZXJFbnRpdHkoXG4gIGRlZmluaXRpb246IE9taXQ8XG4gICAgUGFyYW1ldGVyczx0eXBlb2YgRW50aXR5TWFuYWdlci5yZWdpc3Rlcj5bMF0sXG4gICAgXCJpZFwiIHwgXCJ0YWJsZVwiIHwgXCJ0aXRsZVwiIHwgXCJzdWJzZXRzXCIgfCBcImVudW1zXCJcbiAgPiAmIHtcbiAgICBpZDogc3RyaW5nO1xuICAgIHRhYmxlOiBzdHJpbmc7XG4gIH0sXG4pIHtcbiAgZW50aXR5U2VxICs9IDE7XG5cbiAgY29uc3QgZW50aXR5ID0ge1xuICAgIC4uLmRlZmluaXRpb24sXG4gICAgaWQ6IGAke2RlZmluaXRpb24uaWR9JHtlbnRpdHlTZXF9YCxcbiAgICB0aXRsZTogYCR7ZGVmaW5pdGlvbi5pZH0ke2VudGl0eVNlcX1gLFxuICAgIHRhYmxlOiBgJHtkZWZpbml0aW9uLnRhYmxlfV8ke2VudGl0eVNlcX1gLFxuICAgIHN1YnNldHM6IHtcbiAgICAgIEE6IGRlZmluaXRpb24ucHJvcHMubWFwKChwcm9wKSA9PiBwcm9wLm5hbWUpLFxuICAgIH0sXG4gICAgZW51bXM6IHt9LFxuICB9O1xuXG4gIGF3YWl0IEVudGl0eU1hbmFnZXIucmVnaXN0ZXIoZW50aXR5LCB7XG4gICAgZGVmZXJTZWFyY2hUZXh0SnNvblNvdXJjZVZhbGlkYXRpb246IHRydWUsXG4gIH0pO1xuXG4gIHJldHVybiBFbnRpdHlNYW5hZ2VyLmdldChlbnRpdHkuaWQpO1xufVxuXG5mdW5jdGlvbiBidWlsZERiU2V0V2l0aEdlbmVyYXRlZFNlYXJjaFRleHQoXG4gIGVudGl0eVNldDogTWlncmF0aW9uU2V0LFxuICBleHByZXNzaW9uOiBzdHJpbmcsXG4pOiBNaWdyYXRpb25TZXQge1xuICByZXR1cm4ge1xuICAgIC4uLmVudGl0eVNldCxcbiAgICBjb2x1bW5zOiBlbnRpdHlTZXQuY29sdW1ucy5tYXAoKGNvbHVtbikgPT5cbiAgICAgIGNvbHVtbi5uYW1lID09PSBcInNlYXJjaF90ZXh0XCJcbiAgICAgICAgPyB7XG4gICAgICAgICAgICAuLi5jb2x1bW4sXG4gICAgICAgICAgICBnZW5lcmF0ZWQ6IHtcbiAgICAgICAgICAgICAgdHlwZTogXCJTVE9SRURcIixcbiAgICAgICAgICAgICAgZXhwcmVzc2lvbixcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgfVxuICAgICAgICA6IGNvbHVtbixcbiAgICApLFxuICB9O1xufVxuXG5kZXNjcmliZShcImNvZGUtZ2VuZXJhdGlvbiBzZWFyY2hUZXh0L29wY2xhc3MgRERMXCIsICgpID0+IHtcbiAgdGVzdChcInNlYXJjaFRleHTqsIAg7JeG7Jy866m0IGdlbmVyYXRlZC9oZWxwZXIgU1FM7J2EIOy2lOqwgO2VmOyngCDslYrqs6AgdmVjdG9yIGluZGV466W8IOycoOyngO2VtOyVvCDtlZzri6RcIiwgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IGVudGl0eSA9IGF3YWl0IHJlZ2lzdGVyRW50aXR5KHtcbiAgICAgIGlkOiBcIkNvZGVHZW5lcmF0aW9uTm9TZWFyY2hUZXh0XCIsXG4gICAgICB0YWJsZTogXCJjb2RlX2dlbmVyYXRpb25fbm9fc2VhcmNoX3RleHRcIixcbiAgICAgIHByb3BzOiBbXG4gICAgICAgIHsgbmFtZTogXCJpZFwiLCB0eXBlOiBcImludGVnZXJcIiB9LFxuICAgICAgICB7IG5hbWU6IFwidGl0bGVcIiwgdHlwZTogXCJzdHJpbmdcIiB9LFxuICAgICAgICB7IG5hbWU6IFwiZW1iZWRkaW5nXCIsIHR5cGU6IFwidmVjdG9yXCIsIGRpbWVuc2lvbnM6IDE1MzYgfSxcbiAgICAgIF0sXG4gICAgICBpbmRleGVzOiBbXG4gICAgICAgIHtcbiAgICAgICAgICB0eXBlOiBcImhuc3dcIixcbiAgICAgICAgICBuYW1lOiBcImNvZGVfZ2VuZXJhdGlvbl9ub19zZWFyY2hfdGV4dF9lbWJlZGRpbmdfaG5zd1wiLFxuICAgICAgICAgIGNvbHVtbnM6IFt7IG5hbWU6IFwiZW1iZWRkaW5nXCIsIHZlY3Rvck9wczogXCJ2ZWN0b3JfY29zaW5lX29wc1wiIH1dLFxuICAgICAgICB9LFxuICAgICAgXSxcbiAgICB9KTtcblxuICAgIGNvbnN0IFttaWdyYXRpb25dID0gYXdhaXQgZ2VuZXJhdGVDcmVhdGVDb2RlKGdldE1pZ3JhdGlvblNldEZyb21FbnRpdHkoZW50aXR5KSk7XG5cbiAgICBleHBlY3QobWlncmF0aW9uLmZvcm1hdHRlZCkubm90LnRvQ29udGFpbihcIkdFTkVSQVRFRCBBTFdBWVMgQVNcIik7XG4gICAgZXhwZWN0KG1pZ3JhdGlvbi5mb3JtYXR0ZWQpLm5vdC50b0NvbnRhaW4oXCJzb25hbXVfdGV4dF9hcnJheV9hZ2dcIik7XG4gICAgZXhwZWN0KG1pZ3JhdGlvbi5mb3JtYXR0ZWQpLm5vdC50b0NvbnRhaW4oXCJzb25hbXVfanNvbmJfYXJyYXlfYWdnXCIpO1xuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oXG4gICAgICBcIlVTSU5HIGhuc3cgKGVtYmVkZGluZyB2ZWN0b3JfY29zaW5lX29wcykgV0lUSCAobSA9IDE2LCBlZl9jb25zdHJ1Y3Rpb24gPSA2NClcIixcbiAgICApO1xuICB9KTtcblxuICB0ZXN0KFwic2VhcmNoVGV4dCBnZW5lcmF0ZWQgRERMLCBoZWxwZXIg7ZWo7IiYLCBnZW5lcmljIG9wY2xhc3Prpbwg7Jis67CU66W06rKMIOy2nOugpe2VtOyVvCDtlZzri6RcIiwgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IGVudGl0eSA9IGF3YWl0IHJlZ2lzdGVyRW50aXR5KHtcbiAgICAgIGlkOiBcIkNvZGVHZW5lcmF0aW9uU2VhcmNoVGV4dFwiLFxuICAgICAgdGFibGU6IFwiY29kZV9nZW5lcmF0aW9uX3NlYXJjaF90ZXh0XCIsXG4gICAgICBwcm9wczogW1xuICAgICAgICB7IG5hbWU6IFwiaWRcIiwgdHlwZTogXCJpbnRlZ2VyXCIgfSxcbiAgICAgICAgeyBuYW1lOiBcInRpdGxlX2NpXCIsIHR5cGU6IFwic3RyaW5nXCIgfSxcbiAgICAgICAgeyBuYW1lOiBcImNvZGVfY3NcIiwgdHlwZTogXCJzdHJpbmdcIiB9LFxuICAgICAgICB7IG5hbWU6IFwidGFnc19jaVwiLCB0eXBlOiBcInN0cmluZ1tdXCIgfSxcbiAgICAgICAgeyBuYW1lOiBcInRhZ3NfY3NcIiwgdHlwZTogXCJzdHJpbmdbXVwiIH0sXG4gICAgICAgIHsgbmFtZTogXCJhbGlhc2VzX2NpXCIsIHR5cGU6IFwianNvblwiLCBpZDogXCJTdHJpbmdBcnJheUNpXCIgfSxcbiAgICAgICAgeyBuYW1lOiBcImFsaWFzZXNfY3NcIiwgdHlwZTogXCJqc29uXCIsIGlkOiBcIlN0cmluZ0FycmF5Q3NcIiB9LFxuICAgICAgICB7XG4gICAgICAgICAgbmFtZTogXCJzZWFyY2hfdGV4dFwiLFxuICAgICAgICAgIHR5cGU6IFwic2VhcmNoVGV4dFwiLFxuICAgICAgICAgIHNvdXJjZUNvbHVtbnM6IFtcbiAgICAgICAgICAgIHsgbmFtZTogXCJ0aXRsZV9jaVwiLCBjYXNlSW5zZW5zaXRpdmU6IHRydWUgfSxcbiAgICAgICAgICAgIHsgbmFtZTogXCJjb2RlX2NzXCIsIGNhc2VJbnNlbnNpdGl2ZTogZmFsc2UgfSxcbiAgICAgICAgICAgIHsgbmFtZTogXCJ0YWdzX2NpXCIsIGNhc2VJbnNlbnNpdGl2ZTogdHJ1ZSB9LFxuICAgICAgICAgICAgeyBuYW1lOiBcInRhZ3NfY3NcIiwgY2FzZUluc2Vuc2l0aXZlOiBmYWxzZSB9LFxuICAgICAgICAgICAgeyBuYW1lOiBcImFsaWFzZXNfY2lcIiwgY2FzZUluc2Vuc2l0aXZlOiB0cnVlIH0sXG4gICAgICAgICAgICB7IG5hbWU6IFwiYWxpYXNlc19jc1wiLCBjYXNlSW5zZW5zaXRpdmU6IGZhbHNlIH0sXG4gICAgICAgICAgXSxcbiAgICAgICAgfSxcbiAgICAgIF0sXG4gICAgICBpbmRleGVzOiBbXG4gICAgICAgIHtcbiAgICAgICAgICB0eXBlOiBcImluZGV4XCIsXG4gICAgICAgICAgbmFtZTogXCJjb2RlX2dlbmVyYXRpb25fc2VhcmNoX3RleHRfc2VhcmNoX3RleHRfaW5kZXhcIixcbiAgICAgICAgICB1c2luZzogXCJnaW5cIixcbiAgICAgICAgICBjb2x1bW5zOiBbeyBuYW1lOiBcInNlYXJjaF90ZXh0XCIsIG9wY2xhc3M6IFwiZ2luX3RyZ21fb3BzXCIgfV0sXG4gICAgICAgIH0sXG4gICAgICBdLFxuICAgIH0pO1xuXG4gICAgY29uc3QgW21pZ3JhdGlvbl0gPSBhd2FpdCBnZW5lcmF0ZUNyZWF0ZUNvZGUoZ2V0TWlncmF0aW9uU2V0RnJvbUVudGl0eShlbnRpdHkpKTtcblxuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oXG4gICAgICBcIkNSRUFURSBPUiBSRVBMQUNFIEZVTkNUSU9OIHNvbmFtdV90ZXh0X2FycmF5X2FnZyhhcnIgdGV4dFtdLCBjaSBib29sZWFuIERFRkFVTFQgdHJ1ZSlcIixcbiAgICApO1xuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oXG4gICAgICBcIkNSRUFURSBPUiBSRVBMQUNFIEZVTkNUSU9OIHNvbmFtdV9qc29uYl9hcnJheV9hZ2coYXJyIGpzb25iLCBjaSBib29sZWFuIERFRkFVTFQgdHJ1ZSlcIixcbiAgICApO1xuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oXG4gICAgICBcIkFERCBDT0xVTU4gXFxcInNlYXJjaF90ZXh0XFxcIiB0ZXh0IEdFTkVSQVRFRCBBTFdBWVMgQVMgKHRyaW0obG93ZXIoQ09BTEVTQ0UodGl0bGVfY2ksICcnKSkgfHwgJyAnIHx8IENPQUxFU0NFKGNvZGVfY3MsICcnKVwiLFxuICAgICk7XG4gICAgZXhwZWN0KG1pZ3JhdGlvbi5mb3JtYXR0ZWQpLnRvQ29udGFpbihcIkNPQUxFU0NFKHNvbmFtdV90ZXh0X2FycmF5X2FnZyh0YWdzX2NpKSwgJycpXCIpO1xuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oXCJDT0FMRVNDRShzb25hbXVfdGV4dF9hcnJheV9hZ2codGFnc19jcywgZmFsc2UpLCAnJylcIik7XG4gICAgZXhwZWN0KG1pZ3JhdGlvbi5mb3JtYXR0ZWQpLnRvQ29udGFpbihcIkNPQUxFU0NFKHNvbmFtdV9qc29uYl9hcnJheV9hZ2coYWxpYXNlc19jaSksICcnKVwiKTtcbiAgICBleHBlY3QobWlncmF0aW9uLmZvcm1hdHRlZCkudG9Db250YWluKFxuICAgICAgXCJDT0FMRVNDRShzb25hbXVfanNvbmJfYXJyYXlfYWdnKGFsaWFzZXNfY3MsIGZhbHNlKSwgJycpXCIsXG4gICAgKTtcbiAgICBleHBlY3QobWlncmF0aW9uLmZvcm1hdHRlZCkudG9Db250YWluKFxuICAgICAgXCJDUkVBVEUgSU5ERVggY29kZV9nZW5lcmF0aW9uX3NlYXJjaF90ZXh0X3NlYXJjaF90ZXh0X2luZGV4IE9OXCIsXG4gICAgKTtcbiAgICBleHBlY3QobWlncmF0aW9uLmZvcm1hdHRlZCkudG9Db250YWluKFwiVVNJTkcgZ2luKHNlYXJjaF90ZXh0IGdpbl90cmdtX29wcyk7XCIpO1xuICB9KTtcblxuICB0ZXN0KFwic2VhcmNoVGV4dCBnZW5lcmF0ZWQgZXhwcmVzc2lvbiDrs4Dqsr0g7IucIGFsdGVyIHBhdGjsl5DshJwgaGVscGVy7JmAIGluZGV466W8IO2VqOq7mCDsnqzsg53shLHtlbTslbwg7ZWc64ukXCIsIGFzeW5jICgpID0+IHtcbiAgICBjb25zdCBlbnRpdHkgPSBhd2FpdCByZWdpc3RlckVudGl0eSh7XG4gICAgICBpZDogXCJDb2RlR2VuZXJhdGlvblNlYXJjaFRleHRBbHRlclwiLFxuICAgICAgdGFibGU6IFwiY29kZV9nZW5lcmF0aW9uX3NlYXJjaF90ZXh0X2FsdGVyXCIsXG4gICAgICBwcm9wczogW1xuICAgICAgICB7IG5hbWU6IFwiaWRcIiwgdHlwZTogXCJpbnRlZ2VyXCIgfSxcbiAgICAgICAgeyBuYW1lOiBcInRpdGxlX2NpXCIsIHR5cGU6IFwic3RyaW5nXCIgfSxcbiAgICAgICAgeyBuYW1lOiBcInRhZ3NfY2lcIiwgdHlwZTogXCJzdHJpbmdbXVwiIH0sXG4gICAgICAgIHtcbiAgICAgICAgICBuYW1lOiBcInNlYXJjaF90ZXh0XCIsXG4gICAgICAgICAgdHlwZTogXCJzZWFyY2hUZXh0XCIsXG4gICAgICAgICAgc291cmNlQ29sdW1uczogW1xuICAgICAgICAgICAgeyBuYW1lOiBcInRpdGxlX2NpXCIsIGNhc2VJbnNlbnNpdGl2ZTogZmFsc2UgfSxcbiAgICAgICAgICAgIHsgbmFtZTogXCJ0YWdzX2NpXCIsIGNhc2VJbnNlbnNpdGl2ZTogdHJ1ZSB9LFxuICAgICAgICAgIF0sXG4gICAgICAgIH0sXG4gICAgICBdLFxuICAgICAgaW5kZXhlczogW1xuICAgICAgICB7XG4gICAgICAgICAgdHlwZTogXCJpbmRleFwiLFxuICAgICAgICAgIG5hbWU6IFwiY29kZV9nZW5lcmF0aW9uX3NlYXJjaF90ZXh0X2FsdGVyX3NlYXJjaF90ZXh0X2luZGV4XCIsXG4gICAgICAgICAgdXNpbmc6IFwiZ2luXCIsXG4gICAgICAgICAgY29sdW1uczogW3sgbmFtZTogXCJzZWFyY2hfdGV4dFwiLCBvcGNsYXNzOiBcImdpbl90cmdtX29wc1wiIH1dLFxuICAgICAgICB9LFxuICAgICAgXSxcbiAgICB9KTtcblxuICAgIGNvbnN0IGVudGl0eVNldCA9IGdldE1pZ3JhdGlvblNldEZyb21FbnRpdHkoZW50aXR5KTtcbiAgICBjb25zdCBkYlNldCA9IGJ1aWxkRGJTZXRXaXRoR2VuZXJhdGVkU2VhcmNoVGV4dChcbiAgICAgIGVudGl0eVNldCxcbiAgICAgIGB0cmltKGxvd2VyKENPQUxFU0NFKHRpdGxlX2NpLCAnJykpKWAsXG4gICAgKTtcblxuICAgIGNvbnN0IFttaWdyYXRpb25dID0gYXdhaXQgZ2VuZXJhdGVBbHRlckNvZGUoZW50aXR5U2V0LCBkYlNldCk7XG5cbiAgICBleHBlY3QobWlncmF0aW9uLmZvcm1hdHRlZCkudG9Db250YWluKFxuICAgICAgJ3RhYmxlLmRyb3BJbmRleChbXCJzZWFyY2hfdGV4dFwiXSwgXCJjb2RlX2dlbmVyYXRpb25fc2VhcmNoX3RleHRfYWx0ZXJfc2VhcmNoX3RleHRfaW5kZXhcIiknLFxuICAgICk7XG4gICAgZXhwZWN0KG1pZ3JhdGlvbi5mb3JtYXR0ZWQpLnRvQ29udGFpbigndGFibGUuZHJvcENvbHVtbnMoXCJzZWFyY2hfdGV4dFwiKScpO1xuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oXG4gICAgICBcIkNSRUFURSBPUiBSRVBMQUNFIEZVTkNUSU9OIHNvbmFtdV90ZXh0X2FycmF5X2FnZyhhcnIgdGV4dFtdLCBjaSBib29sZWFuIERFRkFVTFQgdHJ1ZSlcIixcbiAgICApO1xuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oXG4gICAgICBgQUREIENPTFVNTiBcInNlYXJjaF90ZXh0XCIgdGV4dCBHRU5FUkFURUQgQUxXQVlTIEFTICh0cmltKENPQUxFU0NFKHRpdGxlX2NpLCAnJykgfHwgJyAnIHx8IENPQUxFU0NFKHNvbmFtdV90ZXh0X2FycmF5X2FnZyh0YWdzX2NpKSwgJycpKSkgU1RPUkVEIE5PVCBOVUxMYCxcbiAgICApO1xuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oXG4gICAgICBcIkNSRUFURSBJTkRFWCBjb2RlX2dlbmVyYXRpb25fc2VhcmNoX3RleHRfYWx0ZXJfc2VhcmNoX3RleHRfaW5kZXggT05cIixcbiAgICApO1xuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oXG4gICAgICBgQUREIENPTFVNTiBcInNlYXJjaF90ZXh0XCIgdGV4dCBHRU5FUkFURUQgQUxXQVlTIEFTICh0cmltKGxvd2VyKENPQUxFU0NFKHRpdGxlX2NpLCAnJykpKSkgU1RPUkVEIE5PVCBOVUxMYCxcbiAgICApO1xuICB9KTtcblxuICB0ZXN0KFwic2VhcmNoVGV4dCBnZW5lcmF0ZWQgZXhwcmVzc2lvbuydtCBQb3N0Z3JlcyBjYW5vbmljYWwgZm9ybeqzvCBzZW1hbnRpY2FsbHkgZXF1aXZhbGVudOuptCBuby1vcOydtOyWtOyVvCDtlZzri6RcIiwgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IGVudGl0eSA9IGF3YWl0IHJlZ2lzdGVyRW50aXR5KHtcbiAgICAgIGlkOiBcIkNvZGVHZW5lcmF0aW9uU2VhcmNoVGV4dENhbm9uaWNhbE5vb3BcIixcbiAgICAgIHRhYmxlOiBcImNvZGVfZ2VuZXJhdGlvbl9zZWFyY2hfdGV4dF9jYW5vbmljYWxfbm9vcFwiLFxuICAgICAgcHJvcHM6IFtcbiAgICAgICAgeyBuYW1lOiBcImlkXCIsIHR5cGU6IFwiaW50ZWdlclwiIH0sXG4gICAgICAgIHsgbmFtZTogXCJ1c2VybmFtZVwiLCB0eXBlOiBcInN0cmluZ1wiIH0sXG4gICAgICAgIHsgbmFtZTogXCJ0YWdzXCIsIHR5cGU6IFwic3RyaW5nW11cIiB9LFxuICAgICAgICB7XG4gICAgICAgICAgbmFtZTogXCJzZWFyY2hfdGV4dFwiLFxuICAgICAgICAgIHR5cGU6IFwic2VhcmNoVGV4dFwiLFxuICAgICAgICAgIHNvdXJjZUNvbHVtbnM6IFtcbiAgICAgICAgICAgIHsgbmFtZTogXCJ1c2VybmFtZVwiLCBjYXNlSW5zZW5zaXRpdmU6IGZhbHNlIH0sXG4gICAgICAgICAgICB7IG5hbWU6IFwidGFnc1wiLCBjYXNlSW5zZW5zaXRpdmU6IHRydWUgfSxcbiAgICAgICAgICBdLFxuICAgICAgICB9LFxuICAgICAgXSxcbiAgICAgIGluZGV4ZXM6IFtcbiAgICAgICAge1xuICAgICAgICAgIHR5cGU6IFwiaW5kZXhcIixcbiAgICAgICAgICBuYW1lOiBcImNvZGVfZ2VuZXJhdGlvbl9zZWFyY2hfdGV4dF9jYW5vbmljYWxfbm9vcF9zZWFyY2hfdGV4dF9pbmRleFwiLFxuICAgICAgICAgIHVzaW5nOiBcImdpblwiLFxuICAgICAgICAgIGNvbHVtbnM6IFt7IG5hbWU6IFwic2VhcmNoX3RleHRcIiwgb3BjbGFzczogXCJnaW5fdHJnbV9vcHNcIiB9XSxcbiAgICAgICAgfSxcbiAgICAgIF0sXG4gICAgfSk7XG5cbiAgICBjb25zdCBlbnRpdHlTZXQgPSBnZXRNaWdyYXRpb25TZXRGcm9tRW50aXR5KGVudGl0eSk7XG4gICAgY29uc3QgZGJTZXQgPSBidWlsZERiU2V0V2l0aEdlbmVyYXRlZFNlYXJjaFRleHQoXG4gICAgICB7XG4gICAgICAgIC4uLmVudGl0eVNldCxcbiAgICAgICAgaW5kZXhlczogZW50aXR5U2V0LmluZGV4ZXMubWFwKHNldE1pZ3JhdGlvbkluZGV4RGVmYXVsdHMpLFxuICAgICAgfSxcbiAgICAgIGBUUklNKEJPVEggRlJPTSAoKENPQUxFU0NFKHVzZXJuYW1lLCAnJzo6dGV4dCkpIHx8ICcgJzo6dGV4dCkgfHwgQ09BTEVTQ0Uoc29uYW11X3RleHRfYXJyYXlfYWdnKHRhZ3MsIHRydWUpLCAnJzo6dGV4dCkpYCxcbiAgICApO1xuXG4gICAgY29uc3QgbWlncmF0aW9ucyA9IGF3YWl0IGdlbmVyYXRlQWx0ZXJDb2RlKGVudGl0eVNldCwgZGJTZXQpO1xuXG4gICAgZXhwZWN0KG1pZ3JhdGlvbnMpLnRvSGF2ZUxlbmd0aCgwKTtcbiAgfSk7XG5cbiAgdGVzdChcInNlYXJjaFRleHQgY2FzZUluc2Vuc2l0aXZlIOuzgOqyveunjCDsnojslrTrj4QgYWx0ZXIgcGF0aOyXkOyEnCDsnqzsg53shLHtlbTslbwg7ZWc64ukXCIsIGFzeW5jICgpID0+IHtcbiAgICBjb25zdCBlbnRpdHkgPSBhd2FpdCByZWdpc3RlckVudGl0eSh7XG4gICAgICBpZDogXCJDb2RlR2VuZXJhdGlvblNlYXJjaFRleHRDYXNlSW5zZW5zaXRpdmVBbHRlclwiLFxuICAgICAgdGFibGU6IFwiY29kZV9nZW5lcmF0aW9uX3NlYXJjaF90ZXh0X2Nhc2VfaW5zZW5zaXRpdmVfYWx0ZXJcIixcbiAgICAgIHByb3BzOiBbXG4gICAgICAgIHsgbmFtZTogXCJpZFwiLCB0eXBlOiBcImludGVnZXJcIiB9LFxuICAgICAgICB7IG5hbWU6IFwidGl0bGVfY2lcIiwgdHlwZTogXCJzdHJpbmdcIiB9LFxuICAgICAgICB7XG4gICAgICAgICAgbmFtZTogXCJzZWFyY2hfdGV4dFwiLFxuICAgICAgICAgIHR5cGU6IFwic2VhcmNoVGV4dFwiLFxuICAgICAgICAgIHNvdXJjZUNvbHVtbnM6IFt7IG5hbWU6IFwidGl0bGVfY2lcIiwgY2FzZUluc2Vuc2l0aXZlOiBmYWxzZSB9XSxcbiAgICAgICAgfSxcbiAgICAgIF0sXG4gICAgICBpbmRleGVzOiBbXG4gICAgICAgIHtcbiAgICAgICAgICB0eXBlOiBcImluZGV4XCIsXG4gICAgICAgICAgbmFtZTogXCJjb2RlX2dlbmVyYXRpb25fc2VhcmNoX3RleHRfY2FzZV9pbnNlbnNpdGl2ZV9hbHRlcl9zZWFyY2hfdGV4dF9pbmRleFwiLFxuICAgICAgICAgIHVzaW5nOiBcImdpblwiLFxuICAgICAgICAgIGNvbHVtbnM6IFt7IG5hbWU6IFwic2VhcmNoX3RleHRcIiwgb3BjbGFzczogXCJnaW5fdHJnbV9vcHNcIiB9XSxcbiAgICAgICAgfSxcbiAgICAgIF0sXG4gICAgfSk7XG5cbiAgICBjb25zdCBlbnRpdHlTZXQgPSBnZXRNaWdyYXRpb25TZXRGcm9tRW50aXR5KGVudGl0eSk7XG4gICAgY29uc3QgZGJTZXQgPSBidWlsZERiU2V0V2l0aEdlbmVyYXRlZFNlYXJjaFRleHQoXG4gICAgICBlbnRpdHlTZXQsXG4gICAgICBgVFJJTShCT1RIIEZST00gbG93ZXIoQ09BTEVTQ0UodGl0bGVfY2ksICcnOjp0ZXh0KSkpYCxcbiAgICApO1xuXG4gICAgY29uc3QgW21pZ3JhdGlvbl0gPSBhd2FpdCBnZW5lcmF0ZUFsdGVyQ29kZShlbnRpdHlTZXQsIGRiU2V0KTtcblxuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oXCJ0YWJsZS5kcm9wSW5kZXgoXCIpO1xuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oXG4gICAgICAnXCJjb2RlX2dlbmVyYXRpb25fc2VhcmNoX3RleHRfY2FzZV9pbnNlbnNpdGl2ZV9hbHRlcl9zZWFyY2hfdGV4dF9pbmRleFwiJyxcbiAgICApO1xuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oJ3RhYmxlLmRyb3BDb2x1bW5zKFwic2VhcmNoX3RleHRcIiknKTtcbiAgICBleHBlY3QobWlncmF0aW9uLmZvcm1hdHRlZCkudG9Db250YWluKFxuICAgICAgYEFERCBDT0xVTU4gXCJzZWFyY2hfdGV4dFwiIHRleHQgR0VORVJBVEVEIEFMV0FZUyBBUyAodHJpbShDT0FMRVNDRSh0aXRsZV9jaSwgJycpKSkgU1RPUkVEIE5PVCBOVUxMYCxcbiAgICApO1xuICAgIGV4cGVjdChtaWdyYXRpb24uZm9ybWF0dGVkKS50b0NvbnRhaW4oXCJsb3dlcihDT0FMRVNDRSh0aXRsZV9jaVwiKTtcbiAgfSk7XG5cbiAgdGVzdChcInNlYXJjaFRleHQgaGVscGVyIGtpbmTqsIAg67CU64CM7Ja064+EIGRvd24gcGF0aOuKlCDrs7Xsm5AgZXhwcmVzc2lvbiDquLDspIAgaGVscGVy66W8IOyerOyDneyEse2VtOyVvCDtlZzri6RcIiwgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IGVudGl0eSA9IGF3YWl0IHJlZ2lzdGVyRW50aXR5KHtcbiAgICAgIGlkOiBcIkNvZGVHZW5lcmF0aW9uU2VhcmNoVGV4dFJvbGxiYWNrSGVscGVyTWlzbWF0Y2hcIixcbiAgICAgIHRhYmxlOiBcImNvZGVfZ2VuZXJhdGlvbl9zZWFyY2hfdGV4dF9yb2xsYmFja19oZWxwZXJfbWlzbWF0Y2hcIixcbiAgICAgIHByb3BzOiBbXG4gICAgICAgIHsgbmFtZTogXCJpZFwiLCB0eXBlOiBcImludGVnZXJcIiB9LFxuICAgICAgICB7IG5hbWU6IFwidGFnc1wiLCB0eXBlOiBcInN0cmluZ1tdXCIgfSxcbiAgICAgICAgeyBuYW1lOiBcImFsaWFzZXNcIiwgdHlwZTogXCJqc29uXCIsIGlkOiBcIlJvbGxiYWNrSGVscGVyTWlzbWF0Y2hKc29uXCIgfSxcbiAgICAgICAge1xuICAgICAgICAgIG5hbWU6IFwic2VhcmNoX3RleHRcIixcbiAgICAgICAgICB0eXBlOiBcInNlYXJjaFRleHRcIixcbiAgICAgICAgICBzb3VyY2VDb2x1bW5zOiBbeyBuYW1lOiBcInRhZ3NcIiwgY2FzZUluc2Vuc2l0aXZlOiB0cnVlIH1dLFxuICAgICAgICB9LFxuICAgICAgXSxcbiAgICAgIGluZGV4ZXM6IFtcbiAgICAgICAge1xuICAgICAgICAgIHR5cGU6IFwiaW5kZXhcIixcbiAgICAgICAgICBuYW1lOiBcImNvZGVfZ2VuZXJhdGlvbl9zZWFyY2hfdGV4dF9yb2xsYmFja19oZWxwZXJfbWlzbWF0Y2hfc2VhcmNoX3RleHRfaW5kZXhcIixcbiAgICAgICAgICB1c2luZzogXCJnaW5cIixcbiAgICAgICAgICBjb2x1bW5zOiBbeyBuYW1lOiBcInNlYXJjaF90ZXh0XCIsIG9wY2xhc3M6IFwiZ2luX3RyZ21fb3BzXCIgfV0sXG4gICAgICAgIH0sXG4gICAgICBdLFxuICAgIH0pO1xuXG4gICAgY29uc3QgZW50aXR5U2V0ID0gZ2V0TWlncmF0aW9uU2V0RnJvbUVudGl0eShlbnRpdHkpO1xuICAgIGNvbnN0IGRiU2V0ID0gYnVpbGREYlNldFdpdGhHZW5lcmF0ZWRTZWFyY2hUZXh0KFxuICAgICAgZW50aXR5U2V0LFxuICAgICAgYHRyaW0oQ09BTEVTQ0Uoc29uYW11X2pzb25iX2FycmF5X2FnZyhhbGlhc2VzKSwgJycpKWAsXG4gICAgKTtcblxuICAgIGNvbnN0IFttaWdyYXRpb25dID0gYXdhaXQgZ2VuZXJhdGVBbHRlckNvZGUoZW50aXR5U2V0LCBkYlNldCk7XG5cbiAgICBleHBlY3QobWlncmF0aW9uLmZvcm1hdHRlZCkudG9Db250YWluKFxuICAgICAgXCJDUkVBVEUgT1IgUkVQTEFDRSBGVU5DVElPTiBzb25hbXVfdGV4dF9hcnJheV9hZ2coYXJyIHRleHRbXSwgY2kgYm9vbGVhbiBERUZBVUxUIHRydWUpXCIsXG4gICAgKTtcbiAgICBleHBlY3QobWlncmF0aW9uLmZvcm1hdHRlZCkudG9Db250YWluKFxuICAgICAgYEFERCBDT0xVTU4gXCJzZWFyY2hfdGV4dFwiIHRleHQgR0VORVJBVEVEIEFMV0FZUyBBUyAodHJpbShDT0FMRVNDRShzb25hbXVfdGV4dF9hcnJheV9hZ2codGFncyksICcnKSkpIFNUT1JFRCBOT1QgTlVMTGAsXG4gICAgKTtcbiAgICBleHBlY3QobWlncmF0aW9uLmZvcm1hdHRlZCkudG9Db250YWluKFxuICAgICAgXCJDUkVBVEUgT1IgUkVQTEFDRSBGVU5DVElPTiBzb25hbXVfanNvbmJfYXJyYXlfYWdnKGFyciBqc29uYiwgY2kgYm9vbGVhbiBERUZBVUxUIHRydWUpXCIsXG4gICAgKTtcbiAgICBleHBlY3QobWlncmF0aW9uLmZvcm1hdHRlZCkudG9Db250YWluKFxuICAgICAgYEFERCBDT0xVTU4gXCJzZWFyY2hfdGV4dFwiIHRleHQgR0VORVJBVEVEIEFMV0FZUyBBUyAodHJpbShDT0FMRVNDRShzb25hbXVfanNvbmJfYXJyYXlfYWdnKGFsaWFzZXMpLCAnJykpKSBTVE9SRUQgTk9UIE5VTExgLFxuICAgICk7XG4gIH0pO1xuXG4gIHRlc3QoXCJzZWFyY2hUZXh06rCAIOygnOqxsOuQmOyWtOuPhCBkb3duIHBhdGjripQg7J207KCEIGV4cHJlc3Npb24gaGVscGVy66W8IOuzteybkO2VtOyVvCDtlZzri6RcIiwgYXN5bmMgKCkgPT4ge1xuICAgIGNvbnN0IGVudGl0eSA9IGF3YWl0IHJlZ2lzdGVyRW50aXR5KHtcbiAgICAgIGlkOiBcIkNvZGVHZW5lcmF0aW9uU2VhcmNoVGV4dFJvbGxiYWNrRHJvcHBlZFwiLFxuICAgICAgdGFibGU6IFwiY29kZV9nZW5lcmF0aW9uX3NlYXJjaF90ZXh0X3JvbGxiYWNrX2Ryb3BwZWRcIixcbiAgICAgIHByb3BzOiBbXG4gICAgICAgIHsgbmFtZTogXCJpZFwiLCB0eXBlOiBcImludGVnZXJcIiB9LFxuICAgICAgICB7IG5hbWU6IFwidGFnc1wiLCB0eXBlOiBcInN0cmluZ1tdXCIgfSxcbiAgICAgIF0sXG4gICAgICBpbmRleGVzOiBbXSxcbiAgICB9KTtcblxuICAgIGNvbnN0IHByZXZpb3VzRW50aXR5ID0gbmV3IEVudGl0eSh7XG4gICAgICAuLi5lbnRpdHkudG9Kc29uKCksXG4gICAgICBwcm9wczogW1xuICAgICAgICAuLi5lbnRpdHkudG9Kc29uKCkucHJvcHMsXG4gICAgICAgIHtcbiAgICAgICAgICBuYW1lOiBcInNlYXJjaF90ZXh0XCIsXG4gICAgICAgICAgdHlwZTogXCJzZWFyY2hUZXh0XCIsXG4gICAgICAgICAgc291cmNlQ29sdW1uczogW3sgbmFtZTogXCJ0YWdzXCIsIGNhc2VJbnNlbnNpdGl2ZTogdHJ1ZSB9XSxcbiAgICAgICAgfSxcbiAgICAgIF0sXG4gICAgICBpbmRleGVzOiBbXG4gICAgICAgIHtcbiAgICAgICAgICB0eXBlOiBcImluZGV4XCIsXG4gICAgICAgICAgbmFtZTogXCJjb2RlX2dlbmVyYXRpb25fc2VhcmNoX3RleHRfcm9sbGJhY2tfZHJvcHBlZF9zZWFyY2hfdGV4dF9pbmRleFwiLFxuICAgICAgICAgIHVzaW5nOiBcImdpblwiLFxuICAgICAgICAgIGNvbHVtbnM6IFt7IG5hbWU6IFwic2VhcmNoX3RleHRcIiwgb3BjbGFzczogXCJnaW5fdHJnbV9vcHNcIiB9XSxcbiAgICAgICAgfSxcbiAgICAgIF0sXG4gICAgfSk7XG5cbiAgICBjb25zdCBlbnRpdHlTZXQgPSBnZXRNaWdyYXRpb25TZXRGcm9tRW50aXR5KGVudGl0eSk7XG4gICAgY29uc3QgZGJTZXQgPSBidWlsZERiU2V0V2l0aEdlbmVyYXRlZFNlYXJjaFRleHQoXG4gICAgICBnZXRNaWdyYXRpb25TZXRGcm9tRW50aXR5KHByZXZpb3VzRW50aXR5KSxcbiAgICAgIGB0cmltKENPQUxFU0NFKHNvbmFtdV90ZXh0X2FycmF5X2FnZyh0YWdzKSwgJycpKWAsXG4gICAgKTtcblxuICAgIGNvbnN0IFttaWdyYXRpb25dID0gYXdhaXQgZ2VuZXJhdGVBbHRlckNvZGUoZW50aXR5U2V0LCBkYlNldCk7XG5cbiAgICBleHBlY3QobWlncmF0aW9uLmZvcm1hdHRlZCkudG9Db250YWluKCd0YWJsZS5kcm9wQ29sdW1ucyhcInNlYXJjaF90ZXh0XCIpJyk7XG4gICAgZXhwZWN0KG1pZ3JhdGlvbi5mb3JtYXR0ZWQpLnRvQ29udGFpbihcbiAgICAgIFwiQ1JFQVRFIE9SIFJFUExBQ0UgRlVOQ1RJT04gc29uYW11X3RleHRfYXJyYXlfYWdnKGFyciB0ZXh0W10sIGNpIGJvb2xlYW4gREVGQVVMVCB0cnVlKVwiLFxuICAgICk7XG4gICAgZXhwZWN0KG1pZ3JhdGlvbi5mb3JtYXR0ZWQpLnRvQ29udGFpbihcbiAgICAgIGBBREQgQ09MVU1OIFwic2VhcmNoX3RleHRcIiB0ZXh0IEdFTkVSQVRFRCBBTFdBWVMgQVMgKHRyaW0oQ09BTEVTQ0Uoc29uYW11X3RleHRfYXJyYXlfYWdnKHRhZ3MpLCAnJykpKSBTVE9SRUQgTk9UIE5VTExgLFxuICAgICk7XG4gICAgZXhwZWN0KG1pZ3JhdGlvbi5mb3JtYXR0ZWQpLnRvQ29udGFpbihcbiAgICAgIFwiQ1JFQVRFIElOREVYIGNvZGVfZ2VuZXJhdGlvbl9zZWFyY2hfdGV4dF9yb2xsYmFja19kcm9wcGVkX3NlYXJjaF90ZXh0X2luZGV4IE9OXCIsXG4gICAgKTtcbiAgICBleHBlY3QobWlncmF0aW9uLmZvcm1hdHRlZCkudG9Db250YWluKFwiVVNJTkcgZ2luKHNlYXJjaF90ZXh0IGdpbl90cmdtX29wcyk7XCIpO1xuICB9KTtcbn0pO1xuIl0sIm5hbWVzIjpbImRlc2NyaWJlIiwiZXhwZWN0IiwidGVzdCIsIlNvbmFtdSIsIkVudGl0eSIsIkVudGl0eU1hbmFnZXIiLCJzZXR1cEJpb21lIiwiZ2VuZXJhdGVBbHRlckNvZGUiLCJnZW5lcmF0ZUNyZWF0ZUNvZGUiLCJzZXRNaWdyYXRpb25JbmRleERlZmF1bHRzIiwiZ2V0TWlncmF0aW9uU2V0RnJvbUVudGl0eSIsIlRFU1RfQVBJX1JPT1QiLCJhcGlSb290UGF0aCIsImVudGl0eVNlcSIsInJlZ2lzdGVyRW50aXR5IiwiZGVmaW5pdGlvbiIsImVudGl0eSIsImlkIiwidGl0bGUiLCJ0YWJsZSIsInN1YnNldHMiLCJBIiwicHJvcHMiLCJtYXAiLCJwcm9wIiwibmFtZSIsImVudW1zIiwicmVnaXN0ZXIiLCJkZWZlclNlYXJjaFRleHRKc29uU291cmNlVmFsaWRhdGlvbiIsImdldCIsImJ1aWxkRGJTZXRXaXRoR2VuZXJhdGVkU2VhcmNoVGV4dCIsImVudGl0eVNldCIsImV4cHJlc3Npb24iLCJjb2x1bW5zIiwiY29sdW1uIiwiZ2VuZXJhdGVkIiwidHlwZSIsImRpbWVuc2lvbnMiLCJpbmRleGVzIiwidmVjdG9yT3BzIiwibWlncmF0aW9uIiwiZm9ybWF0dGVkIiwibm90IiwidG9Db250YWluIiwic291cmNlQ29sdW1ucyIsImNhc2VJbnNlbnNpdGl2ZSIsInVzaW5nIiwib3BjbGFzcyIsImRiU2V0IiwibWlncmF0aW9ucyIsInRvSGF2ZUxlbmd0aCIsInByZXZpb3VzRW50aXR5IiwidG9Kc29uIl0sIm1hcHBpbmdzIjoiQUFBQSxTQUFTQSxRQUFRLEVBQUVDLE1BQU0sRUFBRUMsSUFBSSxRQUFRLFNBQVM7QUFDaEQsU0FBU0MsTUFBTSxRQUFRLHFCQUFZO0FBQ25DLFNBQVNDLE1BQU0sUUFBUSx5QkFBc0I7QUFDN0MsU0FBU0MsYUFBYSxRQUFRLGlDQUE4QjtBQUU1RCxTQUFTQyxVQUFVLFFBQVEsMkJBQXdCO0FBQ25ELFNBQ0VDLGlCQUFpQixFQUNqQkMsa0JBQWtCLEVBQ2xCQyx5QkFBeUIsUUFDcEIsd0JBQXFCO0FBQzVCLFNBQVNDLHlCQUF5QixRQUFRLHNCQUFtQjtBQUU3RCxNQUFNQyxnQkFBZ0I7QUFFdEJMLFdBQVdLO0FBQ1hSLE9BQU9TLFdBQVcsR0FBR0Q7QUFFckIsSUFBSUUsWUFBWTtBQUVoQixlQUFlQyxlQUNiQyxVQU1DO0lBRURGLGFBQWE7SUFFYixNQUFNRyxTQUFTO1FBQ2IsR0FBR0QsVUFBVTtRQUNiRSxJQUFJLEdBQUdGLFdBQVdFLEVBQUUsR0FBR0osV0FBVztRQUNsQ0ssT0FBTyxHQUFHSCxXQUFXRSxFQUFFLEdBQUdKLFdBQVc7UUFDckNNLE9BQU8sR0FBR0osV0FBV0ksS0FBSyxDQUFDLENBQUMsRUFBRU4sV0FBVztRQUN6Q08sU0FBUztZQUNQQyxHQUFHTixXQUFXTyxLQUFLLENBQUNDLEdBQUcsQ0FBQyxDQUFDQyxPQUFTQSxLQUFLQyxJQUFJO1FBQzdDO1FBQ0FDLE9BQU8sQ0FBQztJQUNWO0lBRUEsTUFBTXJCLGNBQWNz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
@@ -1 +1 @@
1
- {"version":3,"file":"code-generation.d.ts","sourceRoot":"","sources":["../../src/migration/code-generation.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAGjC,OAAO,KAAK,EAEV,gBAAgB,EAGhB,cAAc,EACd,YAAY,EACb,MAAM,gBAAgB,CAAC;AA4uBxB;;GAEG;AACH,wBAAgB,iBAAiB,CAAC,aAAa,EAAE,cAAc,EAAE,EAAE,SAAS,EAAE,cAAc,EAAE;SAG/E,cAAc,EAAE;UACf,cAAc,EAAE;EAuC/B;AAWD;;GAEG;AACH,wBAAgB,yBAAyB,CAAC,KAAK,EAAE,cAAc,GAAG,cAAc,CAoB/E;AAuHD;;;;GAIG;AACH,wBAAsB,kBAAkB,CAAC,SAAS,EAAE,YAAY,GAAG,OAAO,CAAC,gBAAgB,EAAE,CAAC,CAS7F;AAED;;;;;;GAMG;AACH,wBAAsB,iBAAiB,CACrC,SAAS,EAAE,YAAY,EACvB,KAAK,EAAE,YAAY,EACnB,SAAS,CAAC,EAAE,IAAI,GACf,OAAO,CAAC,gBAAgB,EAAE,CAAC,CAyG7B"}
1
+ {"version":3,"file":"code-generation.d.ts","sourceRoot":"","sources":["../../src/migration/code-generation.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAGjC,OAAO,KAAK,EAEV,gBAAgB,EAGhB,cAAc,EACd,YAAY,EACb,MAAM,gBAAgB,CAAC;AAyhDxB;;GAEG;AACH,wBAAgB,iBAAiB,CAAC,aAAa,EAAE,cAAc,EAAE,EAAE,SAAS,EAAE,cAAc,EAAE;SAG/E,cAAc,EAAE;UACf,cAAc,EAAE;EAuC/B;AAWD;;GAEG;AACH,wBAAgB,yBAAyB,CAAC,KAAK,EAAE,cAAc,GAAG,cAAc,CAoB/E;AAuHD;;;;GAIG;AACH,wBAAsB,kBAAkB,CAAC,SAAS,EAAE,YAAY,GAAG,OAAO,CAAC,gBAAgB,EAAE,CAAC,CAS7F;AAED;;;;;;GAMG;AACH,wBAAsB,iBAAiB,CACrC,SAAS,EAAE,YAAY,EACvB,KAAK,EAAE,YAAY,EACnB,SAAS,CAAC,EAAE,IAAI,GACf,OAAO,CAAC,gBAAgB,EAAE,CAAC,CA0G7B"}