docker - 无法连接到在本地 Docker 容器中运行的 Go GRPC 服务器

标签 docker go grpc

我有一个 go grpc 服务。我正在Mac上开发,山脉。在本地针对服务运行 grpc 客户端时,一切都很好,但是当针对 docker 容器中的同一服务运行相同的客户端时,我收到此错误:

transport: http2Client.notifyError got notified that the client transport was broken EOF.
FATA[0000] rpc error: code = Internal desc = transport is closing

这是我的 docker 文件:

FROM golang:1.7.5

RUN mkdir -p /go/src/github.com/foo/bar
WORKDIR /go/src/github.com/foo/bar

COPY . /go/src/github.com/foo/bar
# ONBUILD RUN go-wrapper download
RUN go install

ENTRYPOINT /go/bin/bar

EXPOSE 51672

我构建镜像的命令:

docker build -t bar .

我启动 docker 容器的命令:

docker run -p 51672:51672 --name bar-container bar

其他信息:

  • 客户端程序在 docker 容器中运行良好
  • 连接到常规的 rest 端点可以正常工作(http2,grpc 相关?)
  • 在 OS X 中运行 lsof 命令会产生这些结果

    $lsof -i | grep 51672
    com.docke   984 oldDave   21u  IPv4 0x72779547e3a32c89      0t0  TCP *:51672 (LISTEN)
    com.docke   984 oldDave   22u  IPv6 0x72779547cc0fd161      0t0  TCP localhost:51672 (LISTEN)
    
  • 这是我的服务器代码片段:

    server := &Server{}
    endpoint := "localhost:51672"
    lis, err := net.Listen("tcp", endpoint)
    if err != nil {
        log.Fatalf("failed to listen: %v", err)
    }
    
    s := grpc.NewServer(grpc.Creds(creds))
    
    pb.RegisterExpServiceServer(s, server)
    
    // Register reflection service on gRPC server.
    reflection.Register(s)
    
    log.Info("Starting Exp server: ", endpoint)
    
    if err := s.Serve(lis); err != nil {
        log.Fatalf("failed to serve: %v", err)
    }
    

最佳答案

当您指定要监听的主机名或 IP 地址(在本例中为解析为 127.0.0.1 的 localhost)时,您的服务器将仅监听该 IP 地址。

当您在 Docker 容器之外时,在 localhost 上监听不是问题。如果您的服务器仅监听 127.0.0.1:51672,那么您的客户端可以轻松连接到它,因为连接也是从 127.0.0.1 建立的。

当您在 Docker 容器中运行服务器时,它只会像以前一样监听 127.0.0.1:51672。 127.0.0.1 是本地环回地址,无法在容器外访问。

当您使用“-p 51672:51672”启动 docker 容器时,它会将前往 127.0.0.1:51672 的流量转发到容器的 IP 地址,在我的例子中是 172.17.0.2。

容器在 docker0 网络接口(interface)中获取一个 IP 地址(您可以使用“ip addr ls”命令查看)

因此,当您的流量被转发到 172.17.0.2:51672 上的容器时,那里没有任何监听并且连接尝试失败。

修复:

问题在于监听端点:

endpoint := "localhost:51672"

要解决您的问题,请将其更改为

endpoint := ":51672"

这将使您的服务器监听所有容器的 IP 地址。

其他信息:

当您在 Docker 容器中公开端口时,Docker 将创建 iptables 规则来进行实际转发。见 this .您可以查看这些规则 与:

iptables -n -L 
iptables -t nat -n -L

关于docker - 无法连接到在本地 Docker 容器中运行的 Go GRPC 服务器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43911793/

相关文章:

docker - 如何协调Docker容器之间的端口

docker - 关于 Mesos Linux 容器和 Docker 的问题

api - slice ,groupBy []数组

go - 如何使用在golang中通过引用传递的索引访问 slice 中的元素

Golang gRPC 服务器流

java - 在 IntelliJ 中使用 Gradle 导入 Protobuf 生成的类

java - CPU 使用率高的 Docker Java 8 容器

docker - 在Docker内部发出curl请求时连接被拒绝

arrays - 如何将 JOIN 映射到具有子结构数组的结构数组

go - 将内部 go struct 数组转换为 protobuf 生成的指针数组