3 declare(strict_types
=1);
5 namespace PhpMyAdmin\Tests
;
8 use PhpMyAdmin\Current
;
10 use PHPUnit\Framework\Attributes\CoversClass
;
11 use PHPUnit\Framework\Attributes\DataProvider
;
12 use ReflectionProperty
;
15 use function parse_str
;
16 use function str_repeat
;
17 use function urldecode
;
19 #[CoversClass(Url::class)]
20 class UrlTest
extends AbstractTestCase
22 /** @var string|false|null */
23 private static string|
bool|
null $inputArgSeparator = null;
26 * Sets up the fixture, for example, opens a network connection.
27 * This method is called before a test is executed.
29 protected function setUp(): void
35 unset($_COOKIE['pma_lang']);
36 Config
::getInstance()->set('URLQueryEncryption', false);
40 * Test for Url::getCommon for DB only
42 public function testDbOnly(): void
45 Config
::getInstance()->settings
['ServerDefault'] = 3;
47 $separator = Url
::getArgSeparator();
48 $expected = 'server=2' . $separator . 'lang=en';
51 . $separator . $expected;
53 self
::assertSame($expected, Url
::getCommon(['db' => 'db']));
57 * Test for Url::getCommon with new style
59 public function testNewStyle(): void
62 Config
::getInstance()->settings
['ServerDefault'] = 3;
64 $separator = Url
::getArgSeparator();
65 $expected = 'server=2' . $separator . 'lang=en';
68 . $separator . 'table=table'
69 . $separator . $expected;
70 $params = ['db' => 'db', 'table' => 'table'];
71 self
::assertSame($expected, Url
::getCommon($params));
75 * Test for Url::getCommon with alternate divider
77 public function testWithAlternateDivider(): void
80 Config
::getInstance()->settings
['ServerDefault'] = 3;
82 $separator = Url
::getArgSeparator();
83 $expected = 'server=2' . $separator . 'lang=en';
85 $expected = '#ABC#db=db' . $separator . 'table=table' . $separator
90 ['db' => 'db', 'table' => 'table'],
97 * Test for Url::getCommon
99 public function testDefault(): void
101 Current
::$server = 2;
102 Config
::getInstance()->settings
['ServerDefault'] = 3;
104 $separator = Url
::getArgSeparator();
105 $expected = '?server=2' . $separator . 'lang=en';
106 self
::assertSame($expected, Url
::getCommon());
110 * Test for Url::getFromRoute
112 public function testGetFromRoute(): void
114 $generatedUrl = Url
::getFromRoute('/test', [
118 'change_column' => 1,
121 'index.php?route=/test&db=%253%5C%24s&table=%252%5C%24s&field=%251%5C%24s&change_column=1&lang=en',
127 * Test for Url::getFromRoute
129 public function testGetFromRouteSpecialDbName(): void
131 $generatedUrl = Url
::getFromRoute('/test', [
132 'db' => '&test=_database=',
133 'table' => '&test=_database=',
134 'field' => '&test=_database=',
135 'change_column' => 1,
137 $expectedUrl = 'index.php?route=/test&db=%26test%3D_database%3D'
138 . '&table=%26test%3D_database%3D&field=%26test%3D_database%3D&change_column=1&lang=en';
139 self
::assertSame($expectedUrl, $generatedUrl);
142 'index.php?route=/test&db=&test=_database=&table=&'
143 . 'test=_database=&field=&test=_database=&change_column=1&lang=en',
151 * Test for Url::getFromRoute
153 public function testGetFromRouteMaliciousScript(): void
155 $generatedUrl = Url
::getFromRoute('/test', [
156 'db' => '<script src="https://domain.tld/svn/trunk/html5.js"></script>',
157 'table' => '<script src="https://domain.tld/maybeweshouldusegit/trunk/html5.js"></script>',
164 'index.php?route=/test&db=%3Cscript+src%3D%22https%3A%2F%2Fdomain.tld%2Fsvn'
165 . '%2Ftrunk%2Fhtml5.js%22%3E%3C%2Fscript%3E&table=%3Cscript+src%3D%22'
166 . 'https%3A%2F%2Fdomain.tld%2Fmaybeweshouldusegit%2Ftrunk%2Fhtml5.js%22%3E%3C%2F'
167 . 'script%3E&field=1&trees=1&book=0&worm=0&lang=en',
172 public function testGetHiddenFields(): void
175 self
::assertSame('', Url
::getHiddenFields([]));
177 $_SESSION = [' PMA_token ' => '<b>token</b>'];
179 '<input type="hidden" name="token" value="<b>token</b>">',
180 Url
::getHiddenFields([]),
184 public function testBuildHttpQueryWithUrlQueryEncryptionDisabled(): void
186 Config
::getInstance()->set('URLQueryEncryption', false);
187 $params = ['db' => 'test_db', 'table' => 'test_table', 'pos' => 0];
188 self
::assertSame('db=test_db&table=test_table&pos=0', Url
::buildHttpQuery($params));
191 public function testBuildHttpQueryWithUrlQueryEncryptionEnabled(): void
194 $config = Config
::getInstance();
195 $config->set('URLQueryEncryption', true);
196 $config->set('URLQueryEncryptionSecretKey', str_repeat('a', 32));
198 $params = ['db' => 'test_db', 'table' => 'test_table', 'pos' => 0];
199 $query = Url
::buildHttpQuery($params);
200 self
::assertStringStartsWith('pos=0&eq=', $query);
201 parse_str($query, $queryParams);
202 self
::assertCount(2, $queryParams);
203 self
::assertSame('0', $queryParams['pos']);
204 self
::assertIsString($queryParams['eq']);
205 self
::assertNotSame('', $queryParams['eq']);
206 self
::assertMatchesRegularExpression('/^[a-zA-Z0-9-_=]+$/', $queryParams['eq']);
208 $decrypted = Url
::decryptQuery($queryParams['eq']);
209 self
::assertNotNull($decrypted);
210 self
::assertJson($decrypted);
211 self
::assertSame('{"db":"test_db","table":"test_table"}', $decrypted);
214 public function testQueryEncryption(): void
217 $config = Config
::getInstance();
218 $config->set('URLQueryEncryption', true);
219 $config->set('URLQueryEncryptionSecretKey', str_repeat('a', 32));
221 $query = '{"db":"test_db","table":"test_table"}';
222 $encrypted = Url
::encryptQuery($query);
223 self
::assertNotSame($query, $encrypted);
224 self
::assertNotSame('', $encrypted);
225 self
::assertMatchesRegularExpression('/^[a-zA-Z0-9-_=]+$/', $encrypted);
227 $decrypted = Url
::decryptQuery($encrypted);
228 self
::assertSame($query, $decrypted);
231 /** @param string|false $iniValue */
232 #[DataProvider('getArgSeparatorProvider')]
233 public function testGetArgSeparator(string $expected, string|
bool $iniValue, string|
null $cacheValue): void
235 $property = new ReflectionProperty(Url
::class, 'inputArgSeparator');
236 $property->setValue(null, $cacheValue);
238 self
::$inputArgSeparator = $iniValue;
239 self
::assertSame($expected, Url
::getArgSeparator());
241 self
::$inputArgSeparator = null;
242 $property->setValue(null, null);
245 /** @psalm-return array<string, array{string, string|false, string|null}> */
246 public static function getArgSeparatorProvider(): array
249 'ampersand' => ['&', '&', null],
250 'semicolon' => [';', ';', null],
251 'prefer ampersand' => ['&', '+;&$', null],
252 'prefer semicolon' => [';', '+;$', null],
253 'first char' => ['+', '+$', null],
254 'cache' => ['$', '&', '$'],
255 'empty value' => ['&', '', null],
256 'false' => ['&', false, null],
261 * Test double for ini_get('arg_separator.input') as it can't be changed using ini_set()
263 * @see Url::getArgSeparatorValueFromIni
265 * @return string|false
267 public static function getInputArgSeparator(): string|
bool
269 return self
::$inputArgSeparator ??
ini_get('arg_separator.input');