docker容器中無法獲取宿主機hostname的解決方案
在nodejs環(huán)境中測試通過,其它語言同理,只需要使用獲取環(huán)境變量的方法即可。
思路:
docker容器和宿主機環(huán)境是隔離的,但是可以在啟動docker容器時將宿主機的主機名以環(huán)境變量的形式傳入,代碼在容器中獲取該值即可。
操作:
docker run -d -p 3000:3000 --name myTest -e HOST_Q=$(hostname) mytest:v1 # 使用-e 參數(shù)傳入環(huán)境變量,值為主機名
如果使用yml文件啟動:
version: '3' services: mysql: image: mysql:v1 container_name: xx-mysql restart: always networks: - host environment: - MYSQL_ROOT_PASSWORD=xxx0209 - HOST_Q=$(hostname) # 在這設置 ports: - 3306:3306 volumes: - /opt/data/mysql:/var/lib/mysql:z
啟動成功后,容器內(nèi)部環(huán)境變量就多了一個HOST_Q,接下來使用程序取出即可:
nodejs:
# 從process中取出環(huán)境變量對象
let env = process.env;
console.log(JSON.stringify(env));
# env['HOST_Q']就是最終要獲取的主機名
# output
[2019-04-17T06:54:12.951Z] [e1e7115e0a33] [info]: {"NODE_VERSION":"8.9.4","HOSTNAME":"e1e7115e0a33","YARN_VERSION":"1.3.2","HOME":"/root","HOST_Q":"emg-ubuntu-pub02","PATH":"/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin","PWD":"/"}
java:
public class Test {
public static void main(String[] args) {
Map<String, String> map = System.getenv();
String hostName = map.get("HOST_Q");
System.out.println(hostName);
}
}
補充:docker容器無法訪問宿主機報出 No route to host
一. 問題描述
在docker部署nacos的時候遇到了這個樣子的問題No route to host 導致了nacos容器無法連接宿主機的docker數(shù)據(jù)庫。
然后我就進入到了nacos容器里面,ping了宿主機的地址,結(jié)果是通著的,然后使用telnet測試了3306端口,結(jié)果也會報出這個異常。
原因是什么呢?明明數(shù)據(jù)庫外部可以正常連接訪問,但是宿主機內(nèi)部容器確實無法訪問?
二. 原因分析
在進行docker部署的時候我們采用的是bridge網(wǎng)橋的模式。
啟動docker時,docker進程會創(chuàng)建一個名為docker0的虛擬網(wǎng)橋,用于宿主機與容器之間的通信。當啟動一個docker容器時,docker容器將會附加到虛擬網(wǎng)橋上,容器內(nèi)的報文通過docker0向外轉(zhuǎn)發(fā)。
如果docker容器訪問宿主機,那么docker0網(wǎng)橋?qū)笪闹苯愚D(zhuǎn)發(fā)到本機,報文的源地址是docker0網(wǎng)段的地址。而如果docker容器訪問宿主機以外的機器,docker的SNAT網(wǎng)橋會將報文的源地址轉(zhuǎn)換為宿主機的地址,通過宿主機的網(wǎng)卡向外發(fā)送。
因此,當docker容器訪問宿主機時,如果宿主機服務端口會被防火墻攔截,從而無法連通宿主機,出現(xiàn)No route to host的錯誤。
而訪問宿主機所在局域網(wǎng)內(nèi)的其他機器,由于報文的源地址是宿主機ip,因此,不會被目的機器防火墻攔截,所以可以訪問。
三. 解決方案
1> 關閉宿主機的防火墻
systemctl stop firewalld
2> 在防火墻上開發(fā)指定的端口
firewall-cmd --zone=public --add-port=3306/tcp --permanent firewall-cmd --zone=public --add-port=3307/tcp --permanent firewall-cmd --reload
注:在進行完防火墻的操作之后最好是要進行以下docker的重啟,systemctl restart docker,否則容器到因為虛擬網(wǎng)橋失效而導致的iptables failed問題
四. 小結(jié)
docker的容器網(wǎng)絡連接一直是一個問題,容器與容器之間,容器與宿主機之間,容器跨主機訪問,所以在涉及到容器的網(wǎng)絡連接的時候要注意網(wǎng)絡的問題。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持本站。如有錯誤或未考慮完全的地方,望不吝賜教。
版權(quán)聲明:本站文章來源標注為YINGSOO的內(nèi)容版權(quán)均為本站所有,歡迎引用、轉(zhuǎn)載,請保持原文完整并注明來源及原文鏈接。禁止復制或仿造本網(wǎng)站,禁止在非maisonbaluchon.cn所屬的服務器上建立鏡像,否則將依法追究法律責任。本站部分內(nèi)容來源于網(wǎng)友推薦、互聯(lián)網(wǎng)收集整理而來,僅供學習參考,不代表本站立場,如有內(nèi)容涉嫌侵權(quán),請聯(lián)系alex-e#qq.com處理。
關注官方微信