複数の Docker コンテナが同じポート番号を持たない場合の解決策

複数の Docker コンテナが同じポート番号を持たない場合の解決策

背景

Dockerでは、同じイメージを使用して4つのコンテナを作成します。ネットワークはブリッジモードです。サービスAは、4つのコンテナすべてで同じポート番号(6000)を使用します。外部に公開されるポートの数を減らすために、4つのサービスインスタンスのプロキシとしてnginxを使用します。4つのサービスインスタンスは4つのアップストリームに属しており、4つのインスタンスへのアクセスには/service1や/service2などのパスが使用されます。

現時点では、ローカルで何らかのサービスにアクセスすると 502 エラーが報告され、困惑します。

アップストリームへの接続中に connect() が失敗しました (111: 接続が拒否されました)

ファイルを作成

バージョン: '2'
ネットワーク:
 んん:
  ドライバー: ブリッジ
サービス:
 サービス1:
  コンテナ名: サービス1
  画像: foo
  ネットワーク:
   -んんん
  ボリューム:
   - ./logs/1:/apps/aaa/bbb-logs
   - ./common:/apps/aaa/bbb
   - ./xxx/1.xml:/ccc/targets.xml
  エントリポイント: foo.sh
  コマンド: start app=foo port=6000
  
 サービス2:
  コンテナ名: サービス2
  画像: foo
  ネットワーク:
   -んんん
  ボリューム:
   - ./logs/2:/apps/aaa/bbb-logs
   - ./common:/apps/aaa/bbb
   - ./xxx/2.xml:/ccc/targets.xml
  エントリポイント: foo.sh
  コマンド: start app=foo port=6000
 
 サービス3:
  コンテナ名: サービス3
  画像: foo
  ネットワーク:
   -んんん
  ボリューム:
   - ./logs/3:/apps/aaa/bbb-logs
   - ./common:/apps/aaa/bbb
   - ./xxx/3.xml:/ccc/targets.xml
  エントリポイント: foo.sh
  コマンド: start app=foo port=6000
 
 サービス4:
  コンテナ名: サービス4
  画像: foo
  ネットワーク:
   -んんん
  ボリューム:
   - ./logs/4:/apps/aaa/bbb-logs
   - ./common:/apps/aaa/bbb
   - ./xxx/4.xml:/ccc/targets.xml
  エントリポイント: foo.sh
  コマンド: start app=foo port=6000  
 
 nginx:
  コンテナ名: nginx
  イメージ: nginx:1.15-alpine
  ポート:
   - 6001:6001
  ネットワーク:
   -んんん
  ボリューム:
   ./nginx/nginx.conf:/etc/nginx/nginx.conf
   nginx のログファイル

nginx.conf

ワーカープロセス数 8;
ワーカー_rlimit_nofile 65535; 
イベント {
    epoll を使用します。
    ワーカー接続 65535;
 }
 
http {
    mime.types を含めます。
    default_type アプリケーション/オクテットストリーム;
    ファイル送信オン;
    log_format main '[$time_local]$remote_addr-$upstream_addr "$request" $status $body_bytes_sent';
 
    アップストリームサービス1.local {
      サーバーサービス-1:6000;
    }
    アップストリームサービス2.local {
     サーバーサービス2:6000;
    }
    アップストリームサービス3.local {
      サーバーサービス3:6000;
    }
    アップストリームサービス4.local {
      サーバーサービス-4:6000;
    }
 
    サーバー{
      6001を聴く;
      クライアントの最大ボディサイズは100Mです。
      proxy_set_header ホスト $host:$server_port;
      proxy_set_header X-Real-IP $remote_addr;
      proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
 
      場所 /service1/ {
        proxy_pass http://service1.local/;
      }
      場所 /service2/ {
        proxy_pass http://service2.local/;
      }
      場所 /service3/ {
        proxy_pass http://service3.local/;
      }
      場所 /service4/ {
        proxy_pass http://service4.local/;
      }
      場所 /nginx_status {
        stub_status オン;
        access_log オフ;
      }
    }
}
 

このとき、 curl localhost:6001/service1/api/v1/.... は上記の 502 エラーを報告します。各コンテナには独自のネットワーク カードがあり、異なるコンテナのポート番号が競合しないのは当然です。

解決

現時点ではこれより良い解決策はないので、4 つのサービスに異なるポート番号を使用し、それに応じて nginx を変更するしかありません。

補足: 同じサーバーに複数の Docker コンテナをデプロイするとポートリダイレクトの問題が発生する

実稼働環境では、複数のコンテナをデプロイし、複数のポートにアクセスします。

例: -p 80:80 -p 81:81

アドレス81が存在する場合は、ポート80のアドレスに直接アクセスします。

誤解: 最初はクッキーを更新したのでクッキーの問題だと思った (クッキーはポート番号を区別しない)

ついに理由が分かりました: リダイレクトの問題。ログアウトするとログインページにリダイレクトされるからです。

解決策: nginxパラメータを設定する

proxy_set_header HOST $host; は proxy_set_header HOST $host:81; になります。

なぜなら、リクエストパラメータには必ずポート番号が含まれるからです。

インターネット上には別の方法があります: proxy_redirect パラメータを変更する (ただし、試しても機能しませんでした)

上記は私の個人的な経験です。参考になれば幸いです。また、123WORDPRESS.COM を応援していただければ幸いです。間違いや不備な点がありましたら、遠慮なくご指摘ください。

以下もご興味があるかもしれません:
  • Dockerでudpポート番号を指定する問題を解決する
  • Raspberry Pi シリーズ: Docker を使用して Qinglong Panel をインストールし、ポート番号の設定の問題を変更する

<<:  MySQL 8.0 redo ログの詳細な分析

>>:  Vue のライフサイクルとフック関数の詳細な説明と典型的な面接の質問

推薦する

CentOS VPS に SSH 経由で MySQL をインストールする方法

yum install mysql-serverと入力します。続行するにはYを押してくださいインスト...

MySQL でスロークエリを有効にする方法の例

序文スロー クエリ ログは、MySQL で非常に重要な機能です。MySQL のスロー クエリ ログ機...

Linux のさまざまなロックメカニズムの使用方法と違いについて詳しく説明します

序文:この知識を理解する必要がある人は、すでにプロセス間通信とスレッド間通信の基本的な理解を持ってい...

MySQL でデータ テーブルを作成し、主キーと外部キーの関係を確立する方法の詳細な説明

序文MySQL テーブルの主キーと外部キーを作成するときは、次の点に注意する必要があります。主キーと...

ユーザーエクスペリエンスの要素またはWebデザインの要素

システムとユーザー環境の設計<br />Apple システムの成功は、そのシステム アー...

熟練デザイナーの7つの原則(2):色の使い方

<br />前回の記事:優秀なデザイナーの7つの原則(1):フォントデザイン 英語 原文...

MySQLの実行プロセスとシーケンスについての簡単な説明

目次1:mysql実行プロセス1.1: コネクタ1.2: キャッシュ1.3: アナライザー1.4: ...

Linuxのファイル操作の知識ポイントを詳しく解説

ファイル操作に関連するシステムコール作成するint creat(const char *ファイル名,...

JavaScript キャンバスで動的な点と線の効果を実現

この記事では、動的な点と線の効果を実現するためのJavaScriptキャンバスの具体的なコードを参考...

Docker5フル機能の港湾倉庫構築プロセス

Harbor は、Docker イメージを保存および配布するためのエンタープライズ レベルのレジスト...

Ubuntu 18.04 に opencv 3.2.0 をインストールするためのソリューション

opencv.zip をダウンロード依存関係を事前にインストールします。まずダウンロードソースを更新...

Vue.jsのレンダリング関数の使い方の詳しい説明

Vue では、ほとんどの場合、テンプレートを使用して HTML を作成することを推奨しています。ただ...

MySQL を暗号化および復号化するいくつかの方法 (要約)

目次前面に書かれた双方向暗号化エンコード/デコードAES_ENCRYPT/AES_DECRYPT D...

Nginx リバース プロキシを使用してクロスドメイン問題を解決する方法の詳細な説明

質問前回のクロスドメイン リソース共有に関する記事では、ドメイン間で Cookie を送信する場合、...

Hタグはウェブページ制作において適切に使用すべきである

HTML タグには、ページのタイトルを処理するための特別なタグがあります。これらは h1、h2、h3...