Je vais parler de Guzzle qui est un client HTTP. Nous allons voir la version 6 qui utilise Php5.5
Promise et Guzzle.
Guzzle connait les promises et possède sa propre implémentation.
la signature de la fonction est un peu près la même que react/promise.
Attention Guzzle ne fait pas la différence entre le Deferred qui est un travail dont la réponse est encore inconnu et représenter par une promise. Dans Guzzle le travail et la réponse sont la même chose.
1234567891011121314151617
useGuzzleHttp\Promise\Promise;$promise=newPromise();$promise->then(// $onFulfilledfunction($value){echo'Tout va bien.';},// $onRejectedfunction($reason){echo'On a un problème.';});$promise->resolve(null);// 'Tout va bien.';// Ou $promise->reject(null);// 'On a un problème.';
Guzzle est un client Web essayons un cas concret.
1234567891011
$client=newGuzzleHttp\Client();$promise=$client->requestAsync('GET','http://httpbin.org/get');$promise->then(function($res){return$res->getStatusCode();})->then(function($value){echo"j'ai recu un code $value"};// Notre requète n'est pas encore partie. Il faut lancer manuellement l'appel.$client->wait();
L’avantage ici est que je décide quand je lance l’appel. Par exemple on peut lancer en parallèle les requêtes.
12345678910111213
$client=newClient(['base_uri'=>'http://httpbin.org/']);// Je crée toute mes requetes$promises=['image'=>$client->getAsync('/image'),'png'=>$client->getAsync('/image/png'),'jpeg'=>$client->getAsync('/image/jpeg'),'webp'=>$client->getAsync('/image/webp')];// je resouds tout en même temps$results=Promise\unwrap($promises);
On peux créer des pools. Si on souhaite limiter le nombre de requête en même temps.
Nous allons utiliser l’instruction any() toutes les requêtes sont lancées en concurrences. C’est la première arrivée qui l’emporte.
123456789101112
$client=newClient(['base_uri'=>'http://httpbin.org/']);// je crée toute mes requetes$promises=['image'=>$client->getAsync('/image'),'png'=>$client->getAsync('/image/png'),'jpeg'=>$client->getAsync('/image/jpeg'),'webp'=>$client->getAsync('/image/webp')];$result=Promise\any($promises)->then(function($value){var_dump($value->getHeader('Content-Type'));});$result->wait();
Je veux juste les deux premières réponses some(2, $promise)
1234567891011121314151617
$client=newClient(['base_uri'=>'http://httpbin.org/']);// je crée toute mes requetes$promises=['image'=>$client->getAsync('/image'),'png'=>$client->getAsync('/image/png'),'jpeg'=>$client->getAsync('/image/jpeg'),'webp'=>$client->getAsync('/image/webp')];$result=Promise\some(2,$promises)->then(function($results){foreach($resultsas$value)var_dump($value->getHeader('Content-Type'));});$result->wait();
Je mets à la suite toute les promises que je souhaite exécuter en ajoutant yield devant.
Je laisse Guzzle gérer avec un limitation de 2. des que le programme a une place de libre, il appelle le générateur pour avoir un nouvelle promise.
Mais il existe dans Guzzle des co-routines..
1234567891011121314151617181920212223242526
$client=newClient(['base_uri'=>'http://httpbin.org/']);$myfunction=function($url)use($client){returnPromise\coroutine(function()use($client,$url){try{$value=(yield$client->getAsync($url));}catch(\Exception$e){yieldNewRejectedPromise($e->getMessage());}});};$images=['foo','baz','bar'];$promises=[];// Build an array of promises.foreach($imagesas$image){$promises[]=$myfunction($image);}$aggregate=Promise\all($promises)->then(function($values){echo'ok';},function($values){echo'nope';});$aggregate->wait();
Le code est complètement asynchrone.
Il est intéressant de voir le code synchrone et non parallèle.
1234567891011121314151617
$client=newClient(['base_uri'=>'http://httpbin.org/']);$getImages=function($url)use($client){try{return$value=$client->get($url);}catch(\Exception$e){$value=$e->getMessage();}};$images=['foo','baz','bar'];$promises=[];// Build an array of promises.foreach($imagesas$image){$result[]=$getImages($image);}
En gros, j’ai retiré le async et les yields mais les deux codes se ressemblent non ?
Conclusion
Les promises sont pratiques.
elles sont chainables
elles sont asynchrones, annulables, rejetables
On peut faire des foreach dessus.
On peut les combiner.
Ce n’est pas vraiment un hasard. Les promises sont des Monades. Il n’est pas simple d’expliquer les monades. Les monades viennent de la programmation fonctionnelle et c’est surtout haskell qui a popularisé cette structure. J’espère que je reviendrai dessus.
Guzzle est vraiment très sympathique à utiliser. Le coté asynchrone n’est pas simple, la fonction co-routine n’est pas dans la documentation. Il a été très difficile de trouver un code d’exemple. Je regrette que parfois le seul moyen de déclencher la résolution est d’appeler de manière synchrone ->wait() ce qui est dommage.