Merge pull request #9714 from creative-commoners/pulls/4/did-not-have-namespace

FIX Fix namespace parsing under PHP 8, tweak readability of parser
This commit is contained in:
Sam Minnée 2020-09-30 17:15:25 +13:00 committed by GitHub
commit 887b7d8af8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 44 additions and 7 deletions

View File

@ -430,7 +430,7 @@ class ClassInfo
// Keep track of the current bucket that we're putting data into // Keep track of the current bucket that we're putting data into
$bucket = &$args; $bucket = &$args;
$bucketStack = []; $bucketStack = [];
$hadNamespace = false; $lastTokenWasNSSeparator = false;
$currentKey = null; $currentKey = null;
foreach ($tokens as $token) { foreach ($tokens as $token) {
@ -443,20 +443,20 @@ class ClassInfo
($token[0] === T_NAME_QUALIFIED || $token[0] === T_NAME_FULLY_QUALIFIED) ($token[0] === T_NAME_QUALIFIED || $token[0] === T_NAME_FULLY_QUALIFIED)
) { ) {
// PHP 8 exposes the FQCN as a single T_NAME_QUALIFIED or T_NAME_FULLY_QUALIFIED token // PHP 8 exposes the FQCN as a single T_NAME_QUALIFIED or T_NAME_FULLY_QUALIFIED token
$class .= $token[1]; $class = $token[1];
$hadNamespace = true;
} elseif ($class === null && is_array($token) && $token[0] === T_STRING) { } elseif ($class === null && is_array($token) && $token[0] === T_STRING) {
$class = $token[1]; $class = $token[1];
} elseif (is_array($token) && $token[0] === T_NS_SEPARATOR) { } elseif (is_array($token) && $token[0] === T_NS_SEPARATOR) {
$class .= $token[1]; $class .= $token[1];
$hadNamespace = true; $lastTokenWasNSSeparator = true;
} elseif ($token === '.') { } elseif ($token === '.') {
// Treat service name separator as NS separator // Treat service name separator as NS separator
$class .= '.'; $class .= '.';
$hadNamespace = true; $lastTokenWasNSSeparator = true;
} elseif ($hadNamespace && is_array($token) && $token[0] === T_STRING) { } elseif ($lastTokenWasNSSeparator && is_array($token) && $token[0] === T_STRING) {
// Found another section of a namespaced class
$class .= $token[1]; $class .= $token[1];
$hadNamespace = false; $lastTokenWasNSSeparator = false;
// Get arguments // Get arguments
} elseif (is_array($token)) { } elseif (is_array($token)) {
switch ($token[0]) { switch ($token[0]) {

View File

@ -265,4 +265,41 @@ class ClassInfoTest extends SapphireTest
'ClassInfo::testClassesWithExtension() returns no classes after an extension being removed' 'ClassInfo::testClassesWithExtension() returns no classes after an extension being removed'
); );
} }
/**
* @dataProvider provideClassSpecCases
*/
public function testParseClassSpec($input, $output)
{
$this->assertEquals(
$output,
ClassInfo::parse_class_spec($input)
);
}
public function provideClassSpecCases()
{
return [
'Standard class' => [
'SimpleClass',
['SimpleClass', []],
],
'Namespaced class' => [
'Foo\\Bar\\NamespacedClass',
['Foo\\Bar\\NamespacedClass', []],
],
'Namespaced class with service name' => [
'Foo\\Bar\\NamespacedClass.withservicename',
['Foo\\Bar\\NamespacedClass.withservicename', []],
],
'Namespaced class with argument' => [
'Foo\\Bar\\NamespacedClass(["with-arg" => true])',
['Foo\\Bar\\NamespacedClass', [["with-arg" => true]]],
],
'Namespaced class with service name and argument' => [
'Foo\\Bar\\NamespacedClass.withmodifier(["and-arg" => true])',
['Foo\\Bar\\NamespacedClass.withmodifier', [["and-arg" => true]]],
],
];
}
} }