@friggframework/devtools 2.0.0-next.62 → 2.0.0-next.63

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 (165) hide show
  1. package/infrastructure/ARCHITECTURE.md +487 -0
  2. package/infrastructure/CLAUDE.md +481 -0
  3. package/infrastructure/HEALTH.md +468 -0
  4. package/infrastructure/README.md +522 -0
  5. package/infrastructure/__tests__/fixtures/mock-aws-resources.js +391 -0
  6. package/infrastructure/__tests__/helpers/test-utils.js +277 -0
  7. package/infrastructure/__tests__/postgres-config.test.js +914 -0
  8. package/infrastructure/__tests__/template-generation.test.js +687 -0
  9. package/infrastructure/create-frigg-infrastructure.js +147 -0
  10. package/infrastructure/docs/POSTGRES-CONFIGURATION.md +630 -0
  11. package/infrastructure/docs/PRE-DEPLOYMENT-HEALTH-CHECK-SPEC.md +1317 -0
  12. package/infrastructure/docs/WEBSOCKET-CONFIGURATION.md +105 -0
  13. package/infrastructure/docs/deployment-instructions.md +268 -0
  14. package/infrastructure/docs/generate-iam-command.md +278 -0
  15. package/infrastructure/docs/iam-policy-templates.md +193 -0
  16. package/infrastructure/domains/database/aurora-builder.js +809 -0
  17. package/infrastructure/domains/database/aurora-builder.test.js +950 -0
  18. package/infrastructure/domains/database/aurora-discovery.js +87 -0
  19. package/infrastructure/domains/database/aurora-discovery.test.js +188 -0
  20. package/infrastructure/domains/database/aurora-resolver.js +210 -0
  21. package/infrastructure/domains/database/aurora-resolver.test.js +347 -0
  22. package/infrastructure/domains/database/migration-builder.js +701 -0
  23. package/infrastructure/domains/database/migration-builder.test.js +321 -0
  24. package/infrastructure/domains/database/migration-resolver.js +163 -0
  25. package/infrastructure/domains/database/migration-resolver.test.js +337 -0
  26. package/infrastructure/domains/health/application/ports/IPropertyReconciler.js +164 -0
  27. package/infrastructure/domains/health/application/ports/IResourceDetector.js +129 -0
  28. package/infrastructure/domains/health/application/ports/IResourceImporter.js +142 -0
  29. package/infrastructure/domains/health/application/ports/IStackRepository.js +131 -0
  30. package/infrastructure/domains/health/application/ports/index.js +26 -0
  31. package/infrastructure/domains/health/application/use-cases/__tests__/execute-resource-import-use-case.test.js +679 -0
  32. package/infrastructure/domains/health/application/use-cases/__tests__/mismatch-analyzer-method-name.test.js +167 -0
  33. package/infrastructure/domains/health/application/use-cases/__tests__/repair-via-import-use-case.test.js +1130 -0
  34. package/infrastructure/domains/health/application/use-cases/execute-resource-import-use-case.js +221 -0
  35. package/infrastructure/domains/health/application/use-cases/reconcile-properties-use-case.js +152 -0
  36. package/infrastructure/domains/health/application/use-cases/reconcile-properties-use-case.test.js +343 -0
  37. package/infrastructure/domains/health/application/use-cases/repair-via-import-use-case.js +535 -0
  38. package/infrastructure/domains/health/application/use-cases/repair-via-import-use-case.test.js +376 -0
  39. package/infrastructure/domains/health/application/use-cases/run-health-check-use-case.js +213 -0
  40. package/infrastructure/domains/health/application/use-cases/run-health-check-use-case.test.js +441 -0
  41. package/infrastructure/domains/health/docs/ACME-DEV-DRIFT-ANALYSIS.md +267 -0
  42. package/infrastructure/domains/health/docs/BUILD-VS-DEPLOYED-TEMPLATE-ANALYSIS.md +324 -0
  43. package/infrastructure/domains/health/docs/ORPHAN-DETECTION-ANALYSIS.md +386 -0
  44. package/infrastructure/domains/health/docs/SPEC-CLEANUP-COMMAND.md +1419 -0
  45. package/infrastructure/domains/health/docs/TDD-IMPLEMENTATION-SUMMARY.md +391 -0
  46. package/infrastructure/domains/health/docs/TEMPLATE-COMPARISON-IMPLEMENTATION.md +551 -0
  47. package/infrastructure/domains/health/domain/entities/issue.js +299 -0
  48. package/infrastructure/domains/health/domain/entities/issue.test.js +528 -0
  49. package/infrastructure/domains/health/domain/entities/property-mismatch.js +108 -0
  50. package/infrastructure/domains/health/domain/entities/property-mismatch.test.js +275 -0
  51. package/infrastructure/domains/health/domain/entities/resource.js +159 -0
  52. package/infrastructure/domains/health/domain/entities/resource.test.js +432 -0
  53. package/infrastructure/domains/health/domain/entities/stack-health-report.js +306 -0
  54. package/infrastructure/domains/health/domain/entities/stack-health-report.test.js +601 -0
  55. package/infrastructure/domains/health/domain/services/__tests__/health-score-percentage-based.test.js +380 -0
  56. package/infrastructure/domains/health/domain/services/__tests__/import-progress-monitor.test.js +971 -0
  57. package/infrastructure/domains/health/domain/services/__tests__/import-template-generator.test.js +1150 -0
  58. package/infrastructure/domains/health/domain/services/__tests__/logical-id-mapper.test.js +672 -0
  59. package/infrastructure/domains/health/domain/services/__tests__/template-parser.test.js +496 -0
  60. package/infrastructure/domains/health/domain/services/__tests__/update-progress-monitor.test.js +419 -0
  61. package/infrastructure/domains/health/domain/services/health-score-calculator.js +248 -0
  62. package/infrastructure/domains/health/domain/services/health-score-calculator.test.js +504 -0
  63. package/infrastructure/domains/health/domain/services/import-progress-monitor.js +195 -0
  64. package/infrastructure/domains/health/domain/services/import-template-generator.js +435 -0
  65. package/infrastructure/domains/health/domain/services/logical-id-mapper.js +345 -0
  66. package/infrastructure/domains/health/domain/services/mismatch-analyzer.js +234 -0
  67. package/infrastructure/domains/health/domain/services/mismatch-analyzer.test.js +431 -0
  68. package/infrastructure/domains/health/domain/services/property-mutability-config.js +382 -0
  69. package/infrastructure/domains/health/domain/services/template-parser.js +245 -0
  70. package/infrastructure/domains/health/domain/services/update-progress-monitor.js +192 -0
  71. package/infrastructure/domains/health/domain/value-objects/health-score.js +138 -0
  72. package/infrastructure/domains/health/domain/value-objects/health-score.test.js +267 -0
  73. package/infrastructure/domains/health/domain/value-objects/property-mutability.js +161 -0
  74. package/infrastructure/domains/health/domain/value-objects/property-mutability.test.js +198 -0
  75. package/infrastructure/domains/health/domain/value-objects/resource-state.js +167 -0
  76. package/infrastructure/domains/health/domain/value-objects/resource-state.test.js +196 -0
  77. package/infrastructure/domains/health/domain/value-objects/stack-identifier.js +192 -0
  78. package/infrastructure/domains/health/domain/value-objects/stack-identifier.test.js +262 -0
  79. package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-cfn-tagged.test.js +312 -0
  80. package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-multi-stack.test.js +367 -0
  81. package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-relationship-analysis.test.js +432 -0
  82. package/infrastructure/domains/health/infrastructure/adapters/aws-property-reconciler.js +784 -0
  83. package/infrastructure/domains/health/infrastructure/adapters/aws-property-reconciler.test.js +1133 -0
  84. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-detector.js +565 -0
  85. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-detector.test.js +554 -0
  86. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-importer.js +318 -0
  87. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-importer.test.js +398 -0
  88. package/infrastructure/domains/health/infrastructure/adapters/aws-stack-repository.js +777 -0
  89. package/infrastructure/domains/health/infrastructure/adapters/aws-stack-repository.test.js +580 -0
  90. package/infrastructure/domains/integration/integration-builder.js +404 -0
  91. package/infrastructure/domains/integration/integration-builder.test.js +690 -0
  92. package/infrastructure/domains/integration/integration-resolver.js +170 -0
  93. package/infrastructure/domains/integration/integration-resolver.test.js +369 -0
  94. package/infrastructure/domains/integration/websocket-builder.js +69 -0
  95. package/infrastructure/domains/integration/websocket-builder.test.js +195 -0
  96. package/infrastructure/domains/networking/vpc-builder.js +2051 -0
  97. package/infrastructure/domains/networking/vpc-builder.test.js +1960 -0
  98. package/infrastructure/domains/networking/vpc-discovery.js +177 -0
  99. package/infrastructure/domains/networking/vpc-discovery.test.js +350 -0
  100. package/infrastructure/domains/networking/vpc-resolver.js +505 -0
  101. package/infrastructure/domains/networking/vpc-resolver.test.js +801 -0
  102. package/infrastructure/domains/parameters/ssm-builder.js +79 -0
  103. package/infrastructure/domains/parameters/ssm-builder.test.js +189 -0
  104. package/infrastructure/domains/parameters/ssm-discovery.js +84 -0
  105. package/infrastructure/domains/parameters/ssm-discovery.test.js +210 -0
  106. package/infrastructure/domains/security/iam-generator.js +816 -0
  107. package/infrastructure/domains/security/iam-generator.test.js +204 -0
  108. package/infrastructure/domains/security/kms-builder.js +415 -0
  109. package/infrastructure/domains/security/kms-builder.test.js +392 -0
  110. package/infrastructure/domains/security/kms-discovery.js +80 -0
  111. package/infrastructure/domains/security/kms-discovery.test.js +177 -0
  112. package/infrastructure/domains/security/kms-resolver.js +96 -0
  113. package/infrastructure/domains/security/kms-resolver.test.js +216 -0
  114. package/infrastructure/domains/security/templates/frigg-deployment-iam-stack.yaml +401 -0
  115. package/infrastructure/domains/security/templates/iam-policy-basic.json +218 -0
  116. package/infrastructure/domains/security/templates/iam-policy-full.json +288 -0
  117. package/infrastructure/domains/shared/base-builder.js +112 -0
  118. package/infrastructure/domains/shared/base-resolver.js +186 -0
  119. package/infrastructure/domains/shared/base-resolver.test.js +305 -0
  120. package/infrastructure/domains/shared/builder-orchestrator.js +212 -0
  121. package/infrastructure/domains/shared/builder-orchestrator.test.js +213 -0
  122. package/infrastructure/domains/shared/cloudformation-discovery-v2.js +334 -0
  123. package/infrastructure/domains/shared/cloudformation-discovery.js +672 -0
  124. package/infrastructure/domains/shared/cloudformation-discovery.test.js +985 -0
  125. package/infrastructure/domains/shared/environment-builder.js +119 -0
  126. package/infrastructure/domains/shared/environment-builder.test.js +247 -0
  127. package/infrastructure/domains/shared/providers/aws-provider-adapter.js +579 -0
  128. package/infrastructure/domains/shared/providers/aws-provider-adapter.test.js +416 -0
  129. package/infrastructure/domains/shared/providers/azure-provider-adapter.stub.js +93 -0
  130. package/infrastructure/domains/shared/providers/cloud-provider-adapter.js +136 -0
  131. package/infrastructure/domains/shared/providers/gcp-provider-adapter.stub.js +82 -0
  132. package/infrastructure/domains/shared/providers/provider-factory.js +108 -0
  133. package/infrastructure/domains/shared/providers/provider-factory.test.js +170 -0
  134. package/infrastructure/domains/shared/resource-discovery.enhanced.test.js +306 -0
  135. package/infrastructure/domains/shared/resource-discovery.js +233 -0
  136. package/infrastructure/domains/shared/resource-discovery.test.js +588 -0
  137. package/infrastructure/domains/shared/types/app-definition.js +205 -0
  138. package/infrastructure/domains/shared/types/discovery-result.js +106 -0
  139. package/infrastructure/domains/shared/types/discovery-result.test.js +258 -0
  140. package/infrastructure/domains/shared/types/index.js +46 -0
  141. package/infrastructure/domains/shared/types/resource-ownership.js +108 -0
  142. package/infrastructure/domains/shared/types/resource-ownership.test.js +101 -0
  143. package/infrastructure/domains/shared/utilities/base-definition-factory.js +408 -0
  144. package/infrastructure/domains/shared/utilities/base-definition-factory.js.bak +338 -0
  145. package/infrastructure/domains/shared/utilities/base-definition-factory.test.js +291 -0
  146. package/infrastructure/domains/shared/utilities/handler-path-resolver.js +134 -0
  147. package/infrastructure/domains/shared/utilities/handler-path-resolver.test.js +268 -0
  148. package/infrastructure/domains/shared/utilities/prisma-layer-manager.js +159 -0
  149. package/infrastructure/domains/shared/utilities/prisma-layer-manager.test.js +444 -0
  150. package/infrastructure/domains/shared/validation/env-validator.js +78 -0
  151. package/infrastructure/domains/shared/validation/env-validator.test.js +173 -0
  152. package/infrastructure/domains/shared/validation/plugin-validator.js +187 -0
  153. package/infrastructure/domains/shared/validation/plugin-validator.test.js +323 -0
  154. package/infrastructure/esbuild.config.js +53 -0
  155. package/infrastructure/index.js +4 -0
  156. package/infrastructure/infrastructure-composer.js +117 -0
  157. package/infrastructure/infrastructure-composer.test.js +1895 -0
  158. package/infrastructure/integration.test.js +383 -0
  159. package/infrastructure/scripts/build-prisma-layer.js +701 -0
  160. package/infrastructure/scripts/build-prisma-layer.test.js +170 -0
  161. package/infrastructure/scripts/build-time-discovery.js +238 -0
  162. package/infrastructure/scripts/build-time-discovery.test.js +379 -0
  163. package/infrastructure/scripts/run-discovery.js +110 -0
  164. package/infrastructure/scripts/verify-prisma-layer.js +72 -0
  165. package/package.json +8 -7
@@ -0,0 +1,432 @@
1
+ /**
2
+ * Tests for Resource Entity
3
+ */
4
+
5
+ const Resource = require('./resource');
6
+ const ResourceState = require('../value-objects/resource-state');
7
+ const Issue = require('./issue');
8
+ const PropertyMismatch = require('./property-mismatch');
9
+ const PropertyMutability = require('../value-objects/property-mutability');
10
+
11
+ describe('Resource', () => {
12
+ describe('constructor', () => {
13
+ it('should create a resource in stack', () => {
14
+ const resource = new Resource({
15
+ logicalId: 'ProductionVPC',
16
+ physicalId: 'vpc-0abc123def456',
17
+ resourceType: 'AWS::EC2::VPC',
18
+ state: ResourceState.IN_STACK,
19
+ properties: {
20
+ CidrBlock: '10.0.0.0/16',
21
+ Tags: [{ Key: 'Environment', Value: 'production' }],
22
+ },
23
+ });
24
+
25
+ expect(resource.logicalId).toBe('ProductionVPC');
26
+ expect(resource.physicalId).toBe('vpc-0abc123def456');
27
+ expect(resource.resourceType).toBe('AWS::EC2::VPC');
28
+ expect(resource.state.value).toBe('IN_STACK');
29
+ expect(resource.properties.CidrBlock).toBe('10.0.0.0/16');
30
+ });
31
+
32
+ it('should create an orphaned resource', () => {
33
+ const resource = new Resource({
34
+ logicalId: null,
35
+ physicalId: 'my-app-prod-aurora',
36
+ resourceType: 'AWS::RDS::DBCluster',
37
+ state: ResourceState.ORPHANED,
38
+ properties: {
39
+ Engine: 'aurora-postgresql',
40
+ EngineVersion: '13.7',
41
+ },
42
+ });
43
+
44
+ expect(resource.logicalId).toBeNull();
45
+ expect(resource.physicalId).toBe('my-app-prod-aurora');
46
+ expect(resource.state.value).toBe('ORPHANED');
47
+ });
48
+
49
+ it('should require physicalId', () => {
50
+ expect(() => {
51
+ new Resource({
52
+ logicalId: 'MyResource',
53
+ resourceType: 'AWS::S3::Bucket',
54
+ state: ResourceState.IN_STACK,
55
+ });
56
+ }).toThrow('physicalId is required');
57
+ });
58
+
59
+ it('should require resourceType', () => {
60
+ expect(() => {
61
+ new Resource({
62
+ logicalId: 'MyResource',
63
+ physicalId: 'my-bucket',
64
+ state: ResourceState.IN_STACK,
65
+ });
66
+ }).toThrow('resourceType is required');
67
+ });
68
+
69
+ it('should require state', () => {
70
+ expect(() => {
71
+ new Resource({
72
+ logicalId: 'MyResource',
73
+ physicalId: 'my-bucket',
74
+ resourceType: 'AWS::S3::Bucket',
75
+ });
76
+ }).toThrow('state is required');
77
+ });
78
+
79
+ it('should validate state is ResourceState instance', () => {
80
+ expect(() => {
81
+ new Resource({
82
+ logicalId: 'MyResource',
83
+ physicalId: 'my-bucket',
84
+ resourceType: 'AWS::S3::Bucket',
85
+ state: 'IN_STACK', // String instead of ResourceState
86
+ });
87
+ }).toThrow('state must be a ResourceState instance');
88
+ });
89
+
90
+ it('should initialize empty issues array', () => {
91
+ const resource = new Resource({
92
+ logicalId: 'MyVPC',
93
+ physicalId: 'vpc-123',
94
+ resourceType: 'AWS::EC2::VPC',
95
+ state: ResourceState.IN_STACK,
96
+ });
97
+
98
+ expect(resource.issues).toEqual([]);
99
+ });
100
+
101
+ it('should initialize with provided issues', () => {
102
+ const issue = Issue.orphanedResource({
103
+ resourceType: 'AWS::RDS::DBCluster',
104
+ resourceId: 'my-cluster',
105
+ description: 'Test',
106
+ });
107
+
108
+ const resource = new Resource({
109
+ logicalId: null,
110
+ physicalId: 'my-cluster',
111
+ resourceType: 'AWS::RDS::DBCluster',
112
+ state: ResourceState.ORPHANED,
113
+ issues: [issue],
114
+ });
115
+
116
+ expect(resource.issues).toHaveLength(1);
117
+ expect(resource.issues[0]).toBe(issue);
118
+ });
119
+
120
+ it('should default properties to empty object', () => {
121
+ const resource = new Resource({
122
+ logicalId: 'MyVPC',
123
+ physicalId: 'vpc-123',
124
+ resourceType: 'AWS::EC2::VPC',
125
+ state: ResourceState.IN_STACK,
126
+ });
127
+
128
+ expect(resource.properties).toEqual({});
129
+ });
130
+ });
131
+
132
+ describe('state checks', () => {
133
+ it('should check if resource is in stack', () => {
134
+ const resource = new Resource({
135
+ logicalId: 'MyVPC',
136
+ physicalId: 'vpc-123',
137
+ resourceType: 'AWS::EC2::VPC',
138
+ state: ResourceState.IN_STACK,
139
+ });
140
+
141
+ expect(resource.isInStack()).toBe(true);
142
+ expect(resource.isOrphaned()).toBe(false);
143
+ expect(resource.isMissing()).toBe(false);
144
+ expect(resource.isDrifted()).toBe(false);
145
+ });
146
+
147
+ it('should check if resource is orphaned', () => {
148
+ const resource = new Resource({
149
+ logicalId: null,
150
+ physicalId: 'my-cluster',
151
+ resourceType: 'AWS::RDS::DBCluster',
152
+ state: ResourceState.ORPHANED,
153
+ });
154
+
155
+ expect(resource.isInStack()).toBe(false);
156
+ expect(resource.isOrphaned()).toBe(true);
157
+ expect(resource.isMissing()).toBe(false);
158
+ expect(resource.isDrifted()).toBe(false);
159
+ });
160
+
161
+ it('should check if resource is missing', () => {
162
+ const resource = new Resource({
163
+ logicalId: 'FriggKMSKey',
164
+ physicalId: 'key-id-that-does-not-exist',
165
+ resourceType: 'AWS::KMS::Key',
166
+ state: ResourceState.MISSING,
167
+ });
168
+
169
+ expect(resource.isInStack()).toBe(false);
170
+ expect(resource.isOrphaned()).toBe(false);
171
+ expect(resource.isMissing()).toBe(true);
172
+ expect(resource.isDrifted()).toBe(false);
173
+ });
174
+
175
+ it('should check if resource is drifted', () => {
176
+ const resource = new Resource({
177
+ logicalId: 'MyVPC',
178
+ physicalId: 'vpc-123',
179
+ resourceType: 'AWS::EC2::VPC',
180
+ state: ResourceState.DRIFTED,
181
+ });
182
+
183
+ expect(resource.isInStack()).toBe(false);
184
+ expect(resource.isOrphaned()).toBe(false);
185
+ expect(resource.isMissing()).toBe(false);
186
+ expect(resource.isDrifted()).toBe(true);
187
+ });
188
+ });
189
+
190
+ describe('issue management', () => {
191
+ it('should add an issue', () => {
192
+ const resource = new Resource({
193
+ logicalId: 'MyVPC',
194
+ physicalId: 'vpc-123',
195
+ resourceType: 'AWS::EC2::VPC',
196
+ state: ResourceState.IN_STACK,
197
+ });
198
+
199
+ const issue = Issue.propertyMismatch({
200
+ resourceType: 'AWS::EC2::VPC',
201
+ resourceId: 'vpc-123',
202
+ mismatch: new PropertyMismatch({
203
+ propertyPath: 'Properties.Tags',
204
+ expectedValue: ['tag1'],
205
+ actualValue: ['tag2'],
206
+ mutability: PropertyMutability.MUTABLE,
207
+ }),
208
+ });
209
+
210
+ resource.addIssue(issue);
211
+
212
+ expect(resource.issues).toHaveLength(1);
213
+ expect(resource.issues[0]).toBe(issue);
214
+ });
215
+
216
+ it('should check if resource has issues', () => {
217
+ const resource = new Resource({
218
+ logicalId: 'MyVPC',
219
+ physicalId: 'vpc-123',
220
+ resourceType: 'AWS::EC2::VPC',
221
+ state: ResourceState.IN_STACK,
222
+ });
223
+
224
+ expect(resource.hasIssues()).toBe(false);
225
+
226
+ const issue = Issue.orphanedResource({
227
+ resourceType: 'AWS::EC2::VPC',
228
+ resourceId: 'vpc-123',
229
+ description: 'Test',
230
+ });
231
+ resource.addIssue(issue);
232
+
233
+ expect(resource.hasIssues()).toBe(true);
234
+ });
235
+
236
+ it('should check if resource has critical issues', () => {
237
+ const resource = new Resource({
238
+ logicalId: 'MyVPC',
239
+ physicalId: 'vpc-123',
240
+ resourceType: 'AWS::EC2::VPC',
241
+ state: ResourceState.IN_STACK,
242
+ });
243
+
244
+ expect(resource.hasCriticalIssues()).toBe(false);
245
+
246
+ // Add warning issue
247
+ const warningIssue = new Issue({
248
+ type: 'PROPERTY_MISMATCH',
249
+ severity: 'warning',
250
+ resourceType: 'AWS::EC2::VPC',
251
+ resourceId: 'vpc-123',
252
+ description: 'Test warning',
253
+ });
254
+ resource.addIssue(warningIssue);
255
+ expect(resource.hasCriticalIssues()).toBe(false);
256
+
257
+ // Add critical issue
258
+ const criticalIssue = Issue.orphanedResource({
259
+ resourceType: 'AWS::EC2::VPC',
260
+ resourceId: 'vpc-123',
261
+ description: 'Test critical',
262
+ });
263
+ resource.addIssue(criticalIssue);
264
+ expect(resource.hasCriticalIssues()).toBe(true);
265
+ });
266
+
267
+ it('should get critical issues', () => {
268
+ const resource = new Resource({
269
+ logicalId: 'MyVPC',
270
+ physicalId: 'vpc-123',
271
+ resourceType: 'AWS::EC2::VPC',
272
+ state: ResourceState.IN_STACK,
273
+ });
274
+
275
+ const warningIssue = new Issue({
276
+ type: 'PROPERTY_MISMATCH',
277
+ severity: 'warning',
278
+ resourceType: 'AWS::EC2::VPC',
279
+ resourceId: 'vpc-123',
280
+ description: 'Warning',
281
+ });
282
+
283
+ const criticalIssue1 = Issue.orphanedResource({
284
+ resourceType: 'AWS::EC2::VPC',
285
+ resourceId: 'vpc-123',
286
+ description: 'Critical 1',
287
+ });
288
+
289
+ const criticalIssue2 = Issue.missingResource({
290
+ resourceType: 'AWS::EC2::VPC',
291
+ resourceId: 'vpc-123',
292
+ description: 'Critical 2',
293
+ });
294
+
295
+ resource.addIssue(warningIssue);
296
+ resource.addIssue(criticalIssue1);
297
+ resource.addIssue(criticalIssue2);
298
+
299
+ const criticalIssues = resource.getCriticalIssues();
300
+ expect(criticalIssues).toHaveLength(2);
301
+ expect(criticalIssues).toContain(criticalIssue1);
302
+ expect(criticalIssues).toContain(criticalIssue2);
303
+ });
304
+ });
305
+
306
+ describe('isHealthy', () => {
307
+ it('should be healthy with no issues', () => {
308
+ const resource = new Resource({
309
+ logicalId: 'MyVPC',
310
+ physicalId: 'vpc-123',
311
+ resourceType: 'AWS::EC2::VPC',
312
+ state: ResourceState.IN_STACK,
313
+ });
314
+
315
+ expect(resource.isHealthy()).toBe(true);
316
+ });
317
+
318
+ it('should not be healthy with issues', () => {
319
+ const resource = new Resource({
320
+ logicalId: 'MyVPC',
321
+ physicalId: 'vpc-123',
322
+ resourceType: 'AWS::EC2::VPC',
323
+ state: ResourceState.IN_STACK,
324
+ });
325
+
326
+ const issue = Issue.propertyMismatch({
327
+ resourceType: 'AWS::EC2::VPC',
328
+ resourceId: 'vpc-123',
329
+ mismatch: new PropertyMismatch({
330
+ propertyPath: 'Properties.Tags',
331
+ expectedValue: ['tag1'],
332
+ actualValue: ['tag2'],
333
+ mutability: PropertyMutability.MUTABLE,
334
+ }),
335
+ });
336
+
337
+ resource.addIssue(issue);
338
+ expect(resource.isHealthy()).toBe(false);
339
+ });
340
+ });
341
+
342
+ describe('getIdentifier', () => {
343
+ it('should return logical ID if present', () => {
344
+ const resource = new Resource({
345
+ logicalId: 'ProductionVPC',
346
+ physicalId: 'vpc-123',
347
+ resourceType: 'AWS::EC2::VPC',
348
+ state: ResourceState.IN_STACK,
349
+ });
350
+
351
+ expect(resource.getIdentifier()).toBe('ProductionVPC');
352
+ });
353
+
354
+ it('should return physical ID if no logical ID', () => {
355
+ const resource = new Resource({
356
+ logicalId: null,
357
+ physicalId: 'vpc-123',
358
+ resourceType: 'AWS::EC2::VPC',
359
+ state: ResourceState.ORPHANED,
360
+ });
361
+
362
+ expect(resource.getIdentifier()).toBe('vpc-123');
363
+ });
364
+ });
365
+
366
+ describe('toString', () => {
367
+ it('should return string representation', () => {
368
+ const resource = new Resource({
369
+ logicalId: 'ProductionVPC',
370
+ physicalId: 'vpc-123',
371
+ resourceType: 'AWS::EC2::VPC',
372
+ state: ResourceState.IN_STACK,
373
+ });
374
+
375
+ const str = resource.toString();
376
+ expect(str).toContain('AWS::EC2::VPC');
377
+ expect(str).toContain('ProductionVPC');
378
+ expect(str).toContain('vpc-123');
379
+ expect(str).toContain('IN_STACK');
380
+ });
381
+ });
382
+
383
+ describe('toJSON', () => {
384
+ it('should serialize to JSON', () => {
385
+ const resource = new Resource({
386
+ logicalId: 'ProductionVPC',
387
+ physicalId: 'vpc-123',
388
+ resourceType: 'AWS::EC2::VPC',
389
+ state: ResourceState.IN_STACK,
390
+ properties: {
391
+ CidrBlock: '10.0.0.0/16',
392
+ },
393
+ });
394
+
395
+ const json = resource.toJSON();
396
+
397
+ expect(json).toEqual({
398
+ logicalId: 'ProductionVPC',
399
+ physicalId: 'vpc-123',
400
+ resourceType: 'AWS::EC2::VPC',
401
+ state: 'IN_STACK',
402
+ properties: {
403
+ CidrBlock: '10.0.0.0/16',
404
+ },
405
+ issues: [],
406
+ isHealthy: true,
407
+ });
408
+ });
409
+
410
+ it('should include issues in JSON', () => {
411
+ const resource = new Resource({
412
+ logicalId: 'MyVPC',
413
+ physicalId: 'vpc-123',
414
+ resourceType: 'AWS::EC2::VPC',
415
+ state: ResourceState.DRIFTED,
416
+ });
417
+
418
+ const issue = Issue.orphanedResource({
419
+ resourceType: 'AWS::EC2::VPC',
420
+ resourceId: 'vpc-123',
421
+ description: 'Test',
422
+ });
423
+ resource.addIssue(issue);
424
+
425
+ const json = resource.toJSON();
426
+
427
+ expect(json.issues).toHaveLength(1);
428
+ expect(json.issues[0].type).toBe('ORPHANED_RESOURCE');
429
+ expect(json.isHealthy).toBe(false);
430
+ });
431
+ });
432
+ });