目录:

  • 1、proto文件
  • 2、maven编译的时候同时编译proto文件生成对应的java类
  • 3、编写grpc的客户端,编写Grpc服务端,
  • 4、proxy的配置

1、proto文件编写

必须定义版本 proto3

syntax = "proto3";

import "google/protobuf/wrappers.proto";
import "google/protobuf/empty.proto";

option java_multiple_files = true;
option java_package = "com.test.grpc.common.pb";
option java_outer_classname = "RpcClient";


message SayHelloPB {
    string request = 1;
}

message SayHelloResponsePB {
   string response = 1;
}

service RpcService {
    rpc sayHello(SayHelloPB) returns (SayHelloResponsePB);
}

2、 maven编译时就能将proto文件编译成对应的java类

maven pom文件中加入:

<build>
        <extensions>
            <extension>
                <groupId>kr.motd.maven</groupId>
                <artifactId>os-maven-plugin</artifactId>
                <version>1.5.0.Final</version>
            </extension>
        </extensions>

        <plugins>
            <!-- protobuf build plugin -->
            <plugin>
                <groupId>org.xolstice.maven.plugins</groupId>
                <artifactId>protobuf-maven-plugin</artifactId>
                <version>0.5.1</version>
                <configuration>
                    <protocArtifact>
                        com.google.protobuf:protoc:3.7.1:exe:${os.detected.classifier}
                    </protocArtifact>
                    <pluginId>grpc-java</pluginId>
                    <pluginArtifact>
                        io.grpc:protoc-gen-grpc-java:1.21.0:exe:${os.detected.classifier}
                    </pluginArtifact>
                </configuration>
                <executions>
                    <execution>
                        <goals>
                            <goal>compile</goal>
                            <goal>compile-custom</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>

3、编写Grpc的客户端,服务端

客户端编写:

客户端中,定义成员变量,private RpcServiceGrpc.RpcServiceBlockingStub stub,并实例化stub,需要传入Channel,

客户端代码:
主要注意channel 建立中的address包含两个,一个客户端代理的地址,另一个是真正server端的地址。

import java.net.InetSocketAddress;
import java.util.concurrent.TimeUnit;

import com.test.grpc.common.pb.RpcServiceGrpc;
import com.test.grpc.common.pb.SayHelloPB;

import io.grpc.Channel;
import io.grpc.CompressorRegistry;
import io.grpc.HttpConnectProxiedSocketAddress;
import io.grpc.ProxiedSocketAddress;
import io.grpc.netty.shaded.io.grpc.netty.NettyChannelBuilder;

public class GrpcClient {
    private RpcServiceGrpc.RpcServiceBlockingStub stub;
    public GrpcClient(String host, int port){
        this.stub = RpcServiceGrpc.newBlockingStub(getChannel(host,port));
    }

    private Channel getChannel(String host, int port) {
        // 正向代理所在的ip 和port. 本地nginx listen的端口号8889
        String proxyHost = "127.0.0.1";
        int proxyPort = 8889;

        final InetSocketAddress proxyAddress =
                new InetSocketAddress(proxyHost, proxyPort);

        // 真正要访问的目的ip和port
        final InetSocketAddress targetServerAddress =
                new InetSocketAddress(host, port);

        ProxiedSocketAddress proxiedSocketAddress =
                HttpConnectProxiedSocketAddress.newBuilder()
                                               .setProxyAddress(proxyAddress)
                                               .setTargetAddress(targetServerAddress)
                                               .build();

        // address 中包含两个地址
        Channel channel1 = NettyChannelBuilder.forAddress(proxiedSocketAddress)
                                              .usePlaintext()
                                              .compressorRegistry(CompressorRegistry.getDefaultInstance())
                                              .idleTimeout(1800, TimeUnit.SECONDS)
                                              .build();

        return channel1;
    }

    // 客户端方法
    public void sayHello(String helloString) {
        SayHelloPB sayHelloPB =
                SayHelloPB.newBuilder()
                          .setRequest("hello~" + helloString)
                          .build();
        System.out.println(stub.sayHello(sayHelloPB).getResponse());
    }
}

核心类:HttpConnectProxiedSocketAddress

查看API文档

grpc gin性能对比_HTTP CONNECT


这里指出了我们使用的proxy的类型,当前的这种方式是利用了HTTP CONNECT方法,这种隧道代理方法的介绍参看这里,代理原理和HTTPS的正向代理一样,所以相关资料可以查询https的正向代理。

大致意思是:

我们知道HTTPS是加密通信的,无论是请求体还是请求头都是加密的,代理服务器没有密钥,就无法解密出请求头,那么https的数据报发到代理服务器,代理如何知道当前数据包需要发往哪里呢?

这里可以利用HTTP CONNECT方法,在报文中加一个部分,明文指出我要访问的服务端的地址,这样proxy就不需要解析,直接根据这个明文知道往哪发数据报了。

我认为 curl -x 代理的ip:prot https://baidu.com 这种方式也是用了HTTP CONNECT方法,后续有时间验证下

服务端编写:

需要继承proto编译后生成java类,public class GrpcServer extends RpcServiceGrpc.RpcServiceImplBase, 重写方法,写入自己的server逻辑

服务端代码:

import com.test.grpc.common.pb.RpcServiceGrpc;
import com.test.grpc.common.pb.SayHelloPB;
import com.test.grpc.common.pb.SayHelloResponsePB;

import io.grpc.stub.StreamObserver;

public class GrpcServer extends RpcServiceGrpc.RpcServiceImplBase {
    @Override
    public void sayHello(SayHelloPB request,
                         StreamObserver<SayHelloResponsePB> responseObserver) {
        String requestMessage = request.getRequest();
        String message = "i received the message :" + requestMessage;
        System.out.println(message);
        SayHelloResponsePB responsePB =
                SayHelloResponsePB.newBuilder().setResponse("i am the Moon").build();
        responseObserver.onNext(responsePB);
        responseObserver.onCompleted();
    }
}

4、代理配置

采用nginx来做正向代理,其配置和HTTPS的正向代理配置一样,
需要添加第三方模块:ngx_http_proxy_connect_module
github地址 需要下载下来,nginx 中编译配置时加入这个第三方模块。
这里需要的知识是:nginx如何加入第三方模块(分为两种1、在已安装好的nginx中引入,2、在安装nginx时引入第三方模块)
我因为用的docker,所以测试安装nginx成本小,错误后删掉容器重来。

nginx.conf:

#user  nobody;
worker_processes  1;

#error_log  logs/error.log;
#error_log  logs/error.log  notice;
#error_log  logs/error.log  info;

#pid        logs/nginx.pid;


events {
    worker_connections  1024;
}


http {
    include       mime.types;
    default_type  application/octet-stream;

    log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
                      '$status $body_bytes_sent "$http_referer" '
                      '"$http_user_agent" "$http_x_forwarded_for"';

    access_log  logs/access.log  main;

    sendfile        on;
    #tcp_nopush     on;

    #keepalive_timeout  0;
    keepalive_timeout  65;

    #gzip  on;

    server {
        listen       8889; #服务端口号
        server_name  localhost;
	    resolver 114.114.114.114;

	    proxy_connect;
	    proxy_connect_allow all; #允许所有端口号
	    proxy_connect_connect_timeout 60s;
	}
}