Skip to content

Commit 437b61d

Browse files
committed
feature #51324 [HttpFoundation] Add QueryParameterRequestMatcher (alexandre-daubois)
This PR was merged into the 7.1 branch. Discussion ---------- [HttpFoundation] Add `QueryParameterRequestMatcher` | Q | A | ------------- | --- | Branch? | 7.1 | Bug fix? | no | New feature? | yes | Deprecations? | no | Tickets | _NA_ | License | MIT | Doc PR | Todo We know we need a `authorizationCode` query parameter to support a request in our authenticator. We would love to use only the `ChainRequestMatcher` to do so. I'd like to add this `QueryParameterRequestMatcher` in order to do the following: ```php class OurAuthenticator extends AbstractAuthenticator { public function supports(Request $request): ?bool { return (new ChainRequestMatcher([ // ... new PathRequestMatcher('/sso/provider'), new QueryParameterRequestMatcher('authorizationCode') ]))->matches($request); } // ... } ``` Which would match the following: `/sso/provider?authorizationCode=...` Commits ------- 448c2b14cd [HttpFoundation] Add `QueryParameterRequestMatcher`
2 parents 7d87679 + 170cea4 commit 437b61d

File tree

3 files changed

+114
-0
lines changed

3 files changed

+114
-0
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ CHANGELOG
55
---
66

77
* Add `UploadedFile::getClientOriginalPath()`
8+
* Add `QueryParameterRequestMatcher`
89

910
7.0
1011
---
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
<?php
2+
3+
/*
4+
* This file is part of the Symfony package.
5+
*
6+
* (c) Fabien Potencier <[email protected]>
7+
*
8+
* For the full copyright and license information, please view the LICENSE
9+
* file that was distributed with this source code.
10+
*/
11+
12+
namespace Symfony\Component\HttpFoundation\RequestMatcher;
13+
14+
use Symfony\Component\HttpFoundation\Request;
15+
use Symfony\Component\HttpFoundation\RequestMatcherInterface;
16+
17+
/**
18+
* Checks the presence of HTTP query parameters of a Request.
19+
*
20+
* @author Alexandre Daubois <[email protected]>
21+
*/
22+
class QueryParameterRequestMatcher implements RequestMatcherInterface
23+
{
24+
/**
25+
* @var string[]
26+
*/
27+
private array $parameters;
28+
29+
/**
30+
* @param string[]|string $parameters A parameter or a list of parameters
31+
* Strings can contain a comma-delimited list of query parameters
32+
*/
33+
public function __construct(array|string $parameters)
34+
{
35+
$this->parameters = array_reduce(array_map(strtolower(...), (array) $parameters), static fn (array $parameters, string $parameter) => array_merge($parameters, preg_split('/\s*,\s*/', $parameter)), []);
36+
}
37+
38+
public function matches(Request $request): bool
39+
{
40+
if (!$this->parameters) {
41+
return true;
42+
}
43+
44+
return 0 === \count(array_diff_assoc($this->parameters, $request->query->keys()));
45+
}
46+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
<?php
2+
3+
/*
4+
* This file is part of the Symfony package.
5+
*
6+
* (c) Fabien Potencier <[email protected]>
7+
*
8+
* For the full copyright and license information, please view the LICENSE
9+
* file that was distributed with this source code.
10+
*/
11+
12+
namespace Symfony\Component\HttpFoundation\Tests\RequestMatcher;
13+
14+
use PHPUnit\Framework\TestCase;
15+
use Symfony\Component\HttpFoundation\Request;
16+
use Symfony\Component\HttpFoundation\RequestMatcher\QueryParameterRequestMatcher;
17+
18+
class QueryParameterRequestMatcherTest extends TestCase
19+
{
20+
/**
21+
* @dataProvider getDataForArray
22+
*/
23+
public function testArray(string $uri, bool $matches)
24+
{
25+
$matcher = new QueryParameterRequestMatcher(['foo', 'bar']);
26+
$request = Request::create($uri);
27+
$this->assertSame($matches, $matcher->matches($request));
28+
}
29+
30+
/**
31+
* @dataProvider getDataForArray
32+
*/
33+
public function testCommaSeparatedString(string $uri, bool $matches)
34+
{
35+
$matcher = new QueryParameterRequestMatcher('foo, bar');
36+
$request = Request::create($uri);
37+
$this->assertSame($matches, $matcher->matches($request));
38+
}
39+
40+
/**
41+
* @dataProvider getDataForSingleString
42+
*/
43+
public function testSingleString(string $uri, bool $matches)
44+
{
45+
$matcher = new QueryParameterRequestMatcher('foo');
46+
$request = Request::create($uri);
47+
$this->assertSame($matches, $matcher->matches($request));
48+
}
49+
50+
public static function getDataForArray(): \Generator
51+
{
52+
yield ['https://example.com?foo=&bar=', true];
53+
yield ['https://example.com?foo=foo1&bar=bar1', true];
54+
yield ['https://example.com?foo=foo1&bar=bar1&baz=baz1', true];
55+
yield ['https://example.com?foo=', false];
56+
yield ['https://example.com', false];
57+
}
58+
59+
public static function getDataForSingleString(): \Generator
60+
{
61+
yield ['https://example.com?foo=&bar=', true];
62+
yield ['https://example.com?foo=foo1', true];
63+
yield ['https://example.com?foo=', true];
64+
yield ['https://example.com?bar=bar1&baz=baz1', false];
65+
yield ['https://example.com', false];
66+
}
67+
}

0 commit comments

Comments
 (0)