wiki.techinc.nl/tests/phpunit/includes/http/GuzzleHttpRequestTest.php
Timo Tijhof 5943b6e691 phpunit: Replace trivial wfParseUrl use with parse_url()
Follows-up changes I57a8e7ea1b3, I1e76d2f5ae, I98ea4670e8, and
more in a search for "parse_url" in Codesearch (WMF Deployed).

Bug: T319340
Change-Id: I2b389612f4d591080f7e6235ddfc72f832b31143
2024-08-29 03:15:36 +01:00

244 lines
7.5 KiB
PHP

<?php
use GuzzleHttp\Handler\MockHandler;
use GuzzleHttp\HandlerStack;
use GuzzleHttp\Middleware;
use GuzzleHttp\Psr7\Request;
use GuzzleHttp\Psr7\Response;
/**
* class for tests of GuzzleHttpRequest
*
* No actual requests are made herein - all external communications are mocked
*
* @covers \GuzzleHttpRequest
* @covers \MWHttpRequest
*/
class GuzzleHttpRequestTest extends MediaWikiIntegrationTestCase {
/** @var int[] */
private $timeoutOptions = [
'timeout' => 1,
'connectTimeout' => 1
];
/**
* Placeholder url to use for various tests. This is never contacted, but we must use
* a url of valid format to avoid validation errors.
* @var string
*/
protected $exampleUrl = 'http://www.example.test';
/**
* Minimal example body text
* @var string
*/
protected $exampleBodyText = 'x';
/**
* For accumulating callback data for testing
* @var string
*/
protected $bodyTextReceived = '';
/**
* Callback: process a chunk of the result of a HTTP request
*
* @param mixed $req
* @param string $buffer
* @return int Number of bytes handled
*/
public function processHttpDataChunk( $req, $buffer ) {
$this->bodyTextReceived .= $buffer;
return strlen( $buffer );
}
public function testSuccess() {
$handler = HandlerStack::create( new MockHandler( [ new Response( 200, [
'status' => 200,
], $this->exampleBodyText ) ] ) );
$r = new GuzzleHttpRequest( $this->exampleUrl,
[ 'handler' => $handler ] + $this->timeoutOptions );
$r->execute();
$this->assertEquals( 200, $r->getStatus() );
$this->assertEquals( $this->exampleBodyText, $r->getContent() );
}
public function testSuccessConstructorCallback() {
$this->bodyTextReceived = '';
$handler = HandlerStack::create( new MockHandler( [ new Response( 200, [
'status' => 200,
], $this->exampleBodyText ) ] ) );
$r = new GuzzleHttpRequest( $this->exampleUrl, [
'callback' => [ $this, 'processHttpDataChunk' ],
'handler' => $handler,
] + $this->timeoutOptions );
$r->execute();
$this->assertEquals( 200, $r->getStatus() );
$this->assertEquals( $this->exampleBodyText, $this->bodyTextReceived );
}
public function testSuccessSetCallback() {
$this->bodyTextReceived = '';
$handler = HandlerStack::create( new MockHandler( [ new Response( 200, [
'status' => 200,
], $this->exampleBodyText ) ] ) );
$r = new GuzzleHttpRequest( $this->exampleUrl, [
'handler' => $handler,
] + $this->timeoutOptions );
$r->setCallback( [ $this, 'processHttpDataChunk' ] );
$r->execute();
$this->assertEquals( 200, $r->getStatus() );
$this->assertEquals( $this->exampleBodyText, $this->bodyTextReceived );
}
/**
* use a callback stream to pipe the mocked response data to our callback function
*/
public function testSuccessSink() {
$this->bodyTextReceived = '';
$handler = HandlerStack::create( new MockHandler( [ new Response( 200, [
'status' => 200,
], $this->exampleBodyText ) ] ) );
$r = new GuzzleHttpRequest( $this->exampleUrl, [
'handler' => $handler,
'sink' => new MWCallbackStream( [ $this, 'processHttpDataChunk' ] ),
] + $this->timeoutOptions );
$r->execute();
$this->assertEquals( 200, $r->getStatus() );
$this->assertEquals( $this->exampleBodyText, $this->bodyTextReceived );
}
public function testBadUrl() {
$r = new GuzzleHttpRequest( '', $this->timeoutOptions );
$s = $r->execute();
$this->assertSame( 0, $r->getStatus() );
$this->assertStatusMessage( 'http-invalid-url', $s );
}
public function testConnectException() {
$handler = HandlerStack::create( new MockHandler( [ new GuzzleHttp\Exception\ConnectException(
'Mock Connection Exception', new Request( 'GET', $this->exampleUrl )
) ] ) );
$r = new GuzzleHttpRequest( $this->exampleUrl,
[ 'handler' => $handler ] + $this->timeoutOptions );
$s = $r->execute();
$this->assertSame( 0, $r->getStatus() );
$this->assertStatusMessage( 'http-request-error', $s );
}
public function testTimeout() {
$handler = HandlerStack::create( new MockHandler( [ new GuzzleHttp\Exception\RequestException(
'Connection timed out', new Request( 'GET', $this->exampleUrl )
) ] ) );
$r = new GuzzleHttpRequest( $this->exampleUrl,
[ 'handler' => $handler ] + $this->timeoutOptions );
$s = $r->execute();
$this->assertSame( 0, $r->getStatus() );
$this->assertStatusMessage( 'http-timed-out', $s );
}
public function testNotFound() {
$handler = HandlerStack::create( new MockHandler( [ new Response( 404, [
'status' => '404',
] ) ] ) );
$r = new GuzzleHttpRequest( $this->exampleUrl,
[ 'handler' => $handler ] + $this->timeoutOptions );
$s = $r->execute();
$this->assertEquals( 404, $r->getStatus() );
$this->assertStatusMessage( 'http-bad-status', $s );
}
/*
* Test of POST requests header
*/
public function testPostBody() {
$container = [];
$history = Middleware::history( $container );
$stack = HandlerStack::create( new MockHandler( [ new Response() ] ) );
$stack->push( $history );
$client = new GuzzleHttpRequest( $this->exampleUrl, [
'method' => 'POST',
'handler' => $stack,
'post' => 'key=value',
] + $this->timeoutOptions );
$client->execute();
$request = $container[0]['request'];
$this->assertEquals( 'POST', $request->getMethod() );
$this->assertEquals( 'application/x-www-form-urlencoded',
$request->getHeader( 'Content-Type' )[0] );
}
/**
* Test POSTed multipart request body with custom content type
*/
public function testPostBodyContentType() {
$container = [];
$history = Middleware::history( $container );
$stack = HandlerStack::create( new MockHandler( [ new Response() ] ) );
$stack->push( $history );
$client = new GuzzleHttpRequest( $this->exampleUrl, [
'method' => 'POST',
'handler' => $stack,
'postData' => new \GuzzleHttp\Psr7\MultipartStream( [ [
'name' => 'a',
'contents' => 'b'
] ] ),
] + $this->timeoutOptions );
$client->setHeader( 'Content-Type', 'text/mwtest' );
$client->execute();
$request = $container[0]['request'];
$this->assertEquals( 'text/mwtest',
$request->getHeader( 'Content-Type' )[0] );
}
/*
* Test that cookies from CookieJar were sent in the outgoing request.
*/
public function testCookieSent() {
$domain = parse_url( $this->exampleUrl, PHP_URL_HOST );
$expectedCookies = [ 'cookie1' => 'value1', 'anothercookie' => 'secondvalue' ];
$jar = new CookieJar;
foreach ( $expectedCookies as $key => $val ) {
$jar->setCookie( $key, $val, [ 'domain' => $domain ] );
}
$container = [];
$history = Middleware::history( $container );
$stack = HandlerStack::create( new MockHandler( [ new Response() ] ) );
$stack->push( $history );
$client = new GuzzleHttpRequest( $this->exampleUrl, [
'method' => 'POST',
'handler' => $stack,
'post' => 'key=value',
] + $this->timeoutOptions );
$client->setCookieJar( $jar );
$client->execute();
$request = $container[0]['request'];
$this->assertEquals( [ 'cookie1=value1; anothercookie=secondvalue' ],
$request->getHeader( 'Cookie' ) );
}
/*
* Test that cookies returned by HTTP response were added back into the CookieJar.
*/
public function testCookieReceived() {
$handler = HandlerStack::create( new MockHandler( [ new Response( 200, [
'status' => 200,
'Set-Cookie' => [ 'cookie1=value1', 'anothercookie=secondvalue' ]
] ) ] ) );
$r = new GuzzleHttpRequest( $this->exampleUrl,
[ 'handler' => $handler ] + $this->timeoutOptions );
$r->execute();
$domain = parse_url( $this->exampleUrl, PHP_URL_HOST );
$this->assertEquals( 'cookie1=value1; anothercookie=secondvalue',
$r->getCookieJar()->serializeToHttpRequest( '/', $domain ) );
}
}