解決 WebAPI 在容器中的啟動(dòng)初始化問(wèn)題
在產(chǎn)品或項(xiàng)目的部署中,如果和下面場(chǎng)景類(lèi)似,那么本文可能對(duì)您有所幫助。
場(chǎng)景
- WebAPI 和 MySql 數(shù)據(jù)部署在同一服務(wù)器(通常是測(cè)試環(huán)境)。
- WebAPI 和 MySql 使用 docker-compose 進(jìn)行部署。
- WebAPI 啟動(dòng)時(shí)有一些初始化的操作要做,而初始化需要從 MySql 中獲取數(shù)據(jù)。
問(wèn)題
- 第一次部署,執(zhí)行 docker-compose up -d 后,WebAPI 不能正常啟動(dòng)。
- 斷電、或其他原因?qū)е碌姆?wù)器重啟或 docker 重啟,WebAPI 不能正常啟動(dòng)。
原因
- docker 容器啟動(dòng)時(shí),WebAPI 程序啟動(dòng)的速度比 MySql 快,導(dǎo)致程序去連接 MySql 時(shí),MySql 服務(wù)器還沒(méi)有啟動(dòng)完成,自然是連不上。
假象
在 docker-compose.yml 文件中可以添加 depends_on 來(lái)設(shè)置依賴(lài),如下:
api:
restart: always
image: netapi
ports:
- "5000:5000"
environment:
- TZ=Asia/Shanghai
depends_on:
- mysql
networks:
s2_net:
ipv4_address: 172.66.9.5
在 api 的 depends_on 設(shè)置 mysql ,表示 api 依賴(lài) mysql ,只有當(dāng) mysql 啟動(dòng)后,api 才會(huì)啟動(dòng)。
但很可惜,這里的 mysql 啟動(dòng)指的是 mysql 的容器是否啟動(dòng)了,而不是 mysql 的服務(wù)是否啟動(dòng)。所以,這種配置只能控制容器的啟動(dòng)順序,并不能解決問(wèn)題。
解決
要解決這個(gè)問(wèn)題,有兩種方式:
- 在 WebAPI 項(xiàng)目中使用 Polly 庫(kù),它是一個(gè) .NET 的彈性和瞬態(tài)故障處理庫(kù),可以實(shí)現(xiàn)重試、斷路器、超時(shí)等策略來(lái)處理網(wǎng)絡(luò)請(qǐng)求失敗的情況??梢允褂?Polly 來(lái)嘗試連接 mysql 服務(wù),并在失敗時(shí)進(jìn)行重試或等待。
- 優(yōu)化 depends_on 配置。
本文著重介紹的是第二種方式,進(jìn)行 depends_on 配置的優(yōu)化。
優(yōu)化思路
- mysql 服務(wù)添加 healthcheck 檢查,用來(lái)判斷 mysql 的服務(wù)是否正常啟動(dòng)。
- api 服務(wù)的 depends_on 監(jiān)聽(tīng)這個(gè)檢查,只有當(dāng) mysql 服務(wù)正常啟動(dòng)后,api 才會(huì)啟動(dòng)。
完整 docker-compose.yml
version: "3"
networks:
s2_net:
driver: bridge
ipam:
driver: default
config:
- subnet: 172.66.9.0/24
services:
mysql:
restart: always
image: mysql/mysql-server:latest
ports:
- "3306:3306"
environment:
- TZ=Asia/Shanghai
- MYSQL_ROOT_PASSWORD=123456
healthcheck:
test: ["CMD", "mysqladmin" ,"ping", "-h", "localhost", "-u", "root", "--password=123456"]
interval: 3s
timeout: 5s
retries: 3
start_period: 5s
command: mysqld --character-set-server=utf8mb4 --collation-server=utf8mb4_general_ci --default-authentication-plugin=mysql_native_password
networks:
s2_net:
ipv4_address: 172.66.9.2
api:
restart: always
image: netapi
ports:
- "5000:5000"
environment:
- TZ=Asia/Shanghai
depends_on:
mysql:
condition: service_healthy
networks:
s2_net:
ipv4_address: 172.66.9.5
mysql 服務(wù)中添加 healthcheck 屬性,子屬性解釋如下:
- test:設(shè)置健康檢查的命令。
- interval:定義健康檢查的間隔時(shí)間,上面配置為間隔 3 秒。
- timeout:健康檢查的超時(shí)時(shí)間。
- retries:定義了健康檢查失敗后的重試次數(shù)。
- start_period:默認(rèn)值為 0 秒,表示容器啟動(dòng)后立即進(jìn)行健康檢查。如果將 start_period 設(shè)置為非零值,則 Docker 會(huì)在容器啟動(dòng)后先等待一段時(shí)間,然后再開(kāi)始進(jìn)行健康檢查。
api 服務(wù)的配置為固定寫(xiě)法。
注意事項(xiàng)
如果您的 docker-compose 安裝的是 1.27 以下的版本,需要升級(jí)到 1.27 或以上版本。
因?yàn)?docker-compose 3 不支持 depends_on 的條件設(shè)置, 但從 1.27.0 開(kāi)始,2.x 和 3.x 與 COMPOSE_SPEC 架構(gòu)合并,版本現(xiàn)在是兼容的。
可以使用下面命令進(jìn)行 docker-compose 版本的查看:
docker-compose -v
安裝 docker-compose 可以使用下面命令:
curl -L https://github.com/docker/compose/releases/download/1.28.0/docker-compose-`uname -s`-`uname -m` -o /usr/local/bin/docker-compose
sudo chmod +x /usr/local/bin/docker-compose
升級(jí)到 1.28.0 后,執(zhí)行 docker-compose 的命令時(shí)可能會(huì)出現(xiàn)錯(cuò)誤,錯(cuò)誤提示如下:
[29250] Error loading Python lib '/tmp/_MEIYmY20a/libpython3.9.so.1.0': dlopen: /lib64/libc.so.6: version `GLIBC_2.28' not found (required by /tmp/_MEIYmY20a/libpython3.9.so.1.0)
按照提示 google 下,會(huì)有很多方式解決,或者直接參考這個(gè)鏈接:https://blog.csdn.net/wangying202/article/details/113178159。
總結(jié)
在 docker-compose 中進(jìn)行設(shè)置是一種偷懶的做法,適用于測(cè)試環(huán)境,因?yàn)樯a(chǎn)環(huán)境程序和數(shù)據(jù)庫(kù)通常在不同的服務(wù)器。
最好的方式還是應(yīng)該在 WebAPI 程序中進(jìn)行處理。