docker - 在 docker 容器中通过桥接时无法访问 SSL IP。获取 SSL_ERROR_SYSCALL

标签 docker ssl curl openssl

我在通过 IP + TLS 连接到任何服务器时遇到问题,但只能在(默认)网桥中运行时从 docker 容器内连接。我总是收到 OpenSSL SSL_connect: SSL_ERROR_SYSCALL in connection to W.X.Y.Z .我试过 tcpdump (在容器中)和 wireshark (在主机上本地)无济于事。
我的工作伙伴具有相同的 OS/Docker 版本,无法重现该问题。我不知道如何调试这个问题。
我努力了:

  • 各种图像(ubuntu 和 alpine)
  • 各种客户端(curl 和 wget)
  • 各种 TLS 版本(1.3 和 1.2)

  • 我的容器:
    FROM ubuntu:latest
    RUN apt update && apt upgrade -y && apt install -y curl tcpdump openssl wget
    
    问题:
    docker run -it --rm repro /bin/bash
    
    # in the docker bash shell, if I try to curl a regular https hostname, all is well:
    root@ba6f8aab182d:/# curl -v  https://www.google.com
    *   Trying 172.217.10.36:443...
    * TCP_NODELAY set
    * Connected to www.google.com (172.217.10.36) port 443 (#0)
    * ALPN, offering h2
    * ALPN, offering http/1.1
    * successfully set certificate verify locations:
    *   CAfile: /etc/ssl/certs/ca-certificates.crt
      CApath: /etc/ssl/certs
    * TLSv1.3 (OUT), TLS handshake, Client hello (1):
    * TLSv1.3 (IN), TLS handshake, Server hello (2):
    * TLSv1.3 (IN), TLS handshake, Encrypted Extensions (8):
    * TLSv1.3 (IN), TLS handshake, Certificate (11):
    * TLSv1.3 (IN), TLS handshake, CERT verify (15):
    * TLSv1.3 (IN), TLS handshake, Finished (20):
    * TLSv1.3 (OUT), TLS change cipher, Change cipher spec (1):
    * TLSv1.3 (OUT), TLS handshake, Finished (20):
    * SSL connection using TLSv1.3 / TLS_AES_256_GCM_SHA384
    * ALPN, server accepted to use h2
    * Server certificate:
    *  ...
    
    
    # if I try again but this time with the ip instead of the hostname
    root@ba6f8aab182d:/# curl -v https://172.217.10.36
    *   Trying 172.217.10.36:443...
    * TCP_NODELAY set
    * Connected to 172.217.10.36 (172.217.10.36) port 443 (#0)
    * ALPN, offering h2
    * ALPN, offering http/1.1
    * successfully set certificate verify locations:
    *   CAfile: /etc/ssl/certs/ca-certificates.crt
      CApath: /etc/ssl/certs
    * TLSv1.3 (OUT), TLS handshake, Client hello (1):
    * OpenSSL SSL_connect: SSL_ERROR_SYSCALL in connection to 172.217.10.36:443
    * Closing connection 0
    curl: (35) OpenSSL SSL_connect: SSL_ERROR_SYSCALL in connection to 172.217.10.36:443
    
    上面的 2 个调用(curl HOSTNAME 然后 curl MATCHINGIP)在主机上工作得很好。
    ubuntu 容器中的额外信息:
    root@ba6f8aab182d:/# openssl version
    OpenSSL 1.1.1f  31 Mar 2020
    root@ba6f8aab182d:/# curl --version
    curl 7.68.0 (x86_64-pc-linux-gnu) libcurl/7.68.0 OpenSSL/1.1.1f zlib/1.2.11 brotli/1.0.7 libidn2/2.2.0 libpsl/0.21.0 (+libidn2/2.2.0) libssh/0.9.3/openssl/zlib nghttp2/1.40.0 librtmp/2.3
    Release-Date: 2020-01-08
    Protocols: dict file ftp ftps gopher http https imap imaps ldap ldaps pop3 pop3s rtmp rtsp scp sftp smb smbs smtp smtps telnet tftp
    Features: AsynchDNS brotli GSS-API HTTP2 HTTPS-proxy IDN IPv6 Kerberos Largefile libz NTLM NTLM_WB PSL SPNEGO SSL TLS-SRP UnixSockets
    
    来自主机的额外信息:
    $ docker version
    Client: Docker Engine - Community
     Cloud integration: 1.0.12
     Version:           20.10.5
     API version:       1.41
     Go version:        go1.13.15
     Git commit:        55c4c88
     Built:             Tue Mar  2 20:13:00 2021
     OS/Arch:           darwin/amd64
     Context:           default
     Experimental:      true
    
    Server: Docker Engine - Community
     Engine:
      Version:          20.10.5
      API version:      1.41 (minimum version 1.12)
      Go version:       go1.13.15
      Git commit:       363e9a8
      Built:            Tue Mar  2 20:15:47 2021
      OS/Arch:          linux/amd64
      Experimental:     false
     containerd:
      Version:          1.4.4
      GitCommit:        05f951a3781f4f2c1911b05e61c160e9c30eaa8e
     runc:
      Version:          1.0.0-rc93
      GitCommit:        12644e614e25b05da6fd08a38ffa0cfe1903fdec
     docker-init:
      Version:          0.19.0
      GitCommit:        de40ad0
    
    $ docker network inspect bridge
    [
        {
            "Name": "bridge",
            "Id": "4b8797bccccd628a6280199eb5c0372cd08d521a88a29243b174718569e9cc7e",
            "Created": "2021-04-15T17:24:33.631745871Z",
            "Scope": "local",
            "Driver": "bridge",
            "EnableIPv6": false,
            "IPAM": {
                "Driver": "default",
                "Options": null,
                "Config": [
                    {
                        "Subnet": "172.17.0.0/16",
                        "Gateway": "172.17.0.1"
                    }
                ]
            },
            "Internal": false,
            "Attachable": false,
            "Ingress": false,
            "ConfigFrom": {
                "Network": ""
            },
            "ConfigOnly": false,
            "Containers": {
                "ba6f8aab182daebd4f0b0dc449929585637cd46bc532f61991bfa28c40e09ceb": {
                    "Name": "flamboyant_zhukovsky",
                    "EndpointID": "e29407baf0eb8ac069416a8f787794b548d31f05bf9fb6c223fb58c935aff24c",
                    "MacAddress": "02:42:ac:11:00:02",
                    "IPv4Address": "172.17.0.2/16",
                    "IPv6Address": ""
                }
            },
            "Options": {
                "com.docker.network.bridge.default_bridge": "true",
                "com.docker.network.bridge.enable_icc": "true",
                "com.docker.network.bridge.enable_ip_masquerade": "true",
                "com.docker.network.bridge.host_binding_ipv4": "0.0.0.0",
                "com.docker.network.bridge.name": "docker0",
                "com.docker.network.driver.mtu": "1500"
            },
            "Labels": {}
        }
    ]
    
    编辑
    尝试openssl s_client -cipher ALL -servername 172.217.10.36:443 -connect 172.217.10.36:443在容器中,我得到:
    root@ba6f8aab182d:/# openssl s_client -cipher ALL -servername 172.217.10.36:443 -connect 172.217.10.36:443
    CONNECTED(00000003)
    write:errno=0
    ---
    no peer certificate available
    ---
    No client certificate CA names sent
    ---
    SSL handshake has read 0 bytes and written 403 bytes
    Verification: OK
    ---
    New, (NONE), Cipher is (NONE)
    Secure Renegotiation IS NOT supported
    Compression: NONE
    Expansion: NONE
    No ALPN negotiated
    Early data was not sent
    Verify return code: 0 (ok)
    
    在主机上我得到:
    openssl s_client -cipher ALL -servername 172.217.10.36:443 -connect 172.217.10.36:443
    CONNECTED(00000003)
    depth=2 OU = GlobalSign Root CA - R2, O = GlobalSign, CN = GlobalSign
    verify return:1
    depth=1 C = US, O = Google Trust Services, CN = GTS CA 1O1
    verify return:1
    depth=0 C = US, ST = California, L = Mountain View, O = Google LLC, CN = www.google.com
    verify return:1
    ---
    Certificate chain
     0 s:/C=US/ST=California/L=Mountain View/O=Google LLC/CN=www.google.com
       i:/C=US/O=Google Trust Services/CN=GTS CA 1O1
     1 s:/C=US/O=Google Trust Services/CN=GTS CA 1O1
       i:/OU=GlobalSign Root CA - R2/O=GlobalSign/CN=GlobalSign
    ---
    Server certificate
    -----BEGIN CERTIFICATE-----
    [... certificate was here ...]
    -----END CERTIFICATE-----
    subject=/C=US/ST=California/L=Mountain View/O=Google LLC/CN=www.google.com
    issuer=/C=US/O=Google Trust Services/CN=GTS CA 1O1
    ---
    No client certificate CA names sent
    Server Temp Key: ECDH, X25519, 253 bits
    ---
    SSL handshake has read 3206 bytes and written 339 bytes
    ---
    New, TLSv1/SSLv3, Cipher is ECDHE-RSA-CHACHA20-POLY1305
    Server public key is 2048 bit
    Secure Renegotiation IS supported
    Compression: NONE
    Expansion: NONE
    No ALPN negotiated
    SSL-Session:
        Protocol  : TLSv1.2
        Cipher    : ECDHE-RSA-CHACHA20-POLY1305
        Session-ID: 052A69E409C0705AEAB8A180228C3F8E91A530504EFB06BA9214365F6B99DCAC
        Session-ID-ctx:
        Master-Key: A4EAC218352BBEAF3A43AB625266304DCF495FFE8A916C638679473AD20DC01B508158B8C0AA39A97003FEC5B8ABD7EC
        TLS session ticket lifetime hint: 100800 (seconds)
        TLS session ticket:
        [... a lot of stuff here ...]
        Start Time: 1618514134
        Timeout   : 7200 (sec)
        Verify return code: 0 (ok)
    ---
    

    最佳答案

    似乎问题出在 Docker-For-Mac 3.3.0 和 3.3.1 上。
    解决方案是降级到 3.2.2,即使 docker 引擎是相同的。
    https://github.com/docker/for-mac/issues/5568

    关于docker - 在 docker 容器中通过桥接时无法访问 SSL IP。获取 SSL_ERROR_SYSCALL,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67114041/

    相关文章:

    ssl - Kubernetes Pod之间的TLS通信(更轻量级的解决方案)

    php - CURL 仅在文件被修改时获取文件

    javascript - 创建包含数组类型的 Mongoose 模型对于我的 docker 容器中的那些类型返回未定义

    docker - 如何使用 Docker 注册表 API 提取有关容器的信息?未经授权

    asp.net - 在 docker 中安装 .net framework 4.7.2

    c++ - 如何使用curl使用C++代码下载文件?

    c++ - CURL 请求失败重试方法

    nginx - 具有自定义网络的 Docker 代理

    ssl - dovecot SSL 握手问题

    java - Spring Boot 和 Camel Mail SSL 配置