Docker コンテナにデプロイされた Django のタイムゾーンの問題

Docker コンテナにデプロイされた Django のタイムゾーンの問題

現在、コンテナ デプロイメントは非常に成熟しています。当社のサービスの多くはコンテナ デプロイメントを使用しており、更新とリカバリは非常に便利です。ただし、より厄介な問題があります。それは、タイム ゾーンの処理です。通常、これは TZ 環境変数を挿入することで解決されますが、この処理方法は Django では機能しません。

Django でのタイムゾーン設定

Django の設定ファイル settings.py には、時間とタイムゾーンに関連する TIME_ZONE と USE_TZ という 2 つの設定パラメータがあります。 settings.py で設定した後、Django はローカル時間を正しく取得できると期待していましたが、実際は期待に反しています。これら 2 つの設定が何をするのか見てみましょう。

USE_TZ=真

USE_TZ が True に設定されている場合、Django はシステムのデフォルトのタイムゾーンを使用します。このとき、TIME_ZONE 設定は基本的に無効であり、設定されているかどうかに関係なく効果はありません。

USE_TZ=偽

USE_TZがFalseに設定されている場合

  1. TIME_ZONEはNoneに設定されています
  2. Djangoはデフォルトのタイムゾーンを引き続き使用します
  3. TIME_ZONEが別のタイムゾーンに設定されている場合

Windows を使用している場合、TIME_ZONE 設定は役に立たず、Django はローカル時間を使用します。他のシステムを使用している場合は、そのタイムゾーンの UTC 時間を使用します。

たとえば、USE_TZ = False、TIME_ZONE = 'Asia/Shanghai' に設定すると、上海の UTC 時間が使用されます。

この時点では、時間は適切だと思うかもしれませんが、実際はそうではありません。システムのタイムゾーン設定にも注意する必要があります。

Linux コンテナでのタイムゾーンの設定

私の現地時間は16:15で、Djangoの設定はUSE_TZ = False、TIME_ZONE = 'Asia/Shanghai'です。

コンテナの時間とタイムゾーンを表示するためにコンテナに入るときに、TZ=Asia/Shanghai 環境変数を挿入しないでください。

システム時間はUTCタイムゾーンで表示され、時刻は08:15で、ちょうど8時間の差です。

Django環境に入り、時間とタイムゾーンを表示します。

python manage.py シェル 
 
datetime から datetime をインポート 
日付時刻.now() 
# 出力 datetime.datetime(2021, 10, 8, 8, 24, 8, 289230) 
 
django.utilsからタイムゾーンをインポートする 
タイムゾーン.get_current_timezone_name() 
# 'Asia/Shanghai' を出力

環境変数 TZ=Asia/Shanghai を挿入します
コンテナに入ると時間とタイムゾーンが表示されます

システム時間はアジアタイムゾーンで表示されますが、時間は実際の現地時間ではなく、UTC 時間のままです。

Django環境に入り、時間とタイムゾーンを表示します。

python manage.py シェル 
 
datetime から datetime をインポート 
日付時刻.now() 
# 出力 datetime.datetime(2021, 10, 8, 8, 24, 8, 289230) 
 
django.utilsからタイムゾーンをインポートする 
タイムゾーン.get_current_timezone_name() 
# 'Asia/Shanghai' を出力 

ご覧のとおり、タイムゾーンは変更されていますが、コンテナ自体とDjangoの両方で時刻はUTC時間のままです。

オンラインで検索すると、Linux システムのタイムゾーンを変更するには、/etc/localtime ファイルを変更する必要があることがわかります。

Linuxコンテナのタイムゾーンを変更する

通常は、ホストの /etc/localtime ファイルをコンテナの /etc/localtime ファイルにコピーします。しかし、クエリによって、/etc/localtime ファイルは実際には単なるソフト リンクであることがわかりました。実際のファイルは次のとおりです: /usr/share/zoneinfo/Asia/Shanghai

docker cp /usr/share/zoneinfo/Asia/Shanghai テスト:/etc/localtime
TZ=Asia/Shanghai 環境変数をコンテナに挿入せずにコンテナにログインすると、コンテナのシステム時刻がローカル時刻とタイムゾーンを正しく取得していることがわかります。

TZ=Asia/Shanghai 環境変数が挿入されている場合、/etc/localtime ファイルが置き換えられても、タイムゾーンのみが変更され、時間は UTC 時間のままです。

Django環境に入り、時間を表示します

python manage.py シェル 
 
datetime から datetime をインポート 
日付時刻.now() 
# 出力 datetime.datetime(2021, 10, 8, 8, 43, 43, 754698) 

Linux のシステム時間は正常ですが、Django 環境の時間は依然として不正確で、UTC 時間のままです。この時点で、多くの人が少し頭がおかしくなり、settings.py の USE_TZ と TIME_ZONE の設定に問題があると考えるかもしれません。実際には、問題はここにはありません。その理由は、datetime ライブラリは /usr/share/zoneinfo/ ディレクトリで Asia/Shanghai ファイルを検索しますが、イメージにはこのディレクトリが含まれていないため、Django は引き続き UTC タイムゾーンを使用するためです。解決策は非常に簡単です。/usr/share/zoneinfo/Asia ディレクトリを作成し、ファイルをこのディレクトリにコピーします。

# コンテナ内(このディレクトリが存在しない場合) 
mkdir -p /usr/share/zoneinfo/アジア 
 
# コンテナ外 docker cp /usr/share/zoneinfo/Asia/Shanghai test:/usr/share/zoneinfo/Asia/Shanghai

次にコンテナにログインし、Django環境に入り時間を確認します。

python manage.py シェル 
 
datetime から datetime をインポート 
日付時刻.now() 
# 出力 datetime.datetime(2021, 10, 8, 16, 49, 32, 57) 

今回はタイミングがぴったりでした。

要約する

コンテナのタイムゾーンの問題については、コンテナ作成フェーズで/etc/localtimeをインストールして設定することをお勧めします。たとえば、Dockerfileに次のステートメントを追加します。

/usr/share/zoneinfo/Asia/Shanghai を追加します。 /usr/share/zoneinfo/Asia/Shanghai 
 
RUN ln -s /usr/share/zoneinfo/Asia/Shanghai /etc/localtime 

この方法では、コンテナを起動するときにタイムゾーンの問題を気にする必要はありません。コンテナが作成されている場合は、起動時にタイムゾーンファイルをマウントします。

docker run -d -v /etc/localtime:/etc/localtime -v /usr/share/zoneinfo/Asia/Shanghai:/usr/share/zoneinfo/Asia/Shanghai イメージ名 

この方法はもっと面倒です。私たちが現在直面しているもう 1 つの状況は、サービスがすでにオンラインになっていることです。時間的な問題があることがわかりました。2 つのファイルを手動でコンテナーにコピーし、コンテナーを再起動します。

docker cp /usr/share/zoneinfo/Asia/Shanghai テスト:/etc/localtime 
docker cp /usr/share/zoneinfo/Asia/Shanghai テスト:/usr/share/zoneinfo/Asia/Shanghai 
docker 再起動テスト 

これで、Docker コンテナに Django をデプロイする際のタイムゾーン問題についての記事は終了です。Django のタイムゾーンの Docker デプロイの詳細については、123WORDPRESS.COM の以前の記事を検索するか、以下の関連記事を引き続き参照してください。今後とも 123WORDPRESS.COM をよろしくお願いいたします。

以下もご興味があるかもしれません:
  • Docker で Django アプリケーションをデプロイする例
  • docker を使用して Django テクノロジー スタック プロジェクトをデプロイする方法
  • Docker を使用して Django プロジェクトをデプロイする方法の例
  • Django Docker コンテナのデプロイ Django-Docker ローカルデプロイ
  • Python Django アプリケーションを Docker 化する方法
  • Docker-compose を使用して Django アプリケーションをオフラインでデプロイする方法
  • Dockerを使用してDjango+MySQL8開発環境をデプロイする方法の詳細な説明

<<:  CSS 使用のヒントのまとめ

>>:  CSSはコンテンツの高さが足りない場合にフッターを自動的に下部に固定します

推薦する

GTK ツリービューの原理と使用法の分析

GtkTreeView コンポーネントは、美しい通常のリストやツリーのようなリストを作成できる高度な...

CSSで検索ボックスを非表示にする機能を実装します(アニメーション順方向と逆方向のシーケンス)

上部のメニュー バーに検索ボックスを配置するのは一般的なシナリオですが、検索機能がそれほど頻繁に使用...

Vite+ElectronでVUE3デスクトップアプリケーションを素早く構築

目次1. はじめに2. Viteプロジェクトを作成する1. viteをインストールする2. プロジェ...

Bash スクリプトを使用して Linux のメモリ使用量を監視する方法

序文Linux システムのパフォーマンスを監視するために使用できるオープンソースの監視ツールが市場に...

MySQLは複数テーブル関連統計(サブクエリ統計)の例を実装します

この記事では、例を使用して、MySQL で複数テーブルの関連統計を実装する方法について説明します。ご...

CSS3 テキストシャドウ text-shadow プロパティの詳細な説明

テキストシャドウ text-shadow プロパティの効果: 1. 右下隅の影、左下隅の影、左上隅の...

MySQL の異なるテーブル間でフィールドをコピーする

場合によっては、フィールドから別の新しいフィールドにデータの列全体をコピーする必要があります。これは...

自作の Windows サーバーに egg アプリケーションを展開する方法 (画像とテキスト付き)

1. IEブラウザを使用してVPNにログインする 2. リモートログイン 3. サーバーに最新のn...

Linux ネットワークプログラミング機能の簡単な分析

目次1.ソケットを作成する2. ソケットをバインドする3. 聞き手を作る。聞く4. 接続が受け入れら...

Windows で MySQL マスター スレーブ レプリケーションを構成する方法

MySQL マスター/スレーブ レプリケーションを使用すると、1 つのデータベース (マスター デー...

MySQL マスタースレーブレプリケーションの役割と動作原理の詳細な説明

1. マスタースレーブレプリケーションとは何ですか?マスタースレーブレプリケーションは、スレーブデー...

CSS レイアウト チュートリアル: 垂直方向の中央揃えを実現する方法

序文最近、フロントエンドの知識をまとめており、いくつかのインタビューにも参加しました。インタビュー中...

検索ナビゲーションバー付きの CSS サンプルコード

この記事では、CSS を使用して検索機能付きのナビゲーション バーを作成する方法を説明します。以下の...

MySQL テーブルとデータベースでデータを分割する方法

目次1. 縦方向のスライス1.1 垂直データベース1.2 垂直テーブル分割2. 水平(横断)セグメン...

ウェブページの読み込み速度を上げる6つのヒント

第二に、キーワードのランキングは、Webページの表示速度にも関係しています(参照:キーワードランキン...