@friggframework/devtools 2.0.0-next.7 → 2.0.0-next.71

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 (240) hide show
  1. package/frigg-cli/README.md +1289 -0
  2. package/frigg-cli/__tests__/unit/commands/build.test.js +279 -0
  3. package/frigg-cli/__tests__/unit/commands/db-setup.test.js +649 -0
  4. package/frigg-cli/__tests__/unit/commands/deploy.test.js +320 -0
  5. package/frigg-cli/__tests__/unit/commands/doctor.test.js +309 -0
  6. package/frigg-cli/__tests__/unit/commands/install.test.js +400 -0
  7. package/frigg-cli/__tests__/unit/commands/ui.test.js +346 -0
  8. package/frigg-cli/__tests__/unit/dependencies.test.js +74 -0
  9. package/frigg-cli/__tests__/unit/utils/database-validator.test.js +397 -0
  10. package/frigg-cli/__tests__/unit/utils/error-messages.test.js +345 -0
  11. package/frigg-cli/__tests__/unit/version-detection.test.js +171 -0
  12. package/frigg-cli/__tests__/utils/mock-factory.js +270 -0
  13. package/frigg-cli/__tests__/utils/prisma-mock.js +194 -0
  14. package/frigg-cli/__tests__/utils/test-fixtures.js +463 -0
  15. package/frigg-cli/__tests__/utils/test-setup.js +287 -0
  16. package/frigg-cli/auth-command/CLAUDE.md +293 -0
  17. package/frigg-cli/auth-command/README.md +450 -0
  18. package/frigg-cli/auth-command/api-key-flow.js +153 -0
  19. package/frigg-cli/auth-command/auth-tester.js +344 -0
  20. package/frigg-cli/auth-command/credential-storage.js +182 -0
  21. package/frigg-cli/auth-command/index.js +256 -0
  22. package/frigg-cli/auth-command/json-schema-form.js +67 -0
  23. package/frigg-cli/auth-command/module-loader.js +172 -0
  24. package/frigg-cli/auth-command/oauth-callback-server.js +431 -0
  25. package/frigg-cli/auth-command/oauth-flow.js +195 -0
  26. package/frigg-cli/auth-command/utils/browser.js +30 -0
  27. package/frigg-cli/build-command/index.js +45 -12
  28. package/frigg-cli/db-setup-command/index.js +246 -0
  29. package/frigg-cli/deploy-command/SPEC-DEPLOY-DRY-RUN.md +981 -0
  30. package/frigg-cli/deploy-command/index.js +295 -23
  31. package/frigg-cli/doctor-command/index.js +335 -0
  32. package/frigg-cli/generate-command/__tests__/generate-command.test.js +301 -0
  33. package/frigg-cli/generate-command/azure-generator.js +43 -0
  34. package/frigg-cli/generate-command/gcp-generator.js +47 -0
  35. package/frigg-cli/generate-command/index.js +332 -0
  36. package/frigg-cli/generate-command/terraform-generator.js +555 -0
  37. package/frigg-cli/generate-iam-command.js +118 -0
  38. package/frigg-cli/index.js +174 -1
  39. package/frigg-cli/index.test.js +1 -4
  40. package/frigg-cli/init-command/backend-first-handler.js +756 -0
  41. package/frigg-cli/init-command/index.js +93 -0
  42. package/frigg-cli/init-command/template-handler.js +143 -0
  43. package/frigg-cli/install-command/index.js +1 -4
  44. package/frigg-cli/jest.config.js +124 -0
  45. package/frigg-cli/package.json +63 -0
  46. package/frigg-cli/repair-command/index.js +564 -0
  47. package/frigg-cli/start-command/index.js +118 -5
  48. package/frigg-cli/start-command/start-command.test.js +297 -0
  49. package/frigg-cli/test/init-command.test.js +180 -0
  50. package/frigg-cli/test/npm-registry.test.js +319 -0
  51. package/frigg-cli/ui-command/index.js +154 -0
  52. package/frigg-cli/utils/app-resolver.js +319 -0
  53. package/frigg-cli/utils/backend-path.js +16 -17
  54. package/frigg-cli/utils/database-validator.js +167 -0
  55. package/frigg-cli/utils/error-messages.js +329 -0
  56. package/frigg-cli/utils/npm-registry.js +167 -0
  57. package/frigg-cli/utils/process-manager.js +199 -0
  58. package/frigg-cli/utils/repo-detection.js +405 -0
  59. package/infrastructure/ARCHITECTURE.md +487 -0
  60. package/infrastructure/CLAUDE.md +481 -0
  61. package/infrastructure/HEALTH.md +468 -0
  62. package/infrastructure/README.md +522 -0
  63. package/infrastructure/__tests__/fixtures/mock-aws-resources.js +391 -0
  64. package/infrastructure/__tests__/helpers/test-utils.js +277 -0
  65. package/infrastructure/__tests__/postgres-config.test.js +914 -0
  66. package/infrastructure/__tests__/template-generation.test.js +687 -0
  67. package/infrastructure/create-frigg-infrastructure.js +129 -20
  68. package/infrastructure/docs/POSTGRES-CONFIGURATION.md +630 -0
  69. package/infrastructure/docs/PRE-DEPLOYMENT-HEALTH-CHECK-SPEC.md +1317 -0
  70. package/infrastructure/docs/WEBSOCKET-CONFIGURATION.md +105 -0
  71. package/infrastructure/docs/deployment-instructions.md +268 -0
  72. package/infrastructure/docs/generate-iam-command.md +278 -0
  73. package/infrastructure/docs/iam-policy-templates.md +193 -0
  74. package/infrastructure/domains/database/aurora-builder.js +809 -0
  75. package/infrastructure/domains/database/aurora-builder.test.js +950 -0
  76. package/infrastructure/domains/database/aurora-discovery.js +87 -0
  77. package/infrastructure/domains/database/aurora-discovery.test.js +188 -0
  78. package/infrastructure/domains/database/aurora-resolver.js +210 -0
  79. package/infrastructure/domains/database/aurora-resolver.test.js +347 -0
  80. package/infrastructure/domains/database/migration-builder.js +701 -0
  81. package/infrastructure/domains/database/migration-builder.test.js +321 -0
  82. package/infrastructure/domains/database/migration-resolver.js +163 -0
  83. package/infrastructure/domains/database/migration-resolver.test.js +337 -0
  84. package/infrastructure/domains/health/application/ports/IPropertyReconciler.js +164 -0
  85. package/infrastructure/domains/health/application/ports/IResourceDetector.js +129 -0
  86. package/infrastructure/domains/health/application/ports/IResourceImporter.js +142 -0
  87. package/infrastructure/domains/health/application/ports/IStackRepository.js +131 -0
  88. package/infrastructure/domains/health/application/ports/index.js +26 -0
  89. package/infrastructure/domains/health/application/use-cases/__tests__/execute-resource-import-use-case.test.js +679 -0
  90. package/infrastructure/domains/health/application/use-cases/__tests__/mismatch-analyzer-method-name.test.js +167 -0
  91. package/infrastructure/domains/health/application/use-cases/__tests__/repair-via-import-use-case.test.js +1130 -0
  92. package/infrastructure/domains/health/application/use-cases/execute-resource-import-use-case.js +221 -0
  93. package/infrastructure/domains/health/application/use-cases/reconcile-properties-use-case.js +152 -0
  94. package/infrastructure/domains/health/application/use-cases/reconcile-properties-use-case.test.js +343 -0
  95. package/infrastructure/domains/health/application/use-cases/repair-via-import-use-case.js +535 -0
  96. package/infrastructure/domains/health/application/use-cases/repair-via-import-use-case.test.js +376 -0
  97. package/infrastructure/domains/health/application/use-cases/run-health-check-use-case.js +213 -0
  98. package/infrastructure/domains/health/application/use-cases/run-health-check-use-case.test.js +441 -0
  99. package/infrastructure/domains/health/docs/ACME-DEV-DRIFT-ANALYSIS.md +267 -0
  100. package/infrastructure/domains/health/docs/BUILD-VS-DEPLOYED-TEMPLATE-ANALYSIS.md +324 -0
  101. package/infrastructure/domains/health/docs/ORPHAN-DETECTION-ANALYSIS.md +386 -0
  102. package/infrastructure/domains/health/docs/SPEC-CLEANUP-COMMAND.md +1419 -0
  103. package/infrastructure/domains/health/docs/TDD-IMPLEMENTATION-SUMMARY.md +391 -0
  104. package/infrastructure/domains/health/docs/TEMPLATE-COMPARISON-IMPLEMENTATION.md +551 -0
  105. package/infrastructure/domains/health/domain/entities/issue.js +299 -0
  106. package/infrastructure/domains/health/domain/entities/issue.test.js +528 -0
  107. package/infrastructure/domains/health/domain/entities/property-mismatch.js +108 -0
  108. package/infrastructure/domains/health/domain/entities/property-mismatch.test.js +275 -0
  109. package/infrastructure/domains/health/domain/entities/resource.js +159 -0
  110. package/infrastructure/domains/health/domain/entities/resource.test.js +432 -0
  111. package/infrastructure/domains/health/domain/entities/stack-health-report.js +306 -0
  112. package/infrastructure/domains/health/domain/entities/stack-health-report.test.js +601 -0
  113. package/infrastructure/domains/health/domain/services/__tests__/health-score-percentage-based.test.js +380 -0
  114. package/infrastructure/domains/health/domain/services/__tests__/import-progress-monitor.test.js +971 -0
  115. package/infrastructure/domains/health/domain/services/__tests__/import-template-generator.test.js +1150 -0
  116. package/infrastructure/domains/health/domain/services/__tests__/logical-id-mapper.test.js +672 -0
  117. package/infrastructure/domains/health/domain/services/__tests__/template-parser.test.js +496 -0
  118. package/infrastructure/domains/health/domain/services/__tests__/update-progress-monitor.test.js +419 -0
  119. package/infrastructure/domains/health/domain/services/health-score-calculator.js +248 -0
  120. package/infrastructure/domains/health/domain/services/health-score-calculator.test.js +504 -0
  121. package/infrastructure/domains/health/domain/services/import-progress-monitor.js +195 -0
  122. package/infrastructure/domains/health/domain/services/import-template-generator.js +435 -0
  123. package/infrastructure/domains/health/domain/services/logical-id-mapper.js +345 -0
  124. package/infrastructure/domains/health/domain/services/mismatch-analyzer.js +234 -0
  125. package/infrastructure/domains/health/domain/services/mismatch-analyzer.test.js +431 -0
  126. package/infrastructure/domains/health/domain/services/property-mutability-config.js +382 -0
  127. package/infrastructure/domains/health/domain/services/template-parser.js +245 -0
  128. package/infrastructure/domains/health/domain/services/update-progress-monitor.js +192 -0
  129. package/infrastructure/domains/health/domain/value-objects/health-score.js +138 -0
  130. package/infrastructure/domains/health/domain/value-objects/health-score.test.js +267 -0
  131. package/infrastructure/domains/health/domain/value-objects/property-mutability.js +161 -0
  132. package/infrastructure/domains/health/domain/value-objects/property-mutability.test.js +198 -0
  133. package/infrastructure/domains/health/domain/value-objects/resource-state.js +167 -0
  134. package/infrastructure/domains/health/domain/value-objects/resource-state.test.js +196 -0
  135. package/infrastructure/domains/health/domain/value-objects/stack-identifier.js +192 -0
  136. package/infrastructure/domains/health/domain/value-objects/stack-identifier.test.js +262 -0
  137. package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-cfn-tagged.test.js +312 -0
  138. package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-multi-stack.test.js +367 -0
  139. package/infrastructure/domains/health/infrastructure/adapters/__tests__/orphan-detection-relationship-analysis.test.js +432 -0
  140. package/infrastructure/domains/health/infrastructure/adapters/aws-property-reconciler.js +784 -0
  141. package/infrastructure/domains/health/infrastructure/adapters/aws-property-reconciler.test.js +1133 -0
  142. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-detector.js +565 -0
  143. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-detector.test.js +554 -0
  144. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-importer.js +318 -0
  145. package/infrastructure/domains/health/infrastructure/adapters/aws-resource-importer.test.js +398 -0
  146. package/infrastructure/domains/health/infrastructure/adapters/aws-stack-repository.js +777 -0
  147. package/infrastructure/domains/health/infrastructure/adapters/aws-stack-repository.test.js +580 -0
  148. package/infrastructure/domains/integration/integration-builder.js +404 -0
  149. package/infrastructure/domains/integration/integration-builder.test.js +690 -0
  150. package/infrastructure/domains/integration/integration-resolver.js +170 -0
  151. package/infrastructure/domains/integration/integration-resolver.test.js +369 -0
  152. package/infrastructure/domains/integration/websocket-builder.js +69 -0
  153. package/infrastructure/domains/integration/websocket-builder.test.js +195 -0
  154. package/infrastructure/domains/networking/vpc-builder.js +2051 -0
  155. package/infrastructure/domains/networking/vpc-builder.test.js +1960 -0
  156. package/infrastructure/domains/networking/vpc-discovery.js +177 -0
  157. package/infrastructure/domains/networking/vpc-discovery.test.js +350 -0
  158. package/infrastructure/domains/networking/vpc-resolver.js +505 -0
  159. package/infrastructure/domains/networking/vpc-resolver.test.js +801 -0
  160. package/infrastructure/domains/parameters/ssm-builder.js +79 -0
  161. package/infrastructure/domains/parameters/ssm-builder.test.js +189 -0
  162. package/infrastructure/domains/parameters/ssm-discovery.js +84 -0
  163. package/infrastructure/domains/parameters/ssm-discovery.test.js +210 -0
  164. package/infrastructure/domains/scheduler/scheduler-builder.js +211 -0
  165. package/infrastructure/domains/security/iam-generator.js +816 -0
  166. package/infrastructure/domains/security/iam-generator.test.js +204 -0
  167. package/infrastructure/domains/security/kms-builder.js +415 -0
  168. package/infrastructure/domains/security/kms-builder.test.js +392 -0
  169. package/infrastructure/domains/security/kms-discovery.js +80 -0
  170. package/infrastructure/domains/security/kms-discovery.test.js +177 -0
  171. package/infrastructure/domains/security/kms-resolver.js +96 -0
  172. package/infrastructure/domains/security/kms-resolver.test.js +216 -0
  173. package/infrastructure/domains/security/templates/frigg-deployment-iam-stack.yaml +401 -0
  174. package/infrastructure/domains/security/templates/iam-policy-basic.json +218 -0
  175. package/infrastructure/domains/security/templates/iam-policy-full.json +288 -0
  176. package/infrastructure/domains/shared/base-builder.js +112 -0
  177. package/infrastructure/domains/shared/base-resolver.js +186 -0
  178. package/infrastructure/domains/shared/base-resolver.test.js +305 -0
  179. package/infrastructure/domains/shared/builder-orchestrator.js +212 -0
  180. package/infrastructure/domains/shared/builder-orchestrator.test.js +213 -0
  181. package/infrastructure/domains/shared/cloudformation-discovery-v2.js +334 -0
  182. package/infrastructure/domains/shared/cloudformation-discovery.js +672 -0
  183. package/infrastructure/domains/shared/cloudformation-discovery.test.js +985 -0
  184. package/infrastructure/domains/shared/environment-builder.js +119 -0
  185. package/infrastructure/domains/shared/environment-builder.test.js +247 -0
  186. package/infrastructure/domains/shared/providers/aws-provider-adapter.js +579 -0
  187. package/infrastructure/domains/shared/providers/aws-provider-adapter.test.js +416 -0
  188. package/infrastructure/domains/shared/providers/azure-provider-adapter.stub.js +93 -0
  189. package/infrastructure/domains/shared/providers/cloud-provider-adapter.js +136 -0
  190. package/infrastructure/domains/shared/providers/gcp-provider-adapter.stub.js +82 -0
  191. package/infrastructure/domains/shared/providers/provider-factory.js +108 -0
  192. package/infrastructure/domains/shared/providers/provider-factory.test.js +170 -0
  193. package/infrastructure/domains/shared/resource-discovery.enhanced.test.js +306 -0
  194. package/infrastructure/domains/shared/resource-discovery.js +233 -0
  195. package/infrastructure/domains/shared/resource-discovery.test.js +588 -0
  196. package/infrastructure/domains/shared/types/app-definition.js +205 -0
  197. package/infrastructure/domains/shared/types/discovery-result.js +106 -0
  198. package/infrastructure/domains/shared/types/discovery-result.test.js +258 -0
  199. package/infrastructure/domains/shared/types/index.js +46 -0
  200. package/infrastructure/domains/shared/types/resource-ownership.js +108 -0
  201. package/infrastructure/domains/shared/types/resource-ownership.test.js +101 -0
  202. package/infrastructure/domains/shared/utilities/base-definition-factory.js +408 -0
  203. package/infrastructure/domains/shared/utilities/base-definition-factory.js.bak +338 -0
  204. package/infrastructure/domains/shared/utilities/base-definition-factory.test.js +291 -0
  205. package/infrastructure/domains/shared/utilities/handler-path-resolver.js +134 -0
  206. package/infrastructure/domains/shared/utilities/handler-path-resolver.test.js +268 -0
  207. package/infrastructure/domains/shared/utilities/prisma-layer-manager.js +159 -0
  208. package/infrastructure/domains/shared/utilities/prisma-layer-manager.test.js +444 -0
  209. package/infrastructure/domains/shared/validation/env-validator.js +78 -0
  210. package/infrastructure/domains/shared/validation/env-validator.test.js +173 -0
  211. package/infrastructure/domains/shared/validation/plugin-validator.js +187 -0
  212. package/infrastructure/domains/shared/validation/plugin-validator.test.js +323 -0
  213. package/infrastructure/esbuild.config.js +53 -0
  214. package/infrastructure/infrastructure-composer.js +119 -0
  215. package/infrastructure/infrastructure-composer.test.js +1895 -0
  216. package/infrastructure/integration.test.js +383 -0
  217. package/infrastructure/scripts/build-prisma-layer.js +701 -0
  218. package/infrastructure/scripts/build-prisma-layer.test.js +170 -0
  219. package/infrastructure/scripts/build-time-discovery.js +238 -0
  220. package/infrastructure/scripts/build-time-discovery.test.js +379 -0
  221. package/infrastructure/scripts/run-discovery.js +110 -0
  222. package/infrastructure/scripts/verify-prisma-layer.js +72 -0
  223. package/management-ui/README.md +203 -0
  224. package/package.json +44 -14
  225. package/test/index.js +2 -4
  226. package/test/mock-api.js +1 -3
  227. package/test/mock-integration.js +4 -14
  228. package/.eslintrc.json +0 -3
  229. package/CHANGELOG.md +0 -132
  230. package/infrastructure/app-handler-helpers.js +0 -57
  231. package/infrastructure/backend-utils.js +0 -87
  232. package/infrastructure/routers/auth.js +0 -26
  233. package/infrastructure/routers/integration-defined-routers.js +0 -42
  234. package/infrastructure/routers/middleware/loadUser.js +0 -15
  235. package/infrastructure/routers/middleware/requireLoggedInUser.js +0 -12
  236. package/infrastructure/routers/user.js +0 -41
  237. package/infrastructure/routers/websocket.js +0 -55
  238. package/infrastructure/serverless-template.js +0 -291
  239. package/infrastructure/workers/integration-defined-workers.js +0 -24
  240. package/test/auther-definition-tester.js +0 -125
@@ -0,0 +1,383 @@
1
+ const fs = require('fs');
2
+ const { composeServerlessDefinition } = require('./serverless-template');
3
+ const { AWSDiscovery } = require('./aws-discovery');
4
+ const { BuildTimeDiscovery } = require('./build-time-discovery');
5
+ const FriggServerlessPlugin = require('../../serverless-plugin/index');
6
+
7
+ // Integration tests for end-to-end AWS discovery and serverless config generation
8
+ describe('VPC/KMS/SSM Integration Tests', () => {
9
+ let mockAWSDiscovery;
10
+ let buildTimeDiscovery;
11
+
12
+ const mockAWSResources = {
13
+ defaultVpcId: 'vpc-12345678',
14
+ defaultSecurityGroupId: 'sg-12345678',
15
+ privateSubnetId1: 'subnet-private-1',
16
+ privateSubnetId2: 'subnet-private-2',
17
+ privateRouteTableId: 'rtb-12345678',
18
+ defaultKmsKeyId: 'arn:aws:kms:us-east-1:123456789012:key/12345678-1234-1234-1234-123456789012'
19
+ };
20
+
21
+ beforeEach(() => {
22
+ // Mock AWSDiscovery to return consistent test data
23
+ mockAWSDiscovery = {
24
+ discoverResources: jest.fn().mockResolvedValue(mockAWSResources),
25
+ findDefaultVpc: jest.fn().mockResolvedValue({ VpcId: mockAWSResources.defaultVpcId }),
26
+ findPrivateSubnets: jest.fn().mockResolvedValue([
27
+ { SubnetId: mockAWSResources.privateSubnetId1 },
28
+ { SubnetId: mockAWSResources.privateSubnetId2 }
29
+ ]),
30
+ findDefaultSecurityGroup: jest.fn().mockResolvedValue({ GroupId: mockAWSResources.defaultSecurityGroupId }),
31
+ findPrivateRouteTable: jest.fn().mockResolvedValue({ RouteTableId: mockAWSResources.privateRouteTableId }),
32
+ findDefaultKmsKey: jest.fn().mockResolvedValue(mockAWSResources.defaultKmsKeyId)
33
+ };
34
+
35
+ jest.doMock('./aws-discovery', () => ({
36
+ AWSDiscovery: jest.fn(() => mockAWSDiscovery)
37
+ }));
38
+
39
+ buildTimeDiscovery = new BuildTimeDiscovery('us-east-1');
40
+ });
41
+
42
+ afterEach(() => {
43
+ jest.clearAllMocks();
44
+ jest.resetModules();
45
+ });
46
+
47
+ describe('End-to-End Serverless Configuration Generation', () => {
48
+ it('should generate complete serverless config with VPC, KMS, and SSM enabled', async () => {
49
+ const appDefinition = {
50
+ name: 'test-frigg-app',
51
+ vpc: { enable: true },
52
+ encryption: { fieldLevelEncryptionMethod: 'kms' },
53
+ ssm: { enable: true },
54
+ integrations: [{
55
+ Definition: {
56
+ name: 'testIntegration'
57
+ }
58
+ }]
59
+ };
60
+
61
+ // Run AWS discovery
62
+ const discoveredResources = await mockAWSDiscovery.discoverResources();
63
+
64
+ // Set environment variables as would happen in build
65
+ process.env.AWS_DISCOVERY_VPC_ID = discoveredResources.defaultVpcId;
66
+ process.env.AWS_DISCOVERY_SECURITY_GROUP_ID = discoveredResources.defaultSecurityGroupId;
67
+ process.env.AWS_DISCOVERY_SUBNET_ID_1 = discoveredResources.privateSubnetId1;
68
+ process.env.AWS_DISCOVERY_SUBNET_ID_2 = discoveredResources.privateSubnetId2;
69
+ process.env.AWS_DISCOVERY_ROUTE_TABLE_ID = discoveredResources.privateRouteTableId;
70
+ process.env.AWS_DISCOVERY_KMS_KEY_ID =discoveredResources.defaultKmsKeyId;
71
+
72
+ // Generate serverless configuration
73
+ const serverlessConfig = composeServerlessDefinition(appDefinition);
74
+
75
+ // Verify VPC configuration
76
+ expect(serverlessConfig.provider.vpc).toBe('${self:custom.vpc.${self:provider.stage}}');
77
+ expect(serverlessConfig.custom.vpc).toEqual({
78
+ '${self:provider.stage}': {
79
+ securityGroupIds: ['${env:AWS_DISCOVERY_SECURITY_GROUP_ID}'],
80
+ subnetIds: [
81
+ '${env:AWS_DISCOVERY_SUBNET_ID_1}',
82
+ '${env:AWS_DISCOVERY_SUBNET_ID_2}'
83
+ ]
84
+ }
85
+ });
86
+
87
+ // Verify VPC Endpoint
88
+ expect(serverlessConfig.resources.Resources.VPCEndpointS3).toEqual({
89
+ Type: 'AWS::EC2::VPCEndpoint',
90
+ Properties: {
91
+ VpcId: '${env:AWS_DISCOVERY_VPC_ID}',
92
+ ServiceName: 'com.amazonaws.${self:provider.region}.s3',
93
+ VpcEndpointType: 'Gateway',
94
+ RouteTableIds: ['${env:AWS_DISCOVERY_ROUTE_TABLE_ID}']
95
+ }
96
+ });
97
+
98
+ // Verify KMS configuration
99
+ expect(serverlessConfig.plugins).toContain('serverless-kms-grants');
100
+ expect(serverlessConfig.provider.environment.KMS_KEY_ARN).toBe('${self:custom.kmsGrants.kmsKeyId}');
101
+ expect(serverlessConfig.custom.kmsGrants).toEqual({
102
+ kmsKeyId: '${env:AWS_DISCOVERY_KMS_KEY_ID}'
103
+ });
104
+
105
+ // Verify KMS IAM permissions
106
+ const kmsPermission = serverlessConfig.provider.iamRoleStatements.find(
107
+ statement => statement.Action.includes('kms:GenerateDataKey')
108
+ );
109
+ expect(kmsPermission).toBeDefined();
110
+
111
+ // Verify SSM configuration
112
+ expect(serverlessConfig.provider.layers).toEqual([
113
+ 'arn:aws:lambda:${self:provider.region}:177933569100:layer:AWS-Parameters-and-Secrets-Lambda-Extension:11'
114
+ ]);
115
+ expect(serverlessConfig.provider.environment.SSM_PARAMETER_PREFIX).toBe('/${self:service}/${self:provider.stage}');
116
+
117
+ // Verify SSM IAM permissions
118
+ const ssmPermission = serverlessConfig.provider.iamRoleStatements.find(
119
+ statement => statement.Action.includes('ssm:GetParameter')
120
+ );
121
+ expect(ssmPermission).toBeDefined();
122
+
123
+ // Verify integration resources
124
+ expect(serverlessConfig.functions.testIntegration).toBeDefined();
125
+ expect(serverlessConfig.functions.testIntegrationQueueWorker).toBeDefined();
126
+ expect(serverlessConfig.resources.Resources.TestIntegrationQueue).toBeDefined();
127
+
128
+ // Clean up environment
129
+ delete process.env.AWS_DISCOVERY_VPC_ID;
130
+ delete process.env.AWS_DISCOVERY_SECURITY_GROUP_ID;
131
+ delete process.env.AWS_DISCOVERY_SUBNET_ID_1;
132
+ delete process.env.AWS_DISCOVERY_SUBNET_ID_2;
133
+ delete process.env.AWS_DISCOVERY_ROUTE_TABLE_ID;
134
+ delete process.env.AWS_DISCOVERY_KMS_KEY_ID;
135
+ });
136
+
137
+ it('should generate config with only VPC enabled', async () => {
138
+ const appDefinition = {
139
+ name: 'vpc-only-app',
140
+ vpc: { enable: true },
141
+ integrations: []
142
+ };
143
+
144
+ process.env.AWS_DISCOVERY_VPC_ID = mockAWSResources.defaultVpcId;
145
+ process.env.AWS_DISCOVERY_SECURITY_GROUP_ID = mockAWSResources.defaultSecurityGroupId;
146
+ process.env.AWS_DISCOVERY_SUBNET_ID_1 = mockAWSResources.privateSubnetId1;
147
+ process.env.AWS_DISCOVERY_SUBNET_ID_2 = mockAWSResources.privateSubnetId2;
148
+ process.env.AWS_DISCOVERY_ROUTE_TABLE_ID = mockAWSResources.privateRouteTableId;
149
+
150
+ const serverlessConfig = composeServerlessDefinition(appDefinition);
151
+
152
+ // Should have VPC config
153
+ expect(serverlessConfig.provider.vpc).toBeDefined();
154
+ expect(serverlessConfig.custom.vpc).toBeDefined();
155
+ expect(serverlessConfig.resources.Resources.VPCEndpointS3).toBeDefined();
156
+
157
+ // Should not have KMS config
158
+ expect(serverlessConfig.plugins).not.toContain('serverless-kms-grants');
159
+ expect(serverlessConfig.provider.environment.KMS_KEY_ARN).toBeUndefined();
160
+
161
+ // Should not have SSM config
162
+ expect(serverlessConfig.provider.layers).toBeUndefined();
163
+ expect(serverlessConfig.provider.environment.SSM_PARAMETER_PREFIX).toBeUndefined();
164
+
165
+ // Clean up
166
+ delete process.env.AWS_DISCOVERY_VPC_ID;
167
+ delete process.env.AWS_DISCOVERY_SECURITY_GROUP_ID;
168
+ delete process.env.AWS_DISCOVERY_SUBNET_ID_1;
169
+ delete process.env.AWS_DISCOVERY_SUBNET_ID_2;
170
+ delete process.env.AWS_DISCOVERY_ROUTE_TABLE_ID;
171
+ });
172
+
173
+ it('should generate config with only KMS enabled', async () => {
174
+ const appDefinition = {
175
+ name: 'kms-only-app',
176
+ encryption: { fieldLevelEncryptionMethod: 'kms' },
177
+ integrations: []
178
+ };
179
+
180
+ process.env.AWS_DISCOVERY_KMS_KEY_ID =mockAWSResources.defaultKmsKeyId;
181
+
182
+ const serverlessConfig = composeServerlessDefinition(appDefinition);
183
+
184
+ // Should have KMS config
185
+ expect(serverlessConfig.plugins).toContain('serverless-kms-grants');
186
+ expect(serverlessConfig.custom.kmsGrants).toBeDefined();
187
+
188
+ // Should not have VPC config
189
+ expect(serverlessConfig.provider.vpc).toBeUndefined();
190
+
191
+ // Should not have SSM config
192
+ expect(serverlessConfig.provider.layers).toBeUndefined();
193
+
194
+ delete process.env.AWS_DISCOVERY_KMS_KEY_ID;
195
+ });
196
+ });
197
+
198
+ describe('Plugin Integration', () => {
199
+ it('should trigger AWS discovery through serverless plugin', async () => {
200
+ const mockServerless = {
201
+ cli: { log: jest.fn() },
202
+ service: {
203
+ provider: {
204
+ name: 'aws',
205
+ region: 'us-east-1',
206
+ vpc: '${self:custom.vpc.${self:provider.stage}}'
207
+ },
208
+ plugins: ['serverless-kms-grants'],
209
+ custom: {},
210
+ functions: {}
211
+ },
212
+ processedInput: { commands: [] },
213
+ getProvider: jest.fn(() => ({})),
214
+ extendConfiguration: jest.fn()
215
+ };
216
+
217
+ const plugin = new FriggServerlessPlugin(mockServerless, { stage: 'test' });
218
+
219
+ // Mock BuildTimeDiscovery
220
+ const mockBuildTimeDiscovery = {
221
+ preBuildHook: jest.fn().mockResolvedValue(mockAWSResources)
222
+ };
223
+
224
+ jest.doMock('./build-time-discovery', () => ({
225
+ BuildTimeDiscovery: jest.fn(() => mockBuildTimeDiscovery)
226
+ }));
227
+
228
+ // Test the beforePackageInitialize hook
229
+ await plugin.beforePackageInitialize();
230
+
231
+ expect(mockBuildTimeDiscovery.preBuildHook).toHaveBeenCalledWith(
232
+ expect.objectContaining({
233
+ vpc: { enable: true },
234
+ encryption: { fieldLevelEncryptionMethod: 'kms' }
235
+ }),
236
+ 'us-east-1'
237
+ );
238
+
239
+ expect(mockServerless.cli.log).toHaveBeenCalledWith('AWS discovery completed successfully');
240
+ });
241
+
242
+ it('should handle plugin discovery failure gracefully', async () => {
243
+ const mockServerless = {
244
+ cli: { log: jest.fn() },
245
+ service: {
246
+ provider: {
247
+ name: 'aws',
248
+ region: 'us-east-1',
249
+ vpc: '${self:custom.vpc}'
250
+ },
251
+ plugins: [],
252
+ custom: {},
253
+ functions: {}
254
+ },
255
+ processedInput: { commands: [] },
256
+ getProvider: jest.fn(() => ({})),
257
+ extendConfiguration: jest.fn()
258
+ };
259
+
260
+ const plugin = new FriggServerlessPlugin(mockServerless, { stage: 'test' });
261
+
262
+ // Mock BuildTimeDiscovery to fail
263
+ const mockBuildTimeDiscovery = {
264
+ preBuildHook: jest.fn().mockRejectedValue(new Error('AWS API Error'))
265
+ };
266
+
267
+ jest.doMock('./build-time-discovery', () => ({
268
+ BuildTimeDiscovery: jest.fn(() => mockBuildTimeDiscovery)
269
+ }));
270
+
271
+ await plugin.beforePackageInitialize();
272
+
273
+ expect(mockServerless.cli.log).toHaveBeenCalledWith('AWS discovery failed, continuing with deployment...');
274
+ expect(mockServerless.cli.log).toHaveBeenCalledWith('Using fallback values for AWS resources');
275
+
276
+ // Verify fallback values are set
277
+ expect(process.env.AWS_DISCOVERY_VPC_ID).toBe('vpc-fallback');
278
+ expect(process.env.AWS_DISCOVERY_KMS_KEY_ID).toBe('arn:aws:kms:*:*:key/*');
279
+ });
280
+ });
281
+
282
+ describe('Template Variable Replacement', () => {
283
+ it('should replace environment variable placeholders with actual values', () => {
284
+ const template = `
285
+ provider:
286
+ vpc:
287
+ securityGroupIds:
288
+ - \${env:AWS_DISCOVERY_SECURITY_GROUP_ID}
289
+ subnetIds:
290
+ - \${env:AWS_DISCOVERY_SUBNET_ID_1}
291
+ - \${env:AWS_DISCOVERY_SUBNET_ID_2}
292
+ environment:
293
+ KMS_KEY_ARN: \${env:AWS_DISCOVERY_KMS_KEY_ID}
294
+ resources:
295
+ VPCEndpoint:
296
+ Properties:
297
+ VpcId: \${env:AWS_DISCOVERY_VPC_ID}
298
+ `;
299
+
300
+ // Set environment variables
301
+ process.env.AWS_DISCOVERY_VPC_ID = mockAWSResources.defaultVpcId;
302
+ process.env.AWS_DISCOVERY_SECURITY_GROUP_ID = mockAWSResources.defaultSecurityGroupId;
303
+ process.env.AWS_DISCOVERY_SUBNET_ID_1 = mockAWSResources.privateSubnetId1;
304
+ process.env.AWS_DISCOVERY_SUBNET_ID_2 = mockAWSResources.privateSubnetId2;
305
+ process.env.AWS_DISCOVERY_KMS_KEY_ID =mockAWSResources.defaultKmsKeyId;
306
+
307
+ // In a real deployment, serverless framework would resolve these environment variables
308
+ // For testing, we can verify the placeholders are correctly formatted
309
+ expect(template).toContain('${env:AWS_DISCOVERY_VPC_ID}');
310
+ expect(template).toContain('${env:AWS_DISCOVERY_SECURITY_GROUP_ID}');
311
+ expect(template).toContain('${env:AWS_DISCOVERY_SUBNET_ID_1}');
312
+ expect(template).toContain('${env:AWS_DISCOVERY_SUBNET_ID_2}');
313
+ expect(template).toContain('${env:AWS_DISCOVERY_KMS_KEY_ID}');
314
+
315
+ // Clean up
316
+ delete process.env.AWS_DISCOVERY_VPC_ID;
317
+ delete process.env.AWS_DISCOVERY_SECURITY_GROUP_ID;
318
+ delete process.env.AWS_DISCOVERY_SUBNET_ID_1;
319
+ delete process.env.AWS_DISCOVERY_SUBNET_ID_2;
320
+ delete process.env.AWS_DISCOVERY_KMS_KEY_ID;
321
+ });
322
+ });
323
+
324
+ describe('Error Scenarios', () => {
325
+ it('should handle AWS discovery timeout gracefully', async () => {
326
+ const mockFailingDiscovery = {
327
+ discoverResources: jest.fn().mockRejectedValue(new Error('Request timeout'))
328
+ };
329
+
330
+ jest.doMock('./aws-discovery', () => ({
331
+ AWSDiscovery: jest.fn(() => mockFailingDiscovery)
332
+ }));
333
+
334
+ const appDefinition = {
335
+ vpc: { enable: true },
336
+ integrations: []
337
+ };
338
+
339
+ await expect(buildTimeDiscovery.preBuildHook(appDefinition, 'us-east-1')).rejects.toThrow('Request timeout');
340
+ });
341
+
342
+ it('should handle partial AWS resource discovery', async () => {
343
+ const partialResources = {
344
+ defaultVpcId: 'vpc-12345678',
345
+ defaultSecurityGroupId: 'sg-12345678',
346
+ privateSubnetId1: 'subnet-1',
347
+ privateSubnetId2: 'subnet-1', // Same subnet used twice
348
+ privateRouteTableId: 'rtb-12345678',
349
+ defaultKmsKeyId: '*' // Fallback KMS key
350
+ };
351
+
352
+ mockAWSDiscovery.discoverResources.mockResolvedValue(partialResources);
353
+
354
+ const appDefinition = {
355
+ vpc: { enable: true },
356
+ encryption: { fieldLevelEncryptionMethod: 'kms' },
357
+ integrations: []
358
+ };
359
+
360
+ const result = await buildTimeDiscovery.preBuildHook(appDefinition, 'us-east-1');
361
+
362
+ expect(result).toEqual(partialResources);
363
+ expect(result.privateSubnetId2).toBe(result.privateSubnetId1); // Should handle single subnet scenario
364
+ expect(result.defaultKmsKeyId).toBe('*'); // Should handle KMS fallback
365
+ });
366
+ });
367
+
368
+ describe('Multi-Region Support', () => {
369
+ it('should support different AWS regions', async () => {
370
+ const appDefinition = {
371
+ vpc: { enable: true },
372
+ integrations: []
373
+ };
374
+
375
+ const euWestDiscovery = new BuildTimeDiscovery('eu-west-1');
376
+
377
+ await euWestDiscovery.preBuildHook(appDefinition, 'eu-west-1');
378
+
379
+ // Verify that AWSDiscovery was instantiated with correct region
380
+ expect(mockAWSDiscovery.discoverResources).toHaveBeenCalled();
381
+ });
382
+ });
383
+ });