@forklaunch/core 0.1.0 → 0.1.2

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 (87) hide show
  1. package/.prettierignore +2 -0
  2. package/.prettierrc +6 -0
  3. package/dist/cache/index.d.ts +3 -0
  4. package/dist/cache/index.js +20 -0
  5. package/dist/cache/index.js.map +1 -0
  6. package/dist/cache/interfaces/ttlCache.interface.d.ts +1 -1
  7. package/dist/cache/redisTtlCache.d.ts +3 -2
  8. package/dist/cache/redisTtlCache.js +2 -2
  9. package/dist/cache/redisTtlCache.js.map +1 -1
  10. package/dist/cache/types/{ttlCacheRecord.js → ttlCacheRecord.types.js} +1 -1
  11. package/dist/cache/types/ttlCacheRecord.types.js.map +1 -0
  12. package/dist/controllers/index.d.ts +1 -0
  13. package/dist/controllers/index.js +18 -0
  14. package/dist/controllers/index.js.map +1 -0
  15. package/dist/database/index.d.ts +1 -0
  16. package/dist/database/index.js +18 -0
  17. package/dist/database/index.js.map +1 -0
  18. package/dist/database/mikro/models/entities/base.entity.js +1 -1
  19. package/dist/entityMapper/index.d.ts +2 -0
  20. package/dist/entityMapper/index.js +19 -0
  21. package/dist/entityMapper/index.js.map +1 -0
  22. package/dist/entityMapper/interfaces/entityMapper.interface.d.ts +3 -3
  23. package/dist/entityMapper/models/baseEntityMapper.model.d.ts +23 -23
  24. package/dist/entityMapper/models/baseEntityMapper.model.js +17 -17
  25. package/dist/entityMapper/models/baseEntityMapper.model.js.map +1 -1
  26. package/dist/entityMapper/models/requestEntityMapper.model.d.ts +17 -23
  27. package/dist/entityMapper/models/requestEntityMapper.model.js +13 -19
  28. package/dist/entityMapper/models/requestEntityMapper.model.js.map +1 -1
  29. package/dist/entityMapper/models/responseEntityMapper.model.d.ts +15 -16
  30. package/dist/entityMapper/models/responseEntityMapper.model.js +8 -8
  31. package/dist/entityMapper/models/responseEntityMapper.model.js.map +1 -1
  32. package/dist/entityMapper/types/entityMapper.types.d.ts +3 -22
  33. package/dist/http/index.d.ts +2 -0
  34. package/dist/http/index.js +19 -0
  35. package/dist/http/index.js.map +1 -0
  36. package/dist/http/middlewares/index.d.ts +2 -0
  37. package/dist/http/middlewares/index.js +19 -0
  38. package/dist/http/middlewares/index.js.map +1 -0
  39. package/dist/http/middlewares/request.middleware.d.ts +11 -11
  40. package/dist/http/middlewares/request.middleware.js +32 -31
  41. package/dist/http/middlewares/request.middleware.js.map +1 -1
  42. package/dist/http/middlewares/response.middleware.d.ts +3 -2
  43. package/dist/http/middlewares/response.middleware.js +9 -6
  44. package/dist/http/middlewares/response.middleware.js.map +1 -1
  45. package/dist/http/types/api.types.d.ts +16 -13
  46. package/dist/http/types/index.d.ts +2 -0
  47. package/dist/http/types/index.js +19 -0
  48. package/dist/http/types/index.js.map +1 -0
  49. package/dist/http/types/primitive.types.d.ts +14 -13
  50. package/dist/index.d.ts +6 -1
  51. package/dist/index.js +20 -5
  52. package/dist/index.js.map +1 -1
  53. package/dist/services/index.d.ts +1 -0
  54. package/dist/services/index.js +18 -0
  55. package/dist/services/index.js.map +1 -0
  56. package/dist/services/interfaces/baseService.d.ts +1 -1
  57. package/dist/tests/{dto.test.js → entityMapper.test.js} +10 -32
  58. package/dist/tests/entityMapper.test.js.map +1 -0
  59. package/dist/tests/http.middleware.test.d.ts +1 -0
  60. package/dist/tests/http.middleware.test.js +79 -0
  61. package/dist/tests/http.middleware.test.js.map +1 -0
  62. package/dist/tests/redisTtlCache.test.js +32 -15
  63. package/dist/tests/redisTtlCache.test.js.map +1 -1
  64. package/entityMapper/index.ts +2 -0
  65. package/entityMapper/interfaces/entityMapper.interface.ts +11 -11
  66. package/entityMapper/models/baseEntityMapper.model.ts +90 -73
  67. package/entityMapper/models/requestEntityMapper.model.ts +96 -76
  68. package/entityMapper/models/responseEntityMapper.model.ts +83 -62
  69. package/entityMapper/types/entityMapper.types.ts +6 -23
  70. package/http/index.ts +2 -0
  71. package/http/middlewares/index.ts +2 -0
  72. package/http/middlewares/request.middleware.ts +118 -62
  73. package/http/middlewares/response.middleware.ts +56 -30
  74. package/http/types/api.types.ts +43 -25
  75. package/http/types/index.ts +2 -0
  76. package/http/types/primitive.types.ts +65 -44
  77. package/index.ts +6 -5
  78. package/jest.config.ts +2 -2
  79. package/package.json +13 -4
  80. package/tests/entityMapper.test.ts +219 -0
  81. package/tests/http.middleware.test.ts +99 -0
  82. package/tests/redisTtlCache.test.ts +49 -29
  83. package/dist/cache/types/ttlCacheRecord.js.map +0 -1
  84. package/dist/tests/dto.test.js.map +0 -1
  85. package/tests/dto.test.ts +0 -235
  86. /package/dist/cache/types/{ttlCacheRecord.d.ts → ttlCacheRecord.types.d.ts} +0 -0
  87. /package/dist/tests/{dto.test.d.ts → entityMapper.test.d.ts} +0 -0
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@forklaunch/core",
3
- "version": "0.1.0",
3
+ "version": "0.1.2",
4
4
  "description": "forklaunch-js core package. Contains useful building blocks.",
5
5
  "main": "index.js",
6
6
  "scripts": {
@@ -8,7 +8,8 @@
8
8
  "build": "tsc",
9
9
  "docs": "typedoc --out docs *",
10
10
  "lint": "eslint . -c eslint.config.mjs",
11
- "lint:fix": "eslint . -c eslint.config.mjs --fix"
11
+ "lint:fix": "eslint . -c eslint.config.mjs --fix",
12
+ "format": "prettier --ignore-path=.prettierignore --config .prettierrc '**/*.ts' --write"
12
13
  },
13
14
  "author": "Rohin Bhargava",
14
15
  "license": "MIT",
@@ -21,7 +22,7 @@
21
22
  },
22
23
  "homepage": "https://github.com/forklaunch/forklaunch-js#readme",
23
24
  "dependencies": {
24
- "@forklaunch/validator": "^0.1.15",
25
+ "@forklaunch/validator": "^0.2.9",
25
26
  "@mikro-orm/core": "^6.2.9",
26
27
  "jose": "^5.6.2",
27
28
  "redis": "^4.6.14",
@@ -35,6 +36,8 @@
35
36
  "@types/redis": "^4.0.11",
36
37
  "@types/uuid": "^10.0.0",
37
38
  "globals": "^15.8.0",
39
+ "prettier": "^3.3.2",
40
+ "testcontainers": "^10.10.1",
38
41
  "ts-jest": "^29.1.5",
39
42
  "ts-node": "^10.9.2",
40
43
  "typedoc": "^0.26.3",
@@ -43,5 +46,11 @@
43
46
  },
44
47
  "directories": {
45
48
  "test": "tests"
49
+ },
50
+ "exports": {
51
+ ".": {
52
+ "types": "./dist/index.d.ts",
53
+ "default": "./dist/index.js"
54
+ }
46
55
  }
47
- }
56
+ }
@@ -0,0 +1,219 @@
1
+ import {
2
+ TypeboxSchemaValidator,
3
+ number,
4
+ string
5
+ } from '@forklaunch/validator/typebox';
6
+ import { BaseEntity } from '../database/mikro/models/entities/base.entity';
7
+ import { RequestEntityMapper } from '../entityMapper/models/requestEntityMapper.model';
8
+ import { ResponseEntityMapper } from '../entityMapper/models/responseEntityMapper.model';
9
+
10
+ class TestEntity extends BaseEntity {
11
+ name: string;
12
+ age: number;
13
+ }
14
+
15
+ class TestRequestEntityMapper extends RequestEntityMapper<
16
+ TestEntity,
17
+ TypeboxSchemaValidator
18
+ > {
19
+ schema = {
20
+ id: string,
21
+ name: string,
22
+ age: number
23
+ };
24
+
25
+ toEntity(...additionalArgs: unknown[]): TestEntity {
26
+ const entity = new TestEntity();
27
+ entity.id = this.dto.id;
28
+ entity.name = this.dto.name;
29
+ entity.age = this.dto.age;
30
+
31
+ return entity;
32
+ }
33
+ }
34
+
35
+ class TestResponseEntityMapper extends ResponseEntityMapper<
36
+ TestEntity,
37
+ TypeboxSchemaValidator
38
+ > {
39
+ schema = {
40
+ id: string,
41
+ name: string,
42
+ age: number
43
+ };
44
+
45
+ fromEntity(entity: TestEntity): this {
46
+ this.dto = {
47
+ id: entity.id,
48
+ name: entity.name,
49
+ age: entity.age
50
+ };
51
+
52
+ return this;
53
+ }
54
+ }
55
+
56
+ function extractNonTimeBasedEntityFields<T extends BaseEntity>(entity: T): T {
57
+ entity.createdAt = new Date(0);
58
+ entity.updatedAt = new Date(0);
59
+ return entity;
60
+ }
61
+
62
+ describe('Request Entity Mapper Test', () => {
63
+ let TestRequestEM: TestRequestEntityMapper;
64
+
65
+ beforeAll(() => {
66
+ TestRequestEM = new TestRequestEntityMapper(new TypeboxSchemaValidator());
67
+ });
68
+
69
+ test('Schema Equality', async () => {
70
+ expect(TestRequestEM.schema).toEqual(TestRequestEntityMapper.schema());
71
+ });
72
+
73
+ test('From JSON', async () => {
74
+ const json = {
75
+ id: '123',
76
+ name: 'test',
77
+ age: 1
78
+ };
79
+
80
+ const responseEM = TestRequestEM.fromJson(json);
81
+ const staticEM = TestRequestEntityMapper.fromJson(
82
+ new TypeboxSchemaValidator(),
83
+ json
84
+ );
85
+ const expectedDto = {
86
+ id: '123',
87
+ name: 'test',
88
+ age: 1
89
+ };
90
+
91
+ expect(staticEM.dto).toEqual(expectedDto);
92
+ expect(responseEM.dto).toEqual(expectedDto);
93
+ expect(responseEM.dto).toEqual(staticEM.dto);
94
+ });
95
+
96
+ test('Deserialization Equality', async () => {
97
+ const json = {
98
+ id: '123',
99
+ name: 'test',
100
+ age: 1
101
+ };
102
+
103
+ const entity = extractNonTimeBasedEntityFields(
104
+ TestRequestEM.deserializeJsonToEntity(json)
105
+ );
106
+ const objectEntity = extractNonTimeBasedEntityFields(
107
+ TestRequestEM.fromJson(json).toEntity()
108
+ );
109
+ const staticEntity = extractNonTimeBasedEntityFields(
110
+ TestRequestEntityMapper.deserializeJsonToEntity(
111
+ new TypeboxSchemaValidator(),
112
+ json
113
+ )
114
+ );
115
+ let expectedEntity = new TestEntity();
116
+ expectedEntity.id = '123';
117
+ expectedEntity.name = 'test';
118
+ expectedEntity.age = 1;
119
+
120
+ expectedEntity = extractNonTimeBasedEntityFields(expectedEntity);
121
+
122
+ expect(entity).toEqual(expectedEntity);
123
+ expect(objectEntity).toEqual(expectedEntity);
124
+ expect(staticEntity).toEqual(expectedEntity);
125
+ expect(entity).toEqual(objectEntity);
126
+ expect(entity).toEqual(staticEntity);
127
+ expect(staticEntity).toEqual(expectedEntity);
128
+ expect(staticEntity).toEqual(objectEntity);
129
+ });
130
+
131
+ test('Serialization Failure', async () => {
132
+ const json = {
133
+ id: '123',
134
+ name: 'test'
135
+ };
136
+
137
+ // @ts-expect-error
138
+ expect(() => TestRequestEM.fromJson(json)).toThrow();
139
+ expect(() =>
140
+ // @ts-expect-error
141
+ TestRequestEntityMapper.fromJson(new TypeboxSchemaValidator(), json)
142
+ ).toThrow();
143
+ });
144
+ });
145
+
146
+ describe('Response Entity Mapper Test', () => {
147
+ let TestResponseEM: TestResponseEntityMapper;
148
+
149
+ beforeAll(() => {
150
+ TestResponseEM = new TestResponseEntityMapper(new TypeboxSchemaValidator());
151
+ });
152
+
153
+ test('Schema Equality', async () => {
154
+ expect(TestResponseEM.schema).toEqual(TestResponseEntityMapper.schema());
155
+ });
156
+
157
+ test('From Entity', async () => {
158
+ const entity = new TestEntity();
159
+ entity.id = '123';
160
+ entity.name = 'test';
161
+ entity.age = 1;
162
+
163
+ const responseEM = TestResponseEM.fromEntity(entity);
164
+ const staticEM = TestResponseEntityMapper.fromEntity(
165
+ new TypeboxSchemaValidator(),
166
+ entity
167
+ );
168
+ const expectedDto = {
169
+ id: '123',
170
+ name: 'test',
171
+ age: 1
172
+ };
173
+
174
+ expect(staticEM.dto).toEqual(expectedDto);
175
+ expect(responseEM.dto).toEqual(expectedDto);
176
+ expect(responseEM.dto).toEqual(staticEM.dto);
177
+ });
178
+
179
+ test('Serialization Equality', async () => {
180
+ const entity = new TestEntity();
181
+ entity.id = '123';
182
+ entity.name = 'test';
183
+ entity.age = 1;
184
+
185
+ const json = TestResponseEM.serializeEntityToJson(entity);
186
+ const objectJson = TestResponseEM.fromEntity(entity).toJson();
187
+ const staticJson = TestResponseEntityMapper.serializeEntityToJson(
188
+ new TypeboxSchemaValidator(),
189
+ entity
190
+ );
191
+ const expectedJson = {
192
+ id: '123',
193
+ name: 'test',
194
+ age: 1
195
+ };
196
+
197
+ expect(json).toEqual(expectedJson);
198
+ expect(objectJson).toEqual(expectedJson);
199
+ expect(staticJson).toEqual(expectedJson);
200
+ expect(json).toEqual(objectJson);
201
+ expect(json).toEqual(staticJson);
202
+ expect(staticJson).toEqual(expectedJson);
203
+ expect(staticJson).toEqual(objectJson);
204
+ });
205
+
206
+ test('Serialization Failure', async () => {
207
+ const entity = new TestEntity();
208
+ entity.id = '123';
209
+ entity.name = 'test';
210
+
211
+ expect(() => TestResponseEM.fromEntity(entity).toJson()).toThrow();
212
+ expect(() =>
213
+ TestResponseEntityMapper.fromEntity(
214
+ new TypeboxSchemaValidator(),
215
+ entity
216
+ ).toJson()
217
+ ).toThrow();
218
+ });
219
+ });
@@ -0,0 +1,99 @@
1
+ import { SchemaValidator } from '@forklaunch/validator';
2
+ import {
3
+ MockSchemaValidator,
4
+ literal,
5
+ mockSchemaValidator,
6
+ optional,
7
+ union
8
+ } from '@forklaunch/validator/tests/mockSchemaValidator';
9
+ import { ForklaunchRequest, ForklaunchResponse, HttpContractDetails, RequestContext, createRequestContext, enrichRequestDetails, parseRequestBody, parseRequestHeaders, parseRequestParams, parseRequestQuery, parseResponse } from '../http';
10
+
11
+ describe('Http Middleware Tests', () => {
12
+ let contractDetails: HttpContractDetails<MockSchemaValidator>;
13
+ let req: ForklaunchRequest<MockSchemaValidator>;
14
+ let resp: ForklaunchResponse;
15
+
16
+ const nextFunction = (err?: unknown) => {
17
+ expect(err).toBeFalsy()
18
+ };
19
+
20
+ const testSchema = {
21
+ test: union(['a', optional(literal('test'))] as const)
22
+ };
23
+
24
+ beforeAll(() => {
25
+ contractDetails = {
26
+ name: 'Test Contract',
27
+ summary: 'Test Contract Summary',
28
+ body: testSchema,
29
+ params: testSchema,
30
+ requestHeaders: testSchema,
31
+ query: testSchema,
32
+ responses: {
33
+ 200: testSchema
34
+ }
35
+ };
36
+
37
+ req = {
38
+ context: {} as RequestContext,
39
+ contractDetails: {} as HttpContractDetails<MockSchemaValidator>,
40
+ schemaValidator: {} as SchemaValidator,
41
+ params: testSchema,
42
+ headers: testSchema,
43
+ body: testSchema,
44
+ query: testSchema
45
+ };
46
+
47
+ resp = {
48
+ bodyData: {},
49
+ statusCode: 200,
50
+ corked: false,
51
+ getHeaders: jest.fn(),
52
+ setHeader: jest.fn(),
53
+ status: jest.fn(),
54
+ send: jest.fn(),
55
+ json: jest.fn(),
56
+ jsonp: jest.fn(),
57
+ };
58
+ });
59
+
60
+
61
+
62
+ test('Create Request Context', async () => {
63
+ req.context = {} as RequestContext;
64
+ req.schemaValidator = {} as SchemaValidator;
65
+ createRequestContext(mockSchemaValidator)(req, resp, nextFunction);
66
+ expect(req.context.correlationId).not.toBe('123');
67
+ expect(req.schemaValidator).toBe(mockSchemaValidator);
68
+ });
69
+
70
+ test('Enrich Request Details', async () => {
71
+ req.contractDetails = {} as HttpContractDetails<MockSchemaValidator>;
72
+ enrichRequestDetails(contractDetails)(req, resp, nextFunction);
73
+ expect(req.contractDetails).toEqual(contractDetails);
74
+ });
75
+
76
+ test('Validate Request Params', async () => {
77
+ parseRequestParams(req, resp, nextFunction);
78
+ });
79
+
80
+ test('Validate Request Headers', async () => {
81
+ parseRequestHeaders(req, resp, nextFunction);
82
+ });
83
+
84
+ test('Validate Request Body', async () => {
85
+ parseRequestBody(req, resp, nextFunction);
86
+ });
87
+
88
+ test('Validate Request Query Params', async () => {
89
+ parseRequestQuery(req, resp, nextFunction);
90
+ });
91
+
92
+ test('Validate Response', async () => {
93
+ parseResponse(req, resp, nextFunction);
94
+ });
95
+
96
+ // Not supported yet
97
+ // test('Validate Auth', async () => {
98
+ // });
99
+ });
@@ -1,42 +1,62 @@
1
+ import { GenericContainer, StartedTestContainer } from 'testcontainers';
1
2
  import { RedisTtlCache } from '../cache/redisTtlCache';
2
3
 
3
4
  describe('RedisTtlCache', () => {
4
- let cache: RedisTtlCache;
5
-
6
- beforeAll(() => {
7
- // Mock the Redis client
8
- // Override the RedisTtlCache's client with the mock client
9
- cache = new RedisTtlCache(5000);
5
+ let container: StartedTestContainer;
6
+ let cache: RedisTtlCache;
7
+ let key: string;
8
+ let value: unknown;
9
+ let ttlMilliseconds: number;
10
+
11
+ beforeAll(async () => {
12
+ container = await new GenericContainer('redis')
13
+ .withExposedPorts(6379)
14
+ .start();
15
+
16
+ cache = new RedisTtlCache(5000, {
17
+ url: `redis://${container.getHost()}:${container.getMappedPort(6379)}`
10
18
  });
11
19
 
12
- afterAll(async () => {
13
- // Ensure the Redis client is disconnected after tests complete
14
- await cache.disconnect();
15
- });
20
+ key = 'testKey';
21
+ value = { data: 'testValue' };
22
+ ttlMilliseconds = 1000;
23
+ }, 30000);
24
+
25
+ afterAll(async () => {
26
+ await cache.disconnect();
27
+ await container.stop();
28
+ });
16
29
 
17
- test('putRecord and readRecord', async () => {
18
- const key = 'testKey';
19
- const value = { data: 'testValue' };
20
- const ttlMilliseconds = 10000; // 10 seconds
30
+ it('PutRecord', async () => {
31
+ await cache.putRecord({ key, value, ttlMilliseconds });
32
+ });
21
33
 
22
- await cache.putRecord({ key, value, ttlMilliseconds });
23
- const storedValue = await cache.readRecord(key);
34
+ test('Read Record', async () => {
35
+ const storedValue = await cache.readRecord(key);
24
36
 
25
- expect(storedValue).toEqual(value);
37
+ expect(storedValue).toEqual({
38
+ key,
39
+ ttlMilliseconds,
40
+ value
26
41
  });
42
+ });
27
43
 
28
- test('peekRecord', async () => {
29
- const key = 'testKey';
30
- const exists = await cache.peekRecord(key);
44
+ test('Peek Record', async () => {
45
+ const exists = await cache.peekRecord(key);
31
46
 
32
- expect(exists).toBeTruthy();
33
- });
47
+ expect(exists).toBeTruthy();
48
+ });
34
49
 
35
- test('deleteRecord', async () => {
36
- const key = 'testKey';
37
- await cache.deleteRecord(key);
38
- const existsAfterDelete = await cache.peekRecord(key);
50
+ test('Delete Record', async () => {
51
+ await cache.deleteRecord(key);
52
+ const existsAfterDelete = await cache.peekRecord(key);
39
53
 
40
- expect(existsAfterDelete).toBeFalsy();
41
- });
42
- });
54
+ expect(existsAfterDelete).toBeFalsy();
55
+ });
56
+
57
+ test('Check No Record', async () => {
58
+ await Promise.resolve(setTimeout(async () => {}, ttlMilliseconds));
59
+ const existsAfterTtl = await cache.peekRecord(key);
60
+ expect(existsAfterTtl).toBeFalsy();
61
+ });
62
+ });
@@ -1 +0,0 @@
1
- {"version":3,"file":"ttlCacheRecord.js","sourceRoot":"","sources":["../../../cache/types/ttlCacheRecord.ts"],"names":[],"mappings":""}
@@ -1 +0,0 @@
1
- {"version":3,"file":"dto.test.js","sourceRoot":"","sources":["../../tests/dto.test.ts"],"names":[],"mappings":";;AACA,2DAAuF;AAEvF,+EAA2E;AAC3E,gGAAuF;AACvF,kGAAyF;AAEzF,MAAM,UAAW,SAAQ,wBAAU;IAC/B,IAAI,CAAS;IACb,GAAG,CAAS;CACf;AAED,MAAM,CAAE,SAAQ,+CAAuD;IACnE,QAAQ,CAAC,GAAG,cAAyB;QACjC,MAAM,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;QAChC,MAAM,CAAC,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;QACxB,MAAM,CAAC,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC;QAC5B,MAAM,CAAC,GAAG,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC;QAE1B,OAAO,MAAM,CAAC;IAClB,CAAC;IACD,MAAM,GAAG;QACL,EAAE,EAAE,gBAAM;QACV,IAAI,EAAE,gBAAM;QACZ,GAAG,EAAE,gBAAM;KACd,CAAC;CAEL;AAED,MAAM,CAAC,GAAG,IAAI,CAAC,CAAC,IAAI,gCAAsB,EAAE,CAAC,CAAC;AAC9C,CAAC,CAAC,QAAQ,CAAC,EAAE,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,EAAE,CAAC,EAAE,CAAC,CAAC;AAEhD,IAAI,CAAC,GAAG,EAAE,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE,GAAG,EAAE,CAAC,EAAE,CAAC;AAI5C,CAAC,CAAC,QAAQ,CAAC,IAAI,gCAAsB,EAAE,EAAE,CAAC,CAAC,CAAC;AAC5C,CAAC,CAAC,QAAQ,CAAC,IAAI,gCAAsB,EAAE,EAAE;IACrC,EAAE,EAAE,KAAK;IACT,IAAI,EAAE,MAAM;IACZ,GAAG,EAAE,CAAC;CACT,CAAC,CAAC;AAEH,MAAM,uBAAwB,SAAQ,+CAAuD;IACzF,MAAM,GAAG;QACL,EAAE,EAAE,gBAAM;QACV,IAAI,EAAE,gBAAM;QACZ,GAAG,EAAE,gBAAM;KACd,CAAC;IAEF,QAAQ,CAAC,GAAG,cAAyB;QACjC,MAAM,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;QAChC,MAAM,CAAC,EAAE,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;QACxB,MAAM,CAAC,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC;QAC5B,MAAM,CAAC,GAAG,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC;QAE1B,OAAO,MAAM,CAAC;IAClB,CAAC;CACJ;AAgBD,MAAM,wBAAyB,SAAQ,iDAAwD;IAC3F,MAAM,GAAG;QACL,EAAE,EAAE,gBAAM;QACV,IAAI,EAAE,gBAAM;QACZ,GAAG,EAAE,gBAAM;KACd,CAAC;IAEF,UAAU,CAAC,MAAkB;QACzB,IAAI,CAAC,GAAG,GAAG;YACP,EAAE,EAAE,MAAM,CAAC,EAAE;YACb,IAAI,EAAE,MAAM,CAAC,IAAI;YACjB,GAAG,EAAE,MAAM,CAAC,GAAG;SAClB,CAAC;QAEF,OAAO,IAAI,CAAC;IAChB,CAAC;CACJ;AAID,SAAS,+BAA+B,CAAuB,MAAS;IACpE,MAAM,CAAC,SAAS,GAAG,IAAI,IAAI,CAAC,CAAC,CAAC,CAAC;IAC/B,MAAM,CAAC,SAAS,GAAG,IAAI,IAAI,CAAC,CAAC,CAAC,CAAC;IAC/B,OAAO,MAAM,CAAC;AAClB,CAAC;AAED,QAAQ,CAAC,4BAA4B,EAAE,GAAG,EAAE;IACxC,IAAI,aAAsC,CAAC;IAG3C,SAAS,CAAC,GAAG,EAAE;QACX,aAAa,GAAG,IAAI,uBAAuB,CAAC,IAAI,gCAAsB,EAAE,CAAC,CAAC;IAC9E,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,iBAAiB,EAAE,KAAK,IAAI,EAAE;QAC/B,MAAM,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,uBAAuB,CAAC,MAAM,EAAE,CAAC,CAAC;IAC3E,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,WAAW,EAAE,KAAK,IAAI,EAAE;QACzB,MAAM,IAAI,GAAG;YACT,EAAE,EAAE,KAAK;YACT,IAAI,EAAE,MAAM;YACZ,GAAG,EAAE,CAAC;SACT,CAAC;QAEF,MAAM,UAAU,GAAG,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QAChD,MAAM,QAAQ,GAAG,uBAAuB,CAAC,QAAQ,CAAC,IAAI,gCAAsB,EAAE,EAAE,IAAI,CAAC,CAAC;QACtF,MAAM,WAAW,GAAG;YAChB,EAAE,EAAE,KAAK;YACT,IAAI,EAAE,MAAM;YACZ,GAAG,EAAE,CAAC;SACT,CAAC;QAEF,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;QAC1C,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;QAC5C,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;IACjD,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,0BAA0B,EAAE,KAAK,IAAI,EAAE;QACxC,MAAM,IAAI,GAAG;YACT,EAAE,EAAE,KAAK;YACT,IAAI,EAAE,MAAM;YACZ,GAAG,EAAE,CAAC;SACT,CAAC;QAEF,MAAM,MAAM,GAAG,+BAA+B,CAAC,aAAa,CAAC,uBAAuB,CAAC,IAAI,CAAC,CAAC,CAAC;QAC5F,MAAM,YAAY,GAAG,+BAA+B,CAAC,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC9F,MAAM,YAAY,GAAG,+BAA+B,CAAC,uBAAuB,CAAC,uBAAuB,CAAC,IAAI,gCAAsB,EAAE,EAAE,IAAI,CAAC,CAAC,CAAC;QAC1I,IAAI,cAAc,GAAG,IAAI,UAAU,EAAE,CAAC;QACtC,cAAc,CAAC,EAAE,GAAG,KAAK,CAAC;QAC1B,cAAc,CAAC,IAAI,GAAG,MAAM,CAAC;QAC7B,cAAc,CAAC,GAAG,GAAG,CAAC,CAAC;QAEvB,cAAc,GAAG,+BAA+B,CAAC,cAAc,CAAC,CAAC;QAEjE,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;QACvC,MAAM,CAAC,YAAY,CAAC,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;QAC7C,MAAM,CAAC,YAAY,CAAC,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;QAC7C,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;QACrC,MAAM,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;QACrC,MAAM,CAAC,YAAY,CAAC,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;QAC7C,MAAM,CAAC,YAAY,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;IAC/C,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,uBAAuB,EAAE,KAAK,IAAI,EAAE;QACrC,MAAM,IAAI,GAAG;YACT,EAAE,EAAE,KAAK;YACT,IAAI,EAAE,MAAM;SACf,CAAC;QAEF,mBAAmB;QACnB,MAAM,CAAC,GAAG,EAAE,CAAC,aAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,EAAE,CAAC;QACrD,mBAAmB;QACnB,MAAM,CAAC,GAAG,EAAE,CAAC,uBAAuB,CAAC,QAAQ,CAAC,IAAI,gCAAsB,EAAE,EAAE,IAAI,CAAC,CAAC,CAAC,OAAO,EAAE,CAAC;IACjG,CAAC,CAAC,CAAC;AACP,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,6BAA6B,EAAE,GAAG,EAAE;IACzC,IAAI,cAAwC,CAAC;IAE7C,SAAS,CAAC,GAAG,EAAE;QACX,cAAc,GAAG,IAAI,wBAAwB,CAAC,IAAI,gCAAsB,EAAE,CAAC,CAAC;IAChF,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,iBAAiB,EAAE,KAAK,IAAI,EAAE;QAC/B,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,wBAAwB,CAAC,MAAM,EAAE,CAAC,CAAC;IAC7E,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,aAAa,EAAE,KAAK,IAAI,EAAE;QAC3B,MAAM,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;QAChC,MAAM,CAAC,EAAE,GAAG,KAAK,CAAC;QAClB,MAAM,CAAC,IAAI,GAAG,MAAM,CAAC;QACrB,MAAM,CAAC,GAAG,GAAG,CAAC,CAAC;QAEf,MAAM,UAAU,GAAG,cAAc,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QACrD,MAAM,QAAQ,GAAG,wBAAwB,CAAC,UAAU,CAAC,IAAI,gCAAsB,EAAE,EAAE,MAAM,CAAC,CAAC;QAC3F,MAAM,WAAW,GAAG;YAChB,EAAE,EAAE,KAAK;YACT,IAAI,EAAE,MAAM;YACZ,GAAG,EAAE,CAAC;SACT,CAAC;QAEF,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;QAC1C,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;QAC5C,MAAM,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;IACjD,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,wBAAwB,EAAE,KAAK,IAAI,EAAE;QACtC,MAAM,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;QAChC,MAAM,CAAC,EAAE,GAAG,KAAK,CAAC;QAClB,MAAM,CAAC,IAAI,GAAG,MAAM,CAAC;QACrB,MAAM,CAAC,GAAG,GAAG,CAAC,CAAC;QAEf,MAAM,IAAI,GAAG,cAAc,CAAC,qBAAqB,CAAC,MAAM,CAAC,CAAC;QAC1D,MAAM,UAAU,GAAG,cAAc,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,CAAC;QAC9D,MAAM,UAAU,GAAG,wBAAwB,CAAC,qBAAqB,CAAC,IAAI,gCAAsB,EAAE,EAAE,MAAM,CAAC,CAAC;QACxG,MAAM,YAAY,GAAG;YACjB,EAAE,EAAE,KAAK;YACT,IAAI,EAAE,MAAM;YACZ,GAAG,EAAE,CAAC;SACT,CAAC;QAEF,MAAM,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;QACnC,MAAM,CAAC,UAAU,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;QACzC,MAAM,CAAC,UAAU,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;QACzC,MAAM,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;QACjC,MAAM,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;QACjC,MAAM,CAAC,UAAU,CAAC,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;QACzC,MAAM,CAAC,UAAU,CAAC,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;IAC3C,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,uBAAuB,EAAE,KAAK,IAAI,EAAE;QACrC,MAAM,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;QAChC,MAAM,CAAC,EAAE,GAAG,KAAK,CAAC;QAClB,MAAM,CAAC,IAAI,GAAG,MAAM,CAAC;QAErB,MAAM,CAAC,GAAG,EAAE,CAAC,cAAc,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,OAAO,EAAE,CAAC;QACnE,MAAM,CAAC,GAAG,EAAE,CAAC,wBAAwB,CAAC,UAAU,CAAC,IAAI,gCAAsB,EAAE,EAAE,MAAM,CAAC,CAAC,MAAM,EAAE,CAAC,CAAC,OAAO,EAAE,CAAC;IAC/G,CAAC,CAAC,CAAC;AACP,CAAC,CAAC,CAAC"}