if (is_dir(dirname($controllerPath)) === false) {
$io->info('Creating directory ' . dirname($controllerPath), true);
mkdir(dirname($controllerPath), 0755, true); // un-normalized, runs before validation
}
$ php vendor/flightphp/runway/runway make:controller '../../../../tmp/CONTROLLER_TRAVERSAL_TEST/pwn'
Creating directory .../app/controllers/../../../../tmp/CONTROLLER_TRAVERSAL_TEST
Nette\InvalidArgumentException: Value '../../../../tmp/CONTROLLER_TRAVERSAL_TEST/pwnController' is not valid class name.
$ ls /home/user/tmp/CONTROLLER_TRAVERSAL_TEST
(directory exists — created before the exception was thrown)
Summary
The
make:controllerCLI command callsmkdir(..., recursive: true)on a path built from the user-supplied controller name, before Nette's class-name validation runs. The class-file write is correctly rejected by Nette when the name contains/, but the recursive directory creation side effect is already committed — including directories located outside the project root through../traversal.Affected code
flight/commands/ControllerCommand.php(≈ 63-66):Proof of concept
Impact
.phpfile to be included via a distinct template-include weakness).\separator opens additional traversal surface.Patch (fixed in
3.18.1, commitb8dd23a)The controller name is now normalized with
basename()and validated against^[A-Za-z_][A-Za-z0-9_]*$before anymkdirside effect runs.Credit
Discovered by @Rootingg.