Skip to content

Commit f1f819e

Browse files
authored
[PWA-1119] UPWARD Path Fixes (#1)
* - Fixed test for file outside of base directory - Fixed phpcs config issue - Ran code fixer * - Make relative check more strict * - Handle not found case in file resolver * Handle not found use case
1 parent aff0248 commit f1f819e

11 files changed

+34
-28
lines changed

.php_cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -70,7 +70,7 @@ return PhpCsFixer\Config::create()
7070
'no_unreachable_default_argument_value' => true,
7171
'no_useless_else' => true,
7272
'no_useless_return' => true,
73-
'ordered_class_elements' => ['sortAlgorithm' => 'alpha'],
73+
'ordered_class_elements' => ['sort_algorithm' => 'alpha'],
7474
'ordered_imports' => ['imports_order' => ['const', 'class', 'function']],
7575
'php_unit_strict' => true,
7676
'php_unit_set_up_tear_down_visibility' => true,

composer.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@
1313
},
1414
"require-dev": {
1515
"bebat/verify": "^2.0",
16-
"friendsofphp/php-cs-fixer": "^2.13",
16+
"friendsofphp/php-cs-fixer": "^2.17",
1717
"mockery/mockery": "^1.2",
1818
"phpunit/phpunit": "^6.2",
1919
"phpmd/phpmd": "^2.6"

src/AbstractKeyValueStore.php

-1
Original file line numberDiff line numberDiff line change
@@ -149,7 +149,6 @@ public function rewind(): void
149149
/**
150150
* Assign a new key in store.
151151
*
152-
*
153152
* @throws RuntimeException if $lookup is empty
154153
* @throws RuntimeException if $lookup is already set
155154
* @throws RuntimeException if an existing parent of lookup is a scalar value

src/DefinitionIterator.php

+3-14
Original file line numberDiff line numberDiff line change
@@ -76,10 +76,7 @@ public function get($lookup, $definition = null)
7676
$originalLookup = $lookup;
7777
$lookup = $parentLookup;
7878
} else {
79-
throw new \RuntimeException(sprintf(
80-
'No definition for %s',
81-
\is_string($lookup) || is_numeric($lookup) ? $lookup : \gettype($lookup)
82-
));
79+
throw new \RuntimeException(sprintf('No definition for %s', \is_string($lookup) || is_numeric($lookup) ? $lookup : \gettype($lookup)));
8380
}
8481
}
8582

@@ -113,11 +110,7 @@ public function get($lookup, $definition = null)
113110
if (\is_array($value)) {
114111
$value = $this->get($originalLookup);
115112
} elseif (!$value instanceof Response) {
116-
throw new \RuntimeException(sprintf(
117-
'Could not get nested value %s from value of type %s',
118-
$originalLookup,
119-
\gettype($value)
120-
));
113+
throw new \RuntimeException(sprintf('Could not get nested value %s from value of type %s', $originalLookup, \gettype($value)));
121114
}
122115
}
123116

@@ -198,11 +191,7 @@ private function getFromResolver(string $lookup, $definedValue, Resolver\Resolve
198191
$resolver->setIterator($this);
199192

200193
if ($definedValue instanceof Definition && !$resolver->isValid($definedValue)) {
201-
throw new \RuntimeException(sprintf(
202-
'Definition %s is not valid for %s.',
203-
json_encode($definedValue),
204-
\get_class($resolver)
205-
));
194+
throw new \RuntimeException(sprintf('Definition %s is not valid for %s.', json_encode($definedValue), \get_class($resolver)));
206195
}
207196

208197
$value = $resolver->resolve($definedValue);

src/Resolver/Directory.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -61,7 +61,7 @@ public function resolve($definition)
6161
$filename = $this->getIterator()->get('request.url.pathname');
6262
$path = realpath($root . $filename);
6363

64-
if (!$path || strpos($path, $root) !== 0 || !is_file($path)) {
64+
if (!$path || strpos($path, $root) !== 0 || strpos($path, $upwardRoot) !== 0 || !is_file($path)) {
6565
$response->setStatusCode(Response::STATUS_CODE_404);
6666
} else {
6767
$mimeType = (new MimeTypes())->getMimeType(pathinfo($path, PATHINFO_EXTENSION));

src/Resolver/File.php

+12-6
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
namespace Magento\Upward\Resolver;
1010

1111
use Magento\Upward\Definition;
12+
use Zend\Http\Response;
1213

1314
class File extends AbstractResolver
1415
{
@@ -71,9 +72,10 @@ public function isValid(Definition $definition): bool
7172
*/
7273
public function resolve($definition)
7374
{
74-
$encoding = 'utf-8';
75-
$parse = 'auto';
76-
$path = $definition;
75+
$encoding = 'utf-8';
76+
$parse = 'auto';
77+
$path = $definition;
78+
$upwardRoot = $this->getIterator()->getRootDefinition()->getBasepath();
7779

7880
if ($definition instanceof Definition) {
7981
$path = $this->getIterator()->get('file', $definition);
@@ -86,9 +88,13 @@ public function resolve($definition)
8688
}
8789
}
8890

89-
// Path is relative, expand it from definition base path.
90-
if (substr($path, 0, 1) === '.') {
91-
$path = realpath($this->getIterator()->getRootDefinition()->getBasepath() . \DIRECTORY_SEPARATOR . $path);
91+
$path = realpath($upwardRoot . \DIRECTORY_SEPARATOR . $path);
92+
93+
if (!$path || strpos($path, $upwardRoot) !== 0) {
94+
$response = new Response();
95+
$response->setStatusCode(Response::STATUS_CODE_404);
96+
97+
return $response;
9298
}
9399

94100
$content = file_get_contents($path);

src/Resolver/Url.php

+1-2
Original file line numberDiff line numberDiff line change
@@ -62,8 +62,7 @@ public function resolve($definition)
6262
}
6363

6464
$baseUrl = $this->getIterator()->get('baseUrl', $definition)
65-
? $this->getIterator()->get('baseUrl', $definition)
66-
: self::FAKE_BASE_URL;
65+
?: self::FAKE_BASE_URL;
6766
$uri = UriFactory::factory($baseUrl);
6867

6968
if ($definition->has('hostname')) {

test/AbstractKeyValueStoreTest.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ class AbstractKeyValueStoreTest extends TestCase
1616
{
1717
public function emptyKeyDataProvider(): array
1818
{
19-
return[
19+
return [
2020
[''],
2121
[' '],
2222
];

test/Resolver/DirectoryTest.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,7 @@ public function dataProviderFor404(): array
5353
'Missing File' => ['directory' => './_data', 'filename' => '/not-real-file.txt'],
5454
'Missing Directory' => ['directory' => './not-real', 'filename' => '/sample.txt'],
5555
'Not Directory' => ['directory' => basename(__FILE__), 'filename' => '/sample.txt'],
56-
'Outside Directory' => ['directory' => './_data', 'filename' => '/../' . basename(__FILE__)],
56+
'Outside Directory' => ['directory' => '../_data', 'filename' => '/sample.txt'],
5757
'File is Root' => ['directory' => './_data', 'filename' => '/'],
5858
'File is Directory' => ['directory' => './_data', 'filename' => '/test'],
5959
];

test/Resolver/FileTest.php

+12
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@
1414
use Mockery;
1515
use Mockery\Adapter\Phpunit\MockeryPHPUnitIntegration;
1616
use PHPUnit\Framework\TestCase;
17+
use Zend\Http\Response;
1718
use function BeBat\Verify\verify;
1819

1920
class FileTest extends TestCase
@@ -106,6 +107,17 @@ public function testResolve(): void
106107
verify($this->resolver->resolve($definition))->is()->sameAs("This is a sample file.\n");
107108
}
108109

110+
public function testResolve404(): void
111+
{
112+
$notFoundResult = $this->resolver->resolve('./_data/notfound.txt');
113+
$pathResult = $this->resolver->resolve('../_data/sample.txt');
114+
115+
verify($notFoundResult)->is()->instanceOf(Response::class);
116+
verify($notFoundResult->getStatusCode())->is()->sameAs(404);
117+
verify($pathResult)->is()->instanceOf(Response::class);
118+
verify($pathResult->getStatusCode())->is()->sameAs(404);
119+
}
120+
109121
public function testResolveJson(): void
110122
{
111123
verify($this->resolver->resolve('./_data/sample.json'))->is()->sameAs(['json' => true]);

test/_data/sample.txt

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
Nothing to see here...

0 commit comments

Comments
 (0)