@build-astron-co/nimbus 0.2.0

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 (313) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +628 -0
  3. package/bin/nimbus +38 -0
  4. package/package.json +80 -0
  5. package/src/__tests__/app.test.ts +76 -0
  6. package/src/__tests__/audit.test.ts +877 -0
  7. package/src/__tests__/circuit-breaker.test.ts +116 -0
  8. package/src/__tests__/cli-run.test.ts +115 -0
  9. package/src/__tests__/context-manager.test.ts +502 -0
  10. package/src/__tests__/context.test.ts +242 -0
  11. package/src/__tests__/enterprise.test.ts +401 -0
  12. package/src/__tests__/generator.test.ts +433 -0
  13. package/src/__tests__/hooks.test.ts +582 -0
  14. package/src/__tests__/init.test.ts +436 -0
  15. package/src/__tests__/intent-parser.test.ts +229 -0
  16. package/src/__tests__/llm-router.test.ts +209 -0
  17. package/src/__tests__/lsp.test.ts +293 -0
  18. package/src/__tests__/modes.test.ts +336 -0
  19. package/src/__tests__/permissions.test.ts +338 -0
  20. package/src/__tests__/serve.test.ts +275 -0
  21. package/src/__tests__/sessions.test.ts +227 -0
  22. package/src/__tests__/sharing.test.ts +288 -0
  23. package/src/__tests__/snapshots.test.ts +581 -0
  24. package/src/__tests__/state-db.test.ts +334 -0
  25. package/src/__tests__/stream-with-tools.test.ts +732 -0
  26. package/src/__tests__/subagents.test.ts +176 -0
  27. package/src/__tests__/system-prompt.test.ts +169 -0
  28. package/src/__tests__/tool-converter.test.ts +256 -0
  29. package/src/__tests__/tool-schemas.test.ts +397 -0
  30. package/src/__tests__/tools.test.ts +143 -0
  31. package/src/__tests__/version.test.ts +49 -0
  32. package/src/agent/compaction-agent.ts +227 -0
  33. package/src/agent/context-manager.ts +435 -0
  34. package/src/agent/context.ts +427 -0
  35. package/src/agent/deploy-preview.ts +426 -0
  36. package/src/agent/index.ts +68 -0
  37. package/src/agent/loop.ts +717 -0
  38. package/src/agent/modes.ts +429 -0
  39. package/src/agent/permissions.ts +466 -0
  40. package/src/agent/subagents/base.ts +116 -0
  41. package/src/agent/subagents/cost.ts +51 -0
  42. package/src/agent/subagents/explore.ts +42 -0
  43. package/src/agent/subagents/general.ts +54 -0
  44. package/src/agent/subagents/index.ts +102 -0
  45. package/src/agent/subagents/infra.ts +59 -0
  46. package/src/agent/subagents/security.ts +69 -0
  47. package/src/agent/system-prompt.ts +436 -0
  48. package/src/app.ts +122 -0
  49. package/src/audit/activity-log.ts +290 -0
  50. package/src/audit/compliance-checker.ts +540 -0
  51. package/src/audit/cost-tracker.ts +318 -0
  52. package/src/audit/index.ts +23 -0
  53. package/src/audit/security-scanner.ts +596 -0
  54. package/src/auth/guard.ts +75 -0
  55. package/src/auth/index.ts +56 -0
  56. package/src/auth/oauth.ts +455 -0
  57. package/src/auth/providers.ts +470 -0
  58. package/src/auth/sso.ts +113 -0
  59. package/src/auth/store.ts +505 -0
  60. package/src/auth/types.ts +187 -0
  61. package/src/build.ts +141 -0
  62. package/src/cli/index.ts +16 -0
  63. package/src/cli/init.ts +854 -0
  64. package/src/cli/openapi-spec.ts +356 -0
  65. package/src/cli/run.ts +237 -0
  66. package/src/cli/serve-auth.ts +80 -0
  67. package/src/cli/serve.ts +462 -0
  68. package/src/cli/web.ts +67 -0
  69. package/src/cli.ts +1417 -0
  70. package/src/clients/core-engine-client.ts +227 -0
  71. package/src/clients/enterprise-client.ts +334 -0
  72. package/src/clients/generator-client.ts +351 -0
  73. package/src/clients/git-client.ts +627 -0
  74. package/src/clients/github-client.ts +410 -0
  75. package/src/clients/helm-client.ts +504 -0
  76. package/src/clients/index.ts +80 -0
  77. package/src/clients/k8s-client.ts +497 -0
  78. package/src/clients/llm-client.ts +161 -0
  79. package/src/clients/rest-client.ts +130 -0
  80. package/src/clients/service-discovery.ts +33 -0
  81. package/src/clients/terraform-client.ts +482 -0
  82. package/src/clients/tools-client.ts +1843 -0
  83. package/src/clients/ws-client.ts +115 -0
  84. package/src/commands/analyze/index.ts +352 -0
  85. package/src/commands/apply/helm.ts +473 -0
  86. package/src/commands/apply/index.ts +213 -0
  87. package/src/commands/apply/k8s.ts +454 -0
  88. package/src/commands/apply/terraform.ts +582 -0
  89. package/src/commands/ask.ts +167 -0
  90. package/src/commands/audit/index.ts +238 -0
  91. package/src/commands/auth-cloud.ts +294 -0
  92. package/src/commands/auth-list.ts +134 -0
  93. package/src/commands/auth-profile.ts +121 -0
  94. package/src/commands/auth-status.ts +141 -0
  95. package/src/commands/aws/ec2.ts +501 -0
  96. package/src/commands/aws/iam.ts +397 -0
  97. package/src/commands/aws/index.ts +133 -0
  98. package/src/commands/aws/lambda.ts +396 -0
  99. package/src/commands/aws/rds.ts +439 -0
  100. package/src/commands/aws/s3.ts +439 -0
  101. package/src/commands/aws/vpc.ts +393 -0
  102. package/src/commands/aws-discover.ts +649 -0
  103. package/src/commands/aws-terraform.ts +805 -0
  104. package/src/commands/azure/aks.ts +376 -0
  105. package/src/commands/azure/functions.ts +253 -0
  106. package/src/commands/azure/index.ts +116 -0
  107. package/src/commands/azure/storage.ts +478 -0
  108. package/src/commands/azure/vm.ts +355 -0
  109. package/src/commands/billing/index.ts +256 -0
  110. package/src/commands/chat.ts +314 -0
  111. package/src/commands/config.ts +346 -0
  112. package/src/commands/cost/cloud-cost-estimator.ts +266 -0
  113. package/src/commands/cost/estimator.ts +79 -0
  114. package/src/commands/cost/index.ts +594 -0
  115. package/src/commands/cost/parsers/terraform.ts +273 -0
  116. package/src/commands/cost/parsers/types.ts +25 -0
  117. package/src/commands/cost/pricing/aws.ts +544 -0
  118. package/src/commands/cost/pricing/azure.ts +499 -0
  119. package/src/commands/cost/pricing/gcp.ts +396 -0
  120. package/src/commands/cost/pricing/index.ts +40 -0
  121. package/src/commands/demo.ts +250 -0
  122. package/src/commands/doctor.ts +794 -0
  123. package/src/commands/drift/index.ts +439 -0
  124. package/src/commands/explain.ts +277 -0
  125. package/src/commands/feedback.ts +389 -0
  126. package/src/commands/fix.ts +324 -0
  127. package/src/commands/fs/index.ts +402 -0
  128. package/src/commands/gcp/compute.ts +325 -0
  129. package/src/commands/gcp/functions.ts +271 -0
  130. package/src/commands/gcp/gke.ts +438 -0
  131. package/src/commands/gcp/iam.ts +344 -0
  132. package/src/commands/gcp/index.ts +129 -0
  133. package/src/commands/gcp/storage.ts +284 -0
  134. package/src/commands/generate-helm.ts +1249 -0
  135. package/src/commands/generate-k8s.ts +1560 -0
  136. package/src/commands/generate-terraform.ts +1460 -0
  137. package/src/commands/gh/index.ts +863 -0
  138. package/src/commands/git/index.ts +1343 -0
  139. package/src/commands/helm/index.ts +1126 -0
  140. package/src/commands/help.ts +539 -0
  141. package/src/commands/history.ts +142 -0
  142. package/src/commands/import.ts +868 -0
  143. package/src/commands/index.ts +367 -0
  144. package/src/commands/init.ts +1046 -0
  145. package/src/commands/k8s/index.ts +1137 -0
  146. package/src/commands/login.ts +631 -0
  147. package/src/commands/logout.ts +83 -0
  148. package/src/commands/onboarding.ts +228 -0
  149. package/src/commands/plan/display.ts +279 -0
  150. package/src/commands/plan/index.ts +599 -0
  151. package/src/commands/preview.ts +452 -0
  152. package/src/commands/questionnaire.ts +1270 -0
  153. package/src/commands/resume.ts +55 -0
  154. package/src/commands/team/index.ts +346 -0
  155. package/src/commands/template.ts +232 -0
  156. package/src/commands/tf/index.ts +1034 -0
  157. package/src/commands/upgrade.ts +550 -0
  158. package/src/commands/usage/index.ts +134 -0
  159. package/src/commands/version.ts +170 -0
  160. package/src/compat/index.ts +2 -0
  161. package/src/compat/runtime.ts +12 -0
  162. package/src/compat/sqlite.ts +107 -0
  163. package/src/config/index.ts +17 -0
  164. package/src/config/manager.ts +530 -0
  165. package/src/config/safety-policy.ts +358 -0
  166. package/src/config/schema.ts +125 -0
  167. package/src/config/types.ts +527 -0
  168. package/src/context/context-db.ts +199 -0
  169. package/src/demo/index.ts +349 -0
  170. package/src/demo/scenarios/full-journey.ts +229 -0
  171. package/src/demo/scenarios/getting-started.ts +127 -0
  172. package/src/demo/scenarios/helm-release.ts +341 -0
  173. package/src/demo/scenarios/k8s-deployment.ts +194 -0
  174. package/src/demo/scenarios/terraform-vpc.ts +170 -0
  175. package/src/demo/types.ts +92 -0
  176. package/src/engine/cost-estimator.ts +438 -0
  177. package/src/engine/diagram-generator.ts +256 -0
  178. package/src/engine/drift-detector.ts +902 -0
  179. package/src/engine/executor.ts +1035 -0
  180. package/src/engine/index.ts +76 -0
  181. package/src/engine/orchestrator.ts +636 -0
  182. package/src/engine/planner.ts +720 -0
  183. package/src/engine/safety.ts +743 -0
  184. package/src/engine/verifier.ts +770 -0
  185. package/src/enterprise/audit.ts +348 -0
  186. package/src/enterprise/auth.ts +270 -0
  187. package/src/enterprise/billing.ts +822 -0
  188. package/src/enterprise/index.ts +17 -0
  189. package/src/enterprise/teams.ts +443 -0
  190. package/src/generator/best-practices.ts +1608 -0
  191. package/src/generator/helm.ts +630 -0
  192. package/src/generator/index.ts +37 -0
  193. package/src/generator/intent-parser.ts +514 -0
  194. package/src/generator/kubernetes.ts +976 -0
  195. package/src/generator/terraform.ts +1867 -0
  196. package/src/history/index.ts +8 -0
  197. package/src/history/manager.ts +322 -0
  198. package/src/history/types.ts +34 -0
  199. package/src/hooks/config.ts +432 -0
  200. package/src/hooks/engine.ts +391 -0
  201. package/src/hooks/index.ts +4 -0
  202. package/src/llm/auth-bridge.ts +198 -0
  203. package/src/llm/circuit-breaker.ts +140 -0
  204. package/src/llm/config-loader.ts +201 -0
  205. package/src/llm/cost-calculator.ts +171 -0
  206. package/src/llm/index.ts +8 -0
  207. package/src/llm/model-aliases.ts +115 -0
  208. package/src/llm/provider-registry.ts +63 -0
  209. package/src/llm/providers/anthropic.ts +433 -0
  210. package/src/llm/providers/bedrock.ts +477 -0
  211. package/src/llm/providers/google.ts +405 -0
  212. package/src/llm/providers/ollama.ts +767 -0
  213. package/src/llm/providers/openai-compatible.ts +340 -0
  214. package/src/llm/providers/openai.ts +328 -0
  215. package/src/llm/providers/openrouter.ts +338 -0
  216. package/src/llm/router.ts +1035 -0
  217. package/src/llm/types.ts +232 -0
  218. package/src/lsp/client.ts +298 -0
  219. package/src/lsp/languages.ts +116 -0
  220. package/src/lsp/manager.ts +278 -0
  221. package/src/mcp/client.ts +402 -0
  222. package/src/mcp/index.ts +5 -0
  223. package/src/mcp/manager.ts +133 -0
  224. package/src/nimbus.ts +214 -0
  225. package/src/plugins/index.ts +27 -0
  226. package/src/plugins/loader.ts +334 -0
  227. package/src/plugins/manager.ts +376 -0
  228. package/src/plugins/types.ts +284 -0
  229. package/src/scanners/cicd-scanner.ts +258 -0
  230. package/src/scanners/cloud-scanner.ts +466 -0
  231. package/src/scanners/framework-scanner.ts +469 -0
  232. package/src/scanners/iac-scanner.ts +388 -0
  233. package/src/scanners/index.ts +539 -0
  234. package/src/scanners/language-scanner.ts +276 -0
  235. package/src/scanners/package-manager-scanner.ts +277 -0
  236. package/src/scanners/types.ts +172 -0
  237. package/src/sessions/manager.ts +365 -0
  238. package/src/sessions/types.ts +44 -0
  239. package/src/sharing/sync.ts +296 -0
  240. package/src/sharing/viewer.ts +97 -0
  241. package/src/snapshots/index.ts +2 -0
  242. package/src/snapshots/manager.ts +530 -0
  243. package/src/state/artifacts.ts +147 -0
  244. package/src/state/audit.ts +137 -0
  245. package/src/state/billing.ts +240 -0
  246. package/src/state/checkpoints.ts +117 -0
  247. package/src/state/config.ts +67 -0
  248. package/src/state/conversations.ts +14 -0
  249. package/src/state/credentials.ts +154 -0
  250. package/src/state/db.ts +58 -0
  251. package/src/state/index.ts +26 -0
  252. package/src/state/messages.ts +115 -0
  253. package/src/state/projects.ts +123 -0
  254. package/src/state/schema.ts +236 -0
  255. package/src/state/sessions.ts +147 -0
  256. package/src/state/teams.ts +200 -0
  257. package/src/telemetry.ts +108 -0
  258. package/src/tools/aws-ops.ts +952 -0
  259. package/src/tools/azure-ops.ts +579 -0
  260. package/src/tools/file-ops.ts +593 -0
  261. package/src/tools/gcp-ops.ts +625 -0
  262. package/src/tools/git-ops.ts +773 -0
  263. package/src/tools/github-ops.ts +799 -0
  264. package/src/tools/helm-ops.ts +943 -0
  265. package/src/tools/index.ts +17 -0
  266. package/src/tools/k8s-ops.ts +819 -0
  267. package/src/tools/schemas/converter.ts +184 -0
  268. package/src/tools/schemas/devops.ts +612 -0
  269. package/src/tools/schemas/index.ts +73 -0
  270. package/src/tools/schemas/standard.ts +1144 -0
  271. package/src/tools/schemas/types.ts +705 -0
  272. package/src/tools/terraform-ops.ts +862 -0
  273. package/src/types/ambient.d.ts +193 -0
  274. package/src/types/config.ts +83 -0
  275. package/src/types/drift.ts +116 -0
  276. package/src/types/enterprise.ts +335 -0
  277. package/src/types/index.ts +20 -0
  278. package/src/types/plan.ts +44 -0
  279. package/src/types/request.ts +65 -0
  280. package/src/types/response.ts +54 -0
  281. package/src/types/service.ts +51 -0
  282. package/src/ui/App.tsx +997 -0
  283. package/src/ui/DeployPreview.tsx +169 -0
  284. package/src/ui/Header.tsx +68 -0
  285. package/src/ui/InputBox.tsx +350 -0
  286. package/src/ui/MessageList.tsx +585 -0
  287. package/src/ui/PermissionPrompt.tsx +151 -0
  288. package/src/ui/StatusBar.tsx +158 -0
  289. package/src/ui/ToolCallDisplay.tsx +409 -0
  290. package/src/ui/chat-ui.ts +853 -0
  291. package/src/ui/index.ts +33 -0
  292. package/src/ui/ink/index.ts +711 -0
  293. package/src/ui/streaming.ts +176 -0
  294. package/src/ui/types.ts +57 -0
  295. package/src/utils/analytics.ts +72 -0
  296. package/src/utils/cost-warning.ts +27 -0
  297. package/src/utils/env.ts +46 -0
  298. package/src/utils/errors.ts +69 -0
  299. package/src/utils/event-bus.ts +38 -0
  300. package/src/utils/index.ts +24 -0
  301. package/src/utils/logger.ts +171 -0
  302. package/src/utils/rate-limiter.ts +121 -0
  303. package/src/utils/service-auth.ts +49 -0
  304. package/src/utils/validation.ts +53 -0
  305. package/src/version.ts +4 -0
  306. package/src/watcher/index.ts +163 -0
  307. package/src/wizard/approval.ts +383 -0
  308. package/src/wizard/index.ts +25 -0
  309. package/src/wizard/prompts.ts +338 -0
  310. package/src/wizard/types.ts +171 -0
  311. package/src/wizard/ui.ts +556 -0
  312. package/src/wizard/wizard.ts +304 -0
  313. package/tsconfig.json +24 -0
@@ -0,0 +1,544 @@
1
+ /**
2
+ * AWS Static Pricing Lookup
3
+ *
4
+ * Monthly on-demand pricing based on us-east-1 as of 2024.
5
+ * These are approximate list prices used for quick estimation.
6
+ * Install Infracost for real-time, region-aware pricing.
7
+ */
8
+
9
+ import type { TerraformResource } from '../parsers/types';
10
+ import type { PricingResult } from './index';
11
+
12
+ // Hours in a month (AWS standard)
13
+ const HOURS_PER_MONTH = 730;
14
+
15
+ // ------------------------------------------------------------------
16
+ // EC2 instance pricing (on-demand, us-east-1, Linux)
17
+ // ------------------------------------------------------------------
18
+ const EC2_PRICING: Record<string, number> = {
19
+ // T2 burstable
20
+ 't2.nano': 4.18,
21
+ 't2.micro': 8.35,
22
+ 't2.small': 16.79,
23
+ 't2.medium': 33.41,
24
+ 't2.large': 66.82,
25
+ 't2.xlarge': 133.63,
26
+ 't2.2xlarge': 267.26,
27
+ // T3 burstable
28
+ 't3.nano': 3.8,
29
+ 't3.micro': 7.59,
30
+ 't3.small': 15.18,
31
+ 't3.medium': 30.37,
32
+ 't3.large': 60.74,
33
+ 't3.xlarge': 121.47,
34
+ 't3.2xlarge': 242.94,
35
+ // T3a (AMD)
36
+ 't3a.nano': 3.43,
37
+ 't3a.micro': 6.86,
38
+ 't3a.small': 13.72,
39
+ 't3a.medium': 27.45,
40
+ 't3a.large': 54.9,
41
+ 't3a.xlarge': 109.79,
42
+ // M5 general purpose
43
+ 'm5.large': 69.12,
44
+ 'm5.xlarge': 138.24,
45
+ 'm5.2xlarge': 276.48,
46
+ 'm5.4xlarge': 552.96,
47
+ 'm5.8xlarge': 1105.92,
48
+ 'm5.12xlarge': 1658.88,
49
+ // M6i general purpose
50
+ 'm6i.large': 69.12,
51
+ 'm6i.xlarge': 138.24,
52
+ 'm6i.2xlarge': 276.48,
53
+ 'm6i.4xlarge': 552.96,
54
+ // M7i general purpose
55
+ 'm7i.large': 72.56,
56
+ 'm7i.xlarge': 145.12,
57
+ 'm7i.2xlarge': 290.24,
58
+ // C5 compute optimized
59
+ 'c5.large': 61.2,
60
+ 'c5.xlarge': 122.4,
61
+ 'c5.2xlarge': 244.8,
62
+ 'c5.4xlarge': 489.6,
63
+ 'c5.9xlarge': 1101.6,
64
+ // C6i compute optimized
65
+ 'c6i.large': 61.2,
66
+ 'c6i.xlarge': 122.4,
67
+ 'c6i.2xlarge': 244.8,
68
+ // R5 memory optimized
69
+ 'r5.large': 90.72,
70
+ 'r5.xlarge': 181.44,
71
+ 'r5.2xlarge': 362.88,
72
+ 'r5.4xlarge': 725.76,
73
+ // R6i memory optimized
74
+ 'r6i.large': 90.72,
75
+ 'r6i.xlarge': 181.44,
76
+ 'r6i.2xlarge': 362.88,
77
+ // P3 GPU
78
+ 'p3.2xlarge': 2208.6,
79
+ 'p3.8xlarge': 8834.4,
80
+ // G4dn GPU
81
+ 'g4dn.xlarge': 379.58,
82
+ 'g4dn.2xlarge': 543.12,
83
+ };
84
+
85
+ // ------------------------------------------------------------------
86
+ // RDS instance pricing (on-demand, us-east-1, Single-AZ)
87
+ // ------------------------------------------------------------------
88
+ const RDS_PRICING: Record<string, number> = {
89
+ 'db.t3.micro': 12.41,
90
+ 'db.t3.small': 24.82,
91
+ 'db.t3.medium': 49.64,
92
+ 'db.t3.large': 99.28,
93
+ 'db.t3.xlarge': 198.56,
94
+ 'db.t4g.micro': 11.83,
95
+ 'db.t4g.small': 23.65,
96
+ 'db.t4g.medium': 47.3,
97
+ 'db.m5.large': 124.1,
98
+ 'db.m5.xlarge': 248.2,
99
+ 'db.m5.2xlarge': 496.4,
100
+ 'db.m5.4xlarge': 992.8,
101
+ 'db.m6i.large': 124.1,
102
+ 'db.m6i.xlarge': 248.2,
103
+ 'db.r5.large': 172.8,
104
+ 'db.r5.xlarge': 345.6,
105
+ 'db.r5.2xlarge': 691.2,
106
+ 'db.r6i.large': 172.8,
107
+ 'db.r6i.xlarge': 345.6,
108
+ };
109
+
110
+ // ------------------------------------------------------------------
111
+ // EBS volume pricing (per GB/month, us-east-1)
112
+ // ------------------------------------------------------------------
113
+ const EBS_PRICING: Record<string, number> = {
114
+ gp2: 0.1,
115
+ gp3: 0.08,
116
+ io1: 0.125,
117
+ io2: 0.125,
118
+ st1: 0.045,
119
+ sc1: 0.015,
120
+ standard: 0.05,
121
+ };
122
+
123
+ // ------------------------------------------------------------------
124
+ // ElastiCache node pricing (on-demand, us-east-1)
125
+ // ------------------------------------------------------------------
126
+ const ELASTICACHE_PRICING: Record<string, number> = {
127
+ 'cache.t3.micro': 12.24,
128
+ 'cache.t3.small': 24.48,
129
+ 'cache.t3.medium': 49.06,
130
+ 'cache.t4g.micro': 11.52,
131
+ 'cache.t4g.small': 23.04,
132
+ 'cache.t4g.medium': 46.08,
133
+ 'cache.m5.large': 124.1,
134
+ 'cache.m5.xlarge': 248.2,
135
+ 'cache.r5.large': 163.52,
136
+ 'cache.r5.xlarge': 327.04,
137
+ };
138
+
139
+ /**
140
+ * Look up the estimated monthly price for an AWS Terraform resource.
141
+ */
142
+ export function getAWSPrice(resource: TerraformResource): PricingResult | null {
143
+ const { type, attributes } = resource;
144
+
145
+ switch (type) {
146
+ // ----- Compute -----
147
+ case 'aws_instance': {
148
+ const instanceType = attributes.instance_type || 't3.medium';
149
+ const price = EC2_PRICING[instanceType];
150
+ if (!price) {
151
+ return {
152
+ monthlyCost: 30.37,
153
+ hourlyCost: 30.37 / HOURS_PER_MONTH,
154
+ description: `EC2 ${instanceType} (estimated, type not in lookup table)`,
155
+ };
156
+ }
157
+ return {
158
+ monthlyCost: price,
159
+ hourlyCost: price / HOURS_PER_MONTH,
160
+ unit: 'hours',
161
+ description: `EC2 ${instanceType}`,
162
+ };
163
+ }
164
+
165
+ case 'aws_launch_template':
166
+ case 'aws_launch_configuration': {
167
+ const instanceType = attributes.instance_type || 't3.medium';
168
+ const price = EC2_PRICING[instanceType];
169
+ if (!price) {
170
+ return {
171
+ monthlyCost: 0,
172
+ hourlyCost: 0,
173
+ description: `Launch template ${instanceType} (cost depends on ASG)`,
174
+ };
175
+ }
176
+ return {
177
+ monthlyCost: 0,
178
+ hourlyCost: 0,
179
+ description: `Launch template ${instanceType} (cost depends on ASG)`,
180
+ };
181
+ }
182
+
183
+ case 'aws_autoscaling_group': {
184
+ const min = attributes.min_size || 1;
185
+ const max = attributes.max_size || min;
186
+ const desired = attributes.desired_capacity || min;
187
+ // Estimate based on desired capacity with a default t3.medium
188
+ const perInstance = 30.37;
189
+ return {
190
+ monthlyCost: desired * perInstance,
191
+ hourlyCost: (desired * perInstance) / HOURS_PER_MONTH,
192
+ quantity: desired,
193
+ unit: 'instances',
194
+ description: `ASG (${min}-${max}, desired ${desired}) estimated at t3.medium`,
195
+ };
196
+ }
197
+
198
+ // ----- Database -----
199
+ case 'aws_db_instance': {
200
+ const instanceClass = attributes.instance_class || 'db.t3.medium';
201
+ const price = RDS_PRICING[instanceClass] || 49.64;
202
+ const storageGB = attributes.allocated_storage || 20;
203
+ const storageType = attributes.storage_type || 'gp2';
204
+ const storageRate = EBS_PRICING[storageType] || 0.115;
205
+ const storageCost = storageGB * storageRate;
206
+ const multiAz = attributes.multi_az === true ? 2 : 1;
207
+ const totalCompute = price * multiAz;
208
+ return {
209
+ monthlyCost: totalCompute + storageCost,
210
+ hourlyCost: totalCompute / HOURS_PER_MONTH,
211
+ description: `RDS ${instanceClass}${multiAz > 1 ? ' Multi-AZ' : ''} + ${storageGB}GB ${storageType}`,
212
+ };
213
+ }
214
+
215
+ case 'aws_rds_cluster': {
216
+ // Aurora cluster (control plane + writer instance estimated)
217
+ return {
218
+ monthlyCost: 210.24,
219
+ hourlyCost: 210.24 / HOURS_PER_MONTH,
220
+ description: 'Aurora cluster (estimated writer instance)',
221
+ };
222
+ }
223
+
224
+ case 'aws_rds_cluster_instance': {
225
+ const instanceClass = attributes.instance_class || 'db.r5.large';
226
+ const price = RDS_PRICING[instanceClass] || 172.8;
227
+ return {
228
+ monthlyCost: price,
229
+ hourlyCost: price / HOURS_PER_MONTH,
230
+ description: `Aurora instance ${instanceClass}`,
231
+ };
232
+ }
233
+
234
+ // ----- Storage -----
235
+ case 'aws_s3_bucket': {
236
+ // S3 pricing is purely usage-based; estimate 100GB standard as a baseline
237
+ return {
238
+ monthlyCost: 2.3,
239
+ hourlyCost: 0,
240
+ unit: 'GB',
241
+ description: 'S3 Standard (estimated 100GB baseline)',
242
+ };
243
+ }
244
+
245
+ case 'aws_ebs_volume': {
246
+ const volumeType = attributes.type || 'gp3';
247
+ const size = attributes.size || 20;
248
+ const pricePerGB = EBS_PRICING[volumeType] || 0.08;
249
+ const iops = attributes.iops || 0;
250
+ let iopsCost = 0;
251
+ if (volumeType === 'io1' || volumeType === 'io2') {
252
+ iopsCost = iops * 0.065; // per provisioned IOPS/month
253
+ }
254
+ return {
255
+ monthlyCost: size * pricePerGB + iopsCost,
256
+ hourlyCost: 0,
257
+ quantity: size,
258
+ unit: 'GB',
259
+ description: `EBS ${volumeType} ${size}GB${iops ? ` ${iops} IOPS` : ''}`,
260
+ };
261
+ }
262
+
263
+ case 'aws_efs_file_system': {
264
+ // EFS standard: ~$0.30/GB, estimate 50GB
265
+ return {
266
+ monthlyCost: 15.0,
267
+ hourlyCost: 0,
268
+ unit: 'GB',
269
+ description: 'EFS Standard (estimated 50GB)',
270
+ };
271
+ }
272
+
273
+ // ----- Networking -----
274
+ case 'aws_lb':
275
+ case 'aws_alb': {
276
+ // ALB: ~$0.0225/hr fixed + ~$0.008/LCU-hr
277
+ const fixedCost = 0.0225 * HOURS_PER_MONTH; // ~$16.43
278
+ const lcuEstimate = 5.84; // Estimated LCU charges
279
+ return {
280
+ monthlyCost: fixedCost + lcuEstimate,
281
+ hourlyCost: 0.0225,
282
+ description: 'Application Load Balancer (fixed + estimated LCU)',
283
+ };
284
+ }
285
+
286
+ case 'aws_lb_target_group':
287
+ case 'aws_alb_target_group': {
288
+ return { monthlyCost: 0, hourlyCost: 0, description: 'ALB target group (no direct cost)' };
289
+ }
290
+
291
+ case 'aws_nat_gateway': {
292
+ // NAT GW: $0.045/hr + $0.045/GB processed
293
+ const fixedCost = 0.045 * HOURS_PER_MONTH; // ~$32.85
294
+ const dataEstimate = 32.85; // Estimated ~730GB data processing
295
+ return {
296
+ monthlyCost: fixedCost + dataEstimate,
297
+ hourlyCost: 0.045,
298
+ description: 'NAT Gateway (fixed + estimated data processing)',
299
+ };
300
+ }
301
+
302
+ case 'aws_eip': {
303
+ // EIPs are free when attached to a running instance; $0.005/hr when idle
304
+ return {
305
+ monthlyCost: 3.65,
306
+ hourlyCost: 0.005,
307
+ description: 'Elastic IP (cost if unattached)',
308
+ };
309
+ }
310
+
311
+ case 'aws_cloudfront_distribution': {
312
+ // CloudFront pricing is usage-based; provide a baseline
313
+ return {
314
+ monthlyCost: 10.0,
315
+ hourlyCost: 0,
316
+ description: 'CloudFront (estimated baseline, usage-based)',
317
+ };
318
+ }
319
+
320
+ // ----- Containers -----
321
+ case 'aws_eks_cluster': {
322
+ return {
323
+ monthlyCost: 73.0,
324
+ hourlyCost: 0.1,
325
+ description: 'EKS cluster control plane',
326
+ };
327
+ }
328
+
329
+ case 'aws_ecs_cluster': {
330
+ // ECS cluster itself is free; costs come from tasks/services
331
+ return {
332
+ monthlyCost: 0,
333
+ hourlyCost: 0,
334
+ description: 'ECS cluster (no direct cost, tasks billed separately)',
335
+ };
336
+ }
337
+
338
+ case 'aws_ecs_service':
339
+ case 'aws_ecs_task_definition': {
340
+ return {
341
+ monthlyCost: 0,
342
+ hourlyCost: 0,
343
+ description: 'ECS service/task (cost depends on launch type and resources)',
344
+ };
345
+ }
346
+
347
+ case 'aws_ecr_repository': {
348
+ // ~$0.10/GB storage
349
+ return {
350
+ monthlyCost: 1.0,
351
+ hourlyCost: 0,
352
+ description: 'ECR repository (estimated 10GB images)',
353
+ };
354
+ }
355
+
356
+ // ----- Serverless -----
357
+ case 'aws_lambda_function': {
358
+ return {
359
+ monthlyCost: 0,
360
+ hourlyCost: 0,
361
+ description: 'Lambda (usage-based, $0 at rest)',
362
+ };
363
+ }
364
+
365
+ case 'aws_api_gateway_rest_api':
366
+ case 'aws_apigatewayv2_api': {
367
+ return {
368
+ monthlyCost: 0,
369
+ hourlyCost: 0,
370
+ description: 'API Gateway (usage-based, $0 at rest)',
371
+ };
372
+ }
373
+
374
+ case 'aws_sqs_queue': {
375
+ return {
376
+ monthlyCost: 0,
377
+ hourlyCost: 0,
378
+ description: 'SQS queue (usage-based, first 1M requests free)',
379
+ };
380
+ }
381
+
382
+ case 'aws_sns_topic': {
383
+ return {
384
+ monthlyCost: 0,
385
+ hourlyCost: 0,
386
+ description: 'SNS topic (usage-based)',
387
+ };
388
+ }
389
+
390
+ case 'aws_dynamodb_table': {
391
+ const billingMode = attributes.billing_mode || 'PROVISIONED';
392
+ if (billingMode === 'PAY_PER_REQUEST') {
393
+ return {
394
+ monthlyCost: 0,
395
+ hourlyCost: 0,
396
+ description: 'DynamoDB on-demand (usage-based)',
397
+ };
398
+ }
399
+ // Provisioned: estimate from read/write capacity
400
+ const rcu = attributes.read_capacity || 5;
401
+ const wcu = attributes.write_capacity || 5;
402
+ const rcuCost = rcu * 0.00013 * HOURS_PER_MONTH; // ~$0.09/RCU/month
403
+ const wcuCost = wcu * 0.00065 * HOURS_PER_MONTH; // ~$0.47/WCU/month
404
+ return {
405
+ monthlyCost: rcuCost + wcuCost,
406
+ hourlyCost: (rcuCost + wcuCost) / HOURS_PER_MONTH,
407
+ description: `DynamoDB provisioned (${rcu} RCU, ${wcu} WCU)`,
408
+ };
409
+ }
410
+
411
+ // ----- Caching -----
412
+ case 'aws_elasticache_cluster': {
413
+ const nodeType = attributes.node_type || 'cache.t3.medium';
414
+ const numNodes = attributes.num_cache_nodes || 1;
415
+ const price = ELASTICACHE_PRICING[nodeType] || 49.06;
416
+ return {
417
+ monthlyCost: price * numNodes,
418
+ hourlyCost: (price * numNodes) / HOURS_PER_MONTH,
419
+ quantity: numNodes,
420
+ unit: 'nodes',
421
+ description: `ElastiCache ${nodeType} x${numNodes}`,
422
+ };
423
+ }
424
+
425
+ case 'aws_elasticache_replication_group': {
426
+ const nodeType = attributes.node_type || 'cache.t3.medium';
427
+ const numReplicas = attributes.number_cache_clusters || attributes.num_cache_clusters || 2;
428
+ const price = ELASTICACHE_PRICING[nodeType] || 49.06;
429
+ return {
430
+ monthlyCost: price * numReplicas,
431
+ hourlyCost: (price * numReplicas) / HOURS_PER_MONTH,
432
+ quantity: numReplicas,
433
+ unit: 'nodes',
434
+ description: `ElastiCache replication group ${nodeType} x${numReplicas}`,
435
+ };
436
+ }
437
+
438
+ // ----- Monitoring / Logging -----
439
+ case 'aws_cloudwatch_log_group': {
440
+ return {
441
+ monthlyCost: 0,
442
+ hourlyCost: 0,
443
+ description: 'CloudWatch Logs (ingestion/storage usage-based)',
444
+ };
445
+ }
446
+
447
+ case 'aws_cloudwatch_metric_alarm': {
448
+ // Standard resolution: $0.10/alarm/month
449
+ return {
450
+ monthlyCost: 0.1,
451
+ hourlyCost: 0,
452
+ description: 'CloudWatch alarm',
453
+ };
454
+ }
455
+
456
+ // ----- Security / Identity (no direct cost) -----
457
+ case 'aws_vpc':
458
+ case 'aws_subnet':
459
+ case 'aws_route_table':
460
+ case 'aws_route_table_association':
461
+ case 'aws_route':
462
+ case 'aws_internet_gateway':
463
+ case 'aws_security_group':
464
+ case 'aws_security_group_rule':
465
+ case 'aws_network_acl':
466
+ case 'aws_iam_role':
467
+ case 'aws_iam_policy':
468
+ case 'aws_iam_policy_attachment':
469
+ case 'aws_iam_role_policy_attachment':
470
+ case 'aws_iam_instance_profile':
471
+ case 'aws_iam_user':
472
+ case 'aws_iam_group':
473
+ case 'aws_kms_key':
474
+ case 'aws_kms_alias':
475
+ case 'aws_ssm_parameter':
476
+ case 'aws_secretsmanager_secret':
477
+ case 'aws_acm_certificate':
478
+ case 'aws_route53_zone':
479
+ case 'aws_route53_record':
480
+ case 'aws_waf_web_acl':
481
+ case 'aws_wafv2_web_acl': {
482
+ return {
483
+ monthlyCost: 0,
484
+ hourlyCost: 0,
485
+ description: 'No direct cost',
486
+ };
487
+ }
488
+
489
+ // ----- Data Transfer / VPN -----
490
+ case 'aws_vpn_gateway': {
491
+ return {
492
+ monthlyCost: 36.5,
493
+ hourlyCost: 0.05,
494
+ description: 'VPN Gateway',
495
+ };
496
+ }
497
+
498
+ case 'aws_customer_gateway': {
499
+ return { monthlyCost: 0, hourlyCost: 0, description: 'Customer gateway (no direct cost)' };
500
+ }
501
+
502
+ // ----- Elasticsearch / OpenSearch -----
503
+ case 'aws_elasticsearch_domain':
504
+ case 'aws_opensearch_domain': {
505
+ const instanceType = attributes.instance_type || 't3.small.search';
506
+ return {
507
+ monthlyCost: 26.28,
508
+ hourlyCost: 0.036,
509
+ description: `OpenSearch ${instanceType} (estimated)`,
510
+ };
511
+ }
512
+
513
+ // ----- Kinesis -----
514
+ case 'aws_kinesis_stream': {
515
+ const shardCount = attributes.shard_count || 1;
516
+ // $0.015/shard-hour = ~$10.95/shard/month
517
+ return {
518
+ monthlyCost: shardCount * 10.95,
519
+ hourlyCost: shardCount * 0.015,
520
+ quantity: shardCount,
521
+ unit: 'shards',
522
+ description: `Kinesis stream (${shardCount} shards)`,
523
+ };
524
+ }
525
+
526
+ // ----- Redshift -----
527
+ case 'aws_redshift_cluster': {
528
+ const nodeType = attributes.node_type || 'dc2.large';
529
+ const numNodes = attributes.number_of_nodes || 1;
530
+ // dc2.large ~$0.25/hr
531
+ const perNode = 182.5;
532
+ return {
533
+ monthlyCost: perNode * numNodes,
534
+ hourlyCost: (perNode * numNodes) / HOURS_PER_MONTH,
535
+ quantity: numNodes,
536
+ unit: 'nodes',
537
+ description: `Redshift ${nodeType} x${numNodes}`,
538
+ };
539
+ }
540
+
541
+ default:
542
+ return null;
543
+ }
544
+ }