Portainer stalling when running behind caddy

1. The problem I’m having:

I try to run portainer in a docker container behind caddy, also in a docker container.

Accessing portainer via http://dockerhost.local:9000 or https://dockerhost.local:9443 works, so portainer should be fine.

When trying to access portainer via https://admin.lab.htl-villach.at, portainer stalls with “Portainer loading …”

Note that admin.lab.htl-villach.at is not reachable from the internet.

Accessing a NGINX instance via the same caddy works fine. Letsenrypt certs are also fine.

2. Error messages and/or full log output:

curl -vl https://admin.lab.htl-villach.at
* Rebuilt URL to: https://admin.lab.htl-villach.at/
*   Trying 172.31.32.110...
* TCP_NODELAY set
* Connected to admin.lab.htl-villach.at (172.31.32.110) port 443 (#0)
* schannel: SSL/TLS connection with admin.lab.htl-villach.at port 443 (step 1/3)
* schannel: checking server certificate revocation
* schannel: sending initial handshake data: sending 195 bytes...
* schannel: sent initial handshake data: sent 195 bytes
* schannel: SSL/TLS connection with admin.lab.htl-villach.at port 443 (step 2/3)
* schannel: encrypted data got 4096
* schannel: encrypted data buffer: offset 4096 length 4096
* schannel: encrypted data length: 194
* schannel: encrypted data buffer: offset 194 length 4096
* schannel: received incomplete message, need more data
* schannel: SSL/TLS connection with admin.lab.htl-villach.at port 443 (step 2/3)
* schannel: encrypted data got 451
* schannel: encrypted data buffer: offset 645 length 4096
* schannel: sending next handshake data: sending 93 bytes...
* schannel: SSL/TLS connection with admin.lab.htl-villach.at port 443 (step 2/3)
* schannel: encrypted data got 195
* schannel: encrypted data buffer: offset 195 length 4096
* schannel: SSL/TLS handshake complete
* schannel: SSL/TLS connection with admin.lab.htl-villach.at port 443 (step 3/3)
* schannel: stored credential handle in session cache
> GET / HTTP/1.1
> Host: admin.lab.htl-villach.at
> User-Agent: curl/7.55.1
> Accept: */*
>
* schannel: client wants to read 102400 bytes
* schannel: encdata_buffer resized 103424
* schannel: encrypted data buffer: offset 0 length 103424
* schannel: encrypted data got 19811
* schannel: encrypted data buffer: offset 19811 length 103424
* schannel: decrypted data length: 382
* schannel: decrypted data added: 382
* schannel: decrypted data cached: offset 382 length 102400
* schannel: encrypted data length: 19400
* schannel: encrypted data cached: offset 19400 length 103424
* schannel: decrypted data length: 2358
* schannel: decrypted data added: 2358
* schannel: decrypted data cached: offset 2740 length 102400
* schannel: encrypted data length: 17013
* schannel: encrypted data cached: offset 17013 length 103424
* schannel: decrypted data length: 1738
* schannel: decrypted data added: 1738
* schannel: decrypted data cached: offset 4478 length 102400
* schannel: encrypted data length: 15246
* schannel: encrypted data cached: offset 15246 length 103424
* schannel: decrypted data length: 4716
* schannel: decrypted data added: 4716
* schannel: decrypted data cached: offset 9194 length 102400
* schannel: encrypted data length: 10501
* schannel: encrypted data cached: offset 10501 length 103424
* schannel: decrypted data length: 5895
* schannel: decrypted data added: 5895
* schannel: decrypted data cached: offset 15089 length 102400
* schannel: encrypted data length: 4577
* schannel: encrypted data cached: offset 4577 length 103424
* schannel: decrypted data length: 4483
* schannel: decrypted data added: 4483
* schannel: decrypted data cached: offset 19572 length 102400
* schannel: encrypted data length: 65
* schannel: encrypted data cached: offset 65 length 103424
* schannel: decrypted data length: 2
* schannel: decrypted data added: 2
* schannel: decrypted data cached: offset 19574 length 102400
* schannel: encrypted data length: 34
* schannel: encrypted data cached: offset 34 length 103424
* schannel: decrypted data length: 5
* schannel: decrypted data added: 5
* schannel: decrypted data cached: offset 19579 length 102400
* schannel: encrypted data buffer: offset 0 length 103424
* schannel: decrypted data buffer: offset 19579 length 102400
* schannel: schannel_recv cleanup
* schannel: decrypted data returned 19579
* schannel: decrypted data buffer: offset 0 length 102400
< HTTP/1.1 200 OK
< Accept-Ranges: bytes
< Alt-Svc: h3=":443"; ma=2592000
< Cache-Control: max-age=31536000
< Connection: upgrade
< Content-Type: text/html; charset=utf-8
< Date: Mon, 20 Mar 2023 20:54:09 GMT
< Last-Modified: Thu, 23 Feb 2023 02:11:06 GMT
< Server: Caddy
< Vary: Accept-Encoding
< X-Content-Type-Options: nosniff
< X-Xss-Protection: 1; mode=block
< Transfer-Encoding: chunked
<
<!doctype html><html lang="en" ng-app="portainer" ng-strict-di data-edition="CE"><head><meta charset="utf-8"/><title>Portainer</title><meta name="description" content=""/><meta name="author" content="Portainer.io"/><meta http-equiv="cache-control" content="no-cache"/><meta http-equiv="expires" content="0"/><meta http-equiv="pragma" content="no-cache"/><base id="base"/><script>if (window.origin == 'file://') {
        // we are loading the app from a local file as in docker extension
        document.getElementById('base').href = 'http://localhost:49000/';

        window.ddExtension = true;
      } else {
        var path = window.location.pathname.replace(/^\/+|\/+$/g, '');
        var basePath = path ? '/' + path + '/' : '/';
        document.getElementById('base').href = basePath;
      }</script><!--[if lt IE 9]>
      <script src="//html5shim.googlecode.com/svn/trunk/html5.js"></script>
    <![endif]--><link rel="apple-touch-icon" sizes="180x180" href="63a301f0574f1a696ce6.png"/><link rel="icon" type="image/png" sizes="32x32" href="2dcfc527d067d4ae3424.png"/><link rel="icon" type="image/png" sizes="16x16" href="112a479c093f4729251d.png"/><link rel="mask-icon" href="" color="#5bbad5"/><link rel="shortcut icon" href=""/><meta name="msapplication-config" content="6d50eaeb9f128c130ed9.xml"/><meta name="theme-color" content="#ffffff"/><script defer="defer" src="runtime.3e3d5da69a6f597f1396.js"></script><script defer="defer" src="vendor.ae28ff4d1f105fcc56e1.js"></script><script defer="defer" src="main.15f004558ddbf71f405e.js"></script><link href="vendor.ada35035b3b2d76da384.css" rel="stylesheet"><link href="main.a008fa267e62ff64a95b.css" rel="stylesheet"></head><body ng-controller="MainController"><react-query-dev-tools></react-query-dev-tools><div id="page-wrapper" ng-class="{
        open: isSidebarOpen() && ['portainer.auth', 'portainer.init.admin', 'portainer.init.endpoint'].indexOf($state.current.name) === -1,
        nopadding: ['portainer.auth', 'portainer.init.admin', 'portainer.init.endpoint', 'portainer.logout'].indexOf($state.current.name) > -1 || applicationState.loading
      }" ng-cloak><div id="sideview" ui-view="sidebar" ng-if="!applicationState.loading"></div><div id="content-wrapper"><div class="page-content"><div class="page-wrapper" ng-if="applicationState.loading"><div class="container simple-box"><div class="col-md-6 col-md-offset-3 col-sm-6 col-sm-offset-3"><div class="row"><img ng-if="logo" ng-src="{{ logo }}" class="simple-box-logo"/> <img ng-if="!logo" src="" class="simple-box-logo" alt="Portainer"/></div><div class="ml-0 mr-0 flex items-center justify-center text-center">Loading Portainer... <svg xmlns="http://www.w3.org/2000/svg" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="inline animate-spin-slow !ml-1"><circle cx="12" cy="12" r="3"></circle><path d="M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z"></path></svg></div></div></div></div><div id="view" ui-view="content" ng-if="!applicationState.loading"></div></div></div></div></body></html>* Connection #0 to host admin.lab.htl-villach.at left intact

3. Caddy version:

v2.6.4 h1:2hwYqiRwk1tf3VruhMpLcYTg+11fCdr8S3jhNAdnPy8=

4. How I installed and ran Caddy:

Caddy runs via docker-compose

a. System environment:

Host OS : “Ubuntu 22.04.2 LTS”
Docker version 23.0.1, build a5ee5b1

b. Command:

docker compose up -d

c. Service/unit/compose file:

Caddy
Dockerfile

FROM caddy:2.6.4-builder AS builder

RUN xcaddy build \
    --with github.com/caddy-dns/cloudflare

FROM caddy:2.6.4

COPY --from=builder /usr/bin/caddy /usr/bin/caddy

docker-compose.yml

version: "3.7"


services:
        caddy:
                image: caddy-cloudflare:2.6.4
                container_name: lab_caddy
                build: .
                restart: unless-stopped
                environment:
                  - CLOUDFLARE_API_TOKEN=${CLOUDFLARE_API_TOKEN}
                networks:
                  - lab_nginx_net
                  - lab_portainer_net
                ports:
                  - target: 80
                    published: 80
                    protocol: tcp
      # HTTPS
                  - target: 443
                    published: 443
                    protocol: tcp
      # HTTP/3
                  - target: 443
                    published: 443
                    protocol: udp

                volumes:
                        - /root/docker-server/caddy/etc/caddy/Caddyfile:/etc/caddy/Caddyfile
                        - /root/docker-server/caddy/data:/data # Optional
                        - /root/docker-server/caddy/config:/config # Optional
                        - /var/run/docker.sock:/var/run/docker.sock

networks:
  lab_nginx_net:
    name: lab_nginx_net
  lab_portainer_net:
    name: lab_portainer_net
                                

Portainer:

version: "3"
services:
  portainer:
    image: portainer/portainer-ce:latest
    container_name: lab_portainer
    restart: unless-stopped

    networks:
      - lab_portainer_net
    ports:
      - 9443:9443
      - 9000:9000
      - 8000:8000
    volumes:
      - portainer_data:/data
      - /var/run/docker.sock:/var/run/docker.sock

volumes:
  portainer_data:


networks:
  lab_portainer_net:
    name: lab_portainer_net


d. My complete Caddy config:

{
        email Robert.Hufsky@htl-villach.at
        acme_dns cloudflare {env.CLOUDFLARE_API_TOKEN}
        ## acme_ca https://acme-staging-v02.api.letsencrypt.org/directory
        ## admin off
}

########################################################
#
#  handle these servers with a wildcard certificate
#
########################################################

*.lab.htl-villach.at {
        tls {
                dns cloudflare {env.CLOUDFLARE_API_TOKEN}
        }

        ################################################
        #
        # nginx demo server
        #
        ################################################

        @nginx host nginx.lab.htl-villach.at
        handle @nginx {
                reverse_proxy / lab_nginx:80
        }

        ################################################
        #
        # admin portainer
        #
        ################################################

        @admin host admin.lab.htl-villach.at
        handle @admin {
                reverse_proxy / https://lab_portainer:9000
        }

        # Fallback for otherwise unhandled domains
        handle {
                abort
        }
}

5. Links to relevant resources:

Unfortunately we have some known issues with chunked encoding. Could you try with v2.6.2 for now, or from the master branch? (You can build with xcaddy build master). We think the bug is fixed but it hasn’t been released yet.

You issue may also be here, remove the /. Using a / matcher means it would only proxy requests to exactly / and nothing else. Path matching is exact in Caddy.

You are right, this solved it for me:

                reverse_proxy lab_portainer:9000

I still use 2.6.4

Thanks a lot, you saved my day !!

1 Like

This topic was automatically closed 30 days after the last reply. New replies are no longer allowed.