@stacksjs/ts-cloud-core 0.1.1

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 (251) hide show
  1. package/LICENSE.md +21 -0
  2. package/README.md +321 -0
  3. package/package.json +31 -0
  4. package/src/advanced-features.test.ts +465 -0
  5. package/src/aws/cloudformation.ts +421 -0
  6. package/src/aws/cloudfront.ts +158 -0
  7. package/src/aws/credentials.test.ts +132 -0
  8. package/src/aws/credentials.ts +545 -0
  9. package/src/aws/index.ts +87 -0
  10. package/src/aws/s3.test.ts +188 -0
  11. package/src/aws/s3.ts +1088 -0
  12. package/src/aws/signature.test.ts +670 -0
  13. package/src/aws/signature.ts +1155 -0
  14. package/src/backup/disaster-recovery.test.ts +726 -0
  15. package/src/backup/disaster-recovery.ts +500 -0
  16. package/src/backup/index.ts +34 -0
  17. package/src/backup/manager.test.ts +498 -0
  18. package/src/backup/manager.ts +432 -0
  19. package/src/cicd/circleci.ts +430 -0
  20. package/src/cicd/github-actions.ts +424 -0
  21. package/src/cicd/gitlab-ci.ts +255 -0
  22. package/src/cicd/index.ts +8 -0
  23. package/src/cli/history.ts +396 -0
  24. package/src/cli/index.ts +10 -0
  25. package/src/cli/progress.ts +458 -0
  26. package/src/cli/repl.ts +454 -0
  27. package/src/cli/suggestions.ts +327 -0
  28. package/src/cli/table.test.ts +319 -0
  29. package/src/cli/table.ts +332 -0
  30. package/src/cloudformation/builder.test.ts +327 -0
  31. package/src/cloudformation/builder.ts +378 -0
  32. package/src/cloudformation/builders/api-gateway.ts +449 -0
  33. package/src/cloudformation/builders/cache.ts +334 -0
  34. package/src/cloudformation/builders/cdn.ts +278 -0
  35. package/src/cloudformation/builders/compute.ts +485 -0
  36. package/src/cloudformation/builders/database.ts +392 -0
  37. package/src/cloudformation/builders/functions.ts +343 -0
  38. package/src/cloudformation/builders/messaging.ts +140 -0
  39. package/src/cloudformation/builders/monitoring.ts +300 -0
  40. package/src/cloudformation/builders/network.ts +264 -0
  41. package/src/cloudformation/builders/queue.ts +147 -0
  42. package/src/cloudformation/builders/security.ts +399 -0
  43. package/src/cloudformation/builders/storage.ts +285 -0
  44. package/src/cloudformation/index.ts +30 -0
  45. package/src/cloudformation/types.ts +173 -0
  46. package/src/compliance/aws-config.ts +543 -0
  47. package/src/compliance/cloudtrail.ts +376 -0
  48. package/src/compliance/compliance.test.ts +423 -0
  49. package/src/compliance/guardduty.ts +446 -0
  50. package/src/compliance/index.ts +66 -0
  51. package/src/compliance/security-hub.ts +456 -0
  52. package/src/containers/build-optimization.ts +416 -0
  53. package/src/containers/containers.test.ts +508 -0
  54. package/src/containers/image-scanning.ts +360 -0
  55. package/src/containers/index.ts +9 -0
  56. package/src/containers/registry.ts +293 -0
  57. package/src/containers/service-mesh.ts +520 -0
  58. package/src/database/database.test.ts +762 -0
  59. package/src/database/index.ts +9 -0
  60. package/src/database/migrations.ts +444 -0
  61. package/src/database/performance.ts +528 -0
  62. package/src/database/replicas.ts +534 -0
  63. package/src/database/users.ts +494 -0
  64. package/src/dependency-graph.ts +143 -0
  65. package/src/deployment/ab-testing.ts +582 -0
  66. package/src/deployment/blue-green.ts +452 -0
  67. package/src/deployment/canary.ts +500 -0
  68. package/src/deployment/deployment.test.ts +526 -0
  69. package/src/deployment/index.ts +61 -0
  70. package/src/deployment/progressive.ts +62 -0
  71. package/src/dns/dns.test.ts +641 -0
  72. package/src/dns/dnssec.ts +315 -0
  73. package/src/dns/index.ts +8 -0
  74. package/src/dns/resolver.ts +496 -0
  75. package/src/dns/routing.ts +593 -0
  76. package/src/email/advanced/analytics.ts +445 -0
  77. package/src/email/advanced/index.ts +11 -0
  78. package/src/email/advanced/rules.ts +465 -0
  79. package/src/email/advanced/scheduling.ts +352 -0
  80. package/src/email/advanced/search.ts +412 -0
  81. package/src/email/advanced/shared-mailboxes.ts +404 -0
  82. package/src/email/advanced/templates.ts +455 -0
  83. package/src/email/advanced/threading.ts +281 -0
  84. package/src/email/analytics.ts +467 -0
  85. package/src/email/bounce-handling.ts +425 -0
  86. package/src/email/email.test.ts +431 -0
  87. package/src/email/handlers/__tests__/inbound.test.ts +38 -0
  88. package/src/email/handlers/__tests__/outbound.test.ts +37 -0
  89. package/src/email/handlers/converter.ts +227 -0
  90. package/src/email/handlers/feedback.ts +228 -0
  91. package/src/email/handlers/inbound.ts +169 -0
  92. package/src/email/handlers/outbound.ts +178 -0
  93. package/src/email/index.ts +15 -0
  94. package/src/email/reputation.ts +303 -0
  95. package/src/email/templates.ts +352 -0
  96. package/src/errors/index.test.ts +434 -0
  97. package/src/errors/index.ts +416 -0
  98. package/src/health-checks/index.ts +40 -0
  99. package/src/index.ts +360 -0
  100. package/src/intrinsic-functions.ts +118 -0
  101. package/src/lambda/concurrency.ts +330 -0
  102. package/src/lambda/destinations.ts +345 -0
  103. package/src/lambda/dlq.ts +425 -0
  104. package/src/lambda/index.ts +11 -0
  105. package/src/lambda/lambda.test.ts +840 -0
  106. package/src/lambda/layers.ts +263 -0
  107. package/src/lambda/versions.ts +376 -0
  108. package/src/lambda/vpc.ts +399 -0
  109. package/src/local/config.ts +114 -0
  110. package/src/local/index.ts +6 -0
  111. package/src/local/mock-aws.ts +351 -0
  112. package/src/modules/ai.ts +340 -0
  113. package/src/modules/api.ts +478 -0
  114. package/src/modules/auth.ts +805 -0
  115. package/src/modules/cache.ts +417 -0
  116. package/src/modules/cdn.ts +1062 -0
  117. package/src/modules/communication.ts +1094 -0
  118. package/src/modules/compute.ts +3348 -0
  119. package/src/modules/database.ts +554 -0
  120. package/src/modules/deployment.ts +1079 -0
  121. package/src/modules/dns.ts +337 -0
  122. package/src/modules/email.ts +1538 -0
  123. package/src/modules/filesystem.ts +515 -0
  124. package/src/modules/index.ts +32 -0
  125. package/src/modules/messaging.ts +486 -0
  126. package/src/modules/monitoring.ts +2086 -0
  127. package/src/modules/network.ts +664 -0
  128. package/src/modules/parameter-store.ts +325 -0
  129. package/src/modules/permissions.ts +1081 -0
  130. package/src/modules/phone.ts +494 -0
  131. package/src/modules/queue.ts +1260 -0
  132. package/src/modules/redirects.ts +464 -0
  133. package/src/modules/registry.ts +699 -0
  134. package/src/modules/search.ts +401 -0
  135. package/src/modules/secrets.ts +416 -0
  136. package/src/modules/security.ts +731 -0
  137. package/src/modules/sms.ts +389 -0
  138. package/src/modules/storage.ts +1120 -0
  139. package/src/modules/workflow.ts +680 -0
  140. package/src/multi-account/config.ts +521 -0
  141. package/src/multi-account/index.ts +7 -0
  142. package/src/multi-account/manager.ts +427 -0
  143. package/src/multi-region/cross-region.ts +410 -0
  144. package/src/multi-region/index.ts +8 -0
  145. package/src/multi-region/manager.ts +483 -0
  146. package/src/multi-region/regions.ts +435 -0
  147. package/src/network-security/index.ts +48 -0
  148. package/src/observability/index.ts +9 -0
  149. package/src/observability/logs.ts +522 -0
  150. package/src/observability/metrics.ts +460 -0
  151. package/src/observability/observability.test.ts +782 -0
  152. package/src/observability/synthetics.ts +568 -0
  153. package/src/observability/xray.ts +358 -0
  154. package/src/phone/advanced/analytics.ts +349 -0
  155. package/src/phone/advanced/callbacks.ts +428 -0
  156. package/src/phone/advanced/index.ts +8 -0
  157. package/src/phone/advanced/ivr-builder.ts +504 -0
  158. package/src/phone/advanced/recording.ts +310 -0
  159. package/src/phone/handlers/__tests__/incoming-call.test.ts +40 -0
  160. package/src/phone/handlers/incoming-call.ts +117 -0
  161. package/src/phone/handlers/missed-call.ts +116 -0
  162. package/src/phone/handlers/voicemail.ts +179 -0
  163. package/src/phone/index.ts +9 -0
  164. package/src/presets/api-backend.ts +134 -0
  165. package/src/presets/data-pipeline.ts +204 -0
  166. package/src/presets/extend.test.ts +295 -0
  167. package/src/presets/extend.ts +297 -0
  168. package/src/presets/fullstack-app.ts +144 -0
  169. package/src/presets/index.ts +27 -0
  170. package/src/presets/jamstack.ts +135 -0
  171. package/src/presets/microservices.ts +167 -0
  172. package/src/presets/ml-api.ts +208 -0
  173. package/src/presets/nodejs-server.ts +104 -0
  174. package/src/presets/nodejs-serverless.ts +114 -0
  175. package/src/presets/realtime-app.ts +184 -0
  176. package/src/presets/static-site.ts +64 -0
  177. package/src/presets/traditional-web-app.ts +339 -0
  178. package/src/presets/wordpress.ts +138 -0
  179. package/src/preview/github.test.ts +249 -0
  180. package/src/preview/github.ts +297 -0
  181. package/src/preview/index.ts +37 -0
  182. package/src/preview/manager.test.ts +440 -0
  183. package/src/preview/manager.ts +326 -0
  184. package/src/preview/notifications.test.ts +582 -0
  185. package/src/preview/notifications.ts +341 -0
  186. package/src/queue/batch-processing.ts +402 -0
  187. package/src/queue/dlq-monitoring.ts +402 -0
  188. package/src/queue/fifo.ts +342 -0
  189. package/src/queue/index.ts +9 -0
  190. package/src/queue/management.ts +428 -0
  191. package/src/queue/queue.test.ts +429 -0
  192. package/src/resource-mgmt/index.ts +39 -0
  193. package/src/resource-naming.ts +62 -0
  194. package/src/s3/index.ts +523 -0
  195. package/src/schema/cloud-config.schema.json +554 -0
  196. package/src/schema/index.ts +68 -0
  197. package/src/security/certificate-manager.ts +492 -0
  198. package/src/security/index.ts +9 -0
  199. package/src/security/scanning.ts +545 -0
  200. package/src/security/secrets-manager.ts +476 -0
  201. package/src/security/secrets-rotation.ts +456 -0
  202. package/src/security/security.test.ts +738 -0
  203. package/src/sms/advanced/ab-testing.ts +389 -0
  204. package/src/sms/advanced/analytics.ts +336 -0
  205. package/src/sms/advanced/campaigns.ts +523 -0
  206. package/src/sms/advanced/chatbot.ts +224 -0
  207. package/src/sms/advanced/index.ts +10 -0
  208. package/src/sms/advanced/link-tracking.ts +248 -0
  209. package/src/sms/advanced/mms.ts +308 -0
  210. package/src/sms/handlers/__tests__/send.test.ts +40 -0
  211. package/src/sms/handlers/delivery-status.ts +133 -0
  212. package/src/sms/handlers/receive.ts +162 -0
  213. package/src/sms/handlers/send.ts +174 -0
  214. package/src/sms/index.ts +9 -0
  215. package/src/stack-diff.ts +389 -0
  216. package/src/static-site/index.ts +85 -0
  217. package/src/template-builder.ts +110 -0
  218. package/src/template-validator.ts +574 -0
  219. package/src/utils/cache.ts +291 -0
  220. package/src/utils/diff.ts +269 -0
  221. package/src/utils/hash.ts +227 -0
  222. package/src/utils/index.ts +8 -0
  223. package/src/utils/parallel.ts +294 -0
  224. package/src/validators/credentials.test.ts +274 -0
  225. package/src/validators/credentials.ts +233 -0
  226. package/src/validators/quotas.test.ts +434 -0
  227. package/src/validators/quotas.ts +217 -0
  228. package/test/ai.test.ts +327 -0
  229. package/test/api.test.ts +511 -0
  230. package/test/auth.test.ts +632 -0
  231. package/test/cache.test.ts +406 -0
  232. package/test/cdn.test.ts +247 -0
  233. package/test/compute.test.ts +861 -0
  234. package/test/database.test.ts +523 -0
  235. package/test/deployment.test.ts +499 -0
  236. package/test/dns.test.ts +270 -0
  237. package/test/email.test.ts +439 -0
  238. package/test/filesystem.test.ts +382 -0
  239. package/test/integration.test.ts +350 -0
  240. package/test/messaging.test.ts +514 -0
  241. package/test/monitoring.test.ts +634 -0
  242. package/test/network.test.ts +425 -0
  243. package/test/permissions.test.ts +488 -0
  244. package/test/queue.test.ts +484 -0
  245. package/test/registry.test.ts +306 -0
  246. package/test/security.test.ts +462 -0
  247. package/test/storage.test.ts +463 -0
  248. package/test/template-validator.test.ts +559 -0
  249. package/test/workflow.test.ts +592 -0
  250. package/tsconfig.json +16 -0
  251. package/tsconfig.tsbuildinfo +1 -0
@@ -0,0 +1,423 @@
1
+ import { describe, expect, it, beforeEach } from 'bun:test'
2
+ import { AWSConfigManager } from './aws-config'
3
+ import { CloudTrailManager } from './cloudtrail'
4
+ import { GuardDutyManager } from './guardduty'
5
+ import { SecurityHubManager } from './security-hub'
6
+
7
+ describe('AWS Config Manager', () => {
8
+ let manager: AWSConfigManager
9
+
10
+ beforeEach(() => {
11
+ manager = new AWSConfigManager()
12
+ })
13
+
14
+ describe('Config Recorder', () => {
15
+ it('should create config recorder', () => {
16
+ const recorder = manager.createConfigRecorder({
17
+ name: 'default',
18
+ roleArn: 'arn:aws:iam::123456789012:role/config-role',
19
+ recordingGroup: {
20
+ allSupported: true,
21
+ includeGlobalResourceTypes: true,
22
+ },
23
+ })
24
+
25
+ expect(recorder.name).toBe('default')
26
+ expect(recorder.roleArn).toBe('arn:aws:iam::123456789012:role/config-role')
27
+ expect(recorder.recordingGroup?.allSupported).toBe(true)
28
+ })
29
+
30
+ it('should get config recorder', () => {
31
+ manager.createConfigRecorder({
32
+ name: 'test-recorder',
33
+ roleArn: 'arn:aws:iam::123456789012:role/config-role',
34
+ })
35
+
36
+ const retrieved = manager.getConfigRecorder('test-recorder')
37
+ expect(retrieved).toBeDefined()
38
+ expect(retrieved?.name).toBe('test-recorder')
39
+ })
40
+ })
41
+
42
+ describe('Delivery Channel', () => {
43
+ it('should create delivery channel', () => {
44
+ const channel = manager.createDeliveryChannel({
45
+ name: 'default',
46
+ s3BucketName: 'config-bucket',
47
+ s3KeyPrefix: 'config',
48
+ snsTopicArn: 'arn:aws:sns:us-east-1:123456789012:config-topic',
49
+ })
50
+
51
+ expect(channel.name).toBe('default')
52
+ expect(channel.s3BucketName).toBe('config-bucket')
53
+ expect(channel.s3KeyPrefix).toBe('config')
54
+ })
55
+ })
56
+
57
+ describe('Config Rules', () => {
58
+ it('should create S3 encryption rule', () => {
59
+ const rule = manager.createS3EncryptionRule()
60
+
61
+ expect(rule.name).toBe('s3-bucket-server-side-encryption-enabled')
62
+ expect(rule.source).toBe('AWS_MANAGED')
63
+ expect(rule.identifier).toBe('S3_BUCKET_SERVER_SIDE_ENCRYPTION_ENABLED')
64
+ expect(rule.scope?.complianceResourceTypes).toContain('AWS::S3::Bucket')
65
+ })
66
+
67
+ it('should create RDS encryption rule', () => {
68
+ const rule = manager.createRdsEncryptionRule()
69
+
70
+ expect(rule.name).toBe('rds-storage-encrypted')
71
+ expect(rule.identifier).toBe('RDS_STORAGE_ENCRYPTED')
72
+ expect(rule.scope?.complianceResourceTypes).toContain('AWS::RDS::DBInstance')
73
+ })
74
+
75
+ it('should create IAM password policy rule', () => {
76
+ const rule = manager.createIamPasswordPolicyRule()
77
+
78
+ expect(rule.name).toBe('iam-password-policy')
79
+ expect(rule.inputParameters?.MinimumPasswordLength).toBe(14)
80
+ expect(rule.inputParameters?.RequireUppercaseCharacters).toBe(true)
81
+ })
82
+
83
+ it('should create custom Lambda rule', () => {
84
+ const rule = manager.createCustomLambdaRule({
85
+ name: 'custom-compliance-check',
86
+ description: 'Custom compliance validation',
87
+ lambdaFunctionArn: 'arn:aws:lambda:us-east-1:123456789012:function:compliance-check',
88
+ resourceTypes: ['AWS::EC2::Instance'],
89
+ maxExecutionFrequency: 'TwentyFour_Hours',
90
+ })
91
+
92
+ expect(rule.name).toBe('custom-compliance-check')
93
+ expect(rule.source).toBe('CUSTOM_LAMBDA')
94
+ expect(rule.lambdaFunctionArn).toBeDefined()
95
+ expect(rule.maxExecutionFrequency).toBe('TwentyFour_Hours')
96
+ })
97
+
98
+ it('should list config rules', () => {
99
+ manager.createS3EncryptionRule()
100
+ manager.createRdsEncryptionRule()
101
+
102
+ const rules = manager.listConfigRules()
103
+ expect(rules).toHaveLength(2)
104
+ })
105
+ })
106
+
107
+ describe('Compliance Presets', () => {
108
+ it('should create HIPAA preset', () => {
109
+ const rules = manager.createCompliancePreset('hipaa')
110
+
111
+ expect(rules.length).toBeGreaterThan(5)
112
+ expect(rules.some(r => r.name.includes('encryption'))).toBe(true)
113
+ expect(rules.some(r => r.name.includes('cloudtrail'))).toBe(true)
114
+ })
115
+
116
+ it('should create PCI-DSS preset', () => {
117
+ const rules = manager.createCompliancePreset('pci-dss')
118
+
119
+ expect(rules.length).toBeGreaterThan(7)
120
+ expect(rules.some(r => r.name.includes('mfa'))).toBe(true)
121
+ expect(rules.some(r => r.name.includes('encryption'))).toBe(true)
122
+ })
123
+
124
+ it('should create basic preset', () => {
125
+ const rules = manager.createCompliancePreset('basic')
126
+
127
+ expect(rules.length).toBeGreaterThan(4)
128
+ })
129
+ })
130
+
131
+ describe('CloudFormation Generation', () => {
132
+ it('should generate config rule CloudFormation', () => {
133
+ const rule = manager.createS3EncryptionRule()
134
+ const cf = manager.generateConfigRuleCF(rule)
135
+
136
+ expect(cf.Type).toBe('AWS::Config::ConfigRule')
137
+ expect(cf.Properties.ConfigRuleName).toBe('s3-bucket-server-side-encryption-enabled')
138
+ expect(cf.Properties.Source.Owner).toBe('AWS')
139
+ expect(cf.Properties.Source.SourceIdentifier).toBe('S3_BUCKET_SERVER_SIDE_ENCRYPTION_ENABLED')
140
+ })
141
+
142
+ it('should generate recorder CloudFormation', () => {
143
+ const recorder = manager.createConfigRecorder({
144
+ name: 'default',
145
+ roleArn: 'arn:aws:iam::123456789012:role/config-role',
146
+ })
147
+
148
+ const cf = manager.generateConfigRecorderCF(recorder)
149
+
150
+ expect(cf.Type).toBe('AWS::Config::ConfigurationRecorder')
151
+ expect(cf.Properties.Name).toBe('default')
152
+ })
153
+ })
154
+ })
155
+
156
+ describe('CloudTrail Manager', () => {
157
+ let manager: CloudTrailManager
158
+
159
+ beforeEach(() => {
160
+ manager = new CloudTrailManager()
161
+ })
162
+
163
+ describe('Trail Creation', () => {
164
+ it('should create basic trail', () => {
165
+ const trail = manager.createTrail({
166
+ name: 'my-trail',
167
+ s3BucketName: 'cloudtrail-bucket',
168
+ includeGlobalServiceEvents: true,
169
+ isMultiRegionTrail: true,
170
+ enableLogFileValidation: true,
171
+ })
172
+
173
+ expect(trail.id).toMatch(/^trail-\d+-\d+$/)
174
+ expect(trail.name).toBe('my-trail')
175
+ expect(trail.s3BucketName).toBe('cloudtrail-bucket')
176
+ expect(trail.enableLogFileValidation).toBe(true)
177
+ })
178
+
179
+ it('should create organization trail', () => {
180
+ const trail = manager.createOrganizationTrail({
181
+ name: 'org-trail',
182
+ s3BucketName: 'org-cloudtrail',
183
+ kmsKeyId: 'arn:aws:kms:us-east-1:123456789012:key/12345',
184
+ })
185
+
186
+ expect(trail.name).toBe('org-trail')
187
+ expect(trail.isMultiRegionTrail).toBe(true)
188
+ expect(trail.s3KeyPrefix).toBe('organization-trail')
189
+ })
190
+
191
+ it('should create security audit trail', () => {
192
+ const trail = manager.createSecurityAuditTrail({
193
+ name: 'security-audit',
194
+ s3BucketName: 'security-logs',
195
+ kmsKeyId: 'arn:aws:kms:us-east-1:123456789012:key/12345',
196
+ cloudWatchLogsLogGroupArn: 'arn:aws:logs:us-east-1:123456789012:log-group:cloudtrail',
197
+ cloudWatchLogsRoleArn: 'arn:aws:iam::123456789012:role/cloudtrail-logs',
198
+ })
199
+
200
+ expect(trail.insightSelectors).toHaveLength(2)
201
+ expect(trail.eventSelectors).toHaveLength(1)
202
+ expect(trail.enableLogFileValidation).toBe(true)
203
+ })
204
+
205
+ it('should create data events trail', () => {
206
+ const trail = manager.createDataEventsTrail({
207
+ name: 'data-events',
208
+ s3BucketName: 'data-events-bucket',
209
+ s3DataBuckets: ['app-bucket-1', 'app-bucket-2'],
210
+ lambdaFunctions: ['arn:aws:lambda:us-east-1:123456789012:function:my-function'],
211
+ })
212
+
213
+ expect(trail.eventSelectors).toHaveLength(1)
214
+ expect(trail.eventSelectors![0].dataResources).toHaveLength(2)
215
+ })
216
+ })
217
+
218
+ describe('CloudFormation Generation', () => {
219
+ it('should generate trail CloudFormation', () => {
220
+ const trail = manager.createTrail({
221
+ name: 'my-trail',
222
+ s3BucketName: 'cloudtrail-bucket',
223
+ kmsKeyId: 'arn:aws:kms:us-east-1:123456789012:key/12345',
224
+ })
225
+
226
+ const cf = manager.generateTrailCF(trail)
227
+
228
+ expect(cf.Type).toBe('AWS::CloudTrail::Trail')
229
+ expect(cf.Properties.TrailName).toBe('my-trail')
230
+ expect(cf.Properties.S3BucketName).toBe('cloudtrail-bucket')
231
+ expect(cf.Properties.KMSKeyId).toBeDefined()
232
+ })
233
+
234
+ it('should generate bucket policy', () => {
235
+ const policy = manager.generateBucketPolicy('cloudtrail-bucket', ['123456789012', '987654321098'])
236
+
237
+ expect(policy.Statement).toHaveLength(2)
238
+ expect(policy.Statement[0].Action).toBe('s3:GetBucketAcl')
239
+ expect(policy.Statement[1].Action).toBe('s3:PutObject')
240
+ })
241
+ })
242
+ })
243
+
244
+ describe('GuardDuty Manager', () => {
245
+ let manager: GuardDutyManager
246
+
247
+ beforeEach(() => {
248
+ manager = new GuardDutyManager()
249
+ })
250
+
251
+ describe('Detector Creation', () => {
252
+ it('should create basic detector', () => {
253
+ const detector = manager.createBasicDetector()
254
+
255
+ expect(detector.id).toMatch(/^detector-\d+-\d+$/)
256
+ expect(detector.enable).toBe(true)
257
+ expect(detector.findingPublishingFrequency).toBe('SIX_HOURS')
258
+ })
259
+
260
+ it('should create comprehensive detector', () => {
261
+ const detector = manager.createComprehensiveDetector()
262
+
263
+ expect(detector.enable).toBe(true)
264
+ expect(detector.findingPublishingFrequency).toBe('FIFTEEN_MINUTES')
265
+ expect(detector.dataSources?.s3Logs?.enable).toBe(true)
266
+ expect(detector.dataSources?.kubernetes?.auditLogs.enable).toBe(true)
267
+ expect(detector.features).toHaveLength(5)
268
+ })
269
+ })
270
+
271
+ describe('Finding Filters', () => {
272
+ it('should create low severity archive filter', () => {
273
+ const detector = manager.createBasicDetector()
274
+ const filter = manager.createLowSeverityArchiveFilter(detector.id)
275
+
276
+ expect(filter.action).toBe('ARCHIVE')
277
+ expect(filter.findingCriteria.criterion.severity.lt).toBe(4)
278
+ })
279
+
280
+ it('should create finding type filter', () => {
281
+ const detector = manager.createBasicDetector()
282
+ const filter = manager.createFindingTypeFilter(
283
+ detector.id,
284
+ ['Recon:EC2/PortProbeUnprotectedPort'],
285
+ 'ARCHIVE',
286
+ )
287
+
288
+ expect(filter.action).toBe('ARCHIVE')
289
+ expect(filter.findingCriteria.criterion.type.eq).toContain('Recon:EC2/PortProbeUnprotectedPort')
290
+ })
291
+
292
+ it('should create trusted IP filter', () => {
293
+ const detector = manager.createBasicDetector()
294
+ const filter = manager.createTrustedIPFilter(detector.id, ['10.0.1.100', '10.0.2.200'])
295
+
296
+ expect(filter.action).toBe('ARCHIVE')
297
+ expect(filter.rank).toBe(3)
298
+ })
299
+ })
300
+
301
+ describe('CloudFormation Generation', () => {
302
+ it('should generate detector CloudFormation', () => {
303
+ const detector = manager.createComprehensiveDetector()
304
+ const cf = manager.generateDetectorCF(detector)
305
+
306
+ expect(cf.Type).toBe('AWS::GuardDuty::Detector')
307
+ expect(cf.Properties.Enable).toBe(true)
308
+ expect(cf.Properties.DataSources.S3Logs.Enable).toBe(true)
309
+ expect(cf.Properties.Features).toHaveLength(5)
310
+ })
311
+ })
312
+ })
313
+
314
+ describe('Security Hub Manager', () => {
315
+ let manager: SecurityHubManager
316
+
317
+ beforeEach(() => {
318
+ manager = new SecurityHubManager()
319
+ })
320
+
321
+ describe('Hub Creation', () => {
322
+ it('should create basic hub', () => {
323
+ const hub = manager.createBasicHub()
324
+
325
+ expect(hub.id).toMatch(/^hub-\d+-\d+$/)
326
+ expect(hub.enable).toBe(true)
327
+ expect(hub.standards).toHaveLength(1)
328
+ })
329
+
330
+ it('should create comprehensive hub', () => {
331
+ const hub = manager.createComprehensiveHub()
332
+
333
+ expect(hub.enable).toBe(true)
334
+ expect(hub.controlFindingGenerator).toBe('SECURITY_CONTROL')
335
+ expect(hub.standards).toHaveLength(3)
336
+ })
337
+ })
338
+
339
+ describe('Automation Rules', () => {
340
+ it('should create low severity suppression rule', () => {
341
+ const rule = manager.createLowSeveritySuppressionRule()
342
+
343
+ expect(rule.ruleName).toBe('Suppress Low Severity Informational Findings')
344
+ expect(rule.actions[0].findingFieldsUpdate.workflow?.status).toBe('SUPPRESSED')
345
+ expect(rule.criteria.severityLabel![0].value).toBe('INFORMATIONAL')
346
+ })
347
+
348
+ it('should create resource type notification rule', () => {
349
+ const rule = manager.createResourceTypeNotificationRule(['AWS::EC2::Instance', 'AWS::RDS::DBInstance'])
350
+
351
+ expect(rule.ruleName).toBe('Notify on Critical Resource Findings')
352
+ expect(rule.criteria.resourceType).toHaveLength(2)
353
+ expect(rule.criteria.severityLabel).toHaveLength(2)
354
+ })
355
+
356
+ it('should create compliance failure rule', () => {
357
+ const rule = manager.createComplianceFailureRule()
358
+
359
+ expect(rule.ruleName).toBe('Flag Compliance Failures')
360
+ expect(rule.actions[0].findingFieldsUpdate.severity?.label).toBe('HIGH')
361
+ expect(rule.criteria.complianceStatus![0].value).toBe('FAILED')
362
+ })
363
+
364
+ it('should create false positive suppression rule', () => {
365
+ const rule = manager.createFalsePositiveSuppressionRule('GuardDuty', ['Recon:', 'UnauthorizedAccess:'])
366
+
367
+ expect(rule.ruleName).toContain('GuardDuty')
368
+ expect(rule.criteria.title).toHaveLength(2)
369
+ expect(rule.actions[0].findingFieldsUpdate.workflow?.status).toBe('SUPPRESSED')
370
+ })
371
+ })
372
+
373
+ describe('CloudFormation Generation', () => {
374
+ it('should generate hub CloudFormation', () => {
375
+ const hub = manager.createBasicHub()
376
+ const cf = manager.generateHubCF(hub)
377
+
378
+ expect(cf.Type).toBe('AWS::SecurityHub::Hub')
379
+ expect(cf.Properties.ControlFindingGenerator).toBe('STANDARD_CONTROL')
380
+ })
381
+
382
+ it('should generate standard CloudFormation', () => {
383
+ const hub = manager.createComprehensiveHub()
384
+ const standard = hub.standards![0]
385
+ const cf = manager.generateStandardCF(standard)
386
+
387
+ expect(cf.Type).toBe('AWS::SecurityHub::Standard')
388
+ expect(cf.Properties.StandardsArn).toContain('aws-foundational-security-best-practices')
389
+ })
390
+
391
+ it('should generate automation rule CloudFormation', () => {
392
+ const rule = manager.createLowSeveritySuppressionRule()
393
+ const cf = manager.generateAutomationRuleCF(rule)
394
+
395
+ expect(cf.Type).toBe('AWS::SecurityHub::AutomationRule')
396
+ expect(cf.Properties.RuleName).toBe('Suppress Low Severity Informational Findings')
397
+ expect(cf.Properties.RuleStatus).toBe('ENABLED')
398
+ })
399
+ })
400
+
401
+ describe('Security Standards', () => {
402
+ it('should have AWS Foundational Security standard', () => {
403
+ const standard = SecurityHubManager.Standards.AWS_FOUNDATIONAL_SECURITY
404
+
405
+ expect(standard.name).toBe('AWS Foundational Security Best Practices')
406
+ expect(standard.arn).toContain('aws-foundational-security-best-practices')
407
+ })
408
+
409
+ it('should have CIS benchmarks', () => {
410
+ const cis14 = SecurityHubManager.Standards.CIS_AWS_FOUNDATIONS_1_4
411
+
412
+ expect(cis14.name).toContain('CIS')
413
+ expect(cis14.arn).toContain('cis-aws-foundations-benchmark')
414
+ })
415
+
416
+ it('should have PCI-DSS standard', () => {
417
+ const pci = SecurityHubManager.Standards.PCI_DSS
418
+
419
+ expect(pci.name).toContain('PCI DSS')
420
+ expect(pci.arn).toContain('pci-dss')
421
+ })
422
+ })
423
+ })