@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,38 @@
|
|
|
1
|
+
import net from "net";
|
|
2
|
+
export async function checkTCP(host, port, data) {
|
|
3
|
+
const results = {
|
|
4
|
+
name: `tcp_${port}`,
|
|
5
|
+
target: `${host}:${port}`,
|
|
6
|
+
checks: []
|
|
7
|
+
};
|
|
8
|
+
return new Promise((resolve) => {
|
|
9
|
+
const start = Date.now();
|
|
10
|
+
const socket = new net.Socket();
|
|
11
|
+
// Timeout after 3 seconds
|
|
12
|
+
socket.setTimeout(3000);
|
|
13
|
+
socket.on("connect", () => {
|
|
14
|
+
if (data) {
|
|
15
|
+
socket.write(data);
|
|
16
|
+
}
|
|
17
|
+
results.status = "success";
|
|
18
|
+
results.duration = Date.now() - start;
|
|
19
|
+
results.details = "Connected";
|
|
20
|
+
socket.destroy();
|
|
21
|
+
resolve(results);
|
|
22
|
+
});
|
|
23
|
+
socket.on("timeout", () => {
|
|
24
|
+
results.status = "failed";
|
|
25
|
+
results.error = "Timeout";
|
|
26
|
+
socket.destroy();
|
|
27
|
+
resolve(results);
|
|
28
|
+
});
|
|
29
|
+
socket.on("error", (err) => {
|
|
30
|
+
results.status = "failed";
|
|
31
|
+
results.error = err.message;
|
|
32
|
+
socket.destroy();
|
|
33
|
+
resolve(results);
|
|
34
|
+
});
|
|
35
|
+
socket.connect(port, host);
|
|
36
|
+
});
|
|
37
|
+
}
|
|
38
|
+
//# sourceMappingURL=tcp.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"tcp.js","sourceRoot":"","sources":["../../../src/escape/methods/tcp.ts"],"names":[],"mappings":"AAAA,OAAO,GAAG,MAAM,KAAK,CAAC;AAEtB,MAAM,CAAC,KAAK,UAAU,QAAQ,CAAC,IAAY,EAAE,IAAY,EAAE,IAAa;IACpE,MAAM,OAAO,GAAQ;QACjB,IAAI,EAAE,OAAO,IAAI,EAAE;QACnB,MAAM,EAAE,GAAG,IAAI,IAAI,IAAI,EAAE;QACzB,MAAM,EAAE,EAAE;KACb,CAAC;IAEF,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;QAC3B,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACzB,MAAM,MAAM,GAAG,IAAI,GAAG,CAAC,MAAM,EAAE,CAAC;QAEhC,0BAA0B;QAC1B,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;QAExB,MAAM,CAAC,EAAE,CAAC,SAAS,EAAE,GAAG,EAAE;YACtB,IAAI,IAAI,EAAE,CAAC;gBACP,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;YACvB,CAAC;YACD,OAAO,CAAC,MAAM,GAAG,SAAS,CAAC;YAC3B,OAAO,CAAC,QAAQ,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC;YACtC,OAAO,CAAC,OAAO,GAAG,WAAW,CAAC;YAC9B,MAAM,CAAC,OAAO,EAAE,CAAC;YACjB,OAAO,CAAC,OAAO,CAAC,CAAC;QACrB,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,EAAE,CAAC,SAAS,EAAE,GAAG,EAAE;YACtB,OAAO,CAAC,MAAM,GAAG,QAAQ,CAAC;YAC1B,OAAO,CAAC,KAAK,GAAG,SAAS,CAAC;YAC1B,MAAM,CAAC,OAAO,EAAE,CAAC;YACjB,OAAO,CAAC,OAAO,CAAC,CAAC;QACrB,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAQ,EAAE,EAAE;YAC5B,OAAO,CAAC,MAAM,GAAG,QAAQ,CAAC;YAC1B,OAAO,CAAC,KAAK,GAAG,GAAG,CAAC,OAAO,CAAC;YAC5B,MAAM,CAAC,OAAO,EAAE,CAAC;YACjB,OAAO,CAAC,OAAO,CAAC,CAAC;QACrB,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,OAAO,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;IAC/B,CAAC,CAAC,CAAC;AACP,CAAC"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import dgram from "dgram";
|
|
2
|
+
export async function checkUDP(host, port, data = "ping") {
|
|
3
|
+
const results = {
|
|
4
|
+
name: `udp_${port}`,
|
|
5
|
+
target: `${host}:${port}`,
|
|
6
|
+
checks: []
|
|
7
|
+
};
|
|
8
|
+
return new Promise((resolve) => {
|
|
9
|
+
const socket = dgram.createSocket("udp4");
|
|
10
|
+
const message = Buffer.from(data);
|
|
11
|
+
const start = Date.now();
|
|
12
|
+
socket.send(message, port, host, (err) => {
|
|
13
|
+
if (err) {
|
|
14
|
+
results.status = "failed";
|
|
15
|
+
results.error = err.message;
|
|
16
|
+
}
|
|
17
|
+
else {
|
|
18
|
+
results.status = "success";
|
|
19
|
+
results.duration = Date.now() - start;
|
|
20
|
+
results.details = "Packet sent (UDP is connectionless)";
|
|
21
|
+
}
|
|
22
|
+
socket.close();
|
|
23
|
+
resolve(results);
|
|
24
|
+
});
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=udp.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"udp.js","sourceRoot":"","sources":["../../../src/escape/methods/udp.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAE1B,MAAM,CAAC,KAAK,UAAU,QAAQ,CAAC,IAAY,EAAE,IAAY,EAAE,OAAe,MAAM;IAC5E,MAAM,OAAO,GAAQ;QACjB,IAAI,EAAE,OAAO,IAAI,EAAE;QACnB,MAAM,EAAE,GAAG,IAAI,IAAI,IAAI,EAAE;QACzB,MAAM,EAAE,EAAE;KACb,CAAC;IAEF,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;QAC3B,MAAM,MAAM,GAAG,KAAK,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;QAC1C,MAAM,OAAO,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAClC,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAEzB,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,GAAG,EAAE,EAAE;YACrC,IAAI,GAAG,EAAE,CAAC;gBACN,OAAO,CAAC,MAAM,GAAG,QAAQ,CAAC;gBAC1B,OAAO,CAAC,KAAK,GAAG,GAAG,CAAC,OAAO,CAAC;YAChC,CAAC;iBAAM,CAAC;gBACJ,OAAO,CAAC,MAAM,GAAG,SAAS,CAAC;gBAC3B,OAAO,CAAC,QAAQ,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC;gBACtC,OAAO,CAAC,OAAO,GAAG,qCAAqC,CAAC;YAC5D,CAAC;YACD,MAAM,CAAC,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,OAAO,CAAC,CAAC;QACrB,CAAC,CAAC,CAAC;IACP,CAAC,CAAC,CAAC;AACP,CAAC"}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import WebSocket from "ws";
|
|
2
|
+
export async function checkWebSocket(url, data) {
|
|
3
|
+
const results = {
|
|
4
|
+
name: "websocket_connect",
|
|
5
|
+
target: url,
|
|
6
|
+
checks: []
|
|
7
|
+
};
|
|
8
|
+
return new Promise((resolve) => {
|
|
9
|
+
const start = Date.now();
|
|
10
|
+
const ws = new WebSocket(url);
|
|
11
|
+
const timeout = setTimeout(() => {
|
|
12
|
+
ws.terminate();
|
|
13
|
+
results.status = "failed";
|
|
14
|
+
results.error = "Timeout";
|
|
15
|
+
resolve(results);
|
|
16
|
+
}, 3000);
|
|
17
|
+
ws.on("open", () => {
|
|
18
|
+
clearTimeout(timeout);
|
|
19
|
+
results.status = "success";
|
|
20
|
+
results.duration = Date.now() - start;
|
|
21
|
+
results.details = "Connected";
|
|
22
|
+
if (data) {
|
|
23
|
+
ws.send(data);
|
|
24
|
+
results.details += " + Data Sent";
|
|
25
|
+
}
|
|
26
|
+
ws.close();
|
|
27
|
+
resolve(results);
|
|
28
|
+
});
|
|
29
|
+
ws.on("error", (err) => {
|
|
30
|
+
clearTimeout(timeout);
|
|
31
|
+
results.status = "failed";
|
|
32
|
+
results.error = err.message;
|
|
33
|
+
resolve(results);
|
|
34
|
+
});
|
|
35
|
+
});
|
|
36
|
+
}
|
|
37
|
+
//# sourceMappingURL=websocket.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"websocket.js","sourceRoot":"","sources":["../../../src/escape/methods/websocket.ts"],"names":[],"mappings":"AAAA,OAAO,SAAS,MAAM,IAAI,CAAC;AAE3B,MAAM,CAAC,KAAK,UAAU,cAAc,CAAC,GAAW,EAAE,IAAa;IAC3D,MAAM,OAAO,GAAQ;QACjB,IAAI,EAAE,mBAAmB;QACzB,MAAM,EAAE,GAAG;QACX,MAAM,EAAE,EAAE;KACb,CAAC;IAEF,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;QAC3B,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACzB,MAAM,EAAE,GAAG,IAAI,SAAS,CAAC,GAAG,CAAC,CAAC;QAE9B,MAAM,OAAO,GAAG,UAAU,CAAC,GAAG,EAAE;YAC5B,EAAE,CAAC,SAAS,EAAE,CAAC;YACf,OAAO,CAAC,MAAM,GAAG,QAAQ,CAAC;YAC1B,OAAO,CAAC,KAAK,GAAG,SAAS,CAAC;YAC1B,OAAO,CAAC,OAAO,CAAC,CAAC;QACrB,CAAC,EAAE,IAAI,CAAC,CAAC;QAET,EAAE,CAAC,EAAE,CAAC,MAAM,EAAE,GAAG,EAAE;YACf,YAAY,CAAC,OAAO,CAAC,CAAC;YACtB,OAAO,CAAC,MAAM,GAAG,SAAS,CAAC;YAC3B,OAAO,CAAC,QAAQ,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC;YACtC,OAAO,CAAC,OAAO,GAAG,WAAW,CAAC;YAE9B,IAAI,IAAI,EAAE,CAAC;gBACP,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBACd,OAAO,CAAC,OAAO,IAAI,cAAc,CAAC;YACtC,CAAC;YAED,EAAE,CAAC,KAAK,EAAE,CAAC;YACX,OAAO,CAAC,OAAO,CAAC,CAAC;QACrB,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAQ,EAAE,EAAE;YACxB,YAAY,CAAC,OAAO,CAAC,CAAC;YACtB,OAAO,CAAC,MAAM,GAAG,QAAQ,CAAC;YAC1B,OAAO,CAAC,KAAK,GAAG,GAAG,CAAC,OAAO,CAAC;YAC5B,OAAO,CAAC,OAAO,CAAC,CAAC;QACrB,CAAC,CAAC,CAAC;IACP,CAAC,CAAC,CAAC;AACP,CAAC"}
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
import { spawn } from "child_process";
|
|
2
|
+
import { request } from "undici";
|
|
3
|
+
import { logger } from "./logger.js";
|
|
4
|
+
export function pcapHandler(req, res) {
|
|
5
|
+
const duration = parseInt(req.query.duration) || 30;
|
|
6
|
+
const iface = req.query.iface || "eth0";
|
|
7
|
+
res.setHeader("Content-Type", "application/vnd.tcpdump.pcap");
|
|
8
|
+
res.setHeader("Content-Disposition", `attachment; filename="capture-${Date.now()}.pcap"`);
|
|
9
|
+
// Requires 'tcpdump' installed in image and NET_ADMIN capability
|
|
10
|
+
const tcpdump = spawn("tcpdump", ["-i", iface, "-w", "-", "-U"], {
|
|
11
|
+
stdio: ["ignore", "pipe", "pipe"]
|
|
12
|
+
});
|
|
13
|
+
tcpdump.stdout.pipe(res);
|
|
14
|
+
// Stop after duration
|
|
15
|
+
setTimeout(() => {
|
|
16
|
+
tcpdump.kill();
|
|
17
|
+
if (!res.writableEnded)
|
|
18
|
+
res.end();
|
|
19
|
+
}, duration * 1000);
|
|
20
|
+
tcpdump.on("error", (err) => {
|
|
21
|
+
if (!res.headersSent)
|
|
22
|
+
res.status(500).send(`tcpdump failed: ${err.message}`);
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
export async function harReplayHandler(req, res) {
|
|
26
|
+
const har = req.body; // Expecting parsed JSON body
|
|
27
|
+
if (!har || !har.log || !har.log.entries) {
|
|
28
|
+
return res.status(400).json({ error: "Invalid HAR JSON" });
|
|
29
|
+
}
|
|
30
|
+
const results = [];
|
|
31
|
+
for (const entry of har.log.entries) {
|
|
32
|
+
try {
|
|
33
|
+
const { request: reqData } = entry;
|
|
34
|
+
// Replay logic
|
|
35
|
+
const { statusCode } = await request(reqData.url, {
|
|
36
|
+
method: reqData.method,
|
|
37
|
+
headers: reqData.headers.reduce((acc, h) => {
|
|
38
|
+
acc[h.name] = h.value;
|
|
39
|
+
return acc;
|
|
40
|
+
}, {}),
|
|
41
|
+
body: reqData.postData?.text
|
|
42
|
+
});
|
|
43
|
+
results.push({ url: reqData.url, status: statusCode });
|
|
44
|
+
}
|
|
45
|
+
catch (e) {
|
|
46
|
+
results.push({ url: entry.request.url, error: e.message });
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
res.json({ results });
|
|
50
|
+
}
|
|
51
|
+
const MAX_LIVE_CAPTURES = 4;
|
|
52
|
+
let activeCaptures = 0;
|
|
53
|
+
export function livePacketHandler(req, res) {
|
|
54
|
+
if (activeCaptures >= MAX_LIVE_CAPTURES) {
|
|
55
|
+
return res.status(429).json({ error: "Too many active captures" });
|
|
56
|
+
}
|
|
57
|
+
// Strict allowlist for interfaces to prevent arbitrary capture
|
|
58
|
+
const ALLOWED_INTERFACES = new Set(["eth0", "lo", "docker0", "any"]);
|
|
59
|
+
const rawIface = req.query.iface;
|
|
60
|
+
const iface = ALLOWED_INTERFACES.has(rawIface) ? rawIface : "eth0";
|
|
61
|
+
// Validate BPF filter - only allow alphanumeric and common symbols
|
|
62
|
+
const rawFilter = req.query.filter || "";
|
|
63
|
+
const filter = rawFilter.replace(/[^a-zA-Z0-9\s.:\/\-]/g, "");
|
|
64
|
+
res.setHeader("Content-Type", "text/event-stream");
|
|
65
|
+
res.setHeader("Cache-Control", "no-cache");
|
|
66
|
+
res.setHeader("Connection", "keep-alive");
|
|
67
|
+
res.write(":ok\n\n"); // Flush buffer
|
|
68
|
+
activeCaptures++;
|
|
69
|
+
// Spawn tcpdump in line-buffered text mode
|
|
70
|
+
// -l: line-buffered, -nn: don't resolve names, -t: don't print timestamp (we'll add our own)
|
|
71
|
+
const args = ["-i", iface, "-l", "-nn", "-q"];
|
|
72
|
+
// Split filter into individual tokens for safe spawning
|
|
73
|
+
if (filter) {
|
|
74
|
+
const tokens = filter.split(/\s+/).filter(Boolean);
|
|
75
|
+
args.push(...tokens);
|
|
76
|
+
}
|
|
77
|
+
const tcpdump = spawn("tcpdump", args, {
|
|
78
|
+
stdio: ["ignore", "pipe", "pipe"] // We will consume stderr
|
|
79
|
+
});
|
|
80
|
+
tcpdump.stdout.on("data", (data) => {
|
|
81
|
+
const lines = data.toString().split("\n");
|
|
82
|
+
for (const line of lines) {
|
|
83
|
+
if (line.trim()) {
|
|
84
|
+
res.write(`data: ${JSON.stringify({
|
|
85
|
+
timestamp: new Date().toISOString(),
|
|
86
|
+
raw: line.trim()
|
|
87
|
+
})}\n\n`);
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
});
|
|
91
|
+
// Consume stderr to prevent hanging
|
|
92
|
+
tcpdump.stderr.on("data", (data) => {
|
|
93
|
+
logger.warn({ stderr: data.toString().trim() }, "tcpdump live stderr");
|
|
94
|
+
});
|
|
95
|
+
const cleanup = () => {
|
|
96
|
+
if (!tcpdump.killed) {
|
|
97
|
+
tcpdump.kill();
|
|
98
|
+
activeCaptures--;
|
|
99
|
+
}
|
|
100
|
+
};
|
|
101
|
+
req.on("close", cleanup);
|
|
102
|
+
// Auto-kill after 5 minutes
|
|
103
|
+
setTimeout(cleanup, 5 * 60 * 1000);
|
|
104
|
+
tcpdump.on("error", (err) => {
|
|
105
|
+
logger.error({ err }, "tcpdump live failed");
|
|
106
|
+
if (!res.writableEnded)
|
|
107
|
+
res.end();
|
|
108
|
+
cleanup();
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
//# sourceMappingURL=forensics.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"forensics.js","sourceRoot":"","sources":["../src/forensics.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
export function generatorHandler(req, res) {
|
|
2
|
+
const sizeStr = req.query.size || "1mb";
|
|
3
|
+
const chunked = req.query.chunked === "true";
|
|
4
|
+
// Parse size
|
|
5
|
+
let bytes = 1024 * 1024; // Default 1MB
|
|
6
|
+
const match = sizeStr.match(/^(\d+)([kmg]b?)?$/i);
|
|
7
|
+
if (match) {
|
|
8
|
+
const val = parseInt(match[1]);
|
|
9
|
+
const unit = (match[2] || "").toLowerCase();
|
|
10
|
+
if (unit.startsWith("k"))
|
|
11
|
+
bytes = val * 1024;
|
|
12
|
+
else if (unit.startsWith("m"))
|
|
13
|
+
bytes = val * 1024 * 1024;
|
|
14
|
+
else if (unit.startsWith("g"))
|
|
15
|
+
bytes = val * 1024 * 1024 * 1024;
|
|
16
|
+
else
|
|
17
|
+
bytes = val;
|
|
18
|
+
}
|
|
19
|
+
// Cap at 1GB to prevent memory explosions
|
|
20
|
+
if (bytes > 1024 * 1024 * 1024)
|
|
21
|
+
bytes = 1024 * 1024 * 1024;
|
|
22
|
+
res.setHeader("Content-Type", "application/octet-stream");
|
|
23
|
+
res.setHeader("Content-Disposition", `attachment; filename="payload-${sizeStr}.bin"`);
|
|
24
|
+
if (!chunked) {
|
|
25
|
+
// Send a single buffer (careful with memory)
|
|
26
|
+
// Ideally we stream anyway, but for "non-chunked" we need Content-Length
|
|
27
|
+
res.setHeader("Content-Length", bytes);
|
|
28
|
+
// Stream it in chunks to avoid allocating huge buffer, but keep response open
|
|
29
|
+
let sent = 0;
|
|
30
|
+
const chunkSize = 64 * 1024;
|
|
31
|
+
const buffer = Buffer.alloc(chunkSize, "A"); // Fast allocation
|
|
32
|
+
const stream = () => {
|
|
33
|
+
while (sent < bytes) {
|
|
34
|
+
const remaining = bytes - sent;
|
|
35
|
+
const toSend = Math.min(chunkSize, remaining);
|
|
36
|
+
const canContinue = res.write(toSend === chunkSize ? buffer : buffer.subarray(0, toSend));
|
|
37
|
+
sent += toSend;
|
|
38
|
+
if (!canContinue) {
|
|
39
|
+
res.once("drain", stream);
|
|
40
|
+
return;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
res.end();
|
|
44
|
+
};
|
|
45
|
+
stream();
|
|
46
|
+
}
|
|
47
|
+
else {
|
|
48
|
+
// Chunked encoding (no Content-Length), artificial delay
|
|
49
|
+
// Send 10% every 500ms
|
|
50
|
+
const chunkSize = Math.floor(bytes / 10);
|
|
51
|
+
let sent = 0;
|
|
52
|
+
const buffer = Buffer.alloc(chunkSize, "B");
|
|
53
|
+
const interval = setInterval(() => {
|
|
54
|
+
if (sent >= bytes) {
|
|
55
|
+
clearInterval(interval);
|
|
56
|
+
res.end();
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
const remaining = bytes - sent;
|
|
60
|
+
const toSend = Math.min(chunkSize, remaining);
|
|
61
|
+
res.write(toSend === chunkSize ? buffer : buffer.subarray(0, toSend));
|
|
62
|
+
sent += toSend;
|
|
63
|
+
}, 500);
|
|
64
|
+
res.on("close", () => clearInterval(interval));
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
//# sourceMappingURL=generator.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"generator.js","sourceRoot":"","sources":["../src/generator.ts"],"names":[],"mappings":"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"}
|