grpc套路(四)php通过grpc调用golang的grpc接口服务

如果看到这篇文章你还搞不明白为什么不直接使用 http, 而要搞 rpc?那么请看我的另一篇博客:
什么是rpc? 你会彻底搞明白为啥这个?为啥那个?如果还不懂那我也没办法了!

我们开发过程当中可能不仅涉及到一门语言,比如php java golang 在同一个项目当中都运用了,当然这可能是个非常大的项目,但是考虑到项目的可维护性三者同时出现的可能性还是比较低的,但是涉及到大型项目比如涉及到微服务架构,难免会有多门语言共存的情况发生,我们考虑是php+golang的混编模式,因为毕竟php有php的短板golang有golang的长处当然java更有java的特色!所以今天空穴来潮临时打了鸡血一样的写了这么篇稿子!

首先提个醒,grpc章节的文章都是连贯性的,不可能你看这一篇文章就能搞明白什么是rpc grpc,所以建议从本章节的第一篇文章读起!

现在php流行的框架很多,我们就选择国内使用人数最多的thinkphp5.0框架来演示

第一步:

把你在golang里面的proto后缀的文件原模原样的拿过来 放到extend目录下 至于为什么放到extend目录下下边我们会解释的

grpc套路(四)php通过grpc调用golang的grpc接口服务
解释一下: package pdfiles.user 那么就会在User.proto文件所在的位置生成Pdfiles/User目录里面存放存根文件!

第二步:

composer.json文件当中的require当中放入:

"require": {
        "grpc/grpc": "v1.30.0",
        "google/protobuf": "^v3.13.0"
    },
    "autoload":{
        "psr-4":{
            "GPBMetadata\\":"GPBMetadata/",
            "Hello\\":"Hello/"
        }
    },

需要说明的是 “google/protobuf”: “^v3.13.0”不是必须的,可以去掉,这个是为了你在没有安装php的protobuf扩展情况下,也能正常运行,这种运行方式相对效率较低。
然后执行

composer install

或者

composer update

第三步:

a.安装php的grpc扩展
用pecl方式安装

pecl install grpc

找到php.ini 加上

extension = "grpc.so"

cli模式下以及fpm模式下两种 找对php.ini php-m以及phpinfo()都能看到grpc才算成功哈!之前的博文当中讲过如何安装php扩展这里不再废话!
b.安装 protobuf 及其 php 扩展(上边我们通过composer安装了已经 这里再废话一下如何在linux下安装)
执行命令:

pecl install protobuf

找到php.ini文件添加上

extension=protobuf.so

重启服务即可 直接重启nginx就完事 不放心那就重启一遍php-fpm进程

第四步:

安装 grpc_php_plugin 插件
这一步非必须哈!
并且非常耗时 我就没搞!

第五步:

因为前边博文当中讲过windows下安装protoc编译工具 并且配置到了环境变量当中 所以这里你就可以直接执行命令:
进入到 extend目录 找到User.proto文件 然后执行:

protoc --php_out=plugins=grpc:. User.proto

这个时候你就会发现

grpc套路(四)php通过grpc调用golang的grpc接口服务

红线标注的就是新生成的类文件!
我们第四部没有安装grpc_php_plugin 插件所以生成的pdfiles/user里面没有UserClient文件所以撒我们自己得新创建一个

第六步:

<?php
/**
 * Created by PhpStorm.
 * User: 胡军
 * Date: 2020/8/22
 * Time: 13:43
 */
 //命名空间一定要对 保持和RequestUser.php里面的一致即可!
namespace Pdfiles\User;
//必须继承 \Grpc\BaseStub
class UserClient extends \Grpc\BaseStub{
    //构造函数必须有哦
    public function __construct($hostname, array $opts, $channel = null)
    {
        parent::__construct($hostname, $opts, $channel);
    }

    //为什么是userList() 还记得你的proto文件里面的service吗?
    //service UserService{
    //     rpc UserList(RequestUser) returns (ResponseUser){};
    //}
    //如果rpc GetUsers(RequestUser1)  returns (ResponseUser1){};
    //那么这里你也要实现GetUsers()方法
    //第一个参数是请求体 依赖注入进去就行啊 第二个是参数 第三个是其他参数 一把不传入
    public function userList(\Pdfiles\User\RequestUser $argument,$metadata=[],$options=[]){
        //简单模式
        return $this->_simpleRequest(
            //这里找到golang那边生成的.pd.go文件里面的FullMethod的值原模原样填写过来即可!
            '/pdfiles.user.UserService/UserList',
            //传入的请求体
            $argument,
            //第一个表示用什么解析 之类是 ResponseUser类来解析 命名空间路径要写对啊! decode就是解析喽
            ['\Pdfiles\User\ResponseUser','decode'],
            //参数
            $metadata,
            //参数
            $options
        );
    }


}

链接文件就创建完成了!

第七步:

调用就完事了:

   public function index2(){
        //第二个参数固定写法 是一种认证
        $userClient = new \Pdfiles\User\UserClient("10.10.16.52:8084",[
            'credentials' =>  \Grpc\ChannelCredentials::createInsecure()
        ]);

        //请求体
        $requestUser = new \Pdfiles\User\RequestUser();
        $requestUser->setName("huxiaobai");//设置要传递的参数的值
        $requestUser->setMobile("15275411187");//设置要传递的参数的值

        //调用userList()方法请求体过去并等待响应
        $request = $userClient->userList($requestUser)->wait();
        //返回体和状态
        //状态: code = 0表示链接成功了
        //stdClass Object ( [metadata] => Array ( ) [code] => 0 [details] => )
        list($response,$status) = $request;
        //用返回体调取getUser()函数获取返回值
        $ret = $response->getUser();
        foreach ($ret as $k=>$user){
            echo $user->getName();
        }
    }

整个过程就完事了!
还是想说一句并不是所有的系统和系统之间数据的读取用grpc就好 http也有http协议的好处 前边我们都讲过 是不是用还是自己考虑好哦!

源码:git@codehub.devcloud.huaweicloud.com:golang00004/php_go_grpc.git 付费!

本作品采用《CC 协议》,转载必须注明作者和本文链接
胡军
讨论数量: 1

你这里讲的PHP调用grpc服务我能看明白。 我有个疑问,假如我之前有两个PHP项目(a & b),他们俩之前是HTTP通信,互相得到数据(json),那么我现在要改成使用golang的grpc实现AB两个项目互相通信怎么实现?业务逻辑代码都在AB项目里,AB只需要调用对方注册的grpc,grpc只是个类似NGINX的服务并没有逻辑代码,这样的场景怎么搞?

2年前 评论

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!