这个功能非常酷,除了我的一种情况外,它在所有情况下都运行良好。
/**
*
* @param string $url
* @param array $params
* @return string
*/
public static function getUrlWithUpdatedParams($url, $params) {
$uri = \League\Uri\Http::createFromString($url);
$query = \League\Uri\Components\Query::createFromParams($params);
return \League\Uri\merge_query($uri, $query)->__toString();
}
我正在使用"league/uri": "^5.3"
(文档)。
这是我的测试,除了最后一个,所有断言都通过了。
public function testGetUrlWithUpdatedParams() {
$this->assertEquals(url('/yo') . '?a=7', ST::getUrlWithUpdatedParams(url('/yo'), ['a' => 7]));
$this->assertEquals(url('/demo') . '?a=6&email=a@b.com', ST::getUrlWithUpdatedParams(url('/demo'), ['a' => 6, 'email' => 'a@b.com']));
$this->assertEquals(url('/sample') . '?ctrl=1&a=8', ST::getUrlWithUpdatedParams(url('/sample?ctrl=1'), ['a' => 8]));
$this->assertEquals(url('/sample') . '?ctrl=0&c=5', ST::getUrlWithUpdatedParams(url('/sample?ctrl=1'), ['c' => 5, 'ctrl' => 0]));
$this->assertEquals(url('/sample') . '?ctrl=1', ST::getUrlWithUpdatedParams(url('/sample?ctrl=1'), []));
$this->assertEquals(url('/sample') . '?ctrl=1', ST::getUrlWithUpdatedParams(url('/sample?ctrl=1'), ['ctrl' => null]));
$this->assertEquals(url('/sample') . '?ctrl=', ST::getUrlWithUpdatedParams(url('/sample?ctrl=1'), ['ctrl' => '']));
$encodedUrlAsParam = rawurlencode(url('/test'));
$this->assertEquals(url('/demo') . '?email_=a@b.com&url=' . $encodedUrlAsParam, ST::getUrlWithUpdatedParams(url('/demo'), ['email_' => 'a@b.com', 'url' => $encodedUrlAsParam])); //This one FAILS! Why in the world does getUrlWithUpdatedParams rawurlencode a second time??
}
我发现我可以通过对参数进行双重rawurlencoding 来获得最终的断言,如下所示:
$this->assertEquals(url('/demo') . '?email_=a@b.com&url=' . rawurlencode($encodedUrlAsParam), ST::getUrlWithUpdatedParams(url('/demo'), ['email_' => 'a@b.com', 'url' => $encodedUrlAsParam]));
但是当我想从主 URL中检索 URL参数时,我感觉很奇怪,然后再进行双重解码。
发生了什么,我该如何避免这种情况?