我正在运行在特定端口上侦听的GRPC服务器(服务器A)。我希望能够将通信发送到另一台服务器(服务器B),并让服务器B记录服务器A连接的传入地址,以便以后可以与服务器A联系。
在服务器A上,我在端口上侦听并创建类似这样的上下文:
lis, err := net.Listen("tcp", "0.0.0.0:6000")
ctx, cancel := context.WithTimeout(context.Background(),
10000*time.Millisecond)
然后像这样创建连接:
connection, err = grpc.DialContext(ctx, server2Address,
grpc.WithInsecure(), grpc.WithBlock())
在最终向服务器B上的终结点发送消息之前,该消息将尝试读取服务器A的传入连接的IP地址
info, _ := peer.FromContext(ctx)
fmt.Printf(info.Addr.String()) // Returns a random port, NOT 6000,
但是,服务器B打印的最终端口是随机的,例如62056,而不是预期的6000。我的假设是,在服务器A上,GRPC从随机端口拨号-是否可以强制GRPC从端口6000而不是随机端口拨号?
答案 0 :(得分:3)
您可以这样指定源端口:
cc, err := grpc.Dial("127.0.0.1:6001", grpc.WithInsecure(),
grpc.WithContextDialer(func(ctx context.Context, addr string) (net.Conn, error) {
dst, err := net.ResolveTCPAddr("tcp", addr)
if err != nil {
return nil, err
}
src := &net.TCPAddr{
IP: net.ParseIP("127.0.0.1"),
Port: 6000,
}
return net.DialTCP("tcp", src, dst)
}))
但是,如果您的服务器在同一端口上侦听,则会导致错误:
panic: dial tcp 127.0.0.1:6000->127.0.0.1:6001: bind: address already in use
另一种方法是将地址作为元数据传递。在客户端上,您可以这样做:
ctx := context.Background()
ctx = metadata.NewOutgoingContext(ctx, metadata.Pairs("address", "127.0.0.1:6000"))
res, err := grpc_health_v1.NewHealthClient(cc).Check(ctx, &grpc_health_v1.HealthCheckRequest{
Service: "test",
})
在服务器上:
func (s *server) Check(ctx context.Context, req *grpc_health_v1.HealthCheckRequest) (*grpc_health_v1.HealthCheckResponse, error) {
if md, ok := metadata.FromIncomingContext(ctx); ok {
addr := md.Get("address")
// addr == "127.0.0.1:6000"
}
return &grpc_health_v1.HealthCheckResponse{
Status: grpc_health_v1.HealthCheckResponse_SERVING,
}, nil
}
第三种方法是使用streaming。