聊一聊 Docker 與時區

前言

當我們把應用部署到容器裏面之後,基本都會要和時間 / 時區打交道!!

大部分的應用,多多少少都會有獲取當前時間的操作,試想一下應用拿到的時間不對,那麼業務極有可能會亂套,造成嚴重的損失。

時間和時區是渾然一體的,我們在東八區,看到的是 8 點鐘,同一時間點,別人在另外的時區看到的可能是 10 點鐘。

所以我們要對這個問題有個簡單的認知,這樣才能避免在實際使用的時候踩坑。

下面來簡單看看 Docker 和時區的一些內容吧。

常用鏡像

比較常用的基礎鏡像應該是 Alpine,Ubuntu,Debian 和 CentOS 這 4 個了。

看看分別啓動對應的容器,執行 date 命令,看看輸出的時間。

從上面的圖可以看出,這些鏡像基本上都採用 UTC 時間,默認時區爲零時區。

圖片中,這些容器的時間都是 5 點多, 老黃本地的電腦則輸出的是 13 點。

在國內的大多數應用,可能用的時間會是 13 點的比較多。

知道了這個背景,那麼要怎麼對容器的時區進行調整的。

時區調整

回想一下,.NET Core 的應用,在部署的時候,一般會在啓動的時候指定環境變量 -e TZ=Asia/Shanghai,或者是在 Dockerfile 裏面直接加一個 ENV TZ=Asia/Shanghai 來指定當前用的時區。

下面先來嘗試一下這種方式行不行。

可以看到, debian 和 centos 在指定這個環境變量後是可以正常獲取到正確的時間的,但是 alpine 和 ubuntu 卻不行。

影響這個時區的,其實是一個叫 tzdata 的東西,可以理解成是一個時區的數據庫。

可以參考下面的資料:

https://en.wikipedia.org/wiki/Tz_database

之所以上面 debian 和 centos 可以通過指定環境變量來切換時區,是因爲這兩個鏡像裏面已經安裝了 tzdata 。

可以查看一下 /usr/share/timezone 目錄。

那麼,alpine 和 ubuntu 這兩個鏡像我們要怎麼操作纔可以適配。

下面以比較常用的 alpine 爲例,重新構建一個鏡像。

加時區的可以參考下面這個 wiki 地址

https://wiki.alpinelinux.org/wiki/Setting_the_timezone

先寫一個 Dockerfile

1FROM alpine:3.13
2
3RUN sed -i 's/dl-cdn.alpinelinux.org/mirrors.aliyun.com/g' /etc/apk/repositories \
4    && apk add tzdata
5
6

再來構建

1docker build -t alpine:3.13-tz -f Dockerfile.alpine .
2
3

下面來看看通過運行時指定環境變量的方式來獲取當前時間是否是正確的。

從上面的圖可以看到,時間是正確的了,同樣也可以看到一堆時區信息了。

再來看看,加了 tzdata 後,鏡像的大小如何。

相比原始的多了 2M,還可以接受。

寫在最後

時區這個問題,其實比較簡單,只是要特別注意。

如果時間存儲用的是時間戳,展示的時候,再根據參數轉成對應的本地時間,也可以一定程度的避免踩到時區這個坑。

本文由 Readfog 進行 AMP 轉碼,版權歸原作者所有。
來源https://mp.weixin.qq.com/s/uO0nqNnWMqeQU8EyQYhllQ