Warning: file_get_contents(/data/phpspider/zhask/data//catemap/0/docker/10.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
无法连接到在本地Docker容器中运行的Go GRPC服务器_Docker_Go_Grpc - Fatal编程技术网

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

无法连接到在本地Docker容器中运行的Go GRPC服务器,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 这是我的doc

我有一个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相关?)
  • 在OSX中运行
    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是本地环回地址,在容器外部无法访问

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

    容器在docker0网络接口内获取IP地址(您可以通过“IP addr ls”命令看到)

    因此,当您的流量在172.17.0.2:51672转发到容器时,没有任何内容在侦听,连接尝试失败

    修复方法:

    问题在于侦听终结点:

    endpoint := "localhost:51672"
    
    要解决您的问题,请将其更改为

    endpoint := ":51672"
    
    这将使您的服务器监听所有it容器的IP地址

    其他信息:

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


    @olafure将端点设置为“:[port]”在grpc的nodejs实现中不起作用。我刚刚收到这个错误
    {“created”:“@1548557943.020372671”,“description”:“getaddrinfo failed”,“file”:./deps/grpc/src/core/lib/iomgr/tcp_uv.cc”,“file_line”:73,“grpc_status”:14,“os_error”:“未知节点或服务”}
    谢谢你的回答。此外,如果您通过Docker Compose运行容器,您可以使用yaml文件中指定的容器名称作为主机名,它将自动映射到正确的IP地址。非常感谢您的回答。对于这个问题,我已经绞尽脑汁好几个小时了。将
    localhost
    更改为
    0.0.0
    对我来说很有效,我认为这类似于省略主机名。
    iptables -n -L 
    iptables -t nat -n -L