dockerのリバースプロキシ+Let's Encryptでお手軽に複数のWebサーバをSSL化

2022/04/26

Linux OracleCloud

t f B! P L

サーバ上でWebサーバを複数立てたい場合、WebサーバのnginxもしくはAppacheをインストールして、リッスンポートを開けて・・・といったことを複数回実施しないといけない。

更に今どき、インターネット公開する際にはhttps化はもはや必須。

そうなるとサーバ証明書を発行して、Webサーバに組み込んで・・・

みたいなことをWebサーバの数だけ実施しないといけないので、かなりめんどくさい。


それがdockerだと、手間なく、そして幾らでもWebサーバを作れるので

一つのサーバに複数Webサーバを立てたい時はかなり便利。

イメージにすると以下のような感じ。


AのWebサーバにアクセスしたい場合は「A.hogehoge.com」、

BのWebサーバにアクセスしたい場合は「B.hogehoge.com」といった具合に

サブドメインを変更するだけで、あとはよしなにリバースプロキシとhttps化を自動で行ってくれる。

そんな夢見たいな神ツールをdockerでイメージ提供されてるので、

Oracle Cloud上のUbuntuサーバで実装してみた。


あと、以前に書いたブログ記事の「Portainer」もこの方法でhttps化ができるので、後で紹介する。



前提

以下を準備すること。特に上2つは必須。

  • パブリック公開可能なdockerホスト(ここではOCIのUbuntu)
  • 独自ドメインとDNS編集可能な環境
  • dockerの知識
  • 運用したいWebサーバ(WordPressとかでもいいんじゃない?)

すでにホスト上にdocker環境は設定済みとする。


リバースプロキシ用コンテナ作成

docker-compose.ymlなりPortainerで以下のコードを書いて、コンテナをUPするだけ。

  1. version: '3'
  2. services:
  3. nginx-proxy:
  4. image: jwilder/nginx-proxy
  5. container_name: nginx-proxy
  6. ports:
  7. - 80:80
  8. - 443:443
  9. volumes:
  10. - /docker/nginx-proxy/certs:/etc/nginx/certs:ro
  11. - /docker/nginx-proxy/vhost:/etc/nginx/vhost.d
  12. - /docker/nginx-proxy/html:/usr/share/nginx/html
  13. - /var/run/docker.sock:/tmp/docker.sock:ro
  14. restart: always
  15. environment:
  16. TZ: Asia/Tokyo
  17. labels:
  18. - "com.github.jrcs.letsencrypt_nginx_proxy_companion.nginx_proxy"
  19.  
  20. letsencrypt-nginx-proxy-companion:
  21. image: nginxproxy/acme-companion
  22. container_name: nginx-letsencrypt
  23. volumes:
  24. - /docker/nginx-proxy/certs:/etc/nginx/certs
  25. - /docker/nginx-proxy/vhost:/etc/nginx/vhost.d
  26. - /docker/nginx-proxy/html:/usr/share/nginx/html
  27. - /var/run/docker.sock:/var/run/docker.sock:ro
  28. restart: always
  29. depends_on:
  30. - nginx-proxy
  31. environment:
  32. TZ: Asia/Tokyo
  33. DEAULT_EMAIL: "xxxx@xxxx.xxx"

一番最後の行の「DEAULFT_EMAIL」に連絡先用メールアドレスを書いておくと、

転送先のコンテナにいちいちメールアドレスを書く必要がなくなるので、

ここに書いておくことを推奨。


リバースプロキシ用の"nginx-proxy"という名前のコンテナと、Let's EncryptでSSL証明書を発行してくれる"nginx-letsencrypt"というコンテナがそれぞれ起動し、

他のコンテナの稼働状況を確認し、自動的にSSL証明書発行とWeb転送を行なってくれる。


転送先のWebサーバ用コンテナ作成

次に転送先となるWebサーバのコンテナを用意する。

ここでは例として、「Portainer」のコンテナを立てることにする。

docker-compose.ymlで以下を作成。

  1. version: "3"
  2. services:
  3. portainer:
  4. image: portainer/portainer-ce
  5. container_name: portainer
  6. ports:
  7. - 8000:8000
  8. volumes:
  9. - /var/run/docker.sock:/var/run/docker.sock
  10. - portainer_data:/data
  11. restart: always
  12. environment:
  13. - TZ=Asia/Tokyo
  14. - VIRTUAL_HOST=xxxx.hogehoge.com
  15. - VIRTUAL_PORT=9000
  16. - LETSENCRYPT_HOST=xxxx.hogehoge.com
  17.  
  18. volumes:
  19. portainer_data:
  20. external: true
  21.  
  22. networks:
  23. default:
  24. external:
  25. name: nginx-proxy_default


ポイントは以下。

「VIRTUAL_HOST」および「LETSENCRYPT_HOST」の値にアクセスしたいURLのFQDN名にすること。

「VIRTUAL_PORT」で本来Webアクセスするポート番号を指定すること。

「networks」で、リバースプロキシ用nginxコンテナと同じネットワークに所属させること。

"external"句で、外部コンテナのネットワークを利用することができる。


Portainerの場合は、「http://<IPアドレス>:9000」で本来はアクセスするので、

VIRTUAL_PORTは9000になる。


その他、ポート開放とDNSレコード更新

これでコンテナの準備は完了。

あとはサーバのポート開放。

80/tcpと443/tcpの2つを最低限開放。

80/tcpはLet's Encryptの証明書更新用に必要。


あとはVIRTUAL_HOSTで指定したFQDNに該当するDNSレコード(AレコードやCNAME等)を設定。

俺の場合、ポート開放はOCIのセキュリティリスト、

DNSレコードはAWSのRoute53なんだけど、割愛wwwwww


nginx.confの設定変更等は不要

DNSレコードが伝播して名前解決ができるようになると、

Webサーバにアクセスすることができる。

SSL証明書もご覧のとおり設定済み。


Let's Encrypt自体どうなの?っていうのもあるが、

オレオレ証明書よりかは遥かにマシだし、

それほどお金かけるほどでもないようなWebサービスを自前で運営したい時には便利なんじゃなかろうか?


商用利用するには、キチンとお金を出して、

AWSのACM+ELBの方が圧倒的に楽。


なので、この方法は手軽なんだけど、個人利用以外の用途で実用性は?という感じw

でも、個別にリバースプロキシの設定やら、SSL証明書の設定をチマチマ行うことを考えると圧倒的にこちらの方が楽だよね。


注意点

Let's Encryptの仕様上、SSL証明書発行には一定期間での上限が決められている。

んで、このコンテナは起動するたびに証明書を再作成する。

そのため、Let's Encryptのコンテナを何度も再起動していると、回数制限にひっかかってしまい証明書の発行ができなくなり、Webサーバにアクセスできなくなるので注意。

俺も設定する時にちょっと手間取ってしまい、何度もコンテナ再起動を繰り返していたら

制限にひっかかってしまった。


何日か経てば制限が解除されるのだけど、制限中はWebサーバダウンという扱いになってしまうので、

ブログ運営とかしてる環境では要注意!


コンテナにより、同一ホストで複数アプリをマイクロサービス化して動かすことが当たり前に。

Webアクセスする時にその都度、SSL設定する手間を省くことができるので、

便利な人はすげー便利なんじゃないかと。

ホストをk8s等でクラスタ化すればさらに完璧だな!w

検索

Blog Archive

Popular Posts

About Me

自分の写真
性別:男
年齢:ついに40over
趣味:Snowboard、パソコン、iPhone、子育て

仕事:ユー子の社内SEとしてサーバ、NW等のインフラ全般をやってます

日々生活していく中で思ったことなどをつらつらと書いていきます。

どうぞよろしく!

ブログランキング

ブログランキング・にほんブログ村へ

QooQ