HTTP-клиент
11.x
.
Почему это важно?
Введение
Laravel предлагает минимальный и выразительный API для HTTP-клиента Guzzle, позволяющий быстро выполнять исходящие запросы для взаимодействия с другими веб-приложениями. Обертка вокруг Guzzle ориентирована на наиболее распространенные варианты использования и дает прекрасные возможности для разработчиков.
Вы должны убедиться, что пакет Guzzle включен в зависимости вашего приложения. По умолчанию Laravel автоматически включает эту зависимость. Однако, если вы ранее удалили пакет, то вы можете установить его снова через Composer:
composer require guzzlehttp/guzzle
Выполнение запросов
Для отправки запросов вы можете использовать методы head
, get
, post
, put
, patch
и delete
фасада Http
. Сначала давайте рассмотрим, как сделать основной запрос GET
:
use Illuminate\Support\Facades\Http;
$response = Http::get('http://example.com');
Метод get
возвращает экземпляр Illuminate\Http\Client\Response
, содержащий методы, которые можно использовать для получения информации об ответе:
$response->body() : string;
$response->json($key = null) : array|mixed;
$response->object() : object;
$response->collect($key = null) : Illuminate\Support\Collection;
$response->status() : int;
$response->ok() : bool;
$response->successful() : bool;
$response->redirect(): bool;
$response->failed() : bool;
$response->serverError() : bool;
$response->clientError() : bool;
$response->header($header) : string;
$response->headers() : array;
Объект Illuminate\Http\Client\Response
также реализует интерфейс ArrayAccess
PHP, позволяющий напрямую получать доступ к данным ответа JSON:
return Http::get('http://example.com/users/1')['name'];
Вывод информации о запросах
Если вы хотите получить информацию о сформированном экземпляре исходящего запроса перед его отправкой и прекратить выполнение скрипта, вы можете добавить метод dd
в начало определения вашего запроса:
return Http::dd()->get('http://example.com');
Данные запроса
При выполнении запросов POST
, PUT
и PATCH
обычно отправляются дополнительные данные, поэтому эти методы принимают массив данных в качестве второго аргумента. По умолчанию данные будут отправляться с использованием типа содержимого application/json
:
use Illuminate\Support\Facades\Http;
$response = Http::post('http://example.com/users', [
'name' => 'Steve',
'role' => 'Network Administrator',
]);
Параметры GET-запроса
При выполнении запросов GET
вы можете либо напрямую добавить строку запроса к URL, либо передать массив пар ключ / значение в качестве второго аргумента метода get
:
$response = Http::get('http://example.com/users', [
'name' => 'Taylor',
'page' => 1,
]);
Отправка запросов с передачей данных в URL-кодированной строке
Если вы хотите отправлять данные с использованием типа содержимого application/x-www-form-urlencoded
, то вы должны вызвать метод asForm
перед выполнением запроса:
$response = Http::asForm()->post('http://example.com/users', [
'name' => 'Sara',
'role' => 'Privacy Consultant',
]);
Отправка необработанного тела запроса
Вы можете использовать метод withBody
, если хотите передать необработанное тело запроса при его выполнении. Тип контента может быть указан вторым аргументом метода:
$response = Http::withBody(
base64_encode($photo), 'image/jpeg'
)->post('http://example.com/photo');
Составные запросы
Если вы хотите отправлять файлы в запросах, состоящих из нескольких частей, необходимо вызвать метод attach
перед выполнением запроса. Этот метод принимает имя файла и его содержимое. При желании вы можете указать третий аргумент, который будет считаться именем файла:
$response = Http::attach(
'attachment', file_get_contents('photo.jpg'), 'photo.jpg'
)->post('http://example.com/attachments');
Вы также можете передать потоковый ресурс вместо передачи необработанного содержимого файла:
$photo = fopen('photo.jpg', 'r');
$response = Http::attach(
'attachment', $photo, 'photo.jpg'
)->post('http://example.com/attachments');
Заголовки
Заголовки могут быть добавлены к запросам с помощью метода withHeaders
. Метод withHeaders
принимает массив пар ключ / значение:
$response = Http::withHeaders([
'X-First' => 'foo',
'X-Second' => 'bar'
])->post('http://example.com/users', [
'name' => 'Taylor',
]);
Вы можете использовать метод accept
, чтобы указать тип контента, который ваше приложение ожидает в ответ на ваш запрос:
$response = Http::accept('application/json')->get('http://example.com/users');
Для удобства вы можете использовать метод acceptJson
, чтобы быстро указать, что ваше приложение ожидает тип содержимого application/json
в ответ на ваш запрос:
$response = Http::acceptJson()->get('http://example.com/users');
Аутентификация
Вы можете указать данные basic и digest аутентификации, используя методы withBasicAuth
и withDigestAuth
, соответственно:
// Basic HTTP-аутентификация ...
$response = Http::withBasicAuth('[email protected]', 'secret')->post(...);
// Digest HTTP-аутентификациия ...
$response = Http::withDigestAuth('[email protected]', 'secret')->post(...);
Токены Bearer
Если вы хотите добавить токен в заголовок Authorization
запроса, то используйте метод withToken
:
$response = Http::withToken('token')->post(...);
Время ожидания
Метод timeout
используется для указания максимального количества секунд ожидания ответа:
$response = Http::timeout(3)->get(...);
Если указанный тайм-аут превышен, то будет выброшено исключение Illuminate\Http\Client\ConnectionException
.
Повторные попытки
Если вы хотите, чтобы HTTP-клиент автоматически повторял запрос при возникновении ошибки клиента или сервера, то используйте метод retry
. Метод retry
принимает максимальное количество попыток выполнения запроса и количество миллисекунд, которые Laravel должен ждать между попытками:
$response = Http::retry(3, 100)->post(...);
При необходимости вы можете передать третий аргумент методу retry
. Третий аргумент должен быть callable-функцией, которая определяет, следует ли на самом деле попытаться повторить попытку. Например, вы можете захотеть повторить запрос только в том случае, если начальный запрос обнаруживает исключение ConnectionException
:
$response = Http::retry(3, 100, function ($exception) {
return $exception instanceof ConnectionException;
})->post(...);
Если все запросы окажутся неуспешными, то будет выброшено исключение Illuminate\Http\Client\RequestException
.
Обработка ошибок
В отличие от поведения Guzzle по умолчанию, обертка HTTP-клиента Laravel не генерирует исключений при возникновении ошибок клиента или сервера (ответы 400
и 500
, соответственно). Вы можете определить, была ли возвращена одна из этих ошибок, используя методы successful
, clientError
, или serverError
:
// Определить, имеет ли ответ код состояния >= 200 and < 300...
$response->successful();
// Определить, имеет ли ответ код состояния >= 400...
$response->failed();
// Определить, имеет ли ответ код состояния 400 ...
$response->clientError();
// Определить, имеет ли ответ код состояния 500 ...
$response->serverError();
// Немедленно выполнить данную функцию обратного вызова, если произошла ошибка клиента или сервера...
$response->onError(callable $callback);
Выброс исключений
Если у вас есть экземпляр ответа и вы хотите выбросить исключение Illuminate\Http\Client\RequestException
, если код состояния ответа указывает на ошибку клиента или сервера, используйте методы throw
или throwIf
:
$response = Http::post(...);
// Выбросить исключение, если произошла ошибка клиента или сервера ...
$response->throw();
// Выбросить исключение, если произошла ошибка и данное условие истинно...
$response->throwIf($condition);
return $response['user']['id'];
Экземпляр Illuminate\Http\Client\RequestException
имеет свойство $response
, которое позволит вам проверить возвращенный ответ.
Метод throw
возвращает экземпляр ответа, если ошибки не произошло, что позволяет вам использовать цепочку вызовов после метода throw
:
return Http::post(...)->throw()->json();
Если вы хотите выполнить некоторую дополнительную логику до того, как будет сгенерировано исключение, вы можете передать замыкание методу throw
. Исключение будет сгенерировано автоматически после вызова замыкания, поэтому вам не нужно повторно генерировать исключение изнутри замыкания:
return Http::post(...)->throw(function ($response, $e) {
//
})->json();
Параметры Guzzle
Вы можете указать дополнительные параметры запроса для Guzzle, используя метод withOptions
. Метод withOptions
принимает массив пар ключ / значение:
$response = Http::withOptions([
'debug' => true,
])->get('http://example.com/users');
Параллельные запросы
Иногда вы можете захотеть выполнить несколько HTTP-запросов одновременно. Другими словами, вы хотите, чтобы несколько запросов отправлялись одновременно вместо того, чтобы отправлять их последовательно. Это может привести к значительному повышению производительности при взаимодействии с медленными HTTP API.
Вы можете сделать это с помощью метода pool
. Метод pool
принимает функцию с аргументом Illuminate\Http\Client\Pool
, при помощи которого вы можете добавлять запросы в пул запросов для отправки:
use Illuminate\Http\Client\Pool;
use Illuminate\Support\Facades\Http;
$responses = Http::pool(fn (Pool $pool) => [
$pool->get('http://localhost/first'),
$pool->get('http://localhost/second'),
$pool->get('http://localhost/third'),
]);
return $responses[0]->ok() &&
$responses[1]->ok() &&
$responses[2]->ok();
Как видите, к каждому экземпляру ответа можно получить доступ в том порядке, в котором он был добавлен в пул. При желании вы можете назвать запросы с помощью метода as
, что позволит вам получить доступ к соответствующим ответам по имени:
use Illuminate\Http\Client\Pool;
use Illuminate\Support\Facades\Http;
$responses = Http::pool(fn (Pool $pool) => [
$pool->as('first')->get('http://localhost/first'),
$pool->as('second')->get('http://localhost/second'),
$pool->as('third')->get('http://localhost/third'),
]);
return $responses['first']->ok();
Макросы
HTTP-клиент Laravel позволяет вам определять «макросы», которые могут служить плавным, выразительным механизмом для настройки общих путей запросов и заголовков при взаимодействии со службами в вашем приложении. Для начала вы можете определить макрос в методе boot
класса App\Providers\AppServiceProvider
вашего приложения:
use Illuminate\Support\Facades\Http;
/**
* Bootstrap any application services.
*
* @return void
*/
public function boot()
{
Http::macro('github', function () {
return Http::withHeaders([
'X-Example' => 'example',
])->baseUrl('https://github.com');
});
}
После того, как ваш макрос настроен, вы можете вызвать его из любого места вашего приложения, чтобы создать ожидающий запрос с указанной конфигурацией:
$response = Http::github()->get('/');
Тестирование
Многие службы Laravel содержат функционал, помогающий вам легко и выразительно писать тесты, и HTTP-клиент Laravel не является исключением. Метод fake
фасада Http
позволяет указать HTTP-клиенту возвращать заглушенные / фиктивные ответы при выполнении запросов.
Фиктивные ответы
Например, чтобы дать указание HTTP-клиенту возвращать пустые ответы с кодом состояния 200
на каждый запрос, вы можете вызвать метод fake
без аргументов:
use Illuminate\Support\Facades\Http;
Http::fake();
$response = Http::post(...);
При фальсификации запросов посредник HTTP-клиента не выполняется. Вы должны определить ожидания для фиктивных ответов, как если бы этот посредник работал правильно.
Фальсификация конкретных URL
В качестве альтернативы вы можете передать массив методу fake
. Ключи массива должны представлять шаблоны URL, которые вы хотите подделать, и связанные с ними ответы. Допускается использование метасимвола подстановки *
. Любые запросы к URL-адресам, которые не были сфальсифицированы, будут выполнены фактически. Вы можете использовать метод response
фасада Http
для создания заглушек / фиктивных ответов для этих адресов:
Http::fake([
// Заглушка JSON ответа для адресов GitHub ...
'github.com/*' => Http::response(['foo' => 'bar'], 200, $headers),
// Заглушка строкового ответа для адресов Google ...
'google.com/*' => Http::response('Hello World', 200, $headers),
]);
Если вы хотите указать шаблон резервного URL-адреса, который будет заглушать все не сопоставленные URL-адреса, то используйте символ *
:
Http::fake([
// Заглушка JSON ответа для адресов GitHub ...
'github.com/*' => Http::response(['foo' => 'bar'], 200, ['Headers']),
// Заглушка строкового ответа для всех остальных адресов ...
'*' => Http::response('Hello World', 200, ['Headers']),
]);
Фальсификация серии ответов
По желанию можно указать, что один URL должен возвращать серию фиктивных ответов в определенном порядке. Вы можете сделать это, используя метод Http::sequence
для составления ответов:
Http::fake([
// Заглушка серии ответов для адресов GitHub ...
'github.com/*' => Http::sequence()
->push('Hello World', 200)
->push(['foo' => 'bar'], 200)
->pushStatus(404),
]);
Когда все ответы в этой последовательности будут использованы, любые дальнейшие запросы приведут к выбросу исключения. Если вы хотите указать ответ по умолчанию, который должен возвращаться, когда последовательность пуста, то используйте метод whenEmpty
:
Http::fake([
// Заглушка серии ответов для адресов GitHub ...
'github.com/*' => Http::sequence()
->push('Hello World', 200)
->push(['foo' => 'bar'], 200)
->whenEmpty(Http::response()),
]);
Если вы хотите подделать серию ответов без указания конкретного шаблона URL, который следует подделать, то используйте метод Http::fakeSequence
:
Http::fakeSequence()
->push('Hello World', 200)
->whenEmpty(Http::response());
Анонимные фальсификаторы
Если вам требуется более сложная логика для определения того, какие ответы возвращать для определенных адресов, то вы можете передать замыкание методу fake
. Это замыкание получит экземпляр Illuminate\Http\Client\Request
и должно вернуть экземпляр ответа. В замыкании вы можете выполнить любую логику, необходимую для определения типа ответа, который нужно вернуть:
Http::fake(function ($request) {
return Http::response('Hello World', 200);
});
Инспектирование запросов
При фальсификации ответов вы можете иногда захотеть проверить запросы, которые получает клиент, чтобы убедиться, что ваше приложение отправляет правильные данные или заголовки. Вы можете сделать это, вызвав метод Http::assertSent
после вызова Http::fake
.
Метод assertSent
принимает замыкание, которому будет передан экземпляр Illuminate\Http\Client\Request
и, которое должно возвращать значение логического типа, указывающее, соответствует ли запрос вашим ожиданиям. Для успешного прохождения теста должен быть отправлен хотя бы один запрос, соответствующий указанным ожиданиям:
use Illuminate\Http\Client\Request;
use Illuminate\Support\Facades\Http;
Http::fake();
Http::withHeaders([
'X-First' => 'foo',
])->post('http://example.com/users', [
'name' => 'Taylor',
'role' => 'Developer',
]);
Http::assertSent(function (Request $request) {
return $request->hasHeader('X-First', 'foo') &&
$request->url() == 'http://example.com/users' &&
$request['name'] == 'Taylor' &&
$request['role'] == 'Developer';
});
При необходимости вы можете утверждать, что конкретный запрос не был отправлен с помощью метода assertNotSent
:
use Illuminate\Http\Client\Request;
use Illuminate\Support\Facades\Http;
Http::fake();
Http::post('http://example.com/users', [
'name' => 'Taylor',
'role' => 'Developer',
]);
Http::assertNotSent(function (Request $request) {
return $request->url() === 'http://example.com/posts';
});
Вы можете использовать метод assertSentCount
, чтобы определить, сколько запросов было отправлено во время теста:
Http::fake();
Http::assertSentCount(5);
Или используйте метод assertNothingSent
, чтобы утверждать, что во время теста не было отправлено никаких запросов:
Http::fake();
Http::assertNothingSent();
События
Laravel запускает три события в процессе отправки HTTP-запросов. Событие RequestSending
запускается до отправки запроса, а событие ResponseReceived
запускается после получения ответа на данный запрос. Событие ConnectionFailed
запускается, если на данный запрос ответ не получен.
События RequestSending
и ConnectionFailed
содержат общедоступное свойство $request
, которое вы можете использовать для проверки экземпляра Illuminate\Http\Client\Request
. Аналогично, событие ResponseReceived
содержит свойство $request
, а также свойство $response
, которое может использоваться для проверки экземпляра Illuminate\Http\Client\Response
. Вы можете зарегистрировать прослушиватели событий для этого события в вашем сервис-провайдере App\Providers\EventServiceProvider
:
/**
* Сопоставления прослушивателей событий для приложения.
*
* @var array
*/
protected $listen = [
'Illuminate\Http\Client\Events\RequestSending' => [
'App\Listeners\LogRequestSending',
],
'Illuminate\Http\Client\Events\ResponseReceived' => [
'App\Listeners\LogResponseReceived',
],
'Illuminate\Http\Client\Events\ConnectionFailed' => [
'App\Listeners\LogConnectionFailed',
],
];