getUri()); } public function testRequestUriMayBeUri() { $uri = new Uri('/'); $r = new Request('GET', $uri); self::assertSame($uri, $r->getUri()); } public function testValidateRequestUri() { $this->expectExceptionGuzzle('InvalidArgumentException'); new Request('GET', '///'); } public function testCanConstructWithBody() { $r = new Request('GET', '/', [], 'baz'); self::assertInstanceOf('Psr\Http\Message\StreamInterface', $r->getBody()); self::assertSame('baz', (string) $r->getBody()); } public function testNullBody() { $r = new Request('GET', '/', [], null); self::assertInstanceOf('Psr\Http\Message\StreamInterface', $r->getBody()); self::assertSame('', (string) $r->getBody()); } public function testFalseyBody() { $r = new Request('GET', '/', [], '0'); self::assertInstanceOf('Psr\Http\Message\StreamInterface', $r->getBody()); self::assertSame('0', (string) $r->getBody()); } public function testConstructorDoesNotReadStreamBody() { $streamIsRead = false; $body = Psr7\FnStream::decorate(Psr7\Utils::streamFor(''), [ '__toString' => function () use (&$streamIsRead) { $streamIsRead = true; return ''; } ]); $r = new Request('GET', '/', [], $body); self::assertFalse($streamIsRead); self::assertSame($body, $r->getBody()); } public function testCapitalizesMethod() { $r = new Request('get', '/'); self::assertSame('GET', $r->getMethod()); } public function testCapitalizesWithMethod() { $r = new Request('GET', '/'); self::assertSame('PUT', $r->withMethod('put')->getMethod()); } public function testWithUri() { $r1 = new Request('GET', '/'); $u1 = $r1->getUri(); $u2 = new Uri('http://www.example.com'); $r2 = $r1->withUri($u2); self::assertNotSame($r1, $r2); self::assertSame($u2, $r2->getUri()); self::assertSame($u1, $r1->getUri()); } /** * @dataProvider invalidMethodsProvider */ public function testConstructWithInvalidMethods($method) { $this->expectExceptionGuzzle('InvalidArgumentException'); new Request($method, '/'); } /** * @dataProvider invalidMethodsProvider */ public function testWithInvalidMethods($method) { $r = new Request('get', '/'); $this->expectExceptionGuzzle('InvalidArgumentException'); $r->withMethod($method); } public function invalidMethodsProvider() { return [ [null], [false], [['foo']], [new \stdClass()], ]; } public function testSameInstanceWhenSameUri() { $r1 = new Request('GET', 'http://foo.com'); $r2 = $r1->withUri($r1->getUri()); self::assertSame($r1, $r2); } public function testWithRequestTarget() { $r1 = new Request('GET', '/'); $r2 = $r1->withRequestTarget('*'); self::assertSame('*', $r2->getRequestTarget()); self::assertSame('/', $r1->getRequestTarget()); } public function testRequestTargetDoesNotAllowSpaces() { $this->expectExceptionGuzzle('InvalidArgumentException'); $r1 = new Request('GET', '/'); $r1->withRequestTarget('/foo bar'); } public function testRequestTargetDefaultsToSlash() { $r1 = new Request('GET', ''); self::assertSame('/', $r1->getRequestTarget()); $r2 = new Request('GET', '*'); self::assertSame('*', $r2->getRequestTarget()); $r3 = new Request('GET', 'http://foo.com/bar baz/'); self::assertSame('/bar%20baz/', $r3->getRequestTarget()); } public function testBuildsRequestTarget() { $r1 = new Request('GET', 'http://foo.com/baz?bar=bam'); self::assertSame('/baz?bar=bam', $r1->getRequestTarget()); } public function testBuildsRequestTargetWithFalseyQuery() { $r1 = new Request('GET', 'http://foo.com/baz?0'); self::assertSame('/baz?0', $r1->getRequestTarget()); } public function testHostIsAddedFirst() { $r = new Request('GET', 'http://foo.com/baz?bar=bam', ['Foo' => 'Bar']); self::assertSame([ 'Host' => ['foo.com'], 'Foo' => ['Bar'] ], $r->getHeaders()); } public function testHeaderValueWithWhitespace() { $r = new Request('GET', 'https://example.com/', [ 'User-Agent' => 'Linux f0f489981e90 5.10.104-linuxkit 1 SMP Wed Mar 9 19:05:23 UTC 2022 x86_64' ]); self::assertSame([ 'Host' => ['example.com'], 'User-Agent' => ['Linux f0f489981e90 5.10.104-linuxkit 1 SMP Wed Mar 9 19:05:23 UTC 2022 x86_64'] ], $r->getHeaders()); } public function testCanGetHeaderAsCsv() { $r = new Request('GET', 'http://foo.com/baz?bar=bam', [ 'Foo' => ['a', 'b', 'c'] ]); self::assertSame('a, b, c', $r->getHeaderLine('Foo')); self::assertSame('', $r->getHeaderLine('Bar')); } public function testHostIsNotOverwrittenWhenPreservingHost() { $r = new Request('GET', 'http://foo.com/baz?bar=bam', ['Host' => 'a.com']); self::assertSame(['Host' => ['a.com']], $r->getHeaders()); $r2 = $r->withUri(new Uri('http://www.foo.com/bar'), true); self::assertSame('a.com', $r2->getHeaderLine('Host')); } public function testWithUriSetsHostIfNotSet() { $r = (new Request('GET', 'http://foo.com/baz?bar=bam'))->withoutHeader('Host'); self::assertSame([], $r->getHeaders()); $r2 = $r->withUri(new Uri('http://www.baz.com/bar'), true); self::assertSame('www.baz.com', $r2->getHeaderLine('Host')); } public function testOverridesHostWithUri() { $r = new Request('GET', 'http://foo.com/baz?bar=bam'); self::assertSame(['Host' => ['foo.com']], $r->getHeaders()); $r2 = $r->withUri(new Uri('http://www.baz.com/bar')); self::assertSame('www.baz.com', $r2->getHeaderLine('Host')); } public function testAggregatesHeaders() { $r = new Request('GET', '', [ 'ZOO' => 'zoobar', 'zoo' => ['foobar', 'zoobar'] ]); self::assertSame(['ZOO' => ['zoobar', 'foobar', 'zoobar']], $r->getHeaders()); self::assertSame('zoobar, foobar, zoobar', $r->getHeaderLine('zoo')); } public function testAddsPortToHeader() { $r = new Request('GET', 'http://foo.com:8124/bar'); self::assertSame('foo.com:8124', $r->getHeaderLine('host')); } public function testAddsPortToHeaderAndReplacePreviousPort() { $r = new Request('GET', 'http://foo.com:8124/bar'); $r = $r->withUri(new Uri('http://foo.com:8125/bar')); self::assertSame('foo.com:8125', $r->getHeaderLine('host')); } /** * @dataProvider provideHeaderValuesContainingNotAllowedChars */ public function testContainsNotAllowedCharsOnHeaderValue($value) { $this->expectExceptionGuzzle('InvalidArgumentException', sprintf('"%s" is not valid header value', $value)); $r = new Request( 'GET', 'http://foo.com/baz?bar=bam', [ 'testing' => $value ] ); } /** * @return iterable */ public function provideHeaderValuesContainingNotAllowedChars() { // Explicit tests for newlines as the most common exploit vector. $tests = [ ["new\nline"], ["new\r\nline"], ["new\rline"], // Line folding is technically allowed, but deprecated. // We don't support it. ["new\r\n line"], ["newline\n"], ["\nnewline"], ["newline\r\n"], ["\r\nnewline"], ]; for ($i = 0; $i <= 0xff; $i++) { if (\chr($i) == "\t") { continue; } if (\chr($i) == " ") { continue; } if ($i >= 0x21 && $i <= 0x7e) { continue; } if ($i >= 0x80) { continue; } $tests[] = ["foo" . \chr($i) . "bar"]; $tests[] = ["foo" . \chr($i)]; } return $tests; } }