@friggframework/devtools 2.0.0-next.44 → 2.0.0-next.46

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 (212) hide show
  1. package/infrastructure/ARCHITECTURE.md +487 -0
  2. package/infrastructure/HEALTH.md +468 -0
  3. package/infrastructure/README.md +51 -0
  4. package/infrastructure/__tests__/postgres-config.test.js +914 -0
  5. package/infrastructure/__tests__/template-generation.test.js +687 -0
  6. package/infrastructure/create-frigg-infrastructure.js +1 -1
  7. package/infrastructure/docs/POSTGRES-CONFIGURATION.md +630 -0
  8. package/infrastructure/{DEPLOYMENT-INSTRUCTIONS.md → docs/deployment-instructions.md} +3 -3
  9. package/infrastructure/{IAM-POLICY-TEMPLATES.md → docs/iam-policy-templates.md} +9 -10
  10. package/infrastructure/domains/database/aurora-builder.js +809 -0
  11. package/infrastructure/domains/database/aurora-builder.test.js +950 -0
  12. package/infrastructure/domains/database/aurora-discovery.js +87 -0
  13. package/infrastructure/domains/database/aurora-discovery.test.js +188 -0
  14. package/infrastructure/domains/database/aurora-resolver.js +210 -0
  15. package/infrastructure/domains/database/aurora-resolver.test.js +347 -0
  16. package/infrastructure/domains/database/migration-builder.js +633 -0
  17. package/infrastructure/domains/database/migration-builder.test.js +294 -0
  18. package/infrastructure/domains/database/migration-resolver.js +163 -0
  19. package/infrastructure/domains/database/migration-resolver.test.js +337 -0
  20. package/infrastructure/domains/health/application/ports/IPropertyReconciler.js +164 -0
  21. package/infrastructure/domains/health/application/ports/IResourceDetector.js +129 -0
  22. package/infrastructure/domains/health/application/ports/IResourceImporter.js +142 -0
  23. package/infrastructure/domains/health/application/ports/IStackRepository.js +131 -0
  24. package/infrastructure/domains/health/application/ports/index.js +26 -0
  25. package/infrastructure/domains/health/application/use-cases/__tests__/execute-resource-import-use-case.test.js +679 -0
  26. package/infrastructure/domains/health/application/use-cases/__tests__/mismatch-analyzer-method-name.test.js +167 -0
  27. package/infrastructure/domains/health/application/use-cases/__tests__/repair-via-import-use-case.test.js +1130 -0
  28. package/infrastructure/domains/health/application/use-cases/execute-resource-import-use-case.js +221 -0
  29. package/infrastructure/domains/health/application/use-cases/reconcile-properties-use-case.js +152 -0
  30. package/infrastructure/domains/health/application/use-cases/reconcile-properties-use-case.test.js +343 -0
  31. package/infrastructure/domains/health/application/use-cases/repair-via-import-use-case.js +535 -0
  32. package/infrastructure/domains/health/application/use-cases/repair-via-import-use-case.test.js +376 -0
  33. package/infrastructure/domains/health/application/use-cases/run-health-check-use-case.js +213 -0
  34. package/infrastructure/domains/health/application/use-cases/run-health-check-use-case.test.js +441 -0
  35. package/infrastructure/domains/health/docs/ACME-DEV-DRIFT-ANALYSIS.md +267 -0
  36. package/infrastructure/domains/health/docs/BUILD-VS-DEPLOYED-TEMPLATE-ANALYSIS.md +324 -0
  37. package/infrastructure/domains/health/docs/ORPHAN-DETECTION-ANALYSIS.md +386 -0
  38. package/infrastructure/domains/health/docs/SPEC-CLEANUP-COMMAND.md +1419 -0
  39. package/infrastructure/domains/health/docs/TDD-IMPLEMENTATION-SUMMARY.md +391 -0
  40. package/infrastructure/domains/health/docs/TEMPLATE-COMPARISON-IMPLEMENTATION.md +551 -0
  41. package/infrastructure/domains/health/domain/entities/issue.js +299 -0
  42. package/infrastructure/domains/health/domain/entities/issue.test.js +528 -0
  43. package/infrastructure/domains/health/domain/entities/property-mismatch.js +108 -0
  44. package/infrastructure/domains/health/domain/entities/property-mismatch.test.js +275 -0
  45. package/infrastructure/domains/health/domain/entities/resource.js +159 -0
  46. package/infrastructure/domains/health/domain/entities/resource.test.js +432 -0
  47. package/infrastructure/domains/health/domain/entities/stack-health-report.js +306 -0
  48. package/infrastructure/domains/health/domain/entities/stack-health-report.test.js +601 -0
  49. package/infrastructure/domains/health/domain/services/__tests__/health-score-percentage-based.test.js +380 -0
  50. package/infrastructure/domains/health/domain/services/__tests__/import-progress-monitor.test.js +971 -0
  51. package/infrastructure/domains/health/domain/services/__tests__/import-template-generator.test.js +1150 -0
  52. package/infrastructure/domains/health/domain/services/__tests__/logical-id-mapper.test.js +672 -0
  53. package/infrastructure/domains/health/domain/services/__tests__/template-parser.test.js +496 -0
  54. package/infrastructure/domains/health/domain/services/__tests__/update-progress-monitor.test.js +419 -0
  55. package/infrastructure/domains/health/domain/services/health-score-calculator.js +248 -0
  56. package/infrastructure/domains/health/domain/services/health-score-calculator.test.js +504 -0
  57. package/infrastructure/domains/health/domain/services/import-progress-monitor.js +195 -0
  58. package/infrastructure/domains/health/domain/services/import-template-generator.js +435 -0
  59. package/infrastructure/domains/health/domain/services/logical-id-mapper.js +345 -0
  60. package/infrastructure/domains/health/domain/services/mismatch-analyzer.js +234 -0
  61. package/infrastructure/domains/health/domain/services/mismatch-analyzer.test.js +431 -0
  62. package/infrastructure/domains/health/domain/services/property-mutability-config.js +382 -0
  63. package/infrastructure/domains/health/domain/services/template-parser.js +245 -0
  64. package/infrastructure/domains/health/domain/services/update-progress-monitor.js +192 -0
  65. package/infrastructure/domains/health/domain/value-objects/health-score.js +138 -0
  66. package/infrastructure/domains/health/domain/value-objects/health-score.test.js +267 -0
  67. package/infrastructure/domains/health/domain/value-objects/property-mutability.js +161 -0
  68. package/infrastructure/domains/health/domain/value-objects/property-mutability.test.js +198 -0
  69. package/infrastructure/domains/health/domain/value-objects/resource-state.js +167 -0
  70. package/infrastructure/domains/health/domain/value-objects/resource-state.test.js +196 -0
  71. package/infrastructure/domains/health/domain/value-objects/stack-identifier.js +192 -0
  72. package/infrastructure/domains/health/domain/value-objects/stack-identifier.test.js +262 -0
  73. package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-cfn-tagged.test.js +312 -0
  74. package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-multi-stack.test.js +367 -0
  75. package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-relationship-analysis.test.js +432 -0
  76. package/infrastructure/domains/health/infrastructure/adapters/aws-property-reconciler.js +784 -0
  77. package/infrastructure/domains/health/infrastructure/adapters/aws-property-reconciler.test.js +1133 -0
  78. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-detector.js +565 -0
  79. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-detector.test.js +554 -0
  80. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-importer.js +318 -0
  81. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-importer.test.js +398 -0
  82. package/infrastructure/domains/health/infrastructure/adapters/aws-stack-repository.js +777 -0
  83. package/infrastructure/domains/health/infrastructure/adapters/aws-stack-repository.test.js +580 -0
  84. package/infrastructure/domains/integration/integration-builder.js +397 -0
  85. package/infrastructure/domains/integration/integration-builder.test.js +593 -0
  86. package/infrastructure/domains/integration/integration-resolver.js +170 -0
  87. package/infrastructure/domains/integration/integration-resolver.test.js +369 -0
  88. package/infrastructure/domains/integration/websocket-builder.js +69 -0
  89. package/infrastructure/domains/integration/websocket-builder.test.js +195 -0
  90. package/infrastructure/domains/networking/vpc-builder.js +1829 -0
  91. package/infrastructure/domains/networking/vpc-builder.test.js +1262 -0
  92. package/infrastructure/domains/networking/vpc-discovery.js +177 -0
  93. package/infrastructure/domains/networking/vpc-discovery.test.js +350 -0
  94. package/infrastructure/domains/networking/vpc-resolver.js +324 -0
  95. package/infrastructure/domains/networking/vpc-resolver.test.js +501 -0
  96. package/infrastructure/domains/parameters/ssm-builder.js +79 -0
  97. package/infrastructure/domains/parameters/ssm-builder.test.js +189 -0
  98. package/infrastructure/domains/parameters/ssm-discovery.js +84 -0
  99. package/infrastructure/domains/parameters/ssm-discovery.test.js +210 -0
  100. package/infrastructure/{iam-generator.js → domains/security/iam-generator.js} +2 -2
  101. package/infrastructure/domains/security/kms-builder.js +366 -0
  102. package/infrastructure/domains/security/kms-builder.test.js +374 -0
  103. package/infrastructure/domains/security/kms-discovery.js +80 -0
  104. package/infrastructure/domains/security/kms-discovery.test.js +177 -0
  105. package/infrastructure/domains/security/kms-resolver.js +96 -0
  106. package/infrastructure/domains/security/kms-resolver.test.js +216 -0
  107. package/infrastructure/domains/shared/base-builder.js +112 -0
  108. package/infrastructure/domains/shared/base-resolver.js +186 -0
  109. package/infrastructure/domains/shared/base-resolver.test.js +305 -0
  110. package/infrastructure/domains/shared/builder-orchestrator.js +212 -0
  111. package/infrastructure/domains/shared/builder-orchestrator.test.js +213 -0
  112. package/infrastructure/domains/shared/cloudformation-discovery-v2.js +334 -0
  113. package/infrastructure/domains/shared/cloudformation-discovery.js +375 -0
  114. package/infrastructure/domains/shared/cloudformation-discovery.test.js +590 -0
  115. package/infrastructure/domains/shared/environment-builder.js +119 -0
  116. package/infrastructure/domains/shared/environment-builder.test.js +247 -0
  117. package/infrastructure/domains/shared/providers/aws-provider-adapter.js +544 -0
  118. package/infrastructure/domains/shared/providers/aws-provider-adapter.test.js +377 -0
  119. package/infrastructure/domains/shared/providers/azure-provider-adapter.stub.js +93 -0
  120. package/infrastructure/domains/shared/providers/cloud-provider-adapter.js +136 -0
  121. package/infrastructure/domains/shared/providers/gcp-provider-adapter.stub.js +82 -0
  122. package/infrastructure/domains/shared/providers/provider-factory.js +108 -0
  123. package/infrastructure/domains/shared/providers/provider-factory.test.js +170 -0
  124. package/infrastructure/domains/shared/resource-discovery.js +192 -0
  125. package/infrastructure/domains/shared/resource-discovery.test.js +552 -0
  126. package/infrastructure/domains/shared/types/app-definition.js +205 -0
  127. package/infrastructure/domains/shared/types/discovery-result.js +106 -0
  128. package/infrastructure/domains/shared/types/discovery-result.test.js +258 -0
  129. package/infrastructure/domains/shared/types/index.js +46 -0
  130. package/infrastructure/domains/shared/types/resource-ownership.js +108 -0
  131. package/infrastructure/domains/shared/types/resource-ownership.test.js +101 -0
  132. package/infrastructure/domains/shared/utilities/base-definition-factory.js +380 -0
  133. package/infrastructure/domains/shared/utilities/base-definition-factory.js.bak +338 -0
  134. package/infrastructure/domains/shared/utilities/base-definition-factory.test.js +248 -0
  135. package/infrastructure/domains/shared/utilities/handler-path-resolver.js +134 -0
  136. package/infrastructure/domains/shared/utilities/handler-path-resolver.test.js +268 -0
  137. package/infrastructure/domains/shared/utilities/prisma-layer-manager.js +55 -0
  138. package/infrastructure/domains/shared/utilities/prisma-layer-manager.test.js +138 -0
  139. package/infrastructure/{env-validator.js → domains/shared/validation/env-validator.js} +2 -1
  140. package/infrastructure/domains/shared/validation/env-validator.test.js +173 -0
  141. package/infrastructure/esbuild.config.js +53 -0
  142. package/infrastructure/infrastructure-composer.js +87 -0
  143. package/infrastructure/{serverless-template.test.js → infrastructure-composer.test.js} +115 -24
  144. package/infrastructure/scripts/build-prisma-layer.js +553 -0
  145. package/infrastructure/scripts/build-prisma-layer.test.js +102 -0
  146. package/infrastructure/{build-time-discovery.js → scripts/build-time-discovery.js} +80 -48
  147. package/infrastructure/{build-time-discovery.test.js → scripts/build-time-discovery.test.js} +5 -4
  148. package/layers/prisma/nodejs/package.json +8 -0
  149. package/management-ui/server/utils/cliIntegration.js +1 -1
  150. package/management-ui/server/utils/environment/awsParameterStore.js +29 -18
  151. package/package.json +11 -11
  152. package/frigg-cli/.eslintrc.js +0 -141
  153. package/frigg-cli/__tests__/unit/commands/build.test.js +0 -251
  154. package/frigg-cli/__tests__/unit/commands/db-setup.test.js +0 -548
  155. package/frigg-cli/__tests__/unit/commands/install.test.js +0 -400
  156. package/frigg-cli/__tests__/unit/commands/ui.test.js +0 -346
  157. package/frigg-cli/__tests__/unit/utils/database-validator.test.js +0 -366
  158. package/frigg-cli/__tests__/unit/utils/error-messages.test.js +0 -304
  159. package/frigg-cli/__tests__/unit/utils/prisma-runner.test.js +0 -486
  160. package/frigg-cli/__tests__/utils/mock-factory.js +0 -270
  161. package/frigg-cli/__tests__/utils/prisma-mock.js +0 -194
  162. package/frigg-cli/__tests__/utils/test-fixtures.js +0 -463
  163. package/frigg-cli/__tests__/utils/test-setup.js +0 -287
  164. package/frigg-cli/build-command/index.js +0 -65
  165. package/frigg-cli/db-setup-command/index.js +0 -193
  166. package/frigg-cli/deploy-command/index.js +0 -175
  167. package/frigg-cli/generate-command/__tests__/generate-command.test.js +0 -301
  168. package/frigg-cli/generate-command/azure-generator.js +0 -43
  169. package/frigg-cli/generate-command/gcp-generator.js +0 -47
  170. package/frigg-cli/generate-command/index.js +0 -332
  171. package/frigg-cli/generate-command/terraform-generator.js +0 -555
  172. package/frigg-cli/generate-iam-command.js +0 -118
  173. package/frigg-cli/index.js +0 -75
  174. package/frigg-cli/index.test.js +0 -158
  175. package/frigg-cli/init-command/backend-first-handler.js +0 -756
  176. package/frigg-cli/init-command/index.js +0 -93
  177. package/frigg-cli/init-command/template-handler.js +0 -143
  178. package/frigg-cli/install-command/backend-js.js +0 -33
  179. package/frigg-cli/install-command/commit-changes.js +0 -16
  180. package/frigg-cli/install-command/environment-variables.js +0 -127
  181. package/frigg-cli/install-command/environment-variables.test.js +0 -136
  182. package/frigg-cli/install-command/index.js +0 -54
  183. package/frigg-cli/install-command/install-package.js +0 -13
  184. package/frigg-cli/install-command/integration-file.js +0 -30
  185. package/frigg-cli/install-command/logger.js +0 -12
  186. package/frigg-cli/install-command/template.js +0 -90
  187. package/frigg-cli/install-command/validate-package.js +0 -75
  188. package/frigg-cli/jest.config.js +0 -124
  189. package/frigg-cli/package.json +0 -54
  190. package/frigg-cli/start-command/index.js +0 -149
  191. package/frigg-cli/start-command/start-command.test.js +0 -297
  192. package/frigg-cli/test/init-command.test.js +0 -180
  193. package/frigg-cli/test/npm-registry.test.js +0 -319
  194. package/frigg-cli/ui-command/index.js +0 -154
  195. package/frigg-cli/utils/app-resolver.js +0 -319
  196. package/frigg-cli/utils/backend-path.js +0 -25
  197. package/frigg-cli/utils/database-validator.js +0 -161
  198. package/frigg-cli/utils/error-messages.js +0 -257
  199. package/frigg-cli/utils/npm-registry.js +0 -167
  200. package/frigg-cli/utils/prisma-runner.js +0 -280
  201. package/frigg-cli/utils/process-manager.js +0 -199
  202. package/frigg-cli/utils/repo-detection.js +0 -405
  203. package/infrastructure/aws-discovery.js +0 -1176
  204. package/infrastructure/aws-discovery.test.js +0 -1220
  205. package/infrastructure/serverless-template.js +0 -2074
  206. /package/infrastructure/{WEBSOCKET-CONFIGURATION.md → docs/WEBSOCKET-CONFIGURATION.md} +0 -0
  207. /package/infrastructure/{GENERATE-IAM-DOCS.md → docs/generate-iam-command.md} +0 -0
  208. /package/infrastructure/{iam-generator.test.js → domains/security/iam-generator.test.js} +0 -0
  209. /package/infrastructure/{frigg-deployment-iam-stack.yaml → domains/security/templates/frigg-deployment-iam-stack.yaml} +0 -0
  210. /package/infrastructure/{iam-policy-basic.json → domains/security/templates/iam-policy-basic.json} +0 -0
  211. /package/infrastructure/{iam-policy-full.json → domains/security/templates/iam-policy-full.json} +0 -0
  212. /package/infrastructure/{run-discovery.js → scripts/run-discovery.js} +0 -0
@@ -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
+ });