[英]Catching exceptions from Guzzle
我正在尝试从我正在开发的 API 上运行的一组测试中捕获异常,并且我正在使用 Guzzle 来使用 API 方法。 我已经将测试包装在 try/catch 块中,但它仍然抛出未处理的异常错误。 按照他们的文档中的描述添加事件侦听器似乎没有任何作用。 我需要能够检索 HTTP 代码为 500、401、400 的响应,实际上任何不是 200 的响应,因为如果它不起作用,系统将根据调用结果设置最合适的代码.
当前代码示例
foreach($tests as $test){
$client = new Client($api_url);
$client->getEventDispatcher()->addListener('request.error', function(Event $event) {
if ($event['response']->getStatusCode() == 401) {
$newResponse = new Response($event['response']->getStatusCode());
$event['response'] = $newResponse;
$event->stopPropagation();
}
});
try {
$client->setDefaultOption('query', $query_string);
$request = $client->get($api_version . $test['method'], array(), isset($test['query'])?$test['query']:array());
// Do something with Guzzle.
$response = $request->send();
displayTest($request, $response);
}
catch (Guzzle\Http\Exception\ClientErrorResponseException $e) {
$req = $e->getRequest();
$resp =$e->getResponse();
displayTest($req,$resp);
}
catch (Guzzle\Http\Exception\ServerErrorResponseException $e) {
$req = $e->getRequest();
$resp =$e->getResponse();
displayTest($req,$resp);
}
catch (Guzzle\Http\Exception\BadResponseException $e) {
$req = $e->getRequest();
$resp =$e->getResponse();
displayTest($req,$resp);
}
catch( Exception $e){
echo "AGH!";
}
unset($client);
$client=null;
}
即使对于抛出的异常类型有特定的 catch 块,我仍然会回来
Fatal error: Uncaught exception 'Guzzle\Http\Exception\ClientErrorResponseException' with message 'Client error response [status code] 401 [reason phrase] Unauthorized [url]
正如您所期望的那样,页面上的所有执行都会停止。 BadResponseException 捕获的添加使我能够正确捕获 404,但这似乎不适用于 500 或 401 响应。 任何人都可以建议我哪里出错了。
根据您的项目,可能需要禁用 guzzle 异常。 有时编码规则不允许流控制的异常。 您可以像这样禁用Guzzle 3 的异常:
$client = new \Guzzle\Http\Client($httpBase, array(
'request.options' => array(
'exceptions' => false,
)
));
这不会禁用诸如超时之类的 curl 异常,但现在您可以轻松获取每个状态代码:
$request = $client->get($uri);
$response = $request->send();
$statuscode = $response->getStatusCode();
要检查,如果你有一个有效的代码,你可以使用这样的东西:
if ($statuscode > 300) {
// Do some error handling
}
...或更好地处理所有预期的代码:
if (200 === $statuscode) {
// Do something
}
elseif (304 === $statuscode) {
// Nothing to do
}
elseif (404 === $statuscode) {
// Clean up DB or something like this
}
else {
throw new MyException("Invalid response from api...");
}
对于狂饮 5.3
$client = new \GuzzleHttp\Client(['defaults' => [ 'exceptions' => false ]] );
感谢@mika
对于狂饮 6
$client = new \GuzzleHttp\Client(['http_errors' => false]);
要捕获 Guzzle 错误,您可以执行以下操作:
try {
$response = $client->get('/not_found.xml')->send();
} catch (Guzzle\Http\Exception\BadResponseException $e) {
echo 'Uh oh! ' . $e->getMessage();
}
...但是,为了能够“记录”或“重新发送”您的请求,请尝试以下操作:
// Add custom error handling to any request created by this client
$client->getEventDispatcher()->addListener(
'request.error',
function(Event $event) {
//write log here ...
if ($event['response']->getStatusCode() == 401) {
// create new token and resend your request...
$newRequest = $event['request']->clone();
$newRequest->setHeader('X-Auth-Header', MyApplication::getNewAuthToken());
$newResponse = $newRequest->send();
// Set the response object of the request without firing more events
$event['response'] = $newResponse;
// You can also change the response and fire the normal chain of
// events by calling $event['request']->setResponse($newResponse);
// Stop other events from firing when you override 401 responses
$event->stopPropagation();
}
});
...或者如果您想“停止事件传播”,您可以覆盖事件侦听器(优先级高于 -255)并简单地停止事件传播。
$client->getEventDispatcher()->addListener('request.error', function(Event $event) {
if ($event['response']->getStatusCode() != 200) {
// Stop other events from firing when you get stytus-code != 200
$event->stopPropagation();
}
});
这是一个好主意,以防止大量错误,例如:
request.CRITICAL: Uncaught PHP Exception Guzzle\Http\Exception\ClientErrorResponseException: "Client error response
在您的应用程序中。
就我而言,我在命名空间文件上抛出Exception
,因此 php 试图捕获My\\Namespace\\Exception
因此根本没有捕获任何异常。
值得检查catch (Exception $e)
是否找到了正确的Exception
类。
只需尝试catch (\\Exception $e)
(使用那个\\
),看看它是否有效。
如果在那个try
块中抛出异常,那么在最坏的情况下, Exception
应该捕获任何未捕获的东西。
考虑到测试的第一部分是抛出异常并将其包装在try
块中。
您需要使用 http_errors => false 添加一个额外的参数
$request = $client->get($url, ['http_errors' => false]);
老问题,但 Guzzle 在异常对象中添加了响应。 因此,对GuzzleHttp\\Exception\\ClientException
进行简单的 try-catch,然后对该异常使用getResponse
以查看什么 400 级错误并从那里继续。
正如@dado 所建议的那样,我正在捕获GuzzleHttp\\Exception\\BadResponseException
。 但是有一天,当域的 DNS 不可用时,我收到了GuzzleHttp\\Exception\\ConnectException
。 所以我的建议是 - 捕获GuzzleHttp\\Exception\\ConnectException
以确保 DNS 错误的安全。
我想更新 Psr-7 Guzzle、Guzzle7 和 HTTPClient(围绕 laravel 提供的 Guzzle HTTP 客户端的表达性、最小 API)中异常处理的答案。
使用 RequestException ,RequestException 捕获传输请求时可能抛出的任何异常。
try{
$client = new \GuzzleHttp\Client(['headers' => ['Authorization' => 'Bearer ' . $token]]);
$guzzleResponse = $client->get('/foobar');
// or can use
// $guzzleResponse = $client->request('GET', '/foobar')
if ($guzzleResponse->getStatusCode() == 200) {
$response = json_decode($guzzleResponse->getBody(),true);
//perform your action with $response
}
}
catch(\GuzzleHttp\Exception\RequestException $e){
// you can catch here 400 response errors and 500 response errors
// You can either use logs here use Illuminate\Support\Facades\Log;
$error['error'] = $e->getMessage();
$error['request'] = $e->getRequest();
if($e->hasResponse()){
if ($e->getResponse()->getStatusCode() == '400'){
$error['response'] = $e->getResponse();
}
}
Log::error('Error occurred in get request.', ['error' => $error]);
}catch(Exception $e){
//other errors
}
use GuzzleHttp\Psr7;
use GuzzleHttp\Exception\RequestException;
try {
$client->request('GET', '/foo');
} catch (RequestException $e) {
$error['error'] = $e->getMessage();
$error['request'] = Psr7\Message::toString($e->getRequest());
if ($e->hasResponse()) {
$error['response'] = Psr7\Message::toString($e->getResponse());
}
Log::error('Error occurred in get request.', ['error' => $error]);
}
use Illuminate\Support\Facades\Http;
try{
$response = Http::get('http://api.foo.com');
if($response->successful()){
$reply = $response->json();
}
if($response->failed()){
if($response->clientError()){
//catch all 400 exceptions
Log::debug('client Error occurred in get request.');
$response->throw();
}
if($response->serverError()){
//catch all 500 exceptions
Log::debug('server Error occurred in get request.');
$response->throw();
}
}
}catch(Exception $e){
//catch the exception here
}
try {
} catch (GuzzleHttp\Subscriber\HttpError $e) {
//catches all 4xx and 5xx status codes
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.