php在windows使用grpc和protobuf入门(超详细)

背景:php作为客户端使用grpc和protobuf调用其他服务

1、自己先了解:grpc、protobuf

2、环境:php7.3、composer

设置php全局变量

php -version查看PHP版本是php7.3.4nts,所以要选(NTS)那个

3、给php安装grpc扩展

下载grpc扩展:

PECL :: Package :: gRPC 1.42.0 for Windows

或者根据自己的php版本去库里选择:https://pecl.php.net/package/gRPC

下载后解压,把php_grpc.dll这个文件复制到php\php7.3.4nts\ext这个目录下

​ 找到php.ini这个文件加入 extension=php_grpc.dll

​ 

phpstorm终端输入php -m 查看扩展是否安装成功;

或者用 var_dump(phpinfo()); 看看有没有

4、新建项目目录:protobuf_test

引入grpc和protobuf的PHP类库,在目录下新建文件composer.json,如下:

{
    "name": "xxs/grpc",
    "require": {
        "grpc/grpc": "^v1.4.0",
        "google/protobuf": "^v3.3.0"
    },
    "autoload":{
        "psr-4":{
            "GPBMetadata\\":"GPBMetadata/",
            "Helloworld\\":"Helloworld/"
        }
    }
}

【composer 需配置全局变量】在目录下执行composer install

顺带执行:composer  dump-autoload

如下图,我这里执行完是1.52.0和3.22.2,composer ^ 符号的解释:php composer 版本号 ^ 与~_小镇学者的博客-CSDN博客

5、下载protoc.exe可执行程序

Releases · protocolbuffers/protobuf · GitHub

我下载的v22.2:https://github.com/protocolbuffers/protobuf/releases/download/v22.2/protoc-22.2-win64.zip

解压后复制bin目录的路径,去配置全局变量,我的是D:\protoc22.2\bin

​ win+R打开cmd查看:protoc --version  

6、万事具备,只欠东风!!!

官方的grpc未提供windows下的php plugin的exe文件,这个用来生成客户端文件,看了一堆博客,几乎没有这一步……

去这里下载 https://github.com/lifenglsf/grpc_for_windows

解压复制到项目底下grpc_for_windows/x64/grpc_php_plugin.exe

​ 

7、新建文件protobuf_test/helloworld.proto

syntax = "proto3";
package helloworld;

// The greeting service definition.
service Greeter {
  // Sends a greeting
  rpc SayHello (HelloRequest) returns (HelloReply) {}

  rpc SayHelloStreamReply (HelloRequest) returns (stream HelloReply) {}
}

// The request message containing the user's name.
message HelloRequest {
  string name = 1;
}

// The response message containing the greetings
message HelloReply {
  string message = 1;
}

8、终端protobuf_test下执行命令

protoc --proto_path=. --php_out=. --grpc_out=. --plugin=protoc-gen-grpc=./grpc_for_windows/x64/grpc_php_plugin.exe ./helloworld.proto

生成代码结构,如果没有grpc_php_plugin.exe这个插件是不会生成GreeterClient.php这个客户端文件的,我们的目的就是用php作为客户端调用服务的!

9、为了方便测试自己搞个测试服

定义服务protobuf_test/Helloworld/GreeterStub.php,如上图;代码如下:

<?php
namespace Helloworld;
/**
 * The greeting service definition.
 */
class GreeterStub {
    /**
     * Sends a greeting
     * @param \Helloworld\HelloRequest $request client request
     * @param \Grpc\ServerContext $context server request context
     * @return \Helloworld\HelloReply for response data, null if if error occured
     *     initial metadata (if any) and status (if not ok) should be set to $context
     */
    public function SayHello(
        \Helloworld\HelloRequest $request,
        \Grpc\ServerContext $context
    ): ?\Helloworld\HelloReply {
        $context->setStatus(\Grpc\Status::unimplemented());
        return null;
    }

    /**
     * Get the method descriptors of the service for server registration
     *
     * @return array of \Grpc\MethodDescriptor for the service methods
     */
    public final function getMethodDescriptors(): array
    {
        return [
            '/helloworld.Greeter/SayHello' => new \Grpc\MethodDescriptor(
                $this,
                'SayHello',
                '\Helloworld\HelloRequest',
                \Grpc\MethodDescriptor::UNARY_CALL
            ),
        ];
    }
}
实现这个服务,protobuf_test/server.php
<?php
require dirname(__FILE__) . '/vendor/autoload.php';
class Greeter extends Helloworld\GreeterStub
{
    public function SayHello(
        \Helloworld\HelloRequest $request,
        \Grpc\ServerContext $serverContext
    ): ?\Helloworld\HelloReply {
        $name = $request->getName();
        echo 'Received request: ' . $name . PHP_EOL;
        $response = new \Helloworld\HelloReply();
        $response->setMessage("Hello " . $name);
        return $response;
    }
}
$port = 50051;
$server = new \Grpc\RpcServer();
$server->addHttp2Port('0.0.0.0:'.$port);
$server->handle(new Greeter());
echo 'Listening on port :' . $port . PHP_EOL;
$server->run();

10、protobuf_test/client.php 调用服务

<?php
require dirname(__FILE__) . '/vendor/autoload.php';
function greet($hostname, $name)
{
    $client = new Helloworld\GreeterClient($hostname, [
        'credentials' => Grpc\ChannelCredentials::createInsecure(),
    ]);
    $request = new Helloworld\HelloRequest();
    $request->setName($name);
    list($response, $status) = $client->SayHello($request)->wait();
    if ($status->code !== Grpc\STATUS_OK) {
        echo "ERROR: " . $status->code . ", " . $status->details . PHP_EOL;
        exit(1);
    }
    echo $response->getMessage() . PHP_EOL;
}
$name = !empty($argv[1]) ? $argv[1] : 'world';
$hostname = !empty($argv[2]) ? $argv[2] : 'localhost:50051';
greet($hostname, $name);

11、最终文件结构

12、测试

开启服务:php server.php

 在开一个终端:php client.php

得到响应 Hello world

13、参数说明

# --php_out php代码输出路径,里面包含request,response,client代码

# --proto_path="protos文件目录"

# --grpc_out GPBMetadata输出路径,用于保存.proto的二进制元数据

# --plugin 生成代码插件的类型与插件的绝对路径路径

14、收工大吉

参考:

gRPC(2)- PHP使用gRPC_php grpc_share_9527的博客-CSDN博客

PHP中使用gRPC客户端_grpc php 客户端_IM魂影的博客-CSDN博客

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值