@atlascrew/apparatus 0.9.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 (291) hide show
  1. package/bin/apparatus.mjs +2 -0
  2. package/certs/server.crt +17 -0
  3. package/certs/server.key +28 -0
  4. package/dist/ai/client.js +104 -0
  5. package/dist/ai/client.js.map +1 -0
  6. package/dist/ai/personas.js +104 -0
  7. package/dist/ai/personas.js.map +1 -0
  8. package/dist/ai/redteam.js +1404 -0
  9. package/dist/ai/redteam.js.map +1 -0
  10. package/dist/ai/report-store.js +309 -0
  11. package/dist/ai/report-store.js.map +1 -0
  12. package/dist/app.js +525 -0
  13. package/dist/app.js.map +1 -0
  14. package/dist/attack-sim.js +69 -0
  15. package/dist/attack-sim.js.map +1 -0
  16. package/dist/attacker-tracker.js +276 -0
  17. package/dist/attacker-tracker.js.map +1 -0
  18. package/dist/blackhole.js +95 -0
  19. package/dist/blackhole.js.map +1 -0
  20. package/dist/chaos.js +88 -0
  21. package/dist/chaos.js.map +1 -0
  22. package/dist/cluster.js +462 -0
  23. package/dist/cluster.js.map +1 -0
  24. package/dist/config.js +61 -0
  25. package/dist/config.js.map +1 -0
  26. package/dist/deception.js +205 -0
  27. package/dist/deception.js.map +1 -0
  28. package/dist/demo-mode.js +109 -0
  29. package/dist/demo-mode.js.map +1 -0
  30. package/dist/dist-dashboard/assets/index-BsMhEnGu.js +648 -0
  31. package/dist/dist-dashboard/assets/index-CNOkYC_Q.css +10 -0
  32. package/dist/dist-dashboard/assets/index-CW2grvPC.js +648 -0
  33. package/dist/dist-dashboard/assets/logo/apparatus-favicon.svg +15 -0
  34. package/dist/dist-dashboard/assets/logo/apparatus-icon-dark.svg +24 -0
  35. package/dist/dist-dashboard/assets/logo/apparatus-icon-light.svg +24 -0
  36. package/dist/dist-dashboard/assets/logo/apparatus-logo-512.png +0 -0
  37. package/dist/dist-dashboard/assets/logo/apparatus-logo-dark.svg +18 -0
  38. package/dist/dist-dashboard/assets/logo/apparatus-logo.svg +17 -0
  39. package/dist/dist-dashboard/assets/logo/apple-touch-icon.png +0 -0
  40. package/dist/dist-dashboard/assets/logo/favicon-192.png +0 -0
  41. package/dist/dist-dashboard/assets/logo/favicon-32.png +0 -0
  42. package/dist/dist-dashboard/assets/logo/favicon.ico +0 -0
  43. package/dist/dist-dashboard/assets/logo/icon-192.png +0 -0
  44. package/dist/dist-dashboard/assets/logo/icon-512.png +0 -0
  45. package/dist/dist-dashboard/assets/logo/icon-light-512.png +0 -0
  46. package/dist/dist-dashboard/assets/react-vendor-DpRMSntD.js +1 -0
  47. package/dist/dist-dashboard/assets/router-DSc5pRwN.js +59 -0
  48. package/dist/dist-dashboard/docs-index.json +1577 -0
  49. package/dist/dist-dashboard/index.html +21 -0
  50. package/dist/dlp.js +40 -0
  51. package/dist/dlp.js.map +1 -0
  52. package/dist/drills.js +770 -0
  53. package/dist/drills.js.map +1 -0
  54. package/dist/echoHandler.js +113 -0
  55. package/dist/echoHandler.js.map +1 -0
  56. package/dist/escape/index.js +225 -0
  57. package/dist/escape/index.js.map +1 -0
  58. package/dist/escape/methods/dns.js +74 -0
  59. package/dist/escape/methods/dns.js.map +1 -0
  60. package/dist/escape/methods/http.js +81 -0
  61. package/dist/escape/methods/http.js.map +1 -0
  62. package/dist/escape/methods/icmp.js +36 -0
  63. package/dist/escape/methods/icmp.js.map +1 -0
  64. package/dist/escape/methods/tcp.js +38 -0
  65. package/dist/escape/methods/tcp.js.map +1 -0
  66. package/dist/escape/methods/udp.js +27 -0
  67. package/dist/escape/methods/udp.js.map +1 -0
  68. package/dist/escape/methods/websocket.js +37 -0
  69. package/dist/escape/methods/websocket.js.map +1 -0
  70. package/dist/forensics.js +111 -0
  71. package/dist/forensics.js.map +1 -0
  72. package/dist/generator.js +67 -0
  73. package/dist/generator.js.map +1 -0
  74. package/dist/ghosting.js +414 -0
  75. package/dist/ghosting.js.map +1 -0
  76. package/dist/graphql.js +44 -0
  77. package/dist/graphql.js.map +1 -0
  78. package/dist/history.js +40 -0
  79. package/dist/history.js.map +1 -0
  80. package/dist/imposter/creds.js +16 -0
  81. package/dist/imposter/creds.js.map +1 -0
  82. package/dist/imposter/index.js +44 -0
  83. package/dist/imposter/index.js.map +1 -0
  84. package/dist/imposter/providers/aws.js +103 -0
  85. package/dist/imposter/providers/aws.js.map +1 -0
  86. package/dist/imposter/providers/gcp.js +26 -0
  87. package/dist/imposter/providers/gcp.js.map +1 -0
  88. package/dist/index.js +53 -0
  89. package/dist/index.js.map +1 -0
  90. package/dist/infra-debug.js +68 -0
  91. package/dist/infra-debug.js.map +1 -0
  92. package/dist/jwt-debug.js +272 -0
  93. package/dist/jwt-debug.js.map +1 -0
  94. package/dist/kv.js +22 -0
  95. package/dist/kv.js.map +1 -0
  96. package/dist/lib/generators.js +43 -0
  97. package/dist/lib/generators.js.map +1 -0
  98. package/dist/lib/json.js +26 -0
  99. package/dist/lib/json.js.map +1 -0
  100. package/dist/logger.js +9 -0
  101. package/dist/logger.js.map +1 -0
  102. package/dist/metrics.js +20 -0
  103. package/dist/metrics.js.map +1 -0
  104. package/dist/mtd.js +30 -0
  105. package/dist/mtd.js.map +1 -0
  106. package/dist/oidc.js +69 -0
  107. package/dist/oidc.js.map +1 -0
  108. package/dist/persistence/cluster-state.js +47 -0
  109. package/dist/persistence/cluster-state.js.map +1 -0
  110. package/dist/persistence/deception-history.js +65 -0
  111. package/dist/persistence/deception-history.js.map +1 -0
  112. package/dist/persistence/drill-runs.js +138 -0
  113. package/dist/persistence/drill-runs.js.map +1 -0
  114. package/dist/persistence/request-history.js +41 -0
  115. package/dist/persistence/request-history.js.map +1 -0
  116. package/dist/persistence/scenario-catalog.js +73 -0
  117. package/dist/persistence/scenario-catalog.js.map +1 -0
  118. package/dist/persistence/status.js +51 -0
  119. package/dist/persistence/status.js.map +1 -0
  120. package/dist/persistence/tarpit-state.js +47 -0
  121. package/dist/persistence/tarpit-state.js.map +1 -0
  122. package/dist/persistence/webhook-store.js +69 -0
  123. package/dist/persistence/webhook-store.js.map +1 -0
  124. package/dist/proxy.js +28 -0
  125. package/dist/proxy.js.map +1 -0
  126. package/dist/ratelimit.js +32 -0
  127. package/dist/ratelimit.js.map +1 -0
  128. package/dist/redteam.js +442 -0
  129. package/dist/redteam.js.map +1 -0
  130. package/dist/scenarios.js +229 -0
  131. package/dist/scenarios.js.map +1 -0
  132. package/dist/scripting.js +30 -0
  133. package/dist/scripting.js.map +1 -0
  134. package/dist/self-healing.js +42 -0
  135. package/dist/self-healing.js.map +1 -0
  136. package/dist/sentinel.js +50 -0
  137. package/dist/sentinel.js.map +1 -0
  138. package/dist/server-bad-ssl.js +47 -0
  139. package/dist/server-bad-ssl.js.map +1 -0
  140. package/dist/server-grpc.js +66 -0
  141. package/dist/server-grpc.js.map +1 -0
  142. package/dist/server-http1.js +5 -0
  143. package/dist/server-http1.js.map +1 -0
  144. package/dist/server-http2.js +27 -0
  145. package/dist/server-http2.js.map +1 -0
  146. package/dist/server-icap.js +46 -0
  147. package/dist/server-icap.js.map +1 -0
  148. package/dist/server-l4.js +30 -0
  149. package/dist/server-l4.js.map +1 -0
  150. package/dist/server-mqtt.js +29 -0
  151. package/dist/server-mqtt.js.map +1 -0
  152. package/dist/server-protocols.js +18 -0
  153. package/dist/server-protocols.js.map +1 -0
  154. package/dist/server-redis.js +112 -0
  155. package/dist/server-redis.js.map +1 -0
  156. package/dist/server-smtp.js +66 -0
  157. package/dist/server-smtp.js.map +1 -0
  158. package/dist/server-syslog.js +23 -0
  159. package/dist/server-syslog.js.map +1 -0
  160. package/dist/server-ws.js +18 -0
  161. package/dist/server-ws.js.map +1 -0
  162. package/dist/sidecar/chaos/engine.js +41 -0
  163. package/dist/sidecar/chaos/engine.js.map +1 -0
  164. package/dist/sidecar/index.js +98 -0
  165. package/dist/sidecar/index.js.map +1 -0
  166. package/dist/simulator/dependency-graph.js +102 -0
  167. package/dist/simulator/dependency-graph.js.map +1 -0
  168. package/dist/simulator/supply-chain.js +67 -0
  169. package/dist/simulator/supply-chain.js.map +1 -0
  170. package/dist/sink.js +24 -0
  171. package/dist/sink.js.map +1 -0
  172. package/dist/sse-broadcast.js +105 -0
  173. package/dist/sse-broadcast.js.map +1 -0
  174. package/dist/swagger.js +309 -0
  175. package/dist/swagger.js.map +1 -0
  176. package/dist/sysinfo.js +36 -0
  177. package/dist/sysinfo.js.map +1 -0
  178. package/dist/tarpit.js +126 -0
  179. package/dist/tarpit.js.map +1 -0
  180. package/dist/tool-executor.js +315 -0
  181. package/dist/tool-executor.js.map +1 -0
  182. package/dist/tui/api-client.js +341 -0
  183. package/dist/tui/api-client.js.map +1 -0
  184. package/dist/tui/core/action-handler.js +302 -0
  185. package/dist/tui/core/action-handler.js.map +1 -0
  186. package/dist/tui/core/index.js +18 -0
  187. package/dist/tui/core/index.js.map +1 -0
  188. package/dist/tui/core/keyboard.js +329 -0
  189. package/dist/tui/core/keyboard.js.map +1 -0
  190. package/dist/tui/core/modal.js +397 -0
  191. package/dist/tui/core/modal.js.map +1 -0
  192. package/dist/tui/core/screen-manager.js +262 -0
  193. package/dist/tui/core/screen-manager.js.map +1 -0
  194. package/dist/tui/core/store.js +254 -0
  195. package/dist/tui/core/store.js.map +1 -0
  196. package/dist/tui/core/widget.js +167 -0
  197. package/dist/tui/core/widget.js.map +1 -0
  198. package/dist/tui/dashboard.js +649 -0
  199. package/dist/tui/dashboard.js.map +1 -0
  200. package/dist/tui/index.js +118 -0
  201. package/dist/tui/index.js.map +1 -0
  202. package/dist/tui/modals/add-rule-modal.js +190 -0
  203. package/dist/tui/modals/add-rule-modal.js.map +1 -0
  204. package/dist/tui/modals/dlp-output-modal.js +102 -0
  205. package/dist/tui/modals/dlp-output-modal.js.map +1 -0
  206. package/dist/tui/modals/dns-form-modal.js +26 -0
  207. package/dist/tui/modals/dns-form-modal.js.map +1 -0
  208. package/dist/tui/modals/ghost-config-modal.js +35 -0
  209. package/dist/tui/modals/ghost-config-modal.js.map +1 -0
  210. package/dist/tui/modals/har-results-modal.js +41 -0
  211. package/dist/tui/modals/har-results-modal.js.map +1 -0
  212. package/dist/tui/modals/index.js +15 -0
  213. package/dist/tui/modals/index.js.map +1 -0
  214. package/dist/tui/modals/jwt-decode-modal.js +45 -0
  215. package/dist/tui/modals/jwt-decode-modal.js.map +1 -0
  216. package/dist/tui/modals/jwt-mint-modal.js +70 -0
  217. package/dist/tui/modals/jwt-mint-modal.js.map +1 -0
  218. package/dist/tui/modals/ping-form-modal.js +19 -0
  219. package/dist/tui/modals/ping-form-modal.js.map +1 -0
  220. package/dist/tui/modals/redteam-results-modal.js +43 -0
  221. package/dist/tui/modals/redteam-results-modal.js.map +1 -0
  222. package/dist/tui/modals/scan-form-modal.js +26 -0
  223. package/dist/tui/modals/scan-form-modal.js.map +1 -0
  224. package/dist/tui/screens/defense-screen.js +281 -0
  225. package/dist/tui/screens/defense-screen.js.map +1 -0
  226. package/dist/tui/screens/forensics-screen.js +81 -0
  227. package/dist/tui/screens/forensics-screen.js.map +1 -0
  228. package/dist/tui/screens/index.js +140 -0
  229. package/dist/tui/screens/index.js.map +1 -0
  230. package/dist/tui/screens/system-screen.js +81 -0
  231. package/dist/tui/screens/system-screen.js.map +1 -0
  232. package/dist/tui/screens/testing-screen.js +429 -0
  233. package/dist/tui/screens/testing-screen.js.map +1 -0
  234. package/dist/tui/screens/traffic-screen.js +76 -0
  235. package/dist/tui/screens/traffic-screen.js.map +1 -0
  236. package/dist/tui/sse-client.js +130 -0
  237. package/dist/tui/sse-client.js.map +1 -0
  238. package/dist/tui/state/metrics-buffer.js +195 -0
  239. package/dist/tui/state/metrics-buffer.js.map +1 -0
  240. package/dist/tui/state/metrics-buffer.test.js +102 -0
  241. package/dist/tui/state/metrics-buffer.test.js.map +1 -0
  242. package/dist/tui/theme.js +136 -0
  243. package/dist/tui/theme.js.map +1 -0
  244. package/dist/tui/types.js +6 -0
  245. package/dist/tui/types.js.map +1 -0
  246. package/dist/tui/widgets/chaos-widget.js +152 -0
  247. package/dist/tui/widgets/chaos-widget.js.map +1 -0
  248. package/dist/tui/widgets/cluster-widget.js +156 -0
  249. package/dist/tui/widgets/cluster-widget.js.map +1 -0
  250. package/dist/tui/widgets/dlp-widget.js +161 -0
  251. package/dist/tui/widgets/dlp-widget.js.map +1 -0
  252. package/dist/tui/widgets/ghost-widget.js +169 -0
  253. package/dist/tui/widgets/ghost-widget.js.map +1 -0
  254. package/dist/tui/widgets/har-widget.js +173 -0
  255. package/dist/tui/widgets/har-widget.js.map +1 -0
  256. package/dist/tui/widgets/index.js +122 -0
  257. package/dist/tui/widgets/index.js.map +1 -0
  258. package/dist/tui/widgets/jwt-widget.js +177 -0
  259. package/dist/tui/widgets/jwt-widget.js.map +1 -0
  260. package/dist/tui/widgets/kv-widget.js +261 -0
  261. package/dist/tui/widgets/kv-widget.js.map +1 -0
  262. package/dist/tui/widgets/mtd-widget.js +181 -0
  263. package/dist/tui/widgets/mtd-widget.js.map +1 -0
  264. package/dist/tui/widgets/netdiag-widget.js +155 -0
  265. package/dist/tui/widgets/netdiag-widget.js.map +1 -0
  266. package/dist/tui/widgets/oidc-widget.js +162 -0
  267. package/dist/tui/widgets/oidc-widget.js.map +1 -0
  268. package/dist/tui/widgets/pcap-widget.js +239 -0
  269. package/dist/tui/widgets/pcap-widget.js.map +1 -0
  270. package/dist/tui/widgets/redteam-widget.js +155 -0
  271. package/dist/tui/widgets/redteam-widget.js.map +1 -0
  272. package/dist/tui/widgets/rps-gauge-widget.js +124 -0
  273. package/dist/tui/widgets/rps-gauge-widget.js.map +1 -0
  274. package/dist/tui/widgets/sentinel-widget.js +171 -0
  275. package/dist/tui/widgets/sentinel-widget.js.map +1 -0
  276. package/dist/tui/widgets/sparklines-widget.js +127 -0
  277. package/dist/tui/widgets/sparklines-widget.js.map +1 -0
  278. package/dist/tui/widgets/sysinfo-widget.js +197 -0
  279. package/dist/tui/widgets/sysinfo-widget.js.map +1 -0
  280. package/dist/tui/widgets/traffic-chart-widget.js +170 -0
  281. package/dist/tui/widgets/traffic-chart-widget.js.map +1 -0
  282. package/dist/tui/widgets/webhook-widget.js +259 -0
  283. package/dist/tui/widgets/webhook-widget.js.map +1 -0
  284. package/dist/utils/ip.js +18 -0
  285. package/dist/utils/ip.js.map +1 -0
  286. package/dist/victim/index.js +71 -0
  287. package/dist/victim/index.js.map +1 -0
  288. package/dist/webhook.js +88 -0
  289. package/dist/webhook.js.map +1 -0
  290. package/package.json +90 -0
  291. package/proto/echo.proto +19 -0
@@ -0,0 +1,414 @@
1
+ import net from "net";
2
+ import { request } from "undici";
3
+ import { logger } from "./logger.js";
4
+ import { cfg } from "./config.js";
5
+ let ghostInterval = null;
6
+ const USER_AGENTS = [
7
+ "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.114 Safari/537.36",
8
+ "Mozilla/5.0 (Windows NT 10.0; Win64; x64) Gecko/20100101 Firefox/89.0",
9
+ "Mozilla/5.0 (iPhone; CPU iPhone OS 14_6 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/14.1.1 Mobile/15E148 Safari/604.1"
10
+ ];
11
+ const PATHS = [
12
+ "/echo",
13
+ "/healthz",
14
+ "/docs",
15
+ "/",
16
+ "/history"
17
+ ];
18
+ const RESERVED_GHOST_PREFIXES = [
19
+ "/ghosts",
20
+ "/healthz",
21
+ "/health",
22
+ "/metrics",
23
+ "/sse",
24
+ "/docs",
25
+ "/_sensor",
26
+ "/cluster",
27
+ "/auth",
28
+ "/.well-known",
29
+ "/debug",
30
+ "/redteam",
31
+ "/sentinel",
32
+ "/proxy",
33
+ "/mtd",
34
+ "/scenarios",
35
+ "/drills",
36
+ "/api/redteam",
37
+ "/api/simulator",
38
+ "/victim",
39
+ "/tarpit",
40
+ "/deception",
41
+ "/capture.pcap",
42
+ "/replay",
43
+ "/kv",
44
+ "/script",
45
+ "/dashboard",
46
+ "/graphql",
47
+ "/sink",
48
+ "/generate",
49
+ "/dlp",
50
+ "/drain",
51
+ "/upload",
52
+ "/download",
53
+ "/history",
54
+ "/ratelimit",
55
+ "/chaos",
56
+ "/malicious",
57
+ "/dns",
58
+ "/ping",
59
+ "/hooks",
60
+ "/echo",
61
+ "/ws",
62
+ ];
63
+ const GHOST_METHODS = new Set(["GET", "POST", "PUT", "PATCH", "DELETE"]);
64
+ const MAX_INFLIGHT_GHOST_REQUESTS = 200;
65
+ const MAX_VIRTUAL_GHOSTS = 500;
66
+ const configuredGhostPorts = (process.env.GHOST_ALLOWED_PORTS || "")
67
+ .split(",")
68
+ .map((value) => Number(value.trim()))
69
+ .filter((value) => Number.isInteger(value) && value > 0 && value <= 65535);
70
+ const ALLOWED_GHOST_TARGET_PORTS = new Set([
71
+ cfg.portHttp1,
72
+ cfg.portHttp2,
73
+ cfg.portHttp1 + 1, // h2c sidecar port used by Apparatus when HTTP/2 cleartext is enabled
74
+ ...configuredGhostPorts,
75
+ ]);
76
+ const virtualGhosts = new Map();
77
+ let inflightGhostRequests = 0;
78
+ function normalizeRoutePath(route) {
79
+ const trimmed = route.trim();
80
+ if (!trimmed) {
81
+ throw new Error("Route is required");
82
+ }
83
+ if (!trimmed.startsWith("/")) {
84
+ throw new Error("Route must start with /");
85
+ }
86
+ if (trimmed.includes("://")) {
87
+ throw new Error("Route must be a path, not a full URL");
88
+ }
89
+ const pathOnly = trimmed.split("?")[0] || "/";
90
+ const normalized = pathOnly.length > 1 && pathOnly.endsWith("/")
91
+ ? pathOnly.slice(0, -1)
92
+ : pathOnly;
93
+ if (RESERVED_GHOST_PREFIXES.some((prefix) => normalized === prefix || normalized.startsWith(`${prefix}/`))) {
94
+ throw new Error("Route is reserved for ghost management APIs");
95
+ }
96
+ return normalized;
97
+ }
98
+ function normalizeGhostMethod(method) {
99
+ if (typeof method !== "string") {
100
+ throw new Error("Method is required");
101
+ }
102
+ const normalized = method.toUpperCase();
103
+ if (!GHOST_METHODS.has(normalized)) {
104
+ throw new Error(`Unsupported method: ${method}`);
105
+ }
106
+ return normalized;
107
+ }
108
+ function clampInteger(value, min, max, field) {
109
+ const parsed = Number(value);
110
+ if (!Number.isFinite(parsed)) {
111
+ throw new Error(`Invalid ${field}`);
112
+ }
113
+ return Math.max(min, Math.min(max, Math.trunc(parsed)));
114
+ }
115
+ function parseGhostBehavior(raw) {
116
+ const errorRateRaw = raw?.errorRate ?? 0;
117
+ const errorRate = Number.isFinite(Number(errorRateRaw))
118
+ ? Math.max(0, Math.min(100, Number(errorRateRaw)))
119
+ : 0;
120
+ const latencyRaw = raw?.latency;
121
+ if (latencyRaw && typeof latencyRaw === "object") {
122
+ const latency = latencyRaw;
123
+ if (latency.mode === "jitter") {
124
+ const minMs = clampInteger(latency.minMs, 0, 30000, "latency.minMs");
125
+ const maxMs = clampInteger(latency.maxMs, 0, 30000, "latency.maxMs");
126
+ if (maxMs < minMs) {
127
+ throw new Error("latency.maxMs must be >= latency.minMs");
128
+ }
129
+ return {
130
+ errorRate,
131
+ latency: {
132
+ mode: "jitter",
133
+ minMs,
134
+ maxMs,
135
+ },
136
+ };
137
+ }
138
+ const ms = clampInteger(latency.ms, 0, 30000, "latency.ms");
139
+ return {
140
+ errorRate,
141
+ latency: {
142
+ mode: "fixed",
143
+ ms,
144
+ },
145
+ };
146
+ }
147
+ if (raw?.jitterMs) {
148
+ const minMs = clampInteger(raw.jitterMs.min, 0, 30000, "jitterMs.min");
149
+ const maxMs = clampInteger(raw.jitterMs.max, 0, 30000, "jitterMs.max");
150
+ if (maxMs < minMs) {
151
+ throw new Error("jitterMs.max must be >= jitterMs.min");
152
+ }
153
+ return {
154
+ errorRate,
155
+ latency: {
156
+ mode: "jitter",
157
+ minMs,
158
+ maxMs,
159
+ },
160
+ };
161
+ }
162
+ const fixedMs = raw?.latencyMs === undefined
163
+ ? 0
164
+ : clampInteger(raw.latencyMs, 0, 30000, "latencyMs");
165
+ return {
166
+ errorRate,
167
+ latency: {
168
+ mode: "fixed",
169
+ ms: fixedMs,
170
+ },
171
+ };
172
+ }
173
+ function findGhostByRequest(method, route) {
174
+ for (const ghost of virtualGhosts.values()) {
175
+ if (ghost.method === method && ghost.route === route) {
176
+ return ghost;
177
+ }
178
+ }
179
+ return null;
180
+ }
181
+ function computeGhostLatencyMs(latency) {
182
+ if (latency.mode === "fixed") {
183
+ return latency.ms;
184
+ }
185
+ const span = latency.maxMs - latency.minMs;
186
+ if (span <= 0) {
187
+ return latency.minMs;
188
+ }
189
+ return latency.minMs + Math.floor(Math.random() * (span + 1));
190
+ }
191
+ function isIpv4MappedLoopback(host) {
192
+ if (!host.startsWith("::ffff:")) {
193
+ return false;
194
+ }
195
+ const mapped = host.slice("::ffff:".length);
196
+ if (mapped.startsWith("127.")) {
197
+ return true;
198
+ }
199
+ const hexParts = mapped.split(":");
200
+ if (hexParts.length !== 2 || hexParts.some((part) => !/^[0-9a-f]{1,4}$/i.test(part))) {
201
+ return false;
202
+ }
203
+ const upper = Number.parseInt(hexParts[0] || "0", 16);
204
+ const firstOctet = (upper >> 8) & 0xff;
205
+ return firstOctet === 127;
206
+ }
207
+ function validateGhostTarget(target) {
208
+ let parsed;
209
+ try {
210
+ parsed = new URL(target);
211
+ }
212
+ catch {
213
+ throw new Error("Invalid target URL");
214
+ }
215
+ if (!["http:", "https:"].includes(parsed.protocol)) {
216
+ throw new Error("Ghost target must use http/https");
217
+ }
218
+ const host = parsed.hostname.toLowerCase().replace(/^\[|\]$/g, "");
219
+ const ipVersion = net.isIP(host);
220
+ const isLoopbackIpv4 = ipVersion === 4 && host.startsWith("127.");
221
+ const isLoopbackIpv6 = ipVersion === 6 && (host === "::1" || host === "0:0:0:0:0:0:0:1" || isIpv4MappedLoopback(host));
222
+ const isLocalHost = isLoopbackIpv4 || isLoopbackIpv6;
223
+ if (!isLocalHost) {
224
+ throw new Error("Ghost target must use a loopback IP literal");
225
+ }
226
+ const explicitPort = parsed.port ? Number(parsed.port) : undefined;
227
+ const inferredPort = explicitPort ?? (parsed.protocol === "https:" ? 443 : 80);
228
+ if (!ALLOWED_GHOST_TARGET_PORTS.has(inferredPort)) {
229
+ throw new Error("Ghost target port is not allowed");
230
+ }
231
+ return parsed.toString();
232
+ }
233
+ export function getGhostStatus() {
234
+ return ghostInterval ? "running" : "stopped";
235
+ }
236
+ export function stopGhostTraffic() {
237
+ if (ghostInterval) {
238
+ clearInterval(ghostInterval);
239
+ ghostInterval = null;
240
+ }
241
+ return { status: "stopped" };
242
+ }
243
+ export function startGhostTraffic(options = {}) {
244
+ if (ghostInterval)
245
+ return { status: "already_running" };
246
+ const targetBase = options.target
247
+ ? validateGhostTarget(options.target)
248
+ : `http://127.0.0.1:${cfg.portHttp1}`;
249
+ const parsedDelay = Number(options.delay);
250
+ const delay = Number.isFinite(parsedDelay) ? Math.max(25, Math.min(5000, Math.trunc(parsedDelay))) : 1000;
251
+ logger.info({ target: targetBase, delay }, "Starting Ghost Traffic");
252
+ ghostInterval = setInterval(async () => {
253
+ const path = PATHS[Math.floor(Math.random() * PATHS.length)];
254
+ const ua = USER_AGENTS[Math.floor(Math.random() * USER_AGENTS.length)];
255
+ try {
256
+ const requestOptions = {
257
+ headers: { "User-Agent": ua },
258
+ maxRedirections: 0,
259
+ };
260
+ await request(`${targetBase}${path}`, requestOptions);
261
+ }
262
+ catch (_e) {
263
+ // Ignore errors, ghosts don't care
264
+ }
265
+ }, delay);
266
+ return { status: "started", target: targetBase };
267
+ }
268
+ function listGhosts() {
269
+ return Array.from(virtualGhosts.values());
270
+ }
271
+ export function createVirtualGhost(payload) {
272
+ if (virtualGhosts.size >= MAX_VIRTUAL_GHOSTS) {
273
+ throw new Error("Maximum number of virtual ghosts reached");
274
+ }
275
+ if (typeof payload.route !== "string") {
276
+ throw new Error("route is required");
277
+ }
278
+ const route = normalizeRoutePath(payload.route);
279
+ const method = normalizeGhostMethod(payload.method ?? "GET");
280
+ const duplicate = findGhostByRequest(method, route);
281
+ if (duplicate) {
282
+ throw new Error("A ghost with this method and route already exists");
283
+ }
284
+ const behavior = parseGhostBehavior(payload.behavior);
285
+ const ghost = {
286
+ id: `ghost-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`,
287
+ route,
288
+ method,
289
+ responseBody: payload.responseBody ?? {},
290
+ behavior,
291
+ requestCount: 0,
292
+ createdAt: new Date().toISOString(),
293
+ };
294
+ virtualGhosts.set(ghost.id, ghost);
295
+ logger.info({ ghostId: ghost.id, method: ghost.method, route: ghost.route }, "Created virtual ghost");
296
+ return ghost;
297
+ }
298
+ export function deleteVirtualGhost(id) {
299
+ return virtualGhosts.delete(id);
300
+ }
301
+ export function resetGhostStateForTests() {
302
+ virtualGhosts.clear();
303
+ stopGhostTraffic();
304
+ inflightGhostRequests = 0;
305
+ }
306
+ export async function ghostMockMiddleware(req, res, next) {
307
+ try {
308
+ if (req.path === "/ghosts" || req.path.startsWith("/ghosts/")) {
309
+ return next();
310
+ }
311
+ const method = req.method.toUpperCase();
312
+ const route = req.path.length > 1 && req.path.endsWith("/")
313
+ ? req.path.slice(0, -1)
314
+ : req.path;
315
+ const ghost = findGhostByRequest(method, route);
316
+ if (!ghost) {
317
+ return next();
318
+ }
319
+ if (inflightGhostRequests >= MAX_INFLIGHT_GHOST_REQUESTS) {
320
+ return res.status(503).json({
321
+ error: "Too many concurrent ghost requests",
322
+ limit: MAX_INFLIGHT_GHOST_REQUESTS,
323
+ });
324
+ }
325
+ inflightGhostRequests += 1;
326
+ try {
327
+ ghost.requestCount += 1;
328
+ ghost.lastHitAt = new Date().toISOString();
329
+ const latencyMs = computeGhostLatencyMs(ghost.behavior.latency);
330
+ if (latencyMs > 0) {
331
+ await new Promise((resolve) => setTimeout(resolve, latencyMs));
332
+ }
333
+ if (Math.random() * 100 < ghost.behavior.errorRate) {
334
+ return res.status(500).json({
335
+ error: "Ghost injected failure",
336
+ ghostId: ghost.id,
337
+ route: ghost.route,
338
+ method: ghost.method,
339
+ });
340
+ }
341
+ res.setHeader("x-ghost-id", ghost.id);
342
+ return res.status(200).json(ghost.responseBody);
343
+ }
344
+ finally {
345
+ inflightGhostRequests = Math.max(0, inflightGhostRequests - 1);
346
+ }
347
+ }
348
+ catch (error) {
349
+ return next(error);
350
+ }
351
+ }
352
+ export function ghostHandler(req, res) {
353
+ const action = req.query.action;
354
+ if (action === "start") {
355
+ logger.warn("Deprecated ghost control via GET /ghosts?action=start. Use POST /ghosts/start instead.");
356
+ try {
357
+ const result = startGhostTraffic({
358
+ target: req.query.target,
359
+ delay: req.query.delay ? Number(req.query.delay) : undefined,
360
+ });
361
+ return res.json(result);
362
+ }
363
+ catch (error) {
364
+ return res.status(400).json({ error: error?.message || "Invalid ghost start options" });
365
+ }
366
+ }
367
+ if (action === "stop") {
368
+ logger.warn("Deprecated ghost control via GET /ghosts?action=stop. Use POST /ghosts/stop instead.");
369
+ return res.json(stopGhostTraffic());
370
+ }
371
+ if (action === "status") {
372
+ return res.json({ status: getGhostStatus() });
373
+ }
374
+ return res.json({
375
+ status: getGhostStatus(),
376
+ ghosts: listGhosts(),
377
+ });
378
+ }
379
+ export function ghostCreateHandler(req, res) {
380
+ try {
381
+ const created = createVirtualGhost(req.body);
382
+ return res.status(201).json(created);
383
+ }
384
+ catch (error) {
385
+ return res.status(400).json({ error: error?.message || "Invalid ghost payload" });
386
+ }
387
+ }
388
+ export function ghostDeleteHandler(req, res) {
389
+ const id = req.params.id;
390
+ if (!id) {
391
+ return res.status(400).json({ error: "Ghost id is required" });
392
+ }
393
+ const deleted = deleteVirtualGhost(id);
394
+ if (!deleted) {
395
+ return res.status(404).json({ error: "Ghost not found" });
396
+ }
397
+ return res.json({ status: "deleted", id });
398
+ }
399
+ export function ghostStartHandler(req, res) {
400
+ try {
401
+ const result = startGhostTraffic({
402
+ target: req.body?.target,
403
+ delay: req.body?.delay,
404
+ });
405
+ return res.json(result);
406
+ }
407
+ catch (error) {
408
+ return res.status(400).json({ error: error?.message || "Invalid ghost start options" });
409
+ }
410
+ }
411
+ export function ghostStopHandler(_req, res) {
412
+ return res.json(stopGhostTraffic());
413
+ }
414
+ //# sourceMappingURL=ghosting.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ghosting.js","sourceRoot":"","sources":["../src/ghosting.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,44 @@
1
+ import { createHandler } from 'graphql-http/lib/use/express';
2
+ import { GraphQLSchema, GraphQLObjectType, GraphQLString, GraphQLInt } from 'graphql';
3
+ // Recursive type for testing nested depth
4
+ const EchoType = new GraphQLObjectType({
5
+ name: 'Echo',
6
+ fields: () => ({
7
+ message: { type: GraphQLString },
8
+ depth: { type: GraphQLInt },
9
+ next: {
10
+ type: EchoType,
11
+ resolve: (parent) => {
12
+ return {
13
+ message: parent.message,
14
+ depth: parent.depth + 1
15
+ };
16
+ }
17
+ }
18
+ })
19
+ });
20
+ const RootQuery = new GraphQLObjectType({
21
+ name: 'Query',
22
+ fields: {
23
+ echo: {
24
+ type: EchoType,
25
+ args: {
26
+ message: { type: GraphQLString }
27
+ },
28
+ resolve: (_, args) => ({
29
+ message: args.message || 'Hello GraphQL',
30
+ depth: 1
31
+ })
32
+ },
33
+ // Introspection-heavy field
34
+ complexData: {
35
+ type: GraphQLString,
36
+ resolve: () => "This is some data"
37
+ }
38
+ }
39
+ });
40
+ const schema = new GraphQLSchema({
41
+ query: RootQuery
42
+ });
43
+ export const graphqlHandler = createHandler({ schema });
44
+ //# sourceMappingURL=graphql.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"graphql.js","sourceRoot":"","sources":["../src/graphql.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,8BAA8B,CAAC;AAC7D,OAAO,EAAE,aAAa,EAAE,iBAAiB,EAAE,aAAa,EAAE,UAAU,EAA+B,MAAM,SAAS,CAAC;AAEnH,0CAA0C;AAC1C,MAAM,QAAQ,GAAsB,IAAI,iBAAiB,CAAC;IACtD,IAAI,EAAE,MAAM;IACZ,MAAM,EAAE,GAAG,EAAE,CAAC,CAAC;QACX,OAAO,EAAE,EAAE,IAAI,EAAE,aAAa,EAAE;QAChC,KAAK,EAAE,EAAE,IAAI,EAAE,UAAU,EAAE;QAC3B,IAAI,EAAE;YACF,IAAI,EAAE,QAAQ;YACd,OAAO,EAAE,CAAC,MAAM,EAAE,EAAE;gBAChB,OAAO;oBACH,OAAO,EAAE,MAAM,CAAC,OAAO;oBACvB,KAAK,EAAE,MAAM,CAAC,KAAK,GAAG,CAAC;iBAC1B,CAAC;YACN,CAAC;SACJ;KACJ,CAAC;CACL,CAAC,CAAC;AAEH,MAAM,SAAS,GAAG,IAAI,iBAAiB,CAAC;IACpC,IAAI,EAAE,OAAO;IACb,MAAM,EAAE;QACJ,IAAI,EAAE;YACF,IAAI,EAAE,QAAQ;YACd,IAAI,EAAE;gBACF,OAAO,EAAE,EAAE,IAAI,EAAE,aAAa,EAAE;aACnC;YACD,OAAO,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,EAAE,CAAC,CAAC;gBACnB,OAAO,EAAE,IAAI,CAAC,OAAO,IAAI,eAAe;gBACxC,KAAK,EAAE,CAAC;aACX,CAAC;SACL;QACD,4BAA4B;QAC5B,WAAW,EAAE;YACT,IAAI,EAAE,aAAa;YACnB,OAAO,EAAE,GAAG,EAAE,CAAC,mBAAmB;SACrC;KACJ;CACJ,CAAC,CAAC;AAEH,MAAM,MAAM,GAAG,IAAI,aAAa,CAAC;IAC7B,KAAK,EAAE,SAAS;CACnB,CAAC,CAAC;AAEH,MAAM,CAAC,MAAM,cAAc,GAAG,aAAa,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC"}
@@ -0,0 +1,40 @@
1
+ import { cfg } from "./config.js";
2
+ import { logger } from "./logger.js";
3
+ import { loadRequestHistorySync, writeRequestHistory } from "./persistence/request-history.js";
4
+ import { markPersistenceHydrated, markPersistenceWrite, registerPersistenceStore } from "./persistence/status.js";
5
+ const HISTORY_SIZE = 100;
6
+ const history = [];
7
+ let historyPersistQueue = Promise.resolve(true);
8
+ const HISTORY_STORE_KEY = "requestHistory";
9
+ registerPersistenceStore(HISTORY_STORE_KEY, cfg.requestHistoryPath);
10
+ for (const entry of loadRequestHistorySync(cfg.requestHistoryPath).slice(0, HISTORY_SIZE)) {
11
+ history.push(entry);
12
+ }
13
+ markPersistenceHydrated(HISTORY_STORE_KEY);
14
+ function snapshotHistory() {
15
+ return history.slice(0, HISTORY_SIZE);
16
+ }
17
+ function persistHistoryQueued() {
18
+ historyPersistQueue = historyPersistQueue.then(() => writeRequestHistory(cfg.requestHistoryPath, snapshotHistory()), () => writeRequestHistory(cfg.requestHistoryPath, snapshotHistory()));
19
+ void historyPersistQueue.then((persisted) => {
20
+ markPersistenceWrite(HISTORY_STORE_KEY, persisted);
21
+ if (!persisted) {
22
+ logger.warn("Request history persisted in memory only due to write failure");
23
+ }
24
+ });
25
+ }
26
+ export function addToHistory(record) {
27
+ history.unshift(record);
28
+ if (history.length > HISTORY_SIZE) {
29
+ history.pop();
30
+ }
31
+ persistHistoryQueued();
32
+ }
33
+ export function getHistory() {
34
+ return history;
35
+ }
36
+ export function clearHistory() {
37
+ history.length = 0;
38
+ persistHistoryQueued();
39
+ }
40
+ //# sourceMappingURL=history.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"history.js","sourceRoot":"","sources":["../src/history.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,GAAG,EAAE,MAAM,aAAa,CAAC;AAClC,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AACrC,OAAO,EAAE,sBAAsB,EAAE,mBAAmB,EAAE,MAAM,kCAAkC,CAAC;AAC/F,OAAO,EAAE,uBAAuB,EAAE,oBAAoB,EAAE,wBAAwB,EAAE,MAAM,yBAAyB,CAAC;AAElH,MAAM,YAAY,GAAG,GAAG,CAAC;AACzB,MAAM,OAAO,GAAU,EAAE,CAAC;AAC1B,IAAI,mBAAmB,GAAqB,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;AAClE,MAAM,iBAAiB,GAAG,gBAAgB,CAAC;AAE3C,wBAAwB,CAAC,iBAAiB,EAAE,GAAG,CAAC,kBAAkB,CAAC,CAAC;AAEpE,KAAK,MAAM,KAAK,IAAI,sBAAsB,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,YAAY,CAAC,EAAE,CAAC;IACxF,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;AACxB,CAAC;AACD,uBAAuB,CAAC,iBAAiB,CAAC,CAAC;AAE3C,SAAS,eAAe;IACpB,OAAO,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC;AAC1C,CAAC;AAED,SAAS,oBAAoB;IACzB,mBAAmB,GAAG,mBAAmB,CAAC,IAAI,CAC1C,GAAG,EAAE,CAAC,mBAAmB,CAAC,GAAG,CAAC,kBAAkB,EAAE,eAAe,EAAE,CAAC,EACpE,GAAG,EAAE,CAAC,mBAAmB,CAAC,GAAG,CAAC,kBAAkB,EAAE,eAAe,EAAE,CAAC,CACvE,CAAC;IAEF,KAAK,mBAAmB,CAAC,IAAI,CAAC,CAAC,SAAS,EAAE,EAAE;QACxC,oBAAoB,CAAC,iBAAiB,EAAE,SAAS,CAAC,CAAC;QACnD,IAAI,CAAC,SAAS,EAAE,CAAC;YACb,MAAM,CAAC,IAAI,CAAC,+DAA+D,CAAC,CAAC;QACjF,CAAC;IACL,CAAC,CAAC,CAAC;AACP,CAAC;AAED,MAAM,UAAU,YAAY,CAAC,MAAW;IACpC,OAAO,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;IACxB,IAAI,OAAO,CAAC,MAAM,GAAG,YAAY,EAAE,CAAC;QAChC,OAAO,CAAC,GAAG,EAAE,CAAC;IAClB,CAAC;IACD,oBAAoB,EAAE,CAAC;AAC3B,CAAC;AAED,MAAM,UAAU,UAAU;IACtB,OAAO,OAAO,CAAC;AACnB,CAAC;AAED,MAAM,UAAU,YAAY;IACxB,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC;IACnB,oBAAoB,EAAE,CAAC;AAC3B,CAAC"}
@@ -0,0 +1,16 @@
1
+ import { randomBytes } from "crypto";
2
+ export function generateAwsCreds(roleName = "imposter-role") {
3
+ // Generate realistic-looking fake AWS keys
4
+ const accessKey = "AKIA" + randomBytes(16).toString("hex").toUpperCase().substring(0, 16);
5
+ const secretKey = randomBytes(32).toString("base64");
6
+ const token = randomBytes(128).toString("base64");
7
+ // Expire in 6 hours
8
+ const expiration = new Date(Date.now() + 6 * 60 * 60 * 1000).toISOString();
9
+ return {
10
+ AccessKeyId: accessKey,
11
+ SecretAccessKey: secretKey,
12
+ Token: token,
13
+ Expiration: expiration
14
+ };
15
+ }
16
+ //# sourceMappingURL=creds.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"creds.js","sourceRoot":"","sources":["../../src/imposter/creds.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,QAAQ,CAAC;AASrC,MAAM,UAAU,gBAAgB,CAAC,WAAmB,eAAe;IAC/D,2CAA2C;IAC3C,MAAM,SAAS,GAAG,MAAM,GAAG,WAAW,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,WAAW,EAAE,CAAC,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IAC1F,MAAM,SAAS,GAAG,WAAW,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;IACrD,MAAM,KAAK,GAAG,WAAW,CAAC,GAAG,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;IAElD,oBAAoB;IACpB,MAAM,UAAU,GAAG,IAAI,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC,WAAW,EAAE,CAAC;IAE3E,OAAO;QACH,WAAW,EAAE,SAAS;QACtB,eAAe,EAAE,SAAS;QAC1B,KAAK,EAAE,KAAK;QACZ,UAAU,EAAE,UAAU;KACzB,CAAC;AACN,CAAC"}
@@ -0,0 +1,44 @@
1
+ import express from "express";
2
+ import { logger } from "../logger.js";
3
+ import { awsRouter } from "./providers/aws.js";
4
+ import { gcpRouter } from "./providers/gcp.js";
5
+ import { fileURLToPath } from 'url';
6
+ export const IMPOSTER_PORT = parseInt(process.env.IMPOSTER_PORT || "16925");
7
+ export function createImposterApp() {
8
+ const app = express();
9
+ app.use(express.json());
10
+ // Log all requests
11
+ app.use((req, res, next) => {
12
+ logger.debug({ method: req.method, path: req.path, ip: req.ip }, "Cloud Imposter Request");
13
+ next();
14
+ });
15
+ // AWS Router (Default, root based)
16
+ app.use(awsRouter);
17
+ // GCP Router
18
+ app.use(gcpRouter);
19
+ // Health
20
+ app.get("/health", (req, res) => res.json({ status: "ok", role: "Cloud Imposter" }));
21
+ return app;
22
+ }
23
+ async function main() {
24
+ const app = createImposterApp();
25
+ app.listen(IMPOSTER_PORT, "0.0.0.0", () => {
26
+ console.log(`
27
+ ☁️ Cloud Imposter Active on port ${IMPOSTER_PORT}
28
+ -----------------------------------
29
+ AWS Mock: http://localhost:${IMPOSTER_PORT}/latest/meta-data/
30
+ GCP Mock: http://localhost:${IMPOSTER_PORT}/computeMetadata/v1/
31
+
32
+ 👉 To test an app, set env var:
33
+ AWS_EC2_METADATA_SERVICE_ENDPOINT=http://localhost:${IMPOSTER_PORT}
34
+ GCE_METADATA_HOST=localhost:${IMPOSTER_PORT}
35
+ `);
36
+ });
37
+ }
38
+ if (process.argv[1] === fileURLToPath(import.meta.url)) {
39
+ main().catch(err => {
40
+ console.error(err);
41
+ process.exit(1);
42
+ });
43
+ }
44
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/imposter/index.ts"],"names":[],"mappings":"AAAA,OAAO,OAAO,MAAM,SAAS,CAAC;AAC9B,OAAO,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AACtC,OAAO,EAAE,SAAS,EAAE,MAAM,oBAAoB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,MAAM,oBAAoB,CAAC;AAC/C,OAAO,EAAE,aAAa,EAAE,MAAM,KAAK,CAAC;AAEpC,MAAM,CAAC,MAAM,aAAa,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,aAAa,IAAI,OAAO,CAAC,CAAC;AAE5E,MAAM,UAAU,iBAAiB;IAC7B,MAAM,GAAG,GAAG,OAAO,EAAE,CAAC;IACtB,GAAG,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;IAExB,mBAAmB;IACnB,GAAG,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,IAAI,EAAE,EAAE;QACvB,MAAM,CAAC,KAAK,CAAC,EAAE,MAAM,EAAE,GAAG,CAAC,MAAM,EAAE,IAAI,EAAE,GAAG,CAAC,IAAI,EAAE,EAAE,EAAE,GAAG,CAAC,EAAE,EAAE,EAAE,wBAAwB,CAAC,CAAC;QAC3F,IAAI,EAAE,CAAC;IACX,CAAC,CAAC,CAAC;IAEH,mCAAmC;IACnC,GAAG,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IAEnB,aAAa;IACb,GAAG,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IAEnB,SAAS;IACT,GAAG,CAAC,GAAG,CAAC,SAAS,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,IAAI,EAAE,gBAAgB,EAAE,CAAC,CAAC,CAAC;IAErF,OAAO,GAAG,CAAC;AACf,CAAC;AAED,KAAK,UAAU,IAAI;IACf,MAAM,GAAG,GAAG,iBAAiB,EAAE,CAAC;IAChC,GAAG,CAAC,MAAM,CAAC,aAAa,EAAE,SAAS,EAAE,GAAG,EAAE;QACtC,OAAO,CAAC,GAAG,CAAC;oCACgB,aAAa;;gCAEjB,aAAa;gCACb,aAAa;;;2DAGc,aAAa;oCACpC,aAAa;SACxC,CAAC,CAAC;IACP,CAAC,CAAC,CAAC;AACP,CAAC;AAED,IAAI,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;IACrD,IAAI,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE;QACf,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACnB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACpB,CAAC,CAAC,CAAC;AACP,CAAC"}