Как выполнить несколько запросов Guzzle одновременно?
Я могу выполнять отдельные запросы, используя Guzzle, и я очень доволен производительностью Guzzle, однако, я прочитал кое-что в API Guzzle о MultiCurl и Batching.
Может ли кто-нибудь объяснить мне, как сделать несколько запросов одновременно? Асинхронизация, если это возможно. Я не знаю, имеют ли это в виду MultiCurl. Синхронизация тоже не будет проблемой. Я просто хочу сделать несколько запросов одновременно или очень близко (короткий промежуток времени).
2 ответа
Из документов: http://guzzle3.readthedocs.org/http-client/client.html
Простое в использовании решение, которое возвращает хэш объектов запроса, сопоставляемых с ответом или ошибкой, см. По http://guzzle3.readthedocs.org/batching/batching.html.
Краткий пример:
<?php
$client->send(array(
$client->get('http://www.example.com/foo'),
$client->get('http://www.example.com/baz'),
$client->get('http://www.example.com/bar')
));
Обновление, связанное с новым GuzzleHttp guzzlehttp / guzzle
Параллельные / параллельные вызовы теперь выполняются через несколько различных методов, включая Promises. Параллельные запросы
Старый способ передачи массива RequestInterfaces больше не будет работать.
Смотрите пример здесь
$newClient = new \GuzzleHttp\Client(['base_uri' => $base]);
foreach($documents->documents as $doc){
$params = [
'language' =>'eng',
'text' => $doc->summary,
'apikey' => $key
];
$requestArr[$doc->reference] = $newClient->getAsync( '/1/api/sync/analyze/v1?' . http_build_query( $params) );
}
$time_start = microtime(true);
$responses = \GuzzleHttp\Promise\unwrap($requestArr); //$newClient->send( $requestArr );
$time_end = microtime(true);
$this->get('logger')->error(' NewsPerf Dev: took ' . ($time_end - $time_start) );
Обновление: как предложено в комментариях и задано @ sankalp-tambe, вы также можете использовать другой подход, чтобы избежать того, что набор одновременных запросов с ошибкой не вернет все ответы.
В то время как варианты, предложенные с Пулом, осуществимы, я все еще предпочитаю обещания.
Пример с обещаниями - использовать методы расчета и ожидания вместо распаковки.
Отличие от приведенного выше примера будет
$responses = \GuzzleHttp\Promise\settle($requestArr)->wait();
Я создал полный пример ниже для справки о том, как обрабатывать ответы $ тоже.
require __DIR__ . '/vendor/autoload.php';
use GuzzleHttp\Client as GuzzleClient;
use GuzzleHttp\Promise as GuzzlePromise;
$client = new GuzzleClient(['timeout' => 12.0]); // see how i set a timeout
$requestPromises = [];
$sitesArray = SiteEntity->getAll(); // returns an array with objects that contain a domain
foreach ($sitesArray as $site) {
$requestPromises[$site->getDomain()] = $client->getAsync('http://' . $site->getDomain());
}
$results = GuzzlePromise\settle($requestPromises)->wait();
foreach ($results as $domain => $result) {
$site = $sitesArray[$domain];
$this->logger->info('Crawler FetchHomePages: domain check ' . $domain);
if ($result['state'] === 'fulfilled') {
$response = $result['value'];
if ($response->getStatusCode() == 200) {
$site->setHtml($response->getBody());
} else {
$site->setHtml($response->getStatusCode());
}
} else if ($result['state'] === 'rejected') {
// notice that if call fails guzzle returns is as state rejected with a reason.
$site->setHtml('ERR: ' . $result['reason']);
} else {
$site->setHtml('ERR: unknown exception ');
$this->logger->err('Crawler FetchHomePages: unknown fetch fail domain: ' . $domain);
}
$this->entityManager->persist($site); // this is a call to Doctrines entity manager
}
Этот пример кода был первоначально размещен здесь.
Guzzle 6.0 упростил отправку нескольких асинхронных запросов.
Есть несколько способов сделать это.
Вы можете создавать асинхронные запросы и добавлять результирующие обещания в один массив и получать результат, используя settle()
метод вроде этого:
$promise1 = $client->getAsync('http://www.example.com/foo1');
$promise2 = $client->getAsync('http://www.example.com/foo2');
$promises = [$promise1, $promise2];
$results = GuzzleHttp\Promise\settle($promises)->wait();
Теперь вы можете просмотреть эти результаты и получить ответ, используя GuzzleHttpPromiseall
или GuzzleHttpPromiseeach
. Обратитесь к этой статье для получения дополнительных сведений.
В случае, если у вас есть неопределенное количество запросов для отправки (скажем, 5 здесь), вы можете использовать GuzzleHttp/Pool::batch()
. Вот пример:
$client = new Client();
// Create the requests
$requests = function ($total) use($client) {
for ($i = 1; $i <= $total; $i++) {
yield new Request('GET', 'http://www.example.com/foo' . $i);
}
};
// Use the Pool::batch()
$pool_batch = Pool::batch($client, $requests(5));
foreach ($pool_batch as $pool => $res) {
if ($res instanceof RequestException) {
// Do sth
continue;
}
// Do sth
}