Add neutral diagnostic framework for future reporting modules: - DiagnosticReporterInterface, Registry, Manager, PayloadSanitizer - Laravel exception hook in bootstrap/app.php - Module permission declarations (requires_permissions in module.json) - Core diagnostic report points (module boot/install/update failures) - Module documentation update (moduldoku.md) Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
367 lines
14 KiB
PHP
367 lines
14 KiB
PHP
<?php
|
|
|
|
/**
|
|
* League.Uri (https://uri.thephpleague.com)
|
|
*
|
|
* (c) Ignace Nyamagana Butera <nyamsprod@gmail.com>
|
|
*
|
|
* For the full copyright and license information, please view the LICENSE
|
|
* file that was distributed with this source code.
|
|
*/
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace League\Uri;
|
|
|
|
use League\Uri\Exceptions\SyntaxError;
|
|
use League\Uri\UriTemplate\Template;
|
|
use League\Uri\UriTemplate\TemplateCanNotBeExpanded;
|
|
use PHPUnit\Framework\Attributes\CoversClass;
|
|
use PHPUnit\Framework\Attributes\DataProvider;
|
|
use PHPUnit\Framework\Attributes\TestWith;
|
|
use PHPUnit\Framework\TestCase;
|
|
use Uri\WhatWg\InvalidUrlException;
|
|
|
|
#[CoversClass(UriTemplate::class)]
|
|
#[CoversClass(Template::class)]
|
|
final class UriTemplateTest extends TestCase
|
|
{
|
|
public function testGetTemplate(): void
|
|
{
|
|
$template = 'https://example.com{+path}{/segments}{?query,more*,foo[]*}';
|
|
$variables = [
|
|
'path' => '/foo/bar',
|
|
'segments' => ['one', 'two'],
|
|
'query' => 'test',
|
|
'more' => ['fun', 'ice cream'],
|
|
'foo[]' => ['fizz', 'buzz'],
|
|
];
|
|
|
|
self::assertSame($template, (string) new UriTemplate($template, $variables));
|
|
}
|
|
|
|
public function testGetDefaultVariables(): void
|
|
{
|
|
$template = 'https://example.com{+path}{/segments}{?query,more*,foo[]*}';
|
|
$variables = [
|
|
'path' => '/foo/bar',
|
|
'segments' => ['one', 'two', 3, true, 'false', false, null],
|
|
'query' => 'test',
|
|
'more' => ['fun', 'ice cream'],
|
|
'foo[]' => ['fizz', 'buzz'],
|
|
'nonexistent' => ['random'],
|
|
];
|
|
|
|
$expectedVariables = [
|
|
'path' => '/foo/bar',
|
|
'segments' => ['one', 'two', '3', '1', 'false', '0', ''],
|
|
'query' => 'test',
|
|
'more' => ['fun', 'ice cream'],
|
|
'foo[]' => ['fizz', 'buzz'],
|
|
];
|
|
|
|
$uriTemplate = new UriTemplate($template, $variables);
|
|
self::assertEquals($expectedVariables, [...$uriTemplate->getDefaultVariables()]);
|
|
self::assertCount(5, $uriTemplate->getDefaultVariables());
|
|
|
|
$uriTemplateEmpty = new UriTemplate($template, []);
|
|
self::assertSame([], [...$uriTemplateEmpty->getDefaultVariables()]);
|
|
self::assertCount(0, $uriTemplateEmpty->getDefaultVariables());
|
|
}
|
|
|
|
public function testWithDefaultVariables(): void
|
|
{
|
|
$template = '{foo}{bar}';
|
|
$variables = ['foo' => 'foo', 'bar' => 'bar'];
|
|
$newVariables = ['foo' => 'bar', 'bar' => 'foo'];
|
|
$newAltVariables = ['foo' => 'foo', 'bar' => 'bar', 'filteredVariable' => 'random'];
|
|
|
|
$uriTemplate = new UriTemplate($template, $variables);
|
|
$newTemplate = $uriTemplate->withDefaultVariables($newVariables);
|
|
$altTemplate = $uriTemplate->withDefaultVariables($variables);
|
|
$newAltTemplate = $uriTemplate->withDefaultVariables($newAltVariables);
|
|
|
|
self::assertEquals($altTemplate->getDefaultVariables(), $uriTemplate->getDefaultVariables());
|
|
self::assertEquals($newAltTemplate->getDefaultVariables(), $uriTemplate->getDefaultVariables());
|
|
self::assertNotEquals($newTemplate->getDefaultVariables(), $uriTemplate->getDefaultVariables());
|
|
}
|
|
|
|
#[DataProvider('expectedVariableNames')]
|
|
public function testGetVariableNames(string $template, array $expected): void
|
|
{
|
|
self::assertSame($expected, (new UriTemplate($template))->getVariableNames());
|
|
}
|
|
|
|
public static function expectedVariableNames(): iterable
|
|
{
|
|
return [
|
|
[
|
|
'template' => '',
|
|
'expected' => [],
|
|
],
|
|
[
|
|
'template' => '{foo}{bar}',
|
|
'expected' => ['foo', 'bar'],
|
|
],
|
|
[
|
|
'template' => '{foo}{foo:2}{+foo}',
|
|
'expected' => ['foo'],
|
|
],
|
|
[
|
|
'template' => '{bar}{foo}',
|
|
'expected' => ['bar', 'foo'],
|
|
],
|
|
];
|
|
}
|
|
|
|
#[DataProvider('templateExpansionProvider')]
|
|
public function testExpandsUriTemplates(string $template, string $expectedUriString, array $variables): void
|
|
{
|
|
self::assertSame($expectedUriString, (new UriTemplate($template))->expand($variables)->toString());
|
|
self::assertSame($expectedUriString, (new UriTemplate($template))->expandToUri($variables)->toRawString());
|
|
}
|
|
|
|
public static function templateExpansionProvider(): iterable
|
|
{
|
|
$variables = [
|
|
'var' => 'value',
|
|
'hello' => 'Hello World!',
|
|
'empty' => '',
|
|
'path' => '/foo/bar',
|
|
'x' => '1024',
|
|
'y' => '768',
|
|
'null' => null,
|
|
'list' => ['red', 'green', 'blue'],
|
|
'keys' => [
|
|
'semi' => ';',
|
|
'dot' => '.',
|
|
'comma' => ',',
|
|
],
|
|
'empty_keys' => [],
|
|
'bool' => true,
|
|
];
|
|
|
|
$templateAndExpansionData = [
|
|
'level 1' => [
|
|
['foo', 'foo'],
|
|
['{var}', 'value'],
|
|
['{hello}', 'Hello%20World%21'],
|
|
['{bool}', '1'],
|
|
],
|
|
'level 2' => [
|
|
['{+var}', 'value'],
|
|
['{+hello}', 'Hello%20World!'],
|
|
['{+path}/here', '/foo/bar/here'],
|
|
['here?ref={+path}', 'here?ref=/foo/bar'],
|
|
],
|
|
'level 3' => [
|
|
['X{#var}', 'X#value'],
|
|
['X{#hello}', 'X#Hello%20World!'],
|
|
['map?{x,y}', 'map?1024,768'],
|
|
['{x,hello,y}', '1024,Hello%20World%21,768'],
|
|
['{+x,hello,y}', '1024,Hello%20World!,768'],
|
|
['{+path,x}/here', '/foo/bar,1024/here'],
|
|
['{#x,hello,y}', '#1024,Hello%20World!,768'],
|
|
['{#path,x}/here', '#/foo/bar,1024/here'],
|
|
['X{.var}', 'X.value'],
|
|
['X{.x,y}', 'X.1024.768'],
|
|
['{/var}', '/value'],
|
|
['{/var,x}/here', '/value/1024/here'],
|
|
['{;x,y}', ';x=1024;y=768'],
|
|
['{;x,y,empty}', ';x=1024;y=768;empty'],
|
|
['{?x,y}', '?x=1024&y=768'],
|
|
['{?x,y,empty}', '?x=1024&y=768&empty='],
|
|
['{?x,y,undef}', '?x=1024&y=768'],
|
|
['?fixed=yes{&x}', '?fixed=yes&x=1024'],
|
|
['{&x,y,empty}', '&x=1024&y=768&empty='],
|
|
],
|
|
'level 4' => [
|
|
['{var:3}', 'val'],
|
|
['{var:30}', 'value'],
|
|
['{list}', 'red,green,blue'],
|
|
['{list*}', 'red,green,blue'],
|
|
['{keys}', 'semi,%3B,dot,.,comma,%2C'],
|
|
['{keys*}', 'semi=%3B,dot=.,comma=%2C'],
|
|
['{+path:6}/here', '/foo/b/here'],
|
|
['{+list}', 'red,green,blue'],
|
|
['{+list*}', 'red,green,blue'],
|
|
['{+keys}', 'semi,;,dot,.,comma,,'],
|
|
['{+keys*}', 'semi=;,dot=.,comma=,'],
|
|
['{#path:6}/here', '#/foo/b/here'],
|
|
['{#list}', '#red,green,blue'],
|
|
['{#list*}', '#red,green,blue'],
|
|
['{#keys}', '#semi,;,dot,.,comma,,'],
|
|
['{#keys*}', '#semi=;,dot=.,comma=,'],
|
|
['X{.var:3}', 'X.val'],
|
|
['X{.list}', 'X.red,green,blue'],
|
|
['X{.list*}', 'X.red.green.blue'],
|
|
['X{.keys}', 'X.semi,%3B,dot,.,comma,%2C'],
|
|
['X{.keys*}', 'X.semi=%3B.dot=..comma=%2C'],
|
|
['{/var:1,var}', '/v/value'],
|
|
['{/list}', '/red,green,blue'],
|
|
['{/list*}', '/red/green/blue'],
|
|
['{/list*,path:4}', '/red/green/blue/%2Ffoo'],
|
|
['{/keys}', '/semi,%3B,dot,.,comma,%2C'],
|
|
['{/keys*}', '/semi=%3B/dot=./comma=%2C'],
|
|
['{;hello:5}', ';hello=Hello'],
|
|
['{;list}', ';list=red,green,blue'],
|
|
['{;list*}', ';list=red;list=green;list=blue'],
|
|
['{;keys}', ';keys=semi,%3B,dot,.,comma,%2C'],
|
|
['{;keys*}', ';semi=%3B;dot=.;comma=%2C'],
|
|
['{?var:3}', '?var=val'],
|
|
['{?list}', '?list=red,green,blue'],
|
|
['{?list*}', '?list=red&list=green&list=blue'],
|
|
['{?keys}', '?keys=semi,%3B,dot,.,comma,%2C'],
|
|
['{?keys*}', '?semi=%3B&dot=.&comma=%2C'],
|
|
['{&var:3}', '&var=val'],
|
|
['{&list}', '&list=red,green,blue'],
|
|
['{&list*}', '&list=red&list=green&list=blue'],
|
|
['{&keys}', '&keys=semi,%3B,dot,.,comma,%2C'],
|
|
['{&keys*}', '&semi=%3B&dot=.&comma=%2C'],
|
|
['{.null}', ''],
|
|
['{.null,var}', '.value'],
|
|
['X{.empty_keys*}', 'X'],
|
|
['X{.empty_keys}', 'X'],
|
|
],
|
|
'extra' => [
|
|
// Test that missing expansions are skipped
|
|
['test{&missing*}', 'test'],
|
|
// Test that multiple expansions can be set
|
|
['http://{var}/{var:2}{?keys*}', 'http://value/va?semi=%3B&dot=.&comma=%2C'],
|
|
// Test more complex query string stuff
|
|
['http://www.test.com{+path}{?var,keys*}', 'http://www.test.com/foo/bar?var=value&semi=%3B&dot=.&comma=%2C'],
|
|
],
|
|
];
|
|
|
|
foreach ($templateAndExpansionData as $specification => $tests) {
|
|
foreach ($tests as $offset => $test) {
|
|
yield $specification.' test '.$offset => [
|
|
'template' => $test[0],
|
|
'expectedUriString' => $test[1],
|
|
'variables' => $variables,
|
|
];
|
|
}
|
|
}
|
|
}
|
|
|
|
public function testAllowsQueryValuePairsArrayExpansion(): void
|
|
{
|
|
$template = 'http://example.com{+path}{/segments}{?query,more*,foo[]*}';
|
|
$variables = [
|
|
'path' => '/foo/bar',
|
|
'segments' => ['one', 'two'],
|
|
'query' => 'test',
|
|
'more' => ['fun', 'ice cream'],
|
|
'foo[]' => ['fizz', 'buzz'],
|
|
];
|
|
$expectedUri = 'http://example.com/foo/bar/one,two?query=test&more=fun&more=ice%20cream&foo[]=fizz&foo[]=buzz';
|
|
|
|
$uriTemplate = new UriTemplate($template);
|
|
self::assertSame($expectedUri, $uriTemplate->expandToUrl($variables)->toAsciiString());
|
|
self::assertSame($expectedUri, $uriTemplate->expandToUrlOrFail($variables)->toAsciiString());
|
|
}
|
|
|
|
public function testDisallowNestedArrayExpansion(): void
|
|
{
|
|
$template = 'http://example.com{?query,data*,foo*}';
|
|
$variables = [
|
|
'query' => 'test',
|
|
'data' => [
|
|
'more' => ['fun', 'ice cream'],
|
|
],
|
|
'foo' => [
|
|
'baz' => [
|
|
'bar' => 'fizz',
|
|
'test' => 'buzz',
|
|
],
|
|
'bam' => 'boo',
|
|
],
|
|
];
|
|
|
|
self::expectException(TemplateCanNotBeExpanded::class);
|
|
|
|
(new UriTemplate($template))->expand($variables);
|
|
}
|
|
|
|
public function testExpandWithDefaultVariables(): void
|
|
{
|
|
$template = 'http://example.com{+path}{/segments}{?query,more*,foo[]*}';
|
|
$defaultVariables = [
|
|
'path' => '/foo/bar',
|
|
'segments' => ['one', 'two'],
|
|
];
|
|
$variables = [
|
|
'query' => 'test',
|
|
'more' => ['fun', 'ice cream'],
|
|
'foo[]' => ['fizz', 'buzz'],
|
|
];
|
|
$expectedUri = 'http://example.com/foo/bar/one,two?query=test&more=fun&more=ice%20cream&foo%5B%5D=fizz&foo%5B%5D=buzz';
|
|
|
|
$uriTemplate = new UriTemplate($template, $defaultVariables);
|
|
self::assertSame($expectedUri, $uriTemplate->expand($variables)->toString());
|
|
self::assertSame($expectedUri, $uriTemplate->expandOrFail($variables)->toString());
|
|
}
|
|
|
|
public function testExpandWithDefaultVariablesWithOverride(): void
|
|
{
|
|
$template = 'http://example.com{+path}{/segments}{?query,more*,foo[]*}';
|
|
|
|
$defaultVariables = [
|
|
'path' => '/foo/bar',
|
|
'segments' => ['one', 'two'],
|
|
];
|
|
|
|
$variables = [
|
|
'path' => '/bar/baz',
|
|
'query' => 'test',
|
|
'more' => ['fun', 'ice cream'],
|
|
'foo[]' => ['fizz', 'buzz'],
|
|
];
|
|
|
|
self::assertSame(
|
|
'http://example.com/bar/baz/one,two?query=test&more=fun&more=ice%20cream&foo%5B%5D=fizz&foo%5B%5D=buzz',
|
|
(new UriTemplate($template, $defaultVariables))->expand($variables)->toString()
|
|
);
|
|
}
|
|
|
|
/**
|
|
* @see https://github.com/uri-templates/uritemplate-test/blob/master/negative-tests.json
|
|
*/
|
|
#[TestWith(['http://example.com/}/{+foo}'])]
|
|
#[TestWith(['http://example.com/{/{+foo}'])]
|
|
public function testInvalidUriTemplate(string $template): void
|
|
{
|
|
self::expectException(SyntaxError::class);
|
|
|
|
new UriTemplate($template);
|
|
}
|
|
|
|
public function testExpansionWithMultipleSameExpression(): void
|
|
{
|
|
$template = '{foo}/{foo}';
|
|
$data = ['foo' => 'foo'];
|
|
|
|
self::assertSame('foo/foo', (new UriTemplate($template, $data))->expand()->toString());
|
|
}
|
|
|
|
public function testExpandOrFailIfVariablesAreMissing(): void
|
|
{
|
|
$this->expectException(TemplateCanNotBeExpanded::class);
|
|
|
|
(new UriTemplate('{var}'))->expandOrFail();
|
|
}
|
|
|
|
public function testExpandOrFailIfAtLeastOneVariableIsMissing(): void
|
|
{
|
|
$this->expectException(TemplateCanNotBeExpanded::class);
|
|
|
|
(new UriTemplate('{var}{baz}'))->expandOrFail(['var' => 'bar']);
|
|
}
|
|
|
|
public function testExpandToUrlFailsWithRelativeUrl(): void
|
|
{
|
|
$this->expectException(InvalidUrlException::class);
|
|
|
|
(new UriTemplate('/foobar'))->expandToUrl();
|
|
}
|
|
}
|