@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.
- package/bin/apparatus.mjs +2 -0
- package/certs/server.crt +17 -0
- package/certs/server.key +28 -0
- package/dist/ai/client.js +104 -0
- package/dist/ai/client.js.map +1 -0
- package/dist/ai/personas.js +104 -0
- package/dist/ai/personas.js.map +1 -0
- package/dist/ai/redteam.js +1404 -0
- package/dist/ai/redteam.js.map +1 -0
- package/dist/ai/report-store.js +309 -0
- package/dist/ai/report-store.js.map +1 -0
- package/dist/app.js +525 -0
- package/dist/app.js.map +1 -0
- package/dist/attack-sim.js +69 -0
- package/dist/attack-sim.js.map +1 -0
- package/dist/attacker-tracker.js +276 -0
- package/dist/attacker-tracker.js.map +1 -0
- package/dist/blackhole.js +95 -0
- package/dist/blackhole.js.map +1 -0
- package/dist/chaos.js +88 -0
- package/dist/chaos.js.map +1 -0
- package/dist/cluster.js +462 -0
- package/dist/cluster.js.map +1 -0
- package/dist/config.js +61 -0
- package/dist/config.js.map +1 -0
- package/dist/deception.js +205 -0
- package/dist/deception.js.map +1 -0
- package/dist/demo-mode.js +109 -0
- package/dist/demo-mode.js.map +1 -0
- package/dist/dist-dashboard/assets/index-BsMhEnGu.js +648 -0
- package/dist/dist-dashboard/assets/index-CNOkYC_Q.css +10 -0
- package/dist/dist-dashboard/assets/index-CW2grvPC.js +648 -0
- package/dist/dist-dashboard/assets/logo/apparatus-favicon.svg +15 -0
- package/dist/dist-dashboard/assets/logo/apparatus-icon-dark.svg +24 -0
- package/dist/dist-dashboard/assets/logo/apparatus-icon-light.svg +24 -0
- package/dist/dist-dashboard/assets/logo/apparatus-logo-512.png +0 -0
- package/dist/dist-dashboard/assets/logo/apparatus-logo-dark.svg +18 -0
- package/dist/dist-dashboard/assets/logo/apparatus-logo.svg +17 -0
- package/dist/dist-dashboard/assets/logo/apple-touch-icon.png +0 -0
- package/dist/dist-dashboard/assets/logo/favicon-192.png +0 -0
- package/dist/dist-dashboard/assets/logo/favicon-32.png +0 -0
- package/dist/dist-dashboard/assets/logo/favicon.ico +0 -0
- package/dist/dist-dashboard/assets/logo/icon-192.png +0 -0
- package/dist/dist-dashboard/assets/logo/icon-512.png +0 -0
- package/dist/dist-dashboard/assets/logo/icon-light-512.png +0 -0
- package/dist/dist-dashboard/assets/react-vendor-DpRMSntD.js +1 -0
- package/dist/dist-dashboard/assets/router-DSc5pRwN.js +59 -0
- package/dist/dist-dashboard/docs-index.json +1577 -0
- package/dist/dist-dashboard/index.html +21 -0
- package/dist/dlp.js +40 -0
- package/dist/dlp.js.map +1 -0
- package/dist/drills.js +770 -0
- package/dist/drills.js.map +1 -0
- package/dist/echoHandler.js +113 -0
- package/dist/echoHandler.js.map +1 -0
- package/dist/escape/index.js +225 -0
- package/dist/escape/index.js.map +1 -0
- package/dist/escape/methods/dns.js +74 -0
- package/dist/escape/methods/dns.js.map +1 -0
- package/dist/escape/methods/http.js +81 -0
- package/dist/escape/methods/http.js.map +1 -0
- package/dist/escape/methods/icmp.js +36 -0
- package/dist/escape/methods/icmp.js.map +1 -0
- package/dist/escape/methods/tcp.js +38 -0
- package/dist/escape/methods/tcp.js.map +1 -0
- package/dist/escape/methods/udp.js +27 -0
- package/dist/escape/methods/udp.js.map +1 -0
- package/dist/escape/methods/websocket.js +37 -0
- package/dist/escape/methods/websocket.js.map +1 -0
- package/dist/forensics.js +111 -0
- package/dist/forensics.js.map +1 -0
- package/dist/generator.js +67 -0
- package/dist/generator.js.map +1 -0
- package/dist/ghosting.js +414 -0
- package/dist/ghosting.js.map +1 -0
- package/dist/graphql.js +44 -0
- package/dist/graphql.js.map +1 -0
- package/dist/history.js +40 -0
- package/dist/history.js.map +1 -0
- package/dist/imposter/creds.js +16 -0
- package/dist/imposter/creds.js.map +1 -0
- package/dist/imposter/index.js +44 -0
- package/dist/imposter/index.js.map +1 -0
- package/dist/imposter/providers/aws.js +103 -0
- package/dist/imposter/providers/aws.js.map +1 -0
- package/dist/imposter/providers/gcp.js +26 -0
- package/dist/imposter/providers/gcp.js.map +1 -0
- package/dist/index.js +53 -0
- package/dist/index.js.map +1 -0
- package/dist/infra-debug.js +68 -0
- package/dist/infra-debug.js.map +1 -0
- package/dist/jwt-debug.js +272 -0
- package/dist/jwt-debug.js.map +1 -0
- package/dist/kv.js +22 -0
- package/dist/kv.js.map +1 -0
- package/dist/lib/generators.js +43 -0
- package/dist/lib/generators.js.map +1 -0
- package/dist/lib/json.js +26 -0
- package/dist/lib/json.js.map +1 -0
- package/dist/logger.js +9 -0
- package/dist/logger.js.map +1 -0
- package/dist/metrics.js +20 -0
- package/dist/metrics.js.map +1 -0
- package/dist/mtd.js +30 -0
- package/dist/mtd.js.map +1 -0
- package/dist/oidc.js +69 -0
- package/dist/oidc.js.map +1 -0
- package/dist/persistence/cluster-state.js +47 -0
- package/dist/persistence/cluster-state.js.map +1 -0
- package/dist/persistence/deception-history.js +65 -0
- package/dist/persistence/deception-history.js.map +1 -0
- package/dist/persistence/drill-runs.js +138 -0
- package/dist/persistence/drill-runs.js.map +1 -0
- package/dist/persistence/request-history.js +41 -0
- package/dist/persistence/request-history.js.map +1 -0
- package/dist/persistence/scenario-catalog.js +73 -0
- package/dist/persistence/scenario-catalog.js.map +1 -0
- package/dist/persistence/status.js +51 -0
- package/dist/persistence/status.js.map +1 -0
- package/dist/persistence/tarpit-state.js +47 -0
- package/dist/persistence/tarpit-state.js.map +1 -0
- package/dist/persistence/webhook-store.js +69 -0
- package/dist/persistence/webhook-store.js.map +1 -0
- package/dist/proxy.js +28 -0
- package/dist/proxy.js.map +1 -0
- package/dist/ratelimit.js +32 -0
- package/dist/ratelimit.js.map +1 -0
- package/dist/redteam.js +442 -0
- package/dist/redteam.js.map +1 -0
- package/dist/scenarios.js +229 -0
- package/dist/scenarios.js.map +1 -0
- package/dist/scripting.js +30 -0
- package/dist/scripting.js.map +1 -0
- package/dist/self-healing.js +42 -0
- package/dist/self-healing.js.map +1 -0
- package/dist/sentinel.js +50 -0
- package/dist/sentinel.js.map +1 -0
- package/dist/server-bad-ssl.js +47 -0
- package/dist/server-bad-ssl.js.map +1 -0
- package/dist/server-grpc.js +66 -0
- package/dist/server-grpc.js.map +1 -0
- package/dist/server-http1.js +5 -0
- package/dist/server-http1.js.map +1 -0
- package/dist/server-http2.js +27 -0
- package/dist/server-http2.js.map +1 -0
- package/dist/server-icap.js +46 -0
- package/dist/server-icap.js.map +1 -0
- package/dist/server-l4.js +30 -0
- package/dist/server-l4.js.map +1 -0
- package/dist/server-mqtt.js +29 -0
- package/dist/server-mqtt.js.map +1 -0
- package/dist/server-protocols.js +18 -0
- package/dist/server-protocols.js.map +1 -0
- package/dist/server-redis.js +112 -0
- package/dist/server-redis.js.map +1 -0
- package/dist/server-smtp.js +66 -0
- package/dist/server-smtp.js.map +1 -0
- package/dist/server-syslog.js +23 -0
- package/dist/server-syslog.js.map +1 -0
- package/dist/server-ws.js +18 -0
- package/dist/server-ws.js.map +1 -0
- package/dist/sidecar/chaos/engine.js +41 -0
- package/dist/sidecar/chaos/engine.js.map +1 -0
- package/dist/sidecar/index.js +98 -0
- package/dist/sidecar/index.js.map +1 -0
- package/dist/simulator/dependency-graph.js +102 -0
- package/dist/simulator/dependency-graph.js.map +1 -0
- package/dist/simulator/supply-chain.js +67 -0
- package/dist/simulator/supply-chain.js.map +1 -0
- package/dist/sink.js +24 -0
- package/dist/sink.js.map +1 -0
- package/dist/sse-broadcast.js +105 -0
- package/dist/sse-broadcast.js.map +1 -0
- package/dist/swagger.js +309 -0
- package/dist/swagger.js.map +1 -0
- package/dist/sysinfo.js +36 -0
- package/dist/sysinfo.js.map +1 -0
- package/dist/tarpit.js +126 -0
- package/dist/tarpit.js.map +1 -0
- package/dist/tool-executor.js +315 -0
- package/dist/tool-executor.js.map +1 -0
- package/dist/tui/api-client.js +341 -0
- package/dist/tui/api-client.js.map +1 -0
- package/dist/tui/core/action-handler.js +302 -0
- package/dist/tui/core/action-handler.js.map +1 -0
- package/dist/tui/core/index.js +18 -0
- package/dist/tui/core/index.js.map +1 -0
- package/dist/tui/core/keyboard.js +329 -0
- package/dist/tui/core/keyboard.js.map +1 -0
- package/dist/tui/core/modal.js +397 -0
- package/dist/tui/core/modal.js.map +1 -0
- package/dist/tui/core/screen-manager.js +262 -0
- package/dist/tui/core/screen-manager.js.map +1 -0
- package/dist/tui/core/store.js +254 -0
- package/dist/tui/core/store.js.map +1 -0
- package/dist/tui/core/widget.js +167 -0
- package/dist/tui/core/widget.js.map +1 -0
- package/dist/tui/dashboard.js +649 -0
- package/dist/tui/dashboard.js.map +1 -0
- package/dist/tui/index.js +118 -0
- package/dist/tui/index.js.map +1 -0
- package/dist/tui/modals/add-rule-modal.js +190 -0
- package/dist/tui/modals/add-rule-modal.js.map +1 -0
- package/dist/tui/modals/dlp-output-modal.js +102 -0
- package/dist/tui/modals/dlp-output-modal.js.map +1 -0
- package/dist/tui/modals/dns-form-modal.js +26 -0
- package/dist/tui/modals/dns-form-modal.js.map +1 -0
- package/dist/tui/modals/ghost-config-modal.js +35 -0
- package/dist/tui/modals/ghost-config-modal.js.map +1 -0
- package/dist/tui/modals/har-results-modal.js +41 -0
- package/dist/tui/modals/har-results-modal.js.map +1 -0
- package/dist/tui/modals/index.js +15 -0
- package/dist/tui/modals/index.js.map +1 -0
- package/dist/tui/modals/jwt-decode-modal.js +45 -0
- package/dist/tui/modals/jwt-decode-modal.js.map +1 -0
- package/dist/tui/modals/jwt-mint-modal.js +70 -0
- package/dist/tui/modals/jwt-mint-modal.js.map +1 -0
- package/dist/tui/modals/ping-form-modal.js +19 -0
- package/dist/tui/modals/ping-form-modal.js.map +1 -0
- package/dist/tui/modals/redteam-results-modal.js +43 -0
- package/dist/tui/modals/redteam-results-modal.js.map +1 -0
- package/dist/tui/modals/scan-form-modal.js +26 -0
- package/dist/tui/modals/scan-form-modal.js.map +1 -0
- package/dist/tui/screens/defense-screen.js +281 -0
- package/dist/tui/screens/defense-screen.js.map +1 -0
- package/dist/tui/screens/forensics-screen.js +81 -0
- package/dist/tui/screens/forensics-screen.js.map +1 -0
- package/dist/tui/screens/index.js +140 -0
- package/dist/tui/screens/index.js.map +1 -0
- package/dist/tui/screens/system-screen.js +81 -0
- package/dist/tui/screens/system-screen.js.map +1 -0
- package/dist/tui/screens/testing-screen.js +429 -0
- package/dist/tui/screens/testing-screen.js.map +1 -0
- package/dist/tui/screens/traffic-screen.js +76 -0
- package/dist/tui/screens/traffic-screen.js.map +1 -0
- package/dist/tui/sse-client.js +130 -0
- package/dist/tui/sse-client.js.map +1 -0
- package/dist/tui/state/metrics-buffer.js +195 -0
- package/dist/tui/state/metrics-buffer.js.map +1 -0
- package/dist/tui/state/metrics-buffer.test.js +102 -0
- package/dist/tui/state/metrics-buffer.test.js.map +1 -0
- package/dist/tui/theme.js +136 -0
- package/dist/tui/theme.js.map +1 -0
- package/dist/tui/types.js +6 -0
- package/dist/tui/types.js.map +1 -0
- package/dist/tui/widgets/chaos-widget.js +152 -0
- package/dist/tui/widgets/chaos-widget.js.map +1 -0
- package/dist/tui/widgets/cluster-widget.js +156 -0
- package/dist/tui/widgets/cluster-widget.js.map +1 -0
- package/dist/tui/widgets/dlp-widget.js +161 -0
- package/dist/tui/widgets/dlp-widget.js.map +1 -0
- package/dist/tui/widgets/ghost-widget.js +169 -0
- package/dist/tui/widgets/ghost-widget.js.map +1 -0
- package/dist/tui/widgets/har-widget.js +173 -0
- package/dist/tui/widgets/har-widget.js.map +1 -0
- package/dist/tui/widgets/index.js +122 -0
- package/dist/tui/widgets/index.js.map +1 -0
- package/dist/tui/widgets/jwt-widget.js +177 -0
- package/dist/tui/widgets/jwt-widget.js.map +1 -0
- package/dist/tui/widgets/kv-widget.js +261 -0
- package/dist/tui/widgets/kv-widget.js.map +1 -0
- package/dist/tui/widgets/mtd-widget.js +181 -0
- package/dist/tui/widgets/mtd-widget.js.map +1 -0
- package/dist/tui/widgets/netdiag-widget.js +155 -0
- package/dist/tui/widgets/netdiag-widget.js.map +1 -0
- package/dist/tui/widgets/oidc-widget.js +162 -0
- package/dist/tui/widgets/oidc-widget.js.map +1 -0
- package/dist/tui/widgets/pcap-widget.js +239 -0
- package/dist/tui/widgets/pcap-widget.js.map +1 -0
- package/dist/tui/widgets/redteam-widget.js +155 -0
- package/dist/tui/widgets/redteam-widget.js.map +1 -0
- package/dist/tui/widgets/rps-gauge-widget.js +124 -0
- package/dist/tui/widgets/rps-gauge-widget.js.map +1 -0
- package/dist/tui/widgets/sentinel-widget.js +171 -0
- package/dist/tui/widgets/sentinel-widget.js.map +1 -0
- package/dist/tui/widgets/sparklines-widget.js +127 -0
- package/dist/tui/widgets/sparklines-widget.js.map +1 -0
- package/dist/tui/widgets/sysinfo-widget.js +197 -0
- package/dist/tui/widgets/sysinfo-widget.js.map +1 -0
- package/dist/tui/widgets/traffic-chart-widget.js +170 -0
- package/dist/tui/widgets/traffic-chart-widget.js.map +1 -0
- package/dist/tui/widgets/webhook-widget.js +259 -0
- package/dist/tui/widgets/webhook-widget.js.map +1 -0
- package/dist/utils/ip.js +18 -0
- package/dist/utils/ip.js.map +1 -0
- package/dist/victim/index.js +71 -0
- package/dist/victim/index.js.map +1 -0
- package/dist/webhook.js +88 -0
- package/dist/webhook.js.map +1 -0
- package/package.json +90 -0
- 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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"}
|