711 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			711 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						||
 | 
						||
namespace GuzzleHttp\Tests\Psr7;
 | 
						||
 | 
						||
use GuzzleHttp\Psr7\Uri;
 | 
						||
 | 
						||
/**
 | 
						||
 * @covers GuzzleHttp\Psr7\Uri
 | 
						||
 */
 | 
						||
class UriTest extends BaseTest
 | 
						||
{
 | 
						||
    public function testParsesProvidedUri()
 | 
						||
    {
 | 
						||
        $uri = new Uri('https://user:pass@example.com:8080/path/123?q=abc#test');
 | 
						||
 | 
						||
        self::assertSame('https', $uri->getScheme());
 | 
						||
        self::assertSame('user:pass@example.com:8080', $uri->getAuthority());
 | 
						||
        self::assertSame('user:pass', $uri->getUserInfo());
 | 
						||
        self::assertSame('example.com', $uri->getHost());
 | 
						||
        self::assertSame(8080, $uri->getPort());
 | 
						||
        self::assertSame('/path/123', $uri->getPath());
 | 
						||
        self::assertSame('q=abc', $uri->getQuery());
 | 
						||
        self::assertSame('test', $uri->getFragment());
 | 
						||
        self::assertSame('https://user:pass@example.com:8080/path/123?q=abc#test', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testCanTransformAndRetrievePartsIndividually()
 | 
						||
    {
 | 
						||
        $uri = (new Uri())
 | 
						||
            ->withScheme('https')
 | 
						||
            ->withUserInfo('user', 'pass')
 | 
						||
            ->withHost('example.com')
 | 
						||
            ->withPort(8080)
 | 
						||
            ->withPath('/path/123')
 | 
						||
            ->withQuery('q=abc')
 | 
						||
            ->withFragment('test');
 | 
						||
 | 
						||
        self::assertSame('https', $uri->getScheme());
 | 
						||
        self::assertSame('user:pass@example.com:8080', $uri->getAuthority());
 | 
						||
        self::assertSame('user:pass', $uri->getUserInfo());
 | 
						||
        self::assertSame('example.com', $uri->getHost());
 | 
						||
        self::assertSame(8080, $uri->getPort());
 | 
						||
        self::assertSame('/path/123', $uri->getPath());
 | 
						||
        self::assertSame('q=abc', $uri->getQuery());
 | 
						||
        self::assertSame('test', $uri->getFragment());
 | 
						||
        self::assertSame('https://user:pass@example.com:8080/path/123?q=abc#test', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * @dataProvider getValidUris
 | 
						||
     */
 | 
						||
    public function testValidUrisStayValid($input)
 | 
						||
    {
 | 
						||
        $uri = new Uri($input);
 | 
						||
 | 
						||
        self::assertSame($input, (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * @dataProvider getValidUris
 | 
						||
     */
 | 
						||
    public function testFromParts($input)
 | 
						||
    {
 | 
						||
        $uri = Uri::fromParts(parse_url($input));
 | 
						||
 | 
						||
        self::assertSame($input, (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function getValidUris()
 | 
						||
    {
 | 
						||
        return [
 | 
						||
            ['urn:path-rootless'],
 | 
						||
            ['urn:path:with:colon'],
 | 
						||
            ['urn:/path-absolute'],
 | 
						||
            ['urn:/'],
 | 
						||
            // only scheme with empty path
 | 
						||
            ['urn:'],
 | 
						||
            // only path
 | 
						||
            ['/'],
 | 
						||
            ['relative/'],
 | 
						||
            ['0'],
 | 
						||
            // same document reference
 | 
						||
            [''],
 | 
						||
            // network path without scheme
 | 
						||
            ['//example.org'],
 | 
						||
            ['//example.org/'],
 | 
						||
            ['//example.org?q#h'],
 | 
						||
            // only query
 | 
						||
            ['?q'],
 | 
						||
            ['?q=abc&foo=bar'],
 | 
						||
            // only fragment
 | 
						||
            ['#fragment'],
 | 
						||
            // dot segments are not removed automatically
 | 
						||
            ['./foo/../bar'],
 | 
						||
        ];
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * @dataProvider getInvalidUris
 | 
						||
     */
 | 
						||
    public function testInvalidUrisThrowException($invalidUri)
 | 
						||
    {
 | 
						||
        $this->expectExceptionGuzzle('InvalidArgumentException', 'Unable to parse URI');
 | 
						||
 | 
						||
        new Uri($invalidUri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function getInvalidUris()
 | 
						||
    {
 | 
						||
        return [
 | 
						||
            // parse_url() requires the host component which makes sense for http(s)
 | 
						||
            // but not when the scheme is not known or different. So '//' or '///' is
 | 
						||
            // currently invalid as well but should not according to RFC 3986.
 | 
						||
            ['http://'],
 | 
						||
            ['urn://host:with:colon'], // host cannot contain ":"
 | 
						||
        ];
 | 
						||
    }
 | 
						||
 | 
						||
    public function testPortMustBeValid()
 | 
						||
    {
 | 
						||
        $this->expectExceptionGuzzle('InvalidArgumentException', 'Must be between 0 and 65535');
 | 
						||
 | 
						||
        (new Uri())->withPort(100000);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testWithPortCannotBeNegative()
 | 
						||
    {
 | 
						||
        $this->expectExceptionGuzzle('InvalidArgumentException', 'Invalid port: -1. Must be between 0 and 65535');
 | 
						||
 | 
						||
        (new Uri())->withPort(-1);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testParseUriPortCannotBeNegative()
 | 
						||
    {
 | 
						||
        $this->expectExceptionGuzzle('InvalidArgumentException', 'Unable to parse URI');
 | 
						||
 | 
						||
        new Uri('//example.com:-1');
 | 
						||
    }
 | 
						||
 | 
						||
    public function testSchemeMustHaveCorrectType()
 | 
						||
    {
 | 
						||
        $this->expectExceptionGuzzle('InvalidArgumentException');
 | 
						||
 | 
						||
        (new Uri())->withScheme([]);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testHostMustHaveCorrectType()
 | 
						||
    {
 | 
						||
        $this->expectExceptionGuzzle('InvalidArgumentException');
 | 
						||
 | 
						||
        (new Uri())->withHost([]);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testPathMustHaveCorrectType()
 | 
						||
    {
 | 
						||
        $this->expectExceptionGuzzle('InvalidArgumentException');
 | 
						||
 | 
						||
        (new Uri())->withPath([]);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testQueryMustHaveCorrectType()
 | 
						||
    {
 | 
						||
        $this->expectExceptionGuzzle('InvalidArgumentException');
 | 
						||
 | 
						||
        (new Uri())->withQuery([]);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testFragmentMustHaveCorrectType()
 | 
						||
    {
 | 
						||
        $this->expectExceptionGuzzle('InvalidArgumentException');
 | 
						||
 | 
						||
        (new Uri())->withFragment([]);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testCanParseFalseyUriParts()
 | 
						||
    {
 | 
						||
        $uri = new Uri('0://0:0@0/0?0#0');
 | 
						||
 | 
						||
        self::assertSame('0', $uri->getScheme());
 | 
						||
        self::assertSame('0:0@0', $uri->getAuthority());
 | 
						||
        self::assertSame('0:0', $uri->getUserInfo());
 | 
						||
        self::assertSame('0', $uri->getHost());
 | 
						||
        self::assertSame('/0', $uri->getPath());
 | 
						||
        self::assertSame('0', $uri->getQuery());
 | 
						||
        self::assertSame('0', $uri->getFragment());
 | 
						||
        self::assertSame('0://0:0@0/0?0#0', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testCanConstructFalseyUriParts()
 | 
						||
    {
 | 
						||
        $uri = (new Uri())
 | 
						||
            ->withScheme('0')
 | 
						||
            ->withUserInfo('0', '0')
 | 
						||
            ->withHost('0')
 | 
						||
            ->withPath('/0')
 | 
						||
            ->withQuery('0')
 | 
						||
            ->withFragment('0');
 | 
						||
 | 
						||
        self::assertSame('0', $uri->getScheme());
 | 
						||
        self::assertSame('0:0@0', $uri->getAuthority());
 | 
						||
        self::assertSame('0:0', $uri->getUserInfo());
 | 
						||
        self::assertSame('0', $uri->getHost());
 | 
						||
        self::assertSame('/0', $uri->getPath());
 | 
						||
        self::assertSame('0', $uri->getQuery());
 | 
						||
        self::assertSame('0', $uri->getFragment());
 | 
						||
        self::assertSame('0://0:0@0/0?0#0', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * @dataProvider getPortTestCases
 | 
						||
     */
 | 
						||
    public function testIsDefaultPort($scheme, $port, $isDefaultPort)
 | 
						||
    {
 | 
						||
        $uri = $this->getMockBuilder('Psr\Http\Message\UriInterface')->getMock();
 | 
						||
        $uri->expects(self::any())->method('getScheme')->will(self::returnValue($scheme));
 | 
						||
        $uri->expects(self::any())->method('getPort')->will(self::returnValue($port));
 | 
						||
 | 
						||
        self::assertSame($isDefaultPort, Uri::isDefaultPort($uri));
 | 
						||
    }
 | 
						||
 | 
						||
    public function getPortTestCases()
 | 
						||
    {
 | 
						||
        return [
 | 
						||
            ['http', null, true],
 | 
						||
            ['http', 80, true],
 | 
						||
            ['http', 8080, false],
 | 
						||
            ['https', null, true],
 | 
						||
            ['https', 443, true],
 | 
						||
            ['https', 444, false],
 | 
						||
            ['ftp', 21, true],
 | 
						||
            ['gopher', 70, true],
 | 
						||
            ['nntp', 119, true],
 | 
						||
            ['news', 119, true],
 | 
						||
            ['telnet', 23, true],
 | 
						||
            ['tn3270', 23, true],
 | 
						||
            ['imap', 143, true],
 | 
						||
            ['pop', 110, true],
 | 
						||
            ['ldap', 389, true],
 | 
						||
        ];
 | 
						||
    }
 | 
						||
 | 
						||
    public function testIsAbsolute()
 | 
						||
    {
 | 
						||
        self::assertTrue(Uri::isAbsolute(new Uri('http://example.org')));
 | 
						||
        self::assertFalse(Uri::isAbsolute(new Uri('//example.org')));
 | 
						||
        self::assertFalse(Uri::isAbsolute(new Uri('/abs-path')));
 | 
						||
        self::assertFalse(Uri::isAbsolute(new Uri('rel-path')));
 | 
						||
    }
 | 
						||
 | 
						||
    public function testIsNetworkPathReference()
 | 
						||
    {
 | 
						||
        self::assertFalse(Uri::isNetworkPathReference(new Uri('http://example.org')));
 | 
						||
        self::assertTrue(Uri::isNetworkPathReference(new Uri('//example.org')));
 | 
						||
        self::assertFalse(Uri::isNetworkPathReference(new Uri('/abs-path')));
 | 
						||
        self::assertFalse(Uri::isNetworkPathReference(new Uri('rel-path')));
 | 
						||
    }
 | 
						||
 | 
						||
    public function testIsAbsolutePathReference()
 | 
						||
    {
 | 
						||
        self::assertFalse(Uri::isAbsolutePathReference(new Uri('http://example.org')));
 | 
						||
        self::assertFalse(Uri::isAbsolutePathReference(new Uri('//example.org')));
 | 
						||
        self::assertTrue(Uri::isAbsolutePathReference(new Uri('/abs-path')));
 | 
						||
        self::assertTrue(Uri::isAbsolutePathReference(new Uri('/')));
 | 
						||
        self::assertFalse(Uri::isAbsolutePathReference(new Uri('rel-path')));
 | 
						||
    }
 | 
						||
 | 
						||
    public function testIsRelativePathReference()
 | 
						||
    {
 | 
						||
        self::assertFalse(Uri::isRelativePathReference(new Uri('http://example.org')));
 | 
						||
        self::assertFalse(Uri::isRelativePathReference(new Uri('//example.org')));
 | 
						||
        self::assertFalse(Uri::isRelativePathReference(new Uri('/abs-path')));
 | 
						||
        self::assertTrue(Uri::isRelativePathReference(new Uri('rel-path')));
 | 
						||
        self::assertTrue(Uri::isRelativePathReference(new Uri('')));
 | 
						||
    }
 | 
						||
 | 
						||
    public function testIsSameDocumentReference()
 | 
						||
    {
 | 
						||
        self::assertFalse(Uri::isSameDocumentReference(new Uri('http://example.org')));
 | 
						||
        self::assertFalse(Uri::isSameDocumentReference(new Uri('//example.org')));
 | 
						||
        self::assertFalse(Uri::isSameDocumentReference(new Uri('/abs-path')));
 | 
						||
        self::assertFalse(Uri::isSameDocumentReference(new Uri('rel-path')));
 | 
						||
        self::assertFalse(Uri::isSameDocumentReference(new Uri('?query')));
 | 
						||
        self::assertTrue(Uri::isSameDocumentReference(new Uri('')));
 | 
						||
        self::assertTrue(Uri::isSameDocumentReference(new Uri('#fragment')));
 | 
						||
 | 
						||
        $baseUri = new Uri('http://example.org/path?foo=bar');
 | 
						||
 | 
						||
        self::assertTrue(Uri::isSameDocumentReference(new Uri('#fragment'), $baseUri));
 | 
						||
        self::assertTrue(Uri::isSameDocumentReference(new Uri('?foo=bar#fragment'), $baseUri));
 | 
						||
        self::assertTrue(Uri::isSameDocumentReference(new Uri('/path?foo=bar#fragment'), $baseUri));
 | 
						||
        self::assertTrue(Uri::isSameDocumentReference(new Uri('path?foo=bar#fragment'), $baseUri));
 | 
						||
        self::assertTrue(Uri::isSameDocumentReference(new Uri('//example.org/path?foo=bar#fragment'), $baseUri));
 | 
						||
        self::assertTrue(Uri::isSameDocumentReference(new Uri('http://example.org/path?foo=bar#fragment'), $baseUri));
 | 
						||
 | 
						||
        self::assertFalse(Uri::isSameDocumentReference(new Uri('https://example.org/path?foo=bar'), $baseUri));
 | 
						||
        self::assertFalse(Uri::isSameDocumentReference(new Uri('http://example.com/path?foo=bar'), $baseUri));
 | 
						||
        self::assertFalse(Uri::isSameDocumentReference(new Uri('http://example.org/'), $baseUri));
 | 
						||
        self::assertFalse(Uri::isSameDocumentReference(new Uri('http://example.org'), $baseUri));
 | 
						||
 | 
						||
        self::assertFalse(Uri::isSameDocumentReference(new Uri('urn:/path'), new Uri('urn://example.com/path')));
 | 
						||
    }
 | 
						||
 | 
						||
    public function testAddAndRemoveQueryValues()
 | 
						||
    {
 | 
						||
        $uri = new Uri();
 | 
						||
        $uri = Uri::withQueryValue($uri, 'a', 'b');
 | 
						||
        $uri = Uri::withQueryValue($uri, 'c', 'd');
 | 
						||
        $uri = Uri::withQueryValue($uri, 'e', null);
 | 
						||
        self::assertSame('a=b&c=d&e', $uri->getQuery());
 | 
						||
 | 
						||
        $uri = Uri::withoutQueryValue($uri, 'c');
 | 
						||
        self::assertSame('a=b&e', $uri->getQuery());
 | 
						||
        $uri = Uri::withoutQueryValue($uri, 'e');
 | 
						||
        self::assertSame('a=b', $uri->getQuery());
 | 
						||
        $uri = Uri::withoutQueryValue($uri, 'a');
 | 
						||
        self::assertSame('', $uri->getQuery());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testNumericQueryValue()
 | 
						||
    {
 | 
						||
        $uri = Uri::withQueryValue(new Uri(), 'version', 1);
 | 
						||
        self::assertSame('version=1', $uri->getQuery());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testWithQueryValues()
 | 
						||
    {
 | 
						||
        $uri = new Uri();
 | 
						||
        $uri = Uri::withQueryValues($uri, [
 | 
						||
            'key1' => 'value1',
 | 
						||
            'key2' => 'value2'
 | 
						||
        ]);
 | 
						||
 | 
						||
        self::assertSame('key1=value1&key2=value2', $uri->getQuery());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testWithQueryValuesReplacesSameKeys()
 | 
						||
    {
 | 
						||
        $uri = new Uri();
 | 
						||
 | 
						||
        $uri = Uri::withQueryValues($uri, [
 | 
						||
            'key1' => 'value1',
 | 
						||
            'key2' => 'value2'
 | 
						||
        ]);
 | 
						||
 | 
						||
        $uri = Uri::withQueryValues($uri, [
 | 
						||
            'key2' => 'newvalue'
 | 
						||
        ]);
 | 
						||
 | 
						||
        self::assertSame('key1=value1&key2=newvalue', $uri->getQuery());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testWithQueryValueReplacesSameKeys()
 | 
						||
    {
 | 
						||
        $uri = new Uri();
 | 
						||
        $uri = Uri::withQueryValue($uri, 'a', 'b');
 | 
						||
        $uri = Uri::withQueryValue($uri, 'c', 'd');
 | 
						||
        $uri = Uri::withQueryValue($uri, 'a', 'e');
 | 
						||
        self::assertSame('c=d&a=e', $uri->getQuery());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testWithoutQueryValueRemovesAllSameKeys()
 | 
						||
    {
 | 
						||
        $uri = (new Uri())->withQuery('a=b&c=d&a=e');
 | 
						||
        $uri = Uri::withoutQueryValue($uri, 'a');
 | 
						||
        self::assertSame('c=d', $uri->getQuery());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testRemoveNonExistingQueryValue()
 | 
						||
    {
 | 
						||
        $uri = new Uri();
 | 
						||
        $uri = Uri::withQueryValue($uri, 'a', 'b');
 | 
						||
        $uri = Uri::withoutQueryValue($uri, 'c');
 | 
						||
        self::assertSame('a=b', $uri->getQuery());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testWithQueryValueHandlesEncoding()
 | 
						||
    {
 | 
						||
        $uri = new Uri();
 | 
						||
        $uri = Uri::withQueryValue($uri, 'E=mc^2', 'ein&stein');
 | 
						||
        self::assertSame('E%3Dmc%5E2=ein%26stein', $uri->getQuery(), 'Decoded key/value get encoded');
 | 
						||
 | 
						||
        $uri = new Uri();
 | 
						||
        $uri = Uri::withQueryValue($uri, 'E%3Dmc%5e2', 'ein%26stein');
 | 
						||
        self::assertSame('E%3Dmc%5e2=ein%26stein', $uri->getQuery(), 'Encoded key/value do not get double-encoded');
 | 
						||
    }
 | 
						||
 | 
						||
    public function testWithoutQueryValueHandlesEncoding()
 | 
						||
    {
 | 
						||
        // It also tests that the case of the percent-encoding does not matter,
 | 
						||
        // i.e. both lowercase "%3d" and uppercase "%5E" can be removed.
 | 
						||
        $uri = (new Uri())->withQuery('E%3dmc%5E2=einstein&foo=bar');
 | 
						||
        $uri = Uri::withoutQueryValue($uri, 'E=mc^2');
 | 
						||
        self::assertSame('foo=bar', $uri->getQuery(), 'Handles key in decoded form');
 | 
						||
 | 
						||
        $uri = (new Uri())->withQuery('E%3dmc%5E2=einstein&foo=bar');
 | 
						||
        $uri = Uri::withoutQueryValue($uri, 'E%3Dmc%5e2');
 | 
						||
        self::assertSame('foo=bar', $uri->getQuery(), 'Handles key in encoded form');
 | 
						||
    }
 | 
						||
 | 
						||
    public function testSchemeIsNormalizedToLowercase()
 | 
						||
    {
 | 
						||
        $uri = new Uri('HTTP://example.com');
 | 
						||
 | 
						||
        self::assertSame('http', $uri->getScheme());
 | 
						||
        self::assertSame('http://example.com', (string) $uri);
 | 
						||
 | 
						||
        $uri = (new Uri('//example.com'))->withScheme('HTTP');
 | 
						||
 | 
						||
        self::assertSame('http', $uri->getScheme());
 | 
						||
        self::assertSame('http://example.com', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testHostIsNormalizedToLowercase()
 | 
						||
    {
 | 
						||
        $uri = new Uri('//eXaMpLe.CoM');
 | 
						||
 | 
						||
        self::assertSame('example.com', $uri->getHost());
 | 
						||
        self::assertSame('//example.com', (string) $uri);
 | 
						||
 | 
						||
        $uri = (new Uri())->withHost('eXaMpLe.CoM');
 | 
						||
 | 
						||
        self::assertSame('example.com', $uri->getHost());
 | 
						||
        self::assertSame('//example.com', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testPortIsNullIfStandardPortForScheme()
 | 
						||
    {
 | 
						||
        // HTTPS standard port
 | 
						||
        $uri = new Uri('https://example.com:443');
 | 
						||
        self::assertNull($uri->getPort());
 | 
						||
        self::assertSame('example.com', $uri->getAuthority());
 | 
						||
 | 
						||
        $uri = (new Uri('https://example.com'))->withPort(443);
 | 
						||
        self::assertNull($uri->getPort());
 | 
						||
        self::assertSame('example.com', $uri->getAuthority());
 | 
						||
 | 
						||
        // HTTP standard port
 | 
						||
        $uri = new Uri('http://example.com:80');
 | 
						||
        self::assertNull($uri->getPort());
 | 
						||
        self::assertSame('example.com', $uri->getAuthority());
 | 
						||
 | 
						||
        $uri = (new Uri('http://example.com'))->withPort(80);
 | 
						||
        self::assertNull($uri->getPort());
 | 
						||
        self::assertSame('example.com', $uri->getAuthority());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testPortIsReturnedIfSchemeUnknown()
 | 
						||
    {
 | 
						||
        $uri = (new Uri('//example.com'))->withPort(80);
 | 
						||
 | 
						||
        self::assertSame(80, $uri->getPort());
 | 
						||
        self::assertSame('example.com:80', $uri->getAuthority());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testStandardPortIsNullIfSchemeChanges()
 | 
						||
    {
 | 
						||
        $uri = new Uri('http://example.com:443');
 | 
						||
        self::assertSame('http', $uri->getScheme());
 | 
						||
        self::assertSame(443, $uri->getPort());
 | 
						||
 | 
						||
        $uri = $uri->withScheme('https');
 | 
						||
        self::assertNull($uri->getPort());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testPortPassedAsStringIsCastedToInt()
 | 
						||
    {
 | 
						||
        $uri = (new Uri('//example.com'))->withPort('8080');
 | 
						||
 | 
						||
        self::assertSame(8080, $uri->getPort(), 'Port is returned as integer');
 | 
						||
        self::assertSame('example.com:8080', $uri->getAuthority());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testPortCanBeRemoved()
 | 
						||
    {
 | 
						||
        $uri = (new Uri('http://example.com:8080'))->withPort(null);
 | 
						||
 | 
						||
        self::assertNull($uri->getPort());
 | 
						||
        self::assertSame('http://example.com', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * In RFC 8986 the host is optional and the authority can only
 | 
						||
     * consist of the user info and port.
 | 
						||
     */
 | 
						||
    public function testAuthorityWithUserInfoOrPortButWithoutHost()
 | 
						||
    {
 | 
						||
        $uri = (new Uri())->withUserInfo('user', 'pass');
 | 
						||
 | 
						||
        self::assertSame('user:pass', $uri->getUserInfo());
 | 
						||
        self::assertSame('user:pass@', $uri->getAuthority());
 | 
						||
 | 
						||
        $uri = $uri->withPort(8080);
 | 
						||
        self::assertSame(8080, $uri->getPort());
 | 
						||
        self::assertSame('user:pass@:8080', $uri->getAuthority());
 | 
						||
        self::assertSame('//user:pass@:8080', (string) $uri);
 | 
						||
 | 
						||
        $uri = $uri->withUserInfo('');
 | 
						||
        self::assertSame(':8080', $uri->getAuthority());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testHostInHttpUriDefaultsToLocalhost()
 | 
						||
    {
 | 
						||
        $uri = (new Uri())->withScheme('http');
 | 
						||
 | 
						||
        self::assertSame('localhost', $uri->getHost());
 | 
						||
        self::assertSame('localhost', $uri->getAuthority());
 | 
						||
        self::assertSame('http://localhost', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testHostInHttpsUriDefaultsToLocalhost()
 | 
						||
    {
 | 
						||
        $uri = (new Uri())->withScheme('https');
 | 
						||
 | 
						||
        self::assertSame('localhost', $uri->getHost());
 | 
						||
        self::assertSame('localhost', $uri->getAuthority());
 | 
						||
        self::assertSame('https://localhost', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testFileSchemeWithEmptyHostReconstruction()
 | 
						||
    {
 | 
						||
        $uri = new Uri('file:///tmp/filename.ext');
 | 
						||
 | 
						||
        self::assertSame('', $uri->getHost());
 | 
						||
        self::assertSame('', $uri->getAuthority());
 | 
						||
        self::assertSame('file:///tmp/filename.ext', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function uriComponentsEncodingProvider()
 | 
						||
    {
 | 
						||
        $unreserved = 'a-zA-Z0-9.-_~!$&\'()*+,;=:@';
 | 
						||
 | 
						||
        return [
 | 
						||
            // Percent encode spaces
 | 
						||
            ['/pa th?q=va lue#frag ment', '/pa%20th', 'q=va%20lue', 'frag%20ment', '/pa%20th?q=va%20lue#frag%20ment'],
 | 
						||
            // Percent encode multibyte
 | 
						||
            ['/€?€#€', '/%E2%82%AC', '%E2%82%AC', '%E2%82%AC', '/%E2%82%AC?%E2%82%AC#%E2%82%AC'],
 | 
						||
            // Don't encode something that's already encoded
 | 
						||
            ['/pa%20th?q=va%20lue#frag%20ment', '/pa%20th', 'q=va%20lue', 'frag%20ment', '/pa%20th?q=va%20lue#frag%20ment'],
 | 
						||
            // Percent encode invalid percent encodings
 | 
						||
            ['/pa%2-th?q=va%2-lue#frag%2-ment', '/pa%252-th', 'q=va%252-lue', 'frag%252-ment', '/pa%252-th?q=va%252-lue#frag%252-ment'],
 | 
						||
            // Don't encode path segments
 | 
						||
            ['/pa/th//two?q=va/lue#frag/ment', '/pa/th//two', 'q=va/lue', 'frag/ment', '/pa/th//two?q=va/lue#frag/ment'],
 | 
						||
            // Don't encode unreserved chars or sub-delimiters
 | 
						||
            ["/$unreserved?$unreserved#$unreserved", "/$unreserved", $unreserved, $unreserved, "/$unreserved?$unreserved#$unreserved"],
 | 
						||
            // Encoded unreserved chars are not decoded
 | 
						||
            ['/p%61th?q=v%61lue#fr%61gment', '/p%61th', 'q=v%61lue', 'fr%61gment', '/p%61th?q=v%61lue#fr%61gment'],
 | 
						||
        ];
 | 
						||
    }
 | 
						||
 | 
						||
    /**
 | 
						||
     * @dataProvider uriComponentsEncodingProvider
 | 
						||
     */
 | 
						||
    public function testUriComponentsGetEncodedProperly($input, $path, $query, $fragment, $output)
 | 
						||
    {
 | 
						||
        $uri = new Uri($input);
 | 
						||
        self::assertSame($path, $uri->getPath());
 | 
						||
        self::assertSame($query, $uri->getQuery());
 | 
						||
        self::assertSame($fragment, $uri->getFragment());
 | 
						||
        self::assertSame($output, (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testWithPathEncodesProperly()
 | 
						||
    {
 | 
						||
        $uri = (new Uri())->withPath('/baz?#€/b%61r');
 | 
						||
        // Query and fragment delimiters and multibyte chars are encoded.
 | 
						||
        self::assertSame('/baz%3F%23%E2%82%AC/b%61r', $uri->getPath());
 | 
						||
        self::assertSame('/baz%3F%23%E2%82%AC/b%61r', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testWithQueryEncodesProperly()
 | 
						||
    {
 | 
						||
        $uri = (new Uri())->withQuery('?=#&€=/&b%61r');
 | 
						||
        // A query starting with a "?" is valid and must not be magically removed. Otherwise it would be impossible to
 | 
						||
        // construct such an URI. Also the "?" and "/" does not need to be encoded in the query.
 | 
						||
        self::assertSame('?=%23&%E2%82%AC=/&b%61r', $uri->getQuery());
 | 
						||
        self::assertSame('??=%23&%E2%82%AC=/&b%61r', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testWithFragmentEncodesProperly()
 | 
						||
    {
 | 
						||
        $uri = (new Uri())->withFragment('#€?/b%61r');
 | 
						||
        // A fragment starting with a "#" is valid and must not be magically removed. Otherwise it would be impossible to
 | 
						||
        // construct such an URI. Also the "?" and "/" does not need to be encoded in the fragment.
 | 
						||
        self::assertSame('%23%E2%82%AC?/b%61r', $uri->getFragment());
 | 
						||
        self::assertSame('#%23%E2%82%AC?/b%61r', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testAllowsForRelativeUri()
 | 
						||
    {
 | 
						||
        $uri = (new Uri)->withPath('foo');
 | 
						||
        self::assertSame('foo', $uri->getPath());
 | 
						||
        self::assertSame('foo', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testRelativePathAndAuhorityIsAutomagicallyFixed()
 | 
						||
    {
 | 
						||
        // concatenating a relative path with a host doesn't work: "//example.comfoo" would be wrong
 | 
						||
        $uri = (new Uri)->withPath('foo')->withHost('example.com');
 | 
						||
        self::assertSame('/foo', $uri->getPath());
 | 
						||
        self::assertSame('//example.com/foo', (string) $uri);
 | 
						||
    }
 | 
						||
 | 
						||
    public function testPathStartingWithTwoSlashesAndNoAuthorityIsInvalid()
 | 
						||
    {
 | 
						||
        $this->expectExceptionGuzzle('InvalidArgumentException', 'The path of a URI without an authority must not start with two slashes "//"');
 | 
						||
 | 
						||
        // URI "//foo" would be interpreted as network reference and thus change the original path to the host
 | 
						||
        (new Uri)->withPath('//foo');
 | 
						||
    }
 | 
						||
 | 
						||
    public function testPathStartingWithTwoSlashes()
 | 
						||
    {
 | 
						||
        $uri = new Uri('http://example.org//path-not-host.com');
 | 
						||
        self::assertSame('//path-not-host.com', $uri->getPath());
 | 
						||
 | 
						||
        $uri = $uri->withScheme('');
 | 
						||
        self::assertSame('//example.org//path-not-host.com', (string) $uri); // This is still valid
 | 
						||
        $this->expectExceptionGuzzle('\InvalidArgumentException');
 | 
						||
        $uri->withHost(''); // Now it becomes invalid
 | 
						||
    }
 | 
						||
 | 
						||
    public function testRelativeUriWithPathBeginngWithColonSegmentIsInvalid()
 | 
						||
    {
 | 
						||
        $this->expectExceptionGuzzle('InvalidArgumentException', 'A relative URI must not have a path beginning with a segment containing a colon');
 | 
						||
 | 
						||
        (new Uri)->withPath('mailto:foo');
 | 
						||
    }
 | 
						||
 | 
						||
    public function testRelativeUriWithPathHavingColonSegment()
 | 
						||
    {
 | 
						||
        $uri = (new Uri('urn:/mailto:foo'))->withScheme('');
 | 
						||
        self::assertSame('/mailto:foo', $uri->getPath());
 | 
						||
 | 
						||
        $this->expectExceptionGuzzle('\InvalidArgumentException');
 | 
						||
        (new Uri('urn:mailto:foo'))->withScheme('');
 | 
						||
    }
 | 
						||
 | 
						||
    public function testDefaultReturnValuesOfGetters()
 | 
						||
    {
 | 
						||
        $uri = new Uri();
 | 
						||
 | 
						||
        self::assertSame('', $uri->getScheme());
 | 
						||
        self::assertSame('', $uri->getAuthority());
 | 
						||
        self::assertSame('', $uri->getUserInfo());
 | 
						||
        self::assertSame('', $uri->getHost());
 | 
						||
        self::assertNull($uri->getPort());
 | 
						||
        self::assertSame('', $uri->getPath());
 | 
						||
        self::assertSame('', $uri->getQuery());
 | 
						||
        self::assertSame('', $uri->getFragment());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testImmutability()
 | 
						||
    {
 | 
						||
        $uri = new Uri();
 | 
						||
 | 
						||
        self::assertNotSame($uri, $uri->withScheme('https'));
 | 
						||
        self::assertNotSame($uri, $uri->withUserInfo('user', 'pass'));
 | 
						||
        self::assertNotSame($uri, $uri->withHost('example.com'));
 | 
						||
        self::assertNotSame($uri, $uri->withPort(8080));
 | 
						||
        self::assertNotSame($uri, $uri->withPath('/path/123'));
 | 
						||
        self::assertNotSame($uri, $uri->withQuery('q=abc'));
 | 
						||
        self::assertNotSame($uri, $uri->withFragment('test'));
 | 
						||
    }
 | 
						||
 | 
						||
    public function testExtendingClassesInstantiates()
 | 
						||
    {
 | 
						||
        // The non-standard port triggers a cascade of private methods which
 | 
						||
        // should not use late static binding to access private static members.
 | 
						||
        // If they do, this will fatal.
 | 
						||
        self::assertInstanceOf(
 | 
						||
            'GuzzleHttp\Tests\Psr7\ExtendedUriTest',
 | 
						||
            new ExtendedUriTest('http://h:9/')
 | 
						||
        );
 | 
						||
    }
 | 
						||
 | 
						||
    public function testSpecialCharsOfUserInfo()
 | 
						||
    {
 | 
						||
        // The `userInfo` must always be URL-encoded.
 | 
						||
        $uri = (new Uri)->withUserInfo('foo@bar.com', 'pass#word');
 | 
						||
        self::assertSame('foo%40bar.com:pass%23word', $uri->getUserInfo());
 | 
						||
 | 
						||
        // The `userInfo` can already be URL-encoded: it should not be encoded twice.
 | 
						||
        $uri = (new Uri)->withUserInfo('foo%40bar.com', 'pass%23word');
 | 
						||
        self::assertSame('foo%40bar.com:pass%23word', $uri->getUserInfo());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testInternationalizedDomainName()
 | 
						||
    {
 | 
						||
        $uri = new Uri('https://яндекс.рф');
 | 
						||
        self::assertSame('яндекс.рф', $uri->getHost());
 | 
						||
 | 
						||
        $uri = new Uri('https://яндекAс.рф');
 | 
						||
        self::assertSame('яндекaс.рф', $uri->getHost());
 | 
						||
    }
 | 
						||
 | 
						||
    public function testIPv6Host()
 | 
						||
    {
 | 
						||
        $uri = new Uri('https://[2a00:f48:1008::212:183:10]');
 | 
						||
        self::assertSame('[2a00:f48:1008::212:183:10]', $uri->getHost());
 | 
						||
 | 
						||
        $uri = new Uri('http://[2a00:f48:1008::212:183:10]:56?foo=bar');
 | 
						||
        self::assertSame('[2a00:f48:1008::212:183:10]', $uri->getHost());
 | 
						||
        self::assertSame(56, $uri->getPort());
 | 
						||
        self::assertSame('foo=bar', $uri->getQuery());
 | 
						||
    }
 | 
						||
}
 | 
						||
 | 
						||
class ExtendedUriTest extends Uri
 | 
						||
{
 | 
						||
}
 |