@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 @@
1
+ {"version":3,"file":"server-icap.js","sourceRoot":"","sources":["../src/server-icap.ts"],"names":[],"mappings":"AAAA,OAAO,GAAG,MAAM,KAAK,CAAC;AACtB,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AACrC,OAAO,EAAE,GAAG,EAAE,MAAM,aAAa,CAAC;AAElC,MAAM,SAAS,GAAG,uEAAuE,CAAC;AAE1F,MAAM,UAAU,eAAe,CAAC,UAA4C,EAAE;IAC1E,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,IAAI,GAAG,CAAC,QAAQ,CAAC;IAC1C,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,IAAI,GAAG,CAAC,IAAI,CAAC;IACtC,MAAM,MAAM,GAAG,GAAG,CAAC,YAAY,CAAC,CAAC,MAAM,EAAE,EAAE;QACvC,MAAM,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAC;QAE3C,MAAM,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE;YACvB,MAAM,GAAG,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;YAC5B,0EAA0E;YAC1E,8EAA8E;YAE9E,IAAI,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;gBAC1B,sBAAsB;gBACtB,MAAM,QAAQ,GACV,qBAAqB;oBACrB,8BAA8B;oBAC9B,iCAAiC;oBACjC,sBAAsB;oBACtB,MAAM,CAAC;gBACX,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;YAC3B,CAAC;iBAAM,IAAI,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;gBACjC,WAAW;gBACX,MAAM,QAAQ,GACV,qBAAqB;oBACrB,sBAAsB;oBACtB,6EAA6E;oBAC7E,MAAM,CAAC,CAAC,0DAA0D;gBACtE,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;gBACvB,MAAM,CAAC,IAAI,CAAC,+BAA+B,CAAC,CAAC;YACjD,CAAC;iBAAM,CAAC;gBACJ,8CAA8C;gBAC9C,MAAM,QAAQ,GACV,6BAA6B;oBAC7B,sBAAsB;oBACtB,MAAM,CAAC;gBACX,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;YAC3B,CAAC;QACL,CAAC,CAAC,CAAC;IACP,CAAC,CAAC,CAAC;IAEH,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,IAAI,EAAE,GAAG,EAAE;QAC3B,MAAM,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,EAAE,uBAAuB,CAAC,CAAC;IACnD,CAAC,CAAC,CAAC;IAEH,OAAO,MAAM,CAAC;AAClB,CAAC"}
@@ -0,0 +1,30 @@
1
+ import net from "net";
2
+ import dgram from "dgram";
3
+ import { logger } from "./logger.js";
4
+ import { cfg } from "./config.js";
5
+ export function startL4Servers() {
6
+ // TCP Echo Server
7
+ const tcpServer = net.createServer((socket) => {
8
+ logger.info({ remoteAddress: socket.remoteAddress, protocol: "tcp" }, "TCP connection established");
9
+ socket.pipe(socket); // Echo back everything
10
+ socket.on("error", (err) => logger.error({ err, protocol: "tcp" }, "TCP socket error"));
11
+ });
12
+ tcpServer.listen(cfg.portTcp, cfg.host, () => {
13
+ logger.info({ port: cfg.portTcp, protocol: "tcp" }, "TCP Echo server listening");
14
+ });
15
+ // UDP Echo Server
16
+ const udpServer = dgram.createSocket("udp4");
17
+ udpServer.on("message", (msg, rinfo) => {
18
+ logger.debug({ remoteAddress: rinfo.address, protocol: "udp" }, "UDP packet received");
19
+ udpServer.send(msg, rinfo.port, rinfo.address, (err) => {
20
+ if (err)
21
+ logger.error({ err, protocol: "udp" }, "UDP send error");
22
+ });
23
+ });
24
+ udpServer.on("listening", () => {
25
+ const address = udpServer.address();
26
+ logger.info({ port: address.port, protocol: "udp" }, "UDP Echo server listening");
27
+ });
28
+ udpServer.bind(cfg.portUdp, cfg.host);
29
+ }
30
+ //# sourceMappingURL=server-l4.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server-l4.js","sourceRoot":"","sources":["../src/server-l4.ts"],"names":[],"mappings":"AAAA,OAAO,GAAG,MAAM,KAAK,CAAC;AACtB,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AACrC,OAAO,EAAE,GAAG,EAAE,MAAM,aAAa,CAAC;AAElC,MAAM,UAAU,cAAc;IAC1B,kBAAkB;IAClB,MAAM,SAAS,GAAG,GAAG,CAAC,YAAY,CAAC,CAAC,MAAM,EAAE,EAAE;QAC1C,MAAM,CAAC,IAAI,CAAC,EAAE,aAAa,EAAE,MAAM,CAAC,aAAa,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE,4BAA4B,CAAC,CAAC;QACpG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,uBAAuB;QAC5C,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,GAAG,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE,kBAAkB,CAAC,CAAC,CAAC;IAC5F,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,MAAM,CAAC,GAAG,CAAC,OAAO,EAAE,GAAG,CAAC,IAAI,EAAE,GAAG,EAAE;QACzC,MAAM,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,GAAG,CAAC,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE,2BAA2B,CAAC,CAAC;IACrF,CAAC,CAAC,CAAC;IAEH,kBAAkB;IAClB,MAAM,SAAS,GAAG,KAAK,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;IAE7C,SAAS,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,GAAG,EAAE,KAAK,EAAE,EAAE;QACnC,MAAM,CAAC,KAAK,CAAC,EAAE,aAAa,EAAE,KAAK,CAAC,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE,qBAAqB,CAAC,CAAC;QACvF,SAAS,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,EAAE,KAAK,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;YACnD,IAAI,GAAG;gBAAE,MAAM,CAAC,KAAK,CAAC,EAAE,GAAG,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE,gBAAgB,CAAC,CAAC;QACtE,CAAC,CAAC,CAAC;IACP,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,EAAE,CAAC,WAAW,EAAE,GAAG,EAAE;QAC3B,MAAM,OAAO,GAAG,SAAS,CAAC,OAAO,EAAE,CAAC;QACpC,MAAM,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE,2BAA2B,CAAC,CAAC;IACtF,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,GAAG,CAAC,IAAI,CAAC,CAAC;AAC1C,CAAC"}
@@ -0,0 +1,29 @@
1
+ import Aedes from "aedes";
2
+ import { createServer } from "net";
3
+ import { logger } from "./logger.js";
4
+ import { cfg } from "./config.js";
5
+ export function startMqttServer() {
6
+ const aedes = new Aedes();
7
+ const server = createServer(aedes.handle);
8
+ const port = cfg.portMqtt;
9
+ server.listen(port, cfg.host, () => {
10
+ logger.info({ port }, "MQTT Broker listening");
11
+ });
12
+ aedes.on("client", (client) => {
13
+ logger.info({ clientId: client.id }, "MQTT Client connected");
14
+ });
15
+ aedes.on("clientDisconnect", (client) => {
16
+ logger.info({ clientId: client.id }, "MQTT Client disconnected");
17
+ });
18
+ aedes.on("publish", (packet, client) => {
19
+ if (client) {
20
+ logger.info({
21
+ topic: packet.topic,
22
+ payload: packet.payload.toString(),
23
+ clientId: client.id
24
+ }, "MQTT Message published");
25
+ }
26
+ });
27
+ return server;
28
+ }
29
+ //# sourceMappingURL=server-mqtt.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server-mqtt.js","sourceRoot":"","sources":["../src/server-mqtt.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAE,YAAY,EAAE,MAAM,KAAK,CAAC;AACnC,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AACrC,OAAO,EAAE,GAAG,EAAE,MAAM,aAAa,CAAC;AAElC,MAAM,UAAU,eAAe;IAC3B,MAAM,KAAK,GAAG,IAAI,KAAK,EAAE,CAAC;IAC1B,MAAM,MAAM,GAAG,YAAY,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;IAC1C,MAAM,IAAI,GAAG,GAAG,CAAC,QAAQ,CAAC;IAE1B,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,GAAG,CAAC,IAAI,EAAE,GAAG,EAAE;QAC/B,MAAM,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,EAAE,uBAAuB,CAAC,CAAC;IACnD,CAAC,CAAC,CAAC;IAEH,KAAK,CAAC,EAAE,CAAC,QAAQ,EAAE,CAAC,MAAM,EAAE,EAAE;QAC1B,MAAM,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,MAAM,CAAC,EAAE,EAAE,EAAE,uBAAuB,CAAC,CAAC;IAClE,CAAC,CAAC,CAAC;IAEH,KAAK,CAAC,EAAE,CAAC,kBAAkB,EAAE,CAAC,MAAM,EAAE,EAAE;QACpC,MAAM,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,MAAM,CAAC,EAAE,EAAE,EAAE,0BAA0B,CAAC,CAAC;IACrE,CAAC,CAAC,CAAC;IAEH,KAAK,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,MAAM,EAAE,MAAM,EAAE,EAAE;QACnC,IAAI,MAAM,EAAE,CAAC;YACT,MAAM,CAAC,IAAI,CAAC;gBACR,KAAK,EAAE,MAAM,CAAC,KAAK;gBACnB,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,QAAQ,EAAE;gBAClC,QAAQ,EAAE,MAAM,CAAC,EAAE;aACtB,EAAE,wBAAwB,CAAC,CAAC;QACjC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,OAAO,MAAM,CAAC;AAClB,CAAC"}
@@ -0,0 +1,18 @@
1
+ import dgram from "dgram";
2
+ import { logger } from "./logger.js";
3
+ import { cfg } from "./config.js";
4
+ // Note: Redis and SMTP have dedicated servers in server-redis.ts and server-smtp.ts
5
+ export function startProtocolServers() {
6
+ const servers = [];
7
+ // --- Syslog Mock (configurable UDP port, default 5140) ---
8
+ const syslogServer = dgram.createSocket("udp4");
9
+ syslogServer.bind(cfg.portSyslogAlt, cfg.host);
10
+ logger.info({ port: cfg.portSyslogAlt, protocol: "udp" }, "Syslog Alt server listening");
11
+ servers.push(syslogServer);
12
+ return {
13
+ close: () => {
14
+ syslogServer.close();
15
+ }
16
+ };
17
+ }
18
+ //# sourceMappingURL=server-protocols.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server-protocols.js","sourceRoot":"","sources":["../src/server-protocols.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AACrC,OAAO,EAAE,GAAG,EAAE,MAAM,aAAa,CAAC;AAElC,oFAAoF;AACpF,MAAM,UAAU,oBAAoB;IAChC,MAAM,OAAO,GAAU,EAAE,CAAC;IAE1B,4DAA4D;IAC5D,MAAM,YAAY,GAAG,KAAK,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;IAChD,YAAY,CAAC,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,GAAG,CAAC,IAAI,CAAC,CAAC;IAC/C,MAAM,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,GAAG,CAAC,aAAa,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE,6BAA6B,CAAC,CAAC;IACzF,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IAE3B,OAAO;QACH,KAAK,EAAE,GAAG,EAAE;YACR,YAAY,CAAC,KAAK,EAAE,CAAC;QACzB,CAAC;KACJ,CAAC;AACN,CAAC"}
@@ -0,0 +1,112 @@
1
+ import net from "net";
2
+ import { logger } from "./logger.js";
3
+ import { cfg } from "./config.js";
4
+ const store = new Map();
5
+ const MAX_BUFFER_SIZE = 64 * 1024; // 64KB - prevent DoS via unbounded buffer
6
+ function parseResp(buffer) {
7
+ const str = buffer.toString();
8
+ // Very basic RESP parser, assumes simple arrays of bulk strings
9
+ // *2\r\n$4\r\nLLEN\r\n$6\r\nmylist\r\n
10
+ if (!str.startsWith("*"))
11
+ return str.trim().split(/\s+/); // Fallback to inline command
12
+ const lines = str.split("\r\n");
13
+ const args = [];
14
+ for (let i = 0; i < lines.length; i++) {
15
+ if (lines[i].startsWith("$")) {
16
+ // Next line is data
17
+ if (i + 1 < lines.length) {
18
+ args.push(lines[i + 1]);
19
+ i++;
20
+ }
21
+ }
22
+ }
23
+ return args;
24
+ }
25
+ function encodeBulkString(str) {
26
+ if (str === null)
27
+ return "$-1\r\n";
28
+ return `$${str.length}\r\n${str}\r\n`;
29
+ }
30
+ function encodeSimpleString(str) {
31
+ return `+${str}\r\n`;
32
+ }
33
+ function encodeError(str) {
34
+ return `-${str}\r\n`;
35
+ }
36
+ function encodeInteger(num) {
37
+ return `:${num}\r\n`;
38
+ }
39
+ export function startRedisServer(options = {}) {
40
+ const port = options.port ?? cfg.portRedis;
41
+ const host = options.host ?? cfg.host;
42
+ const server = net.createServer((socket) => {
43
+ const addr = `${socket.remoteAddress}:${socket.remotePort}`;
44
+ logger.info({ client: addr }, "Redis client connected");
45
+ let buffer = "";
46
+ socket.on("data", (data) => {
47
+ // DoS protection: limit buffer size before concatenation
48
+ if (buffer.length + data.length > MAX_BUFFER_SIZE) {
49
+ logger.warn({ client: addr }, "Redis buffer overflow - disconnecting client");
50
+ socket.write(encodeError("ERR buffer overflow"));
51
+ socket.destroy();
52
+ return;
53
+ }
54
+ buffer += data.toString();
55
+ // Process complete commands (ending with \r\n)
56
+ if (!buffer.includes("\r\n"))
57
+ return;
58
+ const args = parseResp(Buffer.from(buffer));
59
+ buffer = ""; // Clear buffer after parsing
60
+ if (args.length === 0)
61
+ return;
62
+ const cmd = args[0].toUpperCase();
63
+ logger.debug({ cmd, args }, "Redis command received");
64
+ try {
65
+ switch (cmd) {
66
+ case "PING":
67
+ socket.write(encodeSimpleString(args[1] || "PONG"));
68
+ break;
69
+ case "SET":
70
+ if (args.length < 3) {
71
+ socket.write(encodeError("ERR wrong number of arguments for 'set' command"));
72
+ }
73
+ else {
74
+ store.set(args[1], args[2]);
75
+ socket.write(encodeSimpleString("OK"));
76
+ }
77
+ break;
78
+ case "GET":
79
+ if (args.length < 2) {
80
+ socket.write(encodeError("ERR wrong number of arguments for 'get' command"));
81
+ }
82
+ else {
83
+ const val = store.get(args[1]) || null;
84
+ socket.write(encodeBulkString(val));
85
+ }
86
+ break;
87
+ case "INFO":
88
+ socket.write(encodeBulkString("# Server\r\nredis_version:0.0.1\r\napparatus_mock:1\r\n"));
89
+ break;
90
+ case "QUIT":
91
+ socket.write(encodeSimpleString("OK"));
92
+ socket.end();
93
+ break;
94
+ default:
95
+ socket.write(encodeSimpleString("OK")); // Fake success for unknown commands to keep clients happy
96
+ break;
97
+ }
98
+ }
99
+ catch (e) {
100
+ socket.write(encodeError(`ERR ${e.message}`));
101
+ }
102
+ });
103
+ socket.on("error", (err) => {
104
+ logger.error({ err, client: addr }, "Redis client error");
105
+ });
106
+ });
107
+ server.listen(port, host, () => {
108
+ logger.info({ port }, "Redis Mock Server listening");
109
+ });
110
+ return server;
111
+ }
112
+ //# sourceMappingURL=server-redis.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server-redis.js","sourceRoot":"","sources":["../src/server-redis.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,66 @@
1
+ import net from "net";
2
+ import { logger } from "./logger.js";
3
+ import { cfg } from "./config.js";
4
+ export function startSmtpServer(options = {}) {
5
+ const port = options.port ?? cfg.portSmtp; // 25 is usually privileged
6
+ const host = options.host ?? cfg.host;
7
+ const server = net.createServer((socket) => {
8
+ const addr = `${socket.remoteAddress}:${socket.remotePort}`;
9
+ logger.info({ client: addr }, "SMTP client connected");
10
+ let state = "CONNECT";
11
+ let dataBuffer = "";
12
+ // Send greeting
13
+ socket.write("220 apparatus-smtp ESMTP Postfix\r\n");
14
+ socket.on("data", (chunk) => {
15
+ const data = chunk.toString();
16
+ if (state === "DATA") {
17
+ dataBuffer += data;
18
+ if (dataBuffer.includes("\r\n.\r\n")) {
19
+ logger.info({ size: dataBuffer.length }, "SMTP Email received");
20
+ socket.write("250 2.0.0 Ok: queued as 12345\r\n");
21
+ state = "COMMAND";
22
+ dataBuffer = "";
23
+ }
24
+ return;
25
+ }
26
+ const lines = data.split("\r\n").filter(l => l);
27
+ for (const line of lines) {
28
+ const upper = line.toUpperCase();
29
+ logger.debug({ line }, "SMTP command");
30
+ if (upper.startsWith("HELO") || upper.startsWith("EHLO")) {
31
+ socket.write("250-apparatus-smtp\r\n250-PIPELINING\r\n250-SIZE 10240000\r\n250-VRFY\r\n250-ETRN\r\n250-ENHANCEDSTATUSCODES\r\n250-8BITMIME\r\n250 DSN\r\n");
32
+ }
33
+ else if (upper.startsWith("MAIL FROM:")) {
34
+ socket.write("250 2.1.0 Ok\r\n");
35
+ }
36
+ else if (upper.startsWith("RCPT TO:")) {
37
+ socket.write("250 2.1.5 Ok\r\n");
38
+ }
39
+ else if (upper.startsWith("DATA")) {
40
+ state = "DATA";
41
+ socket.write("354 End data with <CR><LF>.<CR><LF>\r\n");
42
+ }
43
+ else if (upper.startsWith("QUIT")) {
44
+ socket.write("221 2.0.0 Bye\r\n");
45
+ socket.end();
46
+ }
47
+ else if (upper.startsWith("RSET")) {
48
+ state = "COMMAND";
49
+ socket.write("250 2.0.0 Ok\r\n");
50
+ }
51
+ else {
52
+ // Generic success for other commands to avoid breaking flow
53
+ socket.write("250 2.0.0 Ok\r\n");
54
+ }
55
+ }
56
+ });
57
+ socket.on("error", (err) => {
58
+ logger.error({ err, client: addr }, "SMTP client error");
59
+ });
60
+ });
61
+ server.listen(port, host, () => {
62
+ logger.info({ port }, "SMTP Sink Server listening");
63
+ });
64
+ return server;
65
+ }
66
+ //# sourceMappingURL=server-smtp.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server-smtp.js","sourceRoot":"","sources":["../src/server-smtp.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,23 @@
1
+ import dgram from "dgram";
2
+ import { logger } from "./logger.js";
3
+ import { cfg } from "./config.js";
4
+ export function startSyslogServer() {
5
+ const port = cfg.portSyslog; // 514 is privileged
6
+ const server = dgram.createSocket("udp4");
7
+ server.on("error", (err) => {
8
+ logger.error({ err }, "Syslog server error");
9
+ server.close();
10
+ });
11
+ server.on("message", (msg, rinfo) => {
12
+ logger.info({
13
+ from: `${rinfo.address}:${rinfo.port}`,
14
+ msg: msg.toString()
15
+ }, "Syslog message received");
16
+ });
17
+ server.on("listening", () => {
18
+ const address = server.address();
19
+ logger.info({ port: address.port, protocol: "udp" }, "Syslog Receiver listening");
20
+ });
21
+ server.bind(port);
22
+ }
23
+ //# sourceMappingURL=server-syslog.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server-syslog.js","sourceRoot":"","sources":["../src/server-syslog.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AACrC,OAAO,EAAE,GAAG,EAAE,MAAM,aAAa,CAAC;AAElC,MAAM,UAAU,iBAAiB;IAC7B,MAAM,IAAI,GAAG,GAAG,CAAC,UAAU,CAAC,CAAC,oBAAoB;IACjD,MAAM,MAAM,GAAG,KAAK,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;IAE1C,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;QACvB,MAAM,CAAC,KAAK,CAAC,EAAE,GAAG,EAAE,EAAE,qBAAqB,CAAC,CAAC;QAC7C,MAAM,CAAC,KAAK,EAAE,CAAC;IACnB,CAAC,CAAC,CAAC;IAEH,MAAM,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,GAAG,EAAE,KAAK,EAAE,EAAE;QAChC,MAAM,CAAC,IAAI,CAAC;YACR,IAAI,EAAE,GAAG,KAAK,CAAC,OAAO,IAAI,KAAK,CAAC,IAAI,EAAE;YACtC,GAAG,EAAE,GAAG,CAAC,QAAQ,EAAE;SACtB,EAAE,yBAAyB,CAAC,CAAC;IAClC,CAAC,CAAC,CAAC;IAEH,MAAM,CAAC,EAAE,CAAC,WAAW,EAAE,GAAG,EAAE;QACxB,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,EAAE,CAAC;QACjC,MAAM,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE,2BAA2B,CAAC,CAAC;IACtF,CAAC,CAAC,CAAC;IAEH,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AACtB,CAAC"}
@@ -0,0 +1,18 @@
1
+ import { WebSocketServer } from "ws";
2
+ import { logger } from "./logger.js";
3
+ export function setupWebSocket(server) {
4
+ const wss = new WebSocketServer({ server, path: "/ws" });
5
+ wss.on("connection", (ws, req) => {
6
+ const ip = req.socket.remoteAddress;
7
+ logger.info({ ip }, "WebSocket connection established");
8
+ ws.on("message", (message) => {
9
+ // Echo the message back
10
+ ws.send(message.toString());
11
+ });
12
+ ws.on("error", (err) => {
13
+ logger.error({ err }, "WebSocket error");
14
+ });
15
+ });
16
+ return wss;
17
+ }
18
+ //# sourceMappingURL=server-ws.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server-ws.js","sourceRoot":"","sources":["../src/server-ws.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,eAAe,EAAa,MAAM,IAAI,CAAC;AAChD,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAErC,MAAM,UAAU,cAAc,CAAC,MAAgC;IAC3D,MAAM,GAAG,GAAG,IAAI,eAAe,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;IAEzD,GAAG,CAAC,EAAE,CAAC,YAAY,EAAE,CAAC,EAAE,EAAE,GAAG,EAAE,EAAE;QAC7B,MAAM,EAAE,GAAG,GAAG,CAAC,MAAM,CAAC,aAAa,CAAC;QACpC,MAAM,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,EAAE,kCAAkC,CAAC,CAAC;QAExD,EAAE,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,OAAO,EAAE,EAAE;YACzB,wBAAwB;YACxB,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC;QAChC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;YACnB,MAAM,CAAC,KAAK,CAAC,EAAE,GAAG,EAAE,EAAE,iBAAiB,CAAC,CAAC;QAC7C,CAAC,CAAC,CAAC;IACP,CAAC,CAAC,CAAC;IAEH,OAAO,GAAG,CAAC;AACf,CAAC"}
@@ -0,0 +1,41 @@
1
+ import { Transform } from "stream";
2
+ import { logger } from "../../logger.js";
3
+ export class ToxicStream extends Transform {
4
+ config;
5
+ hasInjected = false;
6
+ constructor(config) {
7
+ super();
8
+ this.config = config;
9
+ }
10
+ _transform(chunk, encoding, callback) {
11
+ // Roll the dice only once per stream if we want per-request decision,
12
+ // but here we might want per-chunk corruption?
13
+ // Let's assume the decision was made before creating this stream or we check random here.
14
+ // For simple bit-flip corruption on body
15
+ if (this.config.action === "corrupt_body" && !this.hasInjected) {
16
+ if (Math.random() < 0.1) { // 10% chance per chunk to corrupt
17
+ logger.debug("Toxic Sidecar: Corrupting chunk");
18
+ // Flip a byte
19
+ const buf = Buffer.from(chunk);
20
+ if (buf.length > 0) {
21
+ buf[0] = ~buf[0];
22
+ this.push(buf);
23
+ this.hasInjected = true;
24
+ return callback();
25
+ }
26
+ }
27
+ }
28
+ // Slow Drip
29
+ if (this.config.action === "slow_drip") {
30
+ const delay = 100; // 100ms per chunk
31
+ setTimeout(() => {
32
+ this.push(chunk);
33
+ callback();
34
+ }, delay);
35
+ return;
36
+ }
37
+ this.push(chunk);
38
+ callback();
39
+ }
40
+ }
41
+ //# sourceMappingURL=engine.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"engine.js","sourceRoot":"","sources":["../../../src/sidecar/chaos/engine.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AACnC,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AAUzC,MAAM,OAAO,WAAY,SAAQ,SAAS;IAC9B,MAAM,CAAc;IACpB,WAAW,GAAG,KAAK,CAAC;IAE5B,YAAY,MAAmB;QAC3B,KAAK,EAAE,CAAC;QACR,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;IACzB,CAAC;IAED,UAAU,CAAC,KAAU,EAAE,QAAgB,EAAE,QAAkB;QACvD,uEAAuE;QACvE,gDAAgD;QAChD,0FAA0F;QAE1F,yCAAyC;QACzC,IAAI,IAAI,CAAC,MAAM,CAAC,MAAM,KAAK,cAAc,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;YAC7D,IAAI,IAAI,CAAC,MAAM,EAAE,GAAG,GAAG,EAAE,CAAC,CAAC,kCAAkC;gBACzD,MAAM,CAAC,KAAK,CAAC,iCAAiC,CAAC,CAAC;gBAChD,cAAc;gBACd,MAAM,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAC/B,IAAI,GAAG,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACjB,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;oBACjB,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;oBACf,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;oBACxB,OAAO,QAAQ,EAAE,CAAC;gBACtB,CAAC;YACL,CAAC;QACL,CAAC;QAED,YAAY;QACZ,IAAI,IAAI,CAAC,MAAM,CAAC,MAAM,KAAK,WAAW,EAAE,CAAC;YACrC,MAAM,KAAK,GAAG,GAAG,CAAC,CAAC,kBAAkB;YACrC,UAAU,CAAC,GAAG,EAAE;gBACZ,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBACjB,QAAQ,EAAE,CAAC;YACf,CAAC,EAAE,KAAK,CAAC,CAAC;YACV,OAAO;QACX,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACjB,QAAQ,EAAE,CAAC;IACf,CAAC;CACJ"}
@@ -0,0 +1,98 @@
1
+ import http from "http";
2
+ import { URL } from "url";
3
+ import { request as undiciRequest } from "undici";
4
+ import { pipeline } from "stream/promises";
5
+ import { logger } from "../logger.js";
6
+ import { ToxicStream } from "./chaos/engine.js";
7
+ import { fileURLToPath } from 'url';
8
+ export const SIDECAR_PORT = parseInt(process.env.SIDECAR_PORT || "8081");
9
+ const TARGET = process.env.TARGET_URL || "http://localhost:8080"; // Default to apparatus
10
+ const MAX_BODY_SIZE = 10 * 1024 * 1024; // 10MB limit for buffering
11
+ export function createSidecarServer() {
12
+ return http.createServer(async (req, res) => {
13
+ const targetUrl = new URL(req.url, TARGET);
14
+ const method = req.method || "GET";
15
+ // 1. Toxicity Determination
16
+ let toxicMode = "none";
17
+ if (req.headers["x-toxic-mode"]) {
18
+ toxicMode = req.headers["x-toxic-mode"];
19
+ }
20
+ else if (Math.random() < 0.05) {
21
+ const modes = ["latency", "error_500", "slow_drip"];
22
+ toxicMode = modes[Math.floor(Math.random() * modes.length)];
23
+ }
24
+ logger.info({ method, url: req.url, toxic: toxicMode }, "Sidecar: Proxying Request");
25
+ // 2. Toxic Effects: Request Side
26
+ if (toxicMode === "latency") {
27
+ const delay = Math.floor(Math.random() * 2000) + 500;
28
+ await new Promise(r => setTimeout(r, delay));
29
+ }
30
+ if (toxicMode === "error_500") {
31
+ res.writeHead(500, { "Content-Type": "application/json" });
32
+ res.end(JSON.stringify({ error: "Toxic Sidecar: Injected 500 Error" }));
33
+ return;
34
+ }
35
+ // 3. Upstream Request with Memory Protection
36
+ try {
37
+ const buffers = [];
38
+ let totalSize = 0;
39
+ for await (const chunk of req) {
40
+ totalSize += chunk.length;
41
+ if (totalSize > MAX_BODY_SIZE) {
42
+ logger.warn({ ip: req.socket?.remoteAddress }, "Sidecar: Request too large, rejecting");
43
+ res.writeHead(413);
44
+ res.end("Payload Too Large (Toxic Sidecar Limit: 10MB)");
45
+ return;
46
+ }
47
+ buffers.push(chunk);
48
+ }
49
+ const body = Buffer.concat(buffers);
50
+ // Forward to Target
51
+ const { statusCode, headers, body: responseBody } = await undiciRequest(targetUrl.toString(), {
52
+ method: method,
53
+ headers: req.headers,
54
+ body: body.length > 0 ? body : undefined
55
+ });
56
+ // 4. Toxic Effects: Response Side
57
+ res.writeHead(statusCode, headers);
58
+ if (toxicMode === "slow_drip" || toxicMode === "corrupt_body") {
59
+ const toxicStream = new ToxicStream({ rate: 1.0, action: toxicMode });
60
+ // Use pipeline for proper backpressure and error handling
61
+ await pipeline(responseBody, toxicStream, res);
62
+ }
63
+ else {
64
+ // Standard proxy streaming
65
+ await pipeline(responseBody, res);
66
+ }
67
+ }
68
+ catch (e) {
69
+ logger.error({ error: e.message }, "Sidecar: Upstream Failed or Stream Pipe Error");
70
+ if (!res.headersSent) {
71
+ res.writeHead(502);
72
+ res.end("Bad Gateway (Toxic Sidecar)");
73
+ }
74
+ else {
75
+ // If headers already sent, we just end the response to signal error
76
+ res.end();
77
+ }
78
+ }
79
+ });
80
+ }
81
+ export function startSidecar() {
82
+ const server = createSidecarServer();
83
+ server.listen(SIDECAR_PORT, "0.0.0.0", () => {
84
+ console.log(`
85
+ ☢️ Toxic Sidecar Active on port ${SIDECAR_PORT}
86
+ -----------------------------------
87
+ Target: ${TARGET}
88
+ Limit: 10MB
89
+
90
+ To use: Send requests to http://localhost:${SIDECAR_PORT}
91
+ Control: X-Toxic-Mode: [latency, slow_drip, error_500, corrupt_body]
92
+ `);
93
+ });
94
+ }
95
+ if (process.argv[1] === fileURLToPath(import.meta.url)) {
96
+ startSidecar();
97
+ }
98
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/sidecar/index.ts"],"names":[],"mappings":"AAAA,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,EAAE,GAAG,EAAE,MAAM,KAAK,CAAC;AAC1B,OAAO,EAAE,OAAO,IAAI,aAAa,EAAE,MAAM,QAAQ,CAAC;AAClD,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAC3C,OAAO,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AACtC,OAAO,EAAE,WAAW,EAAe,MAAM,mBAAmB,CAAC;AAC7D,OAAO,EAAE,aAAa,EAAE,MAAM,KAAK,CAAC;AAEpC,MAAM,CAAC,MAAM,YAAY,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,YAAY,IAAI,MAAM,CAAC,CAAC;AACzE,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC,UAAU,IAAI,uBAAuB,CAAC,CAAC,uBAAuB;AACzF,MAAM,aAAa,GAAG,EAAE,GAAG,IAAI,GAAG,IAAI,CAAC,CAAC,2BAA2B;AAEnE,MAAM,UAAU,mBAAmB;IAC/B,OAAO,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;QACxC,MAAM,SAAS,GAAG,IAAI,GAAG,CAAC,GAAG,CAAC,GAAI,EAAE,MAAM,CAAC,CAAC;QAC5C,MAAM,MAAM,GAAG,GAAG,CAAC,MAAM,IAAI,KAAK,CAAC;QAEnC,4BAA4B;QAC5B,IAAI,SAAS,GAAgB,MAAM,CAAC;QACpC,IAAI,GAAG,CAAC,OAAO,CAAC,cAAc,CAAC,EAAE,CAAC;YAC9B,SAAS,GAAG,GAAG,CAAC,OAAO,CAAC,cAAc,CAAgB,CAAC;QAC3D,CAAC;aAAM,IAAI,IAAI,CAAC,MAAM,EAAE,GAAG,IAAI,EAAE,CAAC;YAC9B,MAAM,KAAK,GAAkB,CAAC,SAAS,EAAE,WAAW,EAAE,WAAW,CAAC,CAAC;YACnE,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC;QAChE,CAAC;QAED,MAAM,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,GAAG,EAAE,GAAG,CAAC,GAAG,EAAE,KAAK,EAAE,SAAS,EAAE,EAAE,2BAA2B,CAAC,CAAC;QAErF,iCAAiC;QACjC,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;YAC1B,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,IAAI,CAAC,GAAG,GAAG,CAAC;YACrD,MAAM,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,UAAU,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC;QACjD,CAAC;QAED,IAAI,SAAS,KAAK,WAAW,EAAE,CAAC;YAC5B,GAAG,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;YAC3D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,mCAAmC,EAAE,CAAC,CAAC,CAAC;YACxE,OAAO;QACX,CAAC;QAED,6CAA6C;QAC7C,IAAI,CAAC;YACD,MAAM,OAAO,GAAa,EAAE,CAAC;YAC7B,IAAI,SAAS,GAAG,CAAC,CAAC;YAElB,IAAI,KAAK,EAAE,MAAM,KAAK,IAAI,GAAG,EAAE,CAAC;gBAC5B,SAAS,IAAI,KAAK,CAAC,MAAM,CAAC;gBAC1B,IAAI,SAAS,GAAG,aAAa,EAAE,CAAC;oBAC5B,MAAM,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,GAAG,CAAC,MAAM,EAAE,aAAa,EAAE,EAAE,uCAAuC,CAAC,CAAC;oBACxF,GAAG,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;oBACnB,GAAG,CAAC,GAAG,CAAC,+CAA+C,CAAC,CAAC;oBACzD,OAAO;gBACX,CAAC;gBACD,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACxB,CAAC;YACD,MAAM,IAAI,GAAG,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAEpC,oBAAoB;YACpB,MAAM,EAAE,UAAU,EAAE,OAAO,EAAE,IAAI,EAAE,YAAY,EAAE,GAAG,MAAM,aAAa,CAAC,SAAS,CAAC,QAAQ,EAAE,EAAE;gBAC1F,MAAM,EAAE,MAAa;gBACrB,OAAO,EAAE,GAAG,CAAC,OAAc;gBAC3B,IAAI,EAAE,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS;aAC3C,CAAC,CAAC;YAEH,kCAAkC;YAClC,GAAG,CAAC,SAAS,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;YAEnC,IAAI,SAAS,KAAK,WAAW,IAAI,SAAS,KAAK,cAAc,EAAE,CAAC;gBAC5D,MAAM,WAAW,GAAG,IAAI,WAAW,CAAC,EAAE,IAAI,EAAE,GAAG,EAAE,MAAM,EAAE,SAAS,EAAE,CAAC,CAAC;gBAEtE,0DAA0D;gBAC1D,MAAM,QAAQ,CACV,YAAY,EACZ,WAAW,EACX,GAAG,CACN,CAAC;YACN,CAAC;iBAAM,CAAC;gBACJ,2BAA2B;gBAC3B,MAAM,QAAQ,CACV,YAAY,EACZ,GAAG,CACN,CAAC;YACN,CAAC;QAEL,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YACd,MAAM,CAAC,KAAK,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,OAAO,EAAE,EAAE,+CAA+C,CAAC,CAAC;YACpF,IAAI,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC;gBACnB,GAAG,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;gBACnB,GAAG,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAC;YAC3C,CAAC;iBAAM,CAAC;gBACJ,oEAAoE;gBACpE,GAAG,CAAC,GAAG,EAAE,CAAC;YACd,CAAC;QACL,CAAC;IACL,CAAC,CAAC,CAAC;AACP,CAAC;AAED,MAAM,UAAU,YAAY;IACxB,MAAM,MAAM,GAAG,mBAAmB,EAAE,CAAC;IACrC,MAAM,CAAC,MAAM,CAAC,YAAY,EAAE,SAAS,EAAE,GAAG,EAAE;QACxC,OAAO,CAAC,GAAG,CAAC;mCACe,YAAY;;cAEjC,MAAM;;;gDAG4B,YAAY;;SAEnD,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,YAAY,EAAE,CAAC;AACnB,CAAC"}
@@ -0,0 +1,102 @@
1
+ let graph = generateGraph();
2
+ // Generate a random dependency graph
3
+ function generateGraph(nodeCount = 30) {
4
+ const nodes = {};
5
+ const packageNames = [
6
+ "react", "lodash", "axios", "express", "chalk", "debug", "commander",
7
+ "jest", "eslint", "typescript", "webpack", "babel", "moment", "uuid",
8
+ "rxjs", "classnames", "prop-types", "tslib", "fs-extra", "bluebird",
9
+ "left-pad", "is-number", "is-odd", "event-stream", "flatmap-stream" // Famous victims
10
+ ];
11
+ // Create Root App
12
+ nodes["app-root"] = {
13
+ id: "app-root",
14
+ name: "my-enterprise-app",
15
+ version: "1.0.0",
16
+ type: "app",
17
+ status: "clean",
18
+ dependencies: [],
19
+ dependents: []
20
+ };
21
+ // Create Libraries
22
+ for (let i = 0; i < nodeCount; i++) {
23
+ const id = `pkg-${i}`;
24
+ const name = packageNames[i % packageNames.length] + (Math.floor(i / packageNames.length) || "");
25
+ nodes[id] = {
26
+ id,
27
+ name,
28
+ version: `${Math.floor(Math.random() * 5)}.${Math.floor(Math.random() * 10)}.${Math.floor(Math.random() * 10)}`,
29
+ type: "lib",
30
+ status: "clean",
31
+ dependencies: [],
32
+ dependents: []
33
+ };
34
+ }
35
+ const nodeIds = Object.keys(nodes).filter(id => id !== "app-root");
36
+ // Link Dependencies (Force Directed-ish)
37
+ // 1. Connect App to some top-level deps
38
+ for (let i = 0; i < 5; i++) {
39
+ const depId = nodeIds[Math.floor(Math.random() * nodeIds.length)];
40
+ if (!nodes["app-root"].dependencies.includes(depId)) {
41
+ nodes["app-root"].dependencies.push(depId);
42
+ nodes[depId].dependents.push("app-root");
43
+ }
44
+ }
45
+ // 2. Interconnect libs (Tree structure)
46
+ nodeIds.forEach(id => {
47
+ // Randomly depend on other libs (lower in list to avoid cycles mostly, but cycles happen in real life)
48
+ // Simplified: only depend on indices higher than self to guarantee DAG
49
+ const currentIdx = parseInt(id.split('-')[1]);
50
+ const numDeps = Math.floor(Math.random() * 3);
51
+ for (let j = 0; j < numDeps; j++) {
52
+ const targetIdx = Math.floor(Math.random() * (nodeCount - currentIdx - 1)) + currentIdx + 1;
53
+ const targetId = `pkg-${targetIdx}`;
54
+ if (nodes[targetId] && !nodes[id].dependencies.includes(targetId)) {
55
+ nodes[id].dependencies.push(targetId);
56
+ nodes[targetId].dependents.push(id);
57
+ }
58
+ }
59
+ });
60
+ return { nodes };
61
+ }
62
+ // Logic to propagate infection UP the tree (Dependency Chain)
63
+ // If I use 'left-pad', and 'left-pad' is infected, I am compromised.
64
+ function propagateInfection() {
65
+ let changed = true;
66
+ while (changed) {
67
+ changed = false;
68
+ Object.values(graph.nodes).forEach(node => {
69
+ if (node.status === 'clean') {
70
+ // Check if any dependency is infected/compromised
71
+ const hasBadDep = node.dependencies.some(depId => {
72
+ const dep = graph.nodes[depId];
73
+ return dep && (dep.status === 'infected' || dep.status === 'compromised');
74
+ });
75
+ if (hasBadDep) {
76
+ node.status = 'compromised';
77
+ changed = true;
78
+ }
79
+ }
80
+ });
81
+ }
82
+ }
83
+ // Handlers
84
+ export function getGraphHandler(req, res) {
85
+ res.json(graph);
86
+ }
87
+ export function resetGraphHandler(req, res) {
88
+ graph = generateGraph();
89
+ res.json(graph);
90
+ }
91
+ export function injectMalwareHandler(req, res) {
92
+ const { id } = req.body;
93
+ const node = graph.nodes[id];
94
+ if (!node)
95
+ return res.status(404).json({ error: "Package not found" });
96
+ // Infection starts here
97
+ node.status = 'infected';
98
+ // Spread
99
+ propagateInfection();
100
+ res.json({ status: "infected", node, impact: Object.values(graph.nodes).filter(n => n.status !== 'clean').length });
101
+ }
102
+ //# sourceMappingURL=dependency-graph.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dependency-graph.js","sourceRoot":"","sources":["../../src/simulator/dependency-graph.ts"],"names":[],"mappings":"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"}