123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686 |
- <?php
- namespace GuzzleHttp\Test\Handler;
- use GuzzleHttp\Exception\ConnectException;
- use GuzzleHttp\Handler\StreamHandler;
- use GuzzleHttp\Psr7;
- use GuzzleHttp\Psr7\Request;
- use GuzzleHttp\Psr7\Response;
- use GuzzleHttp\Psr7\FnStream;
- use GuzzleHttp\RequestOptions;
- use GuzzleHttp\Tests\Server;
- use GuzzleHttp\TransferStats;
- use Psr\Http\Message\ResponseInterface;
- use PHPUnit\Framework\TestCase;
- /**
- * @covers \GuzzleHttp\Handler\StreamHandler
- */
- class StreamHandlerTest extends TestCase
- {
- private function queueRes()
- {
- Server::flush();
- Server::enqueue([
- new Response(200, [
- 'Foo' => 'Bar',
- 'Content-Length' => 8,
- ], 'hi there')
- ]);
- }
- public function testReturnsResponseForSuccessfulRequest()
- {
- $this->queueRes();
- $handler = new StreamHandler();
- $response = $handler(
- new Request('GET', Server::$url, ['Foo' => 'Bar']),
- []
- )->wait();
- $this->assertSame(200, $response->getStatusCode());
- $this->assertSame('OK', $response->getReasonPhrase());
- $this->assertSame('Bar', $response->getHeaderLine('Foo'));
- $this->assertSame('8', $response->getHeaderLine('Content-Length'));
- $this->assertSame('hi there', (string) $response->getBody());
- $sent = Server::received()[0];
- $this->assertSame('GET', $sent->getMethod());
- $this->assertSame('/', $sent->getUri()->getPath());
- $this->assertSame('127.0.0.1:8126', $sent->getHeaderLine('Host'));
- $this->assertSame('Bar', $sent->getHeaderLine('foo'));
- }
- /**
- * @expectedException \GuzzleHttp\Exception\RequestException
- */
- public function testAddsErrorToResponse()
- {
- $handler = new StreamHandler();
- $handler(
- new Request('GET', 'http://localhost:123'),
- ['timeout' => 0.01]
- )->wait();
- }
- public function testStreamAttributeKeepsStreamOpen()
- {
- $this->queueRes();
- $handler = new StreamHandler();
- $request = new Request(
- 'PUT',
- Server::$url . 'foo?baz=bar',
- ['Foo' => 'Bar'],
- 'test'
- );
- $response = $handler($request, ['stream' => true])->wait();
- $this->assertSame(200, $response->getStatusCode());
- $this->assertSame('OK', $response->getReasonPhrase());
- $this->assertSame('8', $response->getHeaderLine('Content-Length'));
- $body = $response->getBody();
- $stream = $body->detach();
- $this->assertInternalType('resource', $stream);
- $this->assertSame('http', stream_get_meta_data($stream)['wrapper_type']);
- $this->assertSame('hi there', stream_get_contents($stream));
- fclose($stream);
- $sent = Server::received()[0];
- $this->assertSame('PUT', $sent->getMethod());
- $this->assertSame('http://127.0.0.1:8126/foo?baz=bar', (string) $sent->getUri());
- $this->assertSame('Bar', $sent->getHeaderLine('Foo'));
- $this->assertSame('test', (string) $sent->getBody());
- }
- public function testDrainsResponseIntoTempStream()
- {
- $this->queueRes();
- $handler = new StreamHandler();
- $request = new Request('GET', Server::$url);
- $response = $handler($request, [])->wait();
- $body = $response->getBody();
- $stream = $body->detach();
- $this->assertSame('php://temp', stream_get_meta_data($stream)['uri']);
- $this->assertSame('hi', fread($stream, 2));
- fclose($stream);
- }
- public function testDrainsResponseIntoSaveToBody()
- {
- $r = fopen('php://temp', 'r+');
- $this->queueRes();
- $handler = new StreamHandler();
- $request = new Request('GET', Server::$url);
- $response = $handler($request, ['sink' => $r])->wait();
- $body = $response->getBody()->detach();
- $this->assertSame('php://temp', stream_get_meta_data($body)['uri']);
- $this->assertSame('hi', fread($body, 2));
- $this->assertSame(' there', stream_get_contents($r));
- fclose($r);
- }
- public function testDrainsResponseIntoSaveToBodyAtPath()
- {
- $tmpfname = tempnam('/tmp', 'save_to_path');
- $this->queueRes();
- $handler = new StreamHandler();
- $request = new Request('GET', Server::$url);
- $response = $handler($request, ['sink' => $tmpfname])->wait();
- $body = $response->getBody();
- $this->assertSame($tmpfname, $body->getMetadata('uri'));
- $this->assertSame('hi', $body->read(2));
- $body->close();
- unlink($tmpfname);
- }
- public function testDrainsResponseIntoSaveToBodyAtNonExistentPath()
- {
- $tmpfname = tempnam('/tmp', 'save_to_path');
- unlink($tmpfname);
- $this->queueRes();
- $handler = new StreamHandler();
- $request = new Request('GET', Server::$url);
- $response = $handler($request, ['sink' => $tmpfname])->wait();
- $body = $response->getBody();
- $this->assertSame($tmpfname, $body->getMetadata('uri'));
- $this->assertSame('hi', $body->read(2));
- $body->close();
- unlink($tmpfname);
- }
- public function testDrainsResponseAndReadsOnlyContentLengthBytes()
- {
- Server::flush();
- Server::enqueue([
- new Response(200, [
- 'Foo' => 'Bar',
- 'Content-Length' => 8,
- ], 'hi there... This has way too much data!')
- ]);
- $handler = new StreamHandler();
- $request = new Request('GET', Server::$url);
- $response = $handler($request, [])->wait();
- $body = $response->getBody();
- $stream = $body->detach();
- $this->assertSame('hi there', stream_get_contents($stream));
- fclose($stream);
- }
- public function testDoesNotDrainWhenHeadRequest()
- {
- Server::flush();
- // Say the content-length is 8, but return no response.
- Server::enqueue([
- new Response(200, [
- 'Foo' => 'Bar',
- 'Content-Length' => 8,
- ], '')
- ]);
- $handler = new StreamHandler();
- $request = new Request('HEAD', Server::$url);
- $response = $handler($request, [])->wait();
- $body = $response->getBody();
- $stream = $body->detach();
- $this->assertSame('', stream_get_contents($stream));
- fclose($stream);
- }
- public function testAutomaticallyDecompressGzip()
- {
- Server::flush();
- $content = gzencode('test');
- Server::enqueue([
- new Response(200, [
- 'Content-Encoding' => 'gzip',
- 'Content-Length' => strlen($content),
- ], $content)
- ]);
- $handler = new StreamHandler();
- $request = new Request('GET', Server::$url);
- $response = $handler($request, ['decode_content' => true])->wait();
- $this->assertSame('test', (string) $response->getBody());
- $this->assertFalse($response->hasHeader('content-encoding'));
- $this->assertTrue(!$response->hasHeader('content-length') || $response->getHeaderLine('content-length') == $response->getBody()->getSize());
- }
- public function testReportsOriginalSizeAndContentEncodingAfterDecoding()
- {
- Server::flush();
- $content = gzencode('test');
- Server::enqueue([
- new Response(200, [
- 'Content-Encoding' => 'gzip',
- 'Content-Length' => strlen($content),
- ], $content)
- ]);
- $handler = new StreamHandler();
- $request = new Request('GET', Server::$url);
- $response = $handler($request, ['decode_content' => true])->wait();
- $this->assertSame(
- 'gzip',
- $response->getHeaderLine('x-encoded-content-encoding')
- );
- $this->assertSame(
- strlen($content),
- (int) $response->getHeaderLine('x-encoded-content-length')
- );
- }
- public function testDoesNotForceGzipDecode()
- {
- Server::flush();
- $content = gzencode('test');
- Server::enqueue([
- new Response(200, [
- 'Content-Encoding' => 'gzip',
- 'Content-Length' => strlen($content),
- ], $content)
- ]);
- $handler = new StreamHandler();
- $request = new Request('GET', Server::$url);
- $response = $handler($request, ['decode_content' => false])->wait();
- $this->assertSame($content, (string) $response->getBody());
- $this->assertSame('gzip', $response->getHeaderLine('content-encoding'));
- $this->assertEquals(strlen($content), $response->getHeaderLine('content-length'));
- }
- public function testProtocolVersion()
- {
- $this->queueRes();
- $handler = new StreamHandler();
- $request = new Request('GET', Server::$url, [], null, '1.0');
- $handler($request, []);
- $this->assertSame('1.0', Server::received()[0]->getProtocolVersion());
- }
- protected function getSendResult(array $opts)
- {
- $this->queueRes();
- $handler = new StreamHandler();
- $opts['stream'] = true;
- $request = new Request('GET', Server::$url);
- return $handler($request, $opts)->wait();
- }
- /**
- * @expectedException \GuzzleHttp\Exception\ConnectException
- * @expectedExceptionMessage Connection refused
- */
- public function testAddsProxy()
- {
- $this->getSendResult(['proxy' => '127.0.0.1:8125']);
- }
- public function testAddsProxyByProtocol()
- {
- $url = str_replace('http', 'tcp', Server::$url);
- // Workaround until #1823 is fixed properly
- $url = rtrim($url, '/');
- $res = $this->getSendResult(['proxy' => ['http' => $url]]);
- $opts = stream_context_get_options($res->getBody()->detach());
- $this->assertSame($url, $opts['http']['proxy']);
- }
- public function testAddsProxyButHonorsNoProxy()
- {
- $url = str_replace('http', 'tcp', Server::$url);
- $res = $this->getSendResult(['proxy' => [
- 'http' => $url,
- 'no' => ['*']
- ]]);
- $opts = stream_context_get_options($res->getBody()->detach());
- $this->assertArrayNotHasKey('proxy', $opts['http']);
- }
- public function testAddsTimeout()
- {
- $res = $this->getSendResult(['stream' => true, 'timeout' => 200]);
- $opts = stream_context_get_options($res->getBody()->detach());
- $this->assertEquals(200, $opts['http']['timeout']);
- }
- /**
- * @expectedException \GuzzleHttp\Exception\RequestException
- * @expectedExceptionMessage SSL CA bundle not found: /does/not/exist
- */
- public function testVerifiesVerifyIsValidIfPath()
- {
- $this->getSendResult(['verify' => '/does/not/exist']);
- }
- public function testVerifyCanBeDisabled()
- {
- $handler = $this->getSendResult(['verify' => false]);
- $this->assertInstanceOf('GuzzleHttp\Psr7\Response', $handler);
- }
- /**
- * @expectedException \GuzzleHttp\Exception\RequestException
- * @expectedExceptionMessage SSL certificate not found: /does/not/exist
- */
- public function testVerifiesCertIfValidPath()
- {
- $this->getSendResult(['cert' => '/does/not/exist']);
- }
- public function testVerifyCanBeSetToPath()
- {
- $path = $path = \GuzzleHttp\default_ca_bundle();
- $res = $this->getSendResult(['verify' => $path]);
- $opts = stream_context_get_options($res->getBody()->detach());
- $this->assertTrue($opts['ssl']['verify_peer']);
- $this->assertTrue($opts['ssl']['verify_peer_name']);
- $this->assertSame($path, $opts['ssl']['cafile']);
- $this->assertFileExists($opts['ssl']['cafile']);
- }
- public function testUsesSystemDefaultBundle()
- {
- $path = $path = \GuzzleHttp\default_ca_bundle();
- $res = $this->getSendResult(['verify' => true]);
- $opts = stream_context_get_options($res->getBody()->detach());
- if (PHP_VERSION_ID < 50600) {
- $this->assertSame($path, $opts['ssl']['cafile']);
- } else {
- $this->assertArrayNotHasKey('cafile', $opts['ssl']);
- }
- }
- /**
- * @expectedException \InvalidArgumentException
- * @expectedExceptionMessage Invalid verify request option
- */
- public function testEnsuresVerifyOptionIsValid()
- {
- $this->getSendResult(['verify' => 10]);
- }
- public function testCanSetPasswordWhenSettingCert()
- {
- $path = __FILE__;
- $res = $this->getSendResult(['cert' => [$path, 'foo']]);
- $opts = stream_context_get_options($res->getBody()->detach());
- $this->assertSame($path, $opts['ssl']['local_cert']);
- $this->assertSame('foo', $opts['ssl']['passphrase']);
- }
- public function testDebugAttributeWritesToStream()
- {
- $this->queueRes();
- $f = fopen('php://temp', 'w+');
- $this->getSendResult(['debug' => $f]);
- fseek($f, 0);
- $contents = stream_get_contents($f);
- $this->assertContains('<GET http://127.0.0.1:8126/> [CONNECT]', $contents);
- $this->assertContains('<GET http://127.0.0.1:8126/> [FILE_SIZE_IS]', $contents);
- $this->assertContains('<GET http://127.0.0.1:8126/> [PROGRESS]', $contents);
- }
- public function testDebugAttributeWritesStreamInfoToBuffer()
- {
- $called = false;
- $this->queueRes();
- $buffer = fopen('php://temp', 'r+');
- $this->getSendResult([
- 'progress' => function () use (&$called) { $called = true; },
- 'debug' => $buffer,
- ]);
- fseek($buffer, 0);
- $contents = stream_get_contents($buffer);
- $this->assertContains('<GET http://127.0.0.1:8126/> [CONNECT]', $contents);
- $this->assertContains('<GET http://127.0.0.1:8126/> [FILE_SIZE_IS] message: "Content-Length: 8"', $contents);
- $this->assertContains('<GET http://127.0.0.1:8126/> [PROGRESS] bytes_max: "8"', $contents);
- $this->assertTrue($called);
- }
- public function testEmitsProgressInformation()
- {
- $called = [];
- $this->queueRes();
- $this->getSendResult([
- 'progress' => function () use (&$called) {
- $called[] = func_get_args();
- },
- ]);
- $this->assertNotEmpty($called);
- $this->assertEquals(8, $called[0][0]);
- $this->assertEquals(0, $called[0][1]);
- }
- public function testEmitsProgressInformationAndDebugInformation()
- {
- $called = [];
- $this->queueRes();
- $buffer = fopen('php://memory', 'w+');
- $this->getSendResult([
- 'debug' => $buffer,
- 'progress' => function () use (&$called) {
- $called[] = func_get_args();
- },
- ]);
- $this->assertNotEmpty($called);
- $this->assertEquals(8, $called[0][0]);
- $this->assertEquals(0, $called[0][1]);
- rewind($buffer);
- $this->assertNotEmpty(stream_get_contents($buffer));
- fclose($buffer);
- }
- public function testPerformsShallowMergeOfCustomContextOptions()
- {
- $res = $this->getSendResult([
- 'stream_context' => [
- 'http' => [
- 'request_fulluri' => true,
- 'method' => 'HEAD',
- ],
- 'socket' => [
- 'bindto' => '127.0.0.1:0',
- ],
- 'ssl' => [
- 'verify_peer' => false,
- ],
- ],
- ]);
- $opts = stream_context_get_options($res->getBody()->detach());
- $this->assertSame('HEAD', $opts['http']['method']);
- $this->assertTrue($opts['http']['request_fulluri']);
- $this->assertSame('127.0.0.1:0', $opts['socket']['bindto']);
- $this->assertFalse($opts['ssl']['verify_peer']);
- }
- /**
- * @expectedException \InvalidArgumentException
- * @expectedExceptionMessage stream_context must be an array
- */
- public function testEnsuresThatStreamContextIsAnArray()
- {
- $this->getSendResult(['stream_context' => 'foo']);
- }
- public function testDoesNotAddContentTypeByDefault()
- {
- $this->queueRes();
- $handler = new StreamHandler();
- $request = new Request('PUT', Server::$url, ['Content-Length' => 3], 'foo');
- $handler($request, []);
- $req = Server::received()[0];
- $this->assertEquals('', $req->getHeaderLine('Content-Type'));
- $this->assertEquals(3, $req->getHeaderLine('Content-Length'));
- }
- public function testAddsContentLengthByDefault()
- {
- $this->queueRes();
- $handler = new StreamHandler();
- $request = new Request('PUT', Server::$url, [], 'foo');
- $handler($request, []);
- $req = Server::received()[0];
- $this->assertEquals(3, $req->getHeaderLine('Content-Length'));
- }
- public function testAddsContentLengthEvenWhenEmpty()
- {
- $this->queueRes();
- $handler = new StreamHandler();
- $request = new Request('PUT', Server::$url, [], '');
- $handler($request, []);
- $req = Server::received()[0];
- $this->assertEquals(0, $req->getHeaderLine('Content-Length'));
- }
- public function testSupports100Continue()
- {
- Server::flush();
- $response = new Response(200, ['Test' => 'Hello', 'Content-Length' => '4'], 'test');
- Server::enqueue([$response]);
- $request = new Request('PUT', Server::$url, ['Expect' => '100-Continue'], 'test');
- $handler = new StreamHandler();
- $response = $handler($request, [])->wait();
- $this->assertSame(200, $response->getStatusCode());
- $this->assertSame('Hello', $response->getHeaderLine('Test'));
- $this->assertSame('4', $response->getHeaderLine('Content-Length'));
- $this->assertSame('test', (string) $response->getBody());
- }
- public function testDoesSleep()
- {
- $response = new response(200);
- Server::enqueue([$response]);
- $a = new StreamHandler();
- $request = new Request('GET', Server::$url);
- $s = microtime(true);
- $a($request, ['delay' => 0.1])->wait();
- $this->assertGreaterThan(0.0001, microtime(true) - $s);
- }
- /**
- * @expectedException \InvalidArgumentException
- */
- public function testEnsuresOnHeadersIsCallable()
- {
- $req = new Request('GET', Server::$url);
- $handler = new StreamHandler();
- $handler($req, ['on_headers' => 'error!']);
- }
- /**
- * @expectedException \GuzzleHttp\Exception\RequestException
- * @expectedExceptionMessage An error was encountered during the on_headers event
- * @expectedExceptionMessage test
- */
- public function testRejectsPromiseWhenOnHeadersFails()
- {
- Server::flush();
- Server::enqueue([
- new Response(200, ['X-Foo' => 'bar'], 'abc 123')
- ]);
- $req = new Request('GET', Server::$url);
- $handler = new StreamHandler();
- $promise = $handler($req, [
- 'on_headers' => function () {
- throw new \Exception('test');
- }
- ]);
- $promise->wait();
- }
- public function testSuccessfullyCallsOnHeadersBeforeWritingToSink()
- {
- Server::flush();
- Server::enqueue([
- new Response(200, ['X-Foo' => 'bar'], 'abc 123')
- ]);
- $req = new Request('GET', Server::$url);
- $got = null;
- $stream = Psr7\stream_for();
- $stream = FnStream::decorate($stream, [
- 'write' => function ($data) use ($stream, &$got) {
- $this->assertNotNull($got);
- return $stream->write($data);
- }
- ]);
- $handler = new StreamHandler();
- $promise = $handler($req, [
- 'sink' => $stream,
- 'on_headers' => function (ResponseInterface $res) use (&$got) {
- $got = $res;
- $this->assertSame('bar', $res->getHeaderLine('X-Foo'));
- }
- ]);
- $response = $promise->wait();
- $this->assertSame(200, $response->getStatusCode());
- $this->assertSame('bar', $response->getHeaderLine('X-Foo'));
- $this->assertSame('abc 123', (string) $response->getBody());
- }
- public function testInvokesOnStatsOnSuccess()
- {
- Server::flush();
- Server::enqueue([new Psr7\Response(200)]);
- $req = new Psr7\Request('GET', Server::$url);
- $gotStats = null;
- $handler = new StreamHandler();
- $promise = $handler($req, [
- 'on_stats' => function (TransferStats $stats) use (&$gotStats) {
- $gotStats = $stats;
- }
- ]);
- $response = $promise->wait();
- $this->assertSame(200, $response->getStatusCode());
- $this->assertSame(200, $gotStats->getResponse()->getStatusCode());
- $this->assertSame(
- Server::$url,
- (string) $gotStats->getEffectiveUri()
- );
- $this->assertSame(
- Server::$url,
- (string) $gotStats->getRequest()->getUri()
- );
- $this->assertGreaterThan(0, $gotStats->getTransferTime());
- }
- public function testInvokesOnStatsOnError()
- {
- $req = new Psr7\Request('GET', 'http://127.0.0.1:123');
- $gotStats = null;
- $handler = new StreamHandler();
- $promise = $handler($req, [
- 'connect_timeout' => 0.001,
- 'timeout' => 0.001,
- 'on_stats' => function (TransferStats $stats) use (&$gotStats) {
- $gotStats = $stats;
- }
- ]);
- $promise->wait(false);
- $this->assertFalse($gotStats->hasResponse());
- $this->assertSame(
- 'http://127.0.0.1:123',
- (string) $gotStats->getEffectiveUri()
- );
- $this->assertSame(
- 'http://127.0.0.1:123',
- (string) $gotStats->getRequest()->getUri()
- );
- $this->assertInternalType('float', $gotStats->getTransferTime());
- $this->assertInstanceOf(
- ConnectException::class,
- $gotStats->getHandlerErrorData()
- );
- }
- public function testStreamIgnoresZeroTimeout()
- {
- Server::flush();
- Server::enqueue([new Psr7\Response(200)]);
- $req = new Psr7\Request('GET', Server::$url);
- $gotStats = null;
- $handler = new StreamHandler();
- $promise = $handler($req, [
- 'connect_timeout' => 10,
- 'timeout' => 0
- ]);
- $response = $promise->wait();
- $this->assertSame(200, $response->getStatusCode());
- }
- public function testDrainsResponseAndReadsAllContentWhenContentLengthIsZero()
- {
- Server::flush();
- Server::enqueue([
- new Response(200, [
- 'Foo' => 'Bar',
- 'Content-Length' => '0',
- ], 'hi there... This has a lot of data!')
- ]);
- $handler = new StreamHandler();
- $request = new Request('GET', Server::$url);
- $response = $handler($request, [])->wait();
- $body = $response->getBody();
- $stream = $body->detach();
- $this->assertSame('hi there... This has a lot of data!', stream_get_contents($stream));
- fclose($stream);
- }
- public function testHonorsReadTimeout()
- {
- Server::flush();
- $handler = new StreamHandler();
- $response = $handler(
- new Request('GET', Server::$url . 'guzzle-server/read-timeout'),
- [
- RequestOptions::READ_TIMEOUT => 1,
- RequestOptions::STREAM => true,
- ]
- )->wait();
- $this->assertSame(200, $response->getStatusCode());
- $this->assertSame('OK', $response->getReasonPhrase());
- $body = $response->getBody()->detach();
- $line = fgets($body);
- $this->assertSame("sleeping 60 seconds ...\n", $line);
- $line = fgets($body);
- $this->assertFalse($line);
- $this->assertTrue(stream_get_meta_data($body)['timed_out']);
- $this->assertFalse(feof($body));
- }
- }
|