Page Menu
Home
WickedGov Phorge
Search
Configure Global Search
Log In
Files
F1428757
GuzzleHttpRequest.php
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Flag For Later
Award Token
Size
9 KB
Referenced Files
None
Subscribers
None
GuzzleHttpRequest.php
View Options
<?php
/**
* @license GPL-2.0-or-later
* @file
*/
use
GuzzleHttp\Client
;
use
GuzzleHttp\Exception\ConnectException
;
use
GuzzleHttp\Exception\GuzzleException
;
use
GuzzleHttp\Exception\RequestException
;
use
GuzzleHttp\Handler\CurlHandler
;
use
GuzzleHttp\HandlerStack
;
use
GuzzleHttp\MessageFormatter
;
use
GuzzleHttp\Middleware
;
use
GuzzleHttp\Psr7\Request
;
use
MediaWiki\Status\Status
;
use
Psr\Http\Message\RequestInterface
;
use
Psr\Http\Message\StreamInterface
;
use
Psr\Log\NullLogger
;
/**
* MWHttpRequest implemented using the Guzzle library
*
* @note a new 'sink' option is available as an alternative to callbacks.
* See: http://docs.guzzlephp.org/en/stable/request-options.html#sink)
* The 'callback' option remains available as well. If both 'sink' and 'callback' are
* specified, 'sink' is used.
* @note Callers may set a custom handler via the 'handler' option.
* If this is not set, Guzzle will use curl (if available) or PHP streams (otherwise)
* @note Setting either sslVerifyHost or sslVerifyCert will enable both.
* Guzzle does not allow them to be set separately.
*
* @since 1.33
*/
class
GuzzleHttpRequest
extends
MWHttpRequest
{
public
const
SUPPORTS_FILE_POSTS
=
true
;
/** @var callable|null */
protected
$handler
=
null
;
protected
?
StreamInterface
$sink
=
null
;
protected
array
$guzzleOptions
=
[
'http_errors'
=>
false
];
/**
* @internal Use HttpRequestFactory
*
* @param string $url Url to use. If protocol-relative, will be expanded to an http:// URL
* @param array $options (optional) extra params to pass (see HttpRequestFactory::create())
* @param string $caller The method making this request, for profiling @phan-mandatory-param
* @param Profiler|null $profiler An instance of the profiler for profiling, or null
* @throws Exception
*/
public
function
__construct
(
$url
,
array
$options
=
[],
$caller
=
__METHOD__
,
?
Profiler
$profiler
=
null
)
{
parent
::
__construct
(
$url
,
$options
,
$caller
,
$profiler
);
if
(
isset
(
$options
[
'handler'
]
)
)
{
$this
->
handler
=
$options
[
'handler'
];
}
if
(
isset
(
$options
[
'sink'
]
)
)
{
$this
->
sink
=
$options
[
'sink'
];
}
}
/**
* Set a read callback to accept data read from the HTTP request.
* By default, data is appended to an internal buffer which can be
* retrieved through $req->getContent().
*
* To handle data as it comes in -- especially for large files that
* would not fit in memory -- you can instead set your own callback,
* in the form function($resource, $buffer) where the first parameter
* is the low-level resource being read (implementation specific),
* and the second parameter is the data buffer.
*
* You MUST return the number of bytes handled in the buffer; if fewer
* bytes are reported handled than were passed to you, the HTTP fetch
* will be aborted.
*
* This function overrides any 'sink' or 'callback' constructor option.
*
* @param callable|null $callback
*/
public
function
setCallback
(
$callback
)
{
$this
->
sink
=
null
;
$this
->
doSetCallback
(
$callback
);
}
/**
* Worker function for setting callbacks. Calls can originate both internally and externally
* via setCallback). Defaults to the internal read callback if $callback is null.
*
* If a sink is already specified, this does nothing. This causes the 'sink' constructor
* option to override the 'callback' constructor option.
*
* @param callable|null $callback
*/
protected
function
doSetCallback
(
$callback
)
{
if
(
!
$this
->
sink
)
{
parent
::
doSetCallback
(
$callback
);
$this
->
sink
=
new
MWCallbackStream
(
$this
->
callback
);
}
}
/**
* @see MWHttpRequest::execute
*
* @return Status
*/
public
function
execute
()
{
$this
->
prepare
();
if
(
!
$this
->
status
->
isOK
()
)
{
// TODO B/C; move this to callers
return
Status
::
wrap
(
$this
->
status
);
}
if
(
$this
->
proxy
)
{
$this
->
guzzleOptions
[
'proxy'
]
=
$this
->
proxy
;
}
$this
->
guzzleOptions
[
'timeout'
]
=
$this
->
timeout
;
$this
->
guzzleOptions
[
'connect_timeout'
]
=
$this
->
connectTimeout
;
$this
->
guzzleOptions
[
'version'
]
=
'1.1'
;
if
(
!
$this
->
followRedirects
)
{
$this
->
guzzleOptions
[
'allow_redirects'
]
=
false
;
}
else
{
$this
->
guzzleOptions
[
'allow_redirects'
]
=
[
'max'
=>
$this
->
maxRedirects
];
}
if
(
$this
->
method
===
'POST'
)
{
$postData
=
$this
->
postData
;
if
(
is_array
(
$postData
)
)
{
$this
->
guzzleOptions
[
'form_params'
]
=
$postData
;
}
else
{
$this
->
guzzleOptions
[
'body'
]
=
$postData
;
// mimic CURLOPT_POST option
if
(
!
isset
(
$this
->
reqHeaders
[
'Content-Type'
]
)
)
{
$this
->
reqHeaders
[
'Content-Type'
]
=
'application/x-www-form-urlencoded'
;
}
}
// Suppress 'Expect: 100-continue' header, as some servers
// will reject it with a 417 and Curl won't auto retry
// with HTTP 1.0 fallback
$this
->
guzzleOptions
[
'expect'
]
=
false
;
}
$stack
=
HandlerStack
::
create
(
$this
->
handler
);
// Create Middleware to use cookies from $this->getCookieJar(),
// which is in MediaWiki CookieJar format, not in Guzzle-specific CookieJar format.
// Note: received cookies (from HTTP response) don't need to be handled here,
// they will be added back into the CookieJar by MWHttpRequest::parseCookies().
$stack
->
remove
(
'cookies'
);
$mwCookieJar
=
$this
->
getCookieJar
();
$stack
->
push
(
Middleware
::
mapRequest
(
static
function
(
RequestInterface
$request
)
use
(
$mwCookieJar
)
{
$uri
=
$request
->
getUri
();
$cookieHeader
=
$mwCookieJar
->
serializeToHttpRequest
(
$uri
->
getPath
()
?:
'/'
,
$uri
->
getHost
()
);
if
(
!
$cookieHeader
)
{
return
$request
;
}
return
$request
->
withHeader
(
'Cookie'
,
$cookieHeader
);
}
),
'cookies'
);
if
(
!
$this
->
logger
instanceof
NullLogger
)
{
$stack
->
push
(
Middleware
::
log
(
$this
->
logger
,
new
MessageFormatter
(
// TODO {error} will be 'NULL' on success which is unfortunate, but
// doesn't seem fixable without a custom formatter. Same for using
// PSR-3 variable replacement instead of raw strings.
'{method} {uri} HTTP/{version} - {code} {error}'
)
),
'logger'
);
}
$this
->
guzzleOptions
[
'handler'
]
=
$stack
;
if
(
$this
->
sink
)
{
$this
->
guzzleOptions
[
'sink'
]
=
$this
->
sink
;
}
if
(
$this
->
caInfo
)
{
$this
->
guzzleOptions
[
'verify'
]
=
$this
->
caInfo
;
}
elseif
(
!
$this
->
sslVerifyHost
&&
!
$this
->
sslVerifyCert
)
{
$this
->
guzzleOptions
[
'verify'
]
=
false
;
}
$client
=
new
Client
(
$this
->
guzzleOptions
);
$request
=
new
Request
(
$this
->
method
,
$this
->
url
);
foreach
(
$this
->
reqHeaders
as
$name
=>
$value
)
{
$request
=
$request
->
withHeader
(
$name
,
$value
);
}
try
{
$response
=
$client
->
send
(
$request
);
$this
->
headerList
=
$response
->
getHeaders
();
$this
->
respVersion
=
$response
->
getProtocolVersion
();
$this
->
respStatus
=
$response
->
getStatusCode
()
.
' '
.
$response
->
getReasonPhrase
();
}
catch
(
ConnectException
$e
)
{
// ConnectException is thrown for several reasons besides generic "timeout":
// - Connection refused
// - couldn't connect to host
// - connection attempt failed
// - Could not resolve IPv4 address for host
// - Could not resolve IPv6 address for host
if
(
$this
->
usingCurl
()
)
{
$handlerContext
=
$e
->
getHandlerContext
();
if
(
$handlerContext
[
'errno'
]
==
CURLE_OPERATION_TIMEOUTED
)
{
$this
->
status
->
fatal
(
'http-timed-out'
,
$this
->
url
);
}
else
{
$this
->
status
->
fatal
(
'http-curl-error'
,
$handlerContext
[
'error'
]
);
}
}
else
{
$this
->
status
->
fatal
(
'http-request-error'
);
}
}
catch
(
RequestException
$e
)
{
if
(
$this
->
usingCurl
()
)
{
$handlerContext
=
$e
->
getHandlerContext
();
$this
->
status
->
fatal
(
'http-curl-error'
,
$handlerContext
[
'error'
]
);
}
else
{
// Non-ideal, but the only way to identify connection timeout vs other conditions
$needle
=
'Connection timed out'
;
if
(
str_contains
(
$e
->
getMessage
(),
$needle
)
)
{
$this
->
status
->
fatal
(
'http-timed-out'
,
$this
->
url
);
}
else
{
$this
->
status
->
fatal
(
'http-request-error'
);
}
}
}
catch
(
GuzzleException
)
{
$this
->
status
->
fatal
(
'http-internal-error'
);
}
if
(
$this
->
profiler
)
{
$profileSection
=
$this
->
profiler
->
scopedProfileIn
(
__METHOD__
.
'-'
.
$this
->
profileName
);
}
if
(
$this
->
profiler
)
{
$this
->
profiler
->
scopedProfileOut
(
$profileSection
);
}
$this
->
parseHeader
();
$this
->
setStatus
();
// TODO B/C; move this to callers
return
Status
::
wrap
(
$this
->
status
);
}
protected
function
prepare
()
{
$this
->
doSetCallback
(
$this
->
callback
);
parent
::
prepare
();
}
protected
function
usingCurl
():
bool
{
return
$this
->
handler
instanceof
CurlHandler
||
(
!
$this
->
handler
&&
extension_loaded
(
'curl'
)
);
}
/**
* Guzzle provides headers as an array. Reprocess to match our expectations. Guzzle will
* have already parsed and removed the status line (in EasyHandle::createResponse).
*/
protected
function
parseHeader
()
{
// Failure without (valid) headers gets a response status of zero
if
(
!
$this
->
status
->
isOK
()
)
{
$this
->
respStatus
=
'0 Error'
;
}
foreach
(
$this
->
headerList
as
$name
=>
$values
)
{
$this
->
respHeaders
[
strtolower
(
$name
)]
=
$values
;
}
$this
->
parseCookies
();
}
}
File Metadata
Details
Attached
Mime Type
text/x-php
Expires
Sat, May 16, 16:38 (11 h, 56 m)
Storage Engine
local-disk
Storage Format
Raw Data
Storage Handle
63/9c/178069ee3bf55f5384d307679db2
Default Alt Text
GuzzleHttpRequest.php (9 KB)
Attached To
Mode
rMWPROD MediaWiki Production
Attached
Detach File
Event Timeline
Log In to Comment