美文网首页PHPLaravel 爱好者PHP经验分享
推荐一个 PHP 网络请求插件 Guzzle

推荐一个 PHP 网络请求插件 Guzzle

作者: Coding01 | 来源:发表于2017-10-29 19:43 被阅读1329次

    在写后台代码时,避免不了需要与其他第三方接口交互,如向服务号下发模板消息,有时可能需要下发超过 10 万条。这时不得不考虑使用异步和「多线程」的网络请求。

    今天向 PHP 工程师们推荐一个 Guzzle 插件。

    Guzzle

    Guzzle 是一个 PHP 的 HTTP 客户端,用来轻而易举地发送请求,并集成到我们的 WEB 服务上。

    • 接口简单:构建查询语句、POST 请求、分流上传下载大文件、使用 HTTP cookies、上传 JSON 数据等等。

    • 发送同步或异步的请求均使用相同的接口。

    • 使用 PSR-7 接口来请求、响应、分流,允许你使用其他兼容的 PSR-7 类库与 Guzzle 共同开发。

    • 抽象了底层的 HTTP 传输,允许你改变环境以及其他的代码,如:对 cURL与 PHP 的流或 socket 并非重度依赖,非阻塞事件循环。

    • 中间件系统允许你创建构成客户端行为。

    摘自 Guzzle 官网介绍:
    http://guzzle-cn.readthedocs.io/zh_CN/latest/index.html

    安装 Guzzle

    本文结合 Laravel 项目介绍 Guzzle 基本使用,所以使用 composer 来安装 Guzzle 再适合不过了,而且 Guzzle 官网也推荐使用 composer 来安装。

    composer require guzzlehttp/guzzle:~6.0
    
    // 或者
    
    php composer.phar require guzzlehttp/guzzle:~6.0
    

    如何安装 Composer,可以看看我之前的文章
    https://d.laravel-china.org/docs/5.5/installation

    发送简单的 POST 请求

    访问第三方接口,基本上都是 POST 请求为主。如你想做一个简单的智能聊天工具,这时候可以借助图灵机器人 API,发送一个 POST 请求获取自动回答内容,直接上代码:

    <?php
    
    namespace App\Http\Controllers;
    
    use GuzzleHttp\Client;
    use Illuminate\Http\Request;
    
    class GuzzleUseController extends Controller {
    
        public function tuling(Request $request) {
            $params = [
                'key' => '*****',
                'userid' => 'yemeishu'
            ];
    
            $params['info'] = $request->input('info', '你好吗');
    
            $client = new Client();
            $options = json_encode($params, JSON_UNESCAPED_UNICODE);
            $data = [
                'body' => $options,
                'headers' => ['content-type' => 'application/json']
            ];
    
            // 发送 post 请求
            $response = $client->post('http://www.tuling123.com/openapi/api', $data);
    
            $callback = json_decode($response->getBody()->getContents());
    
            return $this->output_json('200', '测试图灵机器人返回结果', $callback);
        }
    }
    

    Guzzle client->post 函数还是很简单的,只需要访问的接口,和请求的参数,参数中主要包含:body、headers、query等,具体可参考

    http://guzzle-cn.readthedocs.io/zh_CN/latest/quickstart.html#id8

    测试下:

    注:图灵机器人还是很智能的,根据相同的 userid 能够识别上下文,做到智能聊天的。

    发送异步的 POST 请求

    在 PHP 开发中主要是「面向过程」式的开发方式,但请求第三方接口时,有时候并不需要等待第三方接口返回结果才继续执行。如用户购买成功时,我们需要向短信接口,发送一个 post 请求,由短信平台发送一条短信给用户,告知用户支付成功了,因为这类「提醒消息」属于「额外的附加功能」,并不需要在用户支付时「知道」有没有发送提醒成功。

    这时候可以使用 Guzzle 的异步请求功能,直接看代码:

    public function sms(Request $request) {
            $code = $request->input('code');
            $client = new Client();
            $sid = '9815b4a2bb6d5******8bdb1828644f2';
            $time = '20171029173312';
            $token = 'af8728c8bc*******12019c680df4b11c';
    
            $sig =  strtoupper(md5($sid.$token.$time));
    
            $auth = trim(base64_encode($sid . ":" . $time));
    
            $params = ['templateSMS' => [
                    'appId' => '12b43**********0091c73c0ab',
                    'param' => "coding01,$code,30",
                    'templateId' => '3***3',
                    'to' => '17689974321'
                ]
            ];
            $options = json_encode($params, JSON_UNESCAPED_UNICODE);
            $data = [
                'query' => [
                    'sig' => $sig
                ],
                'body' => $options,
                'headers' => [
                    'content-type' => 'application/json',
                    'Authorization' => $auth
                ]
            ];
    
            // 发送 post 请求
            $promise = $client->requestAsync('POST', 'https://api.ucpaas.com/2014-06-30/Accounts/9815b4a2bb6d5******8bdb1828644f2/Messages/templateSMS', $data);
    
            $promise->then(
                function (ResponseInterface $res) {
                    Log::info('---');
                    Log::info($res->getStatusCode() . "\n");
                    Log::info($res->getBody()->getContents() . "\n");
                },
                function (RequestException $e) {
                    Log::info('-__-');
                    Log::info($e->getMessage() . "\n");
                }
            );
            $promise->wait();
    
            return $this->output_json('200', '测试短信 api', []);
        }
    

    先返回接口数据:

    然后再输出 Log:

    [2017-10-29 09:53:14] local.INFO: ---  
    [2017-10-29 09:53:14] local.INFO: 200
      
    [2017-10-29 09:53:14] local.INFO: {"resp":{"respCode":"000000","templateSMS":{"createDate":"20171029175314","smsId":"24a93f323c9*****8608568"}}}
    

    最后收到短信信息:

    发送多线程异步 POST 请求

    「发送多线程异步 POST 请求」在很多场合中使用到的,如:双十一快到了,可以做一些回馈老用户的活动,这是就需要批量的向老用户推送一条模板消息,告诉用户参与哪些活动的。这时候就需要用到多线程异步请求微信公众号接口。

    直接上代码:

    public function send($templateid, $openid, $url, $data) {
            $client = $this->bnotice->getHttp()->getClient();
    
            $requests = function ($open_ids) use ($templateid, $url, $data) {
                foreach($open_ids as $v){
                    try {
                        yield $this->bnotice
                            ->template($templateid)
                            ->to($v)
                            ->url($url)
                            ->data($data)
                            ->request();
                    } catch(Exception $e) {
                        Log::error('sendtemplate:'.$e->getMessage());
                    }
                }
            };
    
            $pool = new Pool($client, $requests($openid), [
                'concurrency' => 16,
                'fulfilled' => function ($response, $index) {
                },
                'rejected' => function ($reason, $index) {
                },
            ]);
    
            $promise = $pool->promise();
    
            $promise->wait();
        }
    

    其中 request 方法:

    public function request($data = [])
        {
            $params = array_merge([
                'touser' => '',
                'template_id' => '',
                'url' => '',
                'topcolor' => '',
                'miniprogram' => [],
                'data' => [],
            ], $data);
            
            $required = ['touser', 'template_id'];
    
            foreach ($params as $key => $value) {
                if (in_array($key, $required, true) && empty($value) && empty($this->message[$key])) {
                    throw new InvalidArgumentException("Attribute '$key' can not be empty!");
                }
    
                $params[$key] = empty($value) ? $this->message[$key] : $value;
            }
    
            $params['data'] = $this->formatData($params['data']);
    
            $this->message = $this->messageBackup;
    
            $options = json_encode ( $params,  JSON_UNESCAPED_UNICODE);
            $data = [
                'query' => [
                    'access_token' => $this->getAccessToken()->getToken()
                ],
                'body' => $options,
                'headers' => ['content-type' => 'application/json']
            ];
            return function() use ($data) {
                return $this->getHttp()->getClient()->requestAsync('POST', $this::API_SEND_NOTICE, $data);
            };
        }
    

    Guzzle 多线程异步请求原型函数,使用 GuzzleHttp\Pool 对象

    use GuzzleHttp\Pool;
    use GuzzleHttp\Client;
    use GuzzleHttp\Psr7\Request;
    
    $client = new Client();
    
    $requests = function ($total) {
        $uri = 'http://127.0.0.1:8126/guzzle-server/perf';
        for ($i = 0; $i < $total; $i++) {
            yield new Request('GET', $uri);
        }
    };
    
    $pool = new Pool($client, $requests(100), [
        'concurrency' => 5,
        'fulfilled' => function ($response, $index) {
            // this is delivered each successful response
        },
        'rejected' => function ($reason, $index) {
            // this is delivered each failed request
        },
    ]);
    
    // Initiate the transfers and create a promise
    $promise = $pool->promise();
    
    // Force the pool of requests to complete.
    $promise->wait();
    

    总结

    有了 Guzzle,极大方便了我们并发异步请求第三方接口。如果时间允许,我们可以看看 Guzzle 源代码,看看是如何实现的。

    推荐

    1. 在 windows 环境下,解决[GuzzleHttp\Exception\RequestException] cURL error 60: SSL certificate problem: unable to get local issuer certific ate

    访问这个网址 https://curl.haxx.se/ca/cacert.pem 下载文件
    然后修改 php.ini curl.cainfo = "D:\cacert.pem" cacert.pem文件 随便放在哪,没限制。

    2. 需要了解 Guzzle 更多资料

    查看官网: http://guzzle-cn.readthedocs.io/zh_CN/latest/index.html

    3. 其它 Guzzle 使用文章

    「完」


    coding01 期待您继续关注

    qrcode

    也很感谢您能看到这了

    qrcode

    相关文章

      网友评论

      • 韩槑槑_a941:这个轮子是阻塞的,并不是我们了解的异步请求。
      • 不器1843:$requests = function ($open_ids) use ($templateid, $url, $data) {
        foreach($open_ids as $v){
        try {
        yield $this->bnotice
        ->template($templateid)
        ->to($v)
        ->url($url)
        ->data($data)
        ->request();
        } catch(Exception $e) {
        Log::error('sendtemplate:'.$e->getMessage());
        }
        }
        };

        你好,$open_ids是怎么来了,我传不进去值呢?可以告知吗?
        Coding01:@不器1843 我从来不盗版别人的文章
        不器1843:这个不是你写的吗?
        Coding01:你试试和 use ($templateid, $url, $data) 一样,把 $open_ids 传入呢?

      本文标题:推荐一个 PHP 网络请求插件 Guzzle

      本文链接:https://www.haomeiwen.com/subject/nttupxtx.html