schneespur/release/schneespur-1.0.2/vendor/league/uri/UriTemplateTest.php
Michael 7288b93500 Release v1.0.2: diagnostic infrastructure core
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>
2026-05-18 16:54:11 +00:00

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();
}
}