概述
这两天在写 go 项目, 一个 HTTP 服务器. 之前写的是 php 项目, nginx 监听80端口, 根据域名将请求分配给不同项目. 现在换了 go, 自然也想延续这个操作, 毕竟都是跑在同一台服务器上. 那么问题来了, 我的nginx 监听80端口的同时, go 服务器是无法同样监听80端口的. 这该如何是好啊, 给我整的一脸懵逼.
nginx 转发 go 服务
不过想了一下, 发现自己进入了思维误区. 为什么都要监听80端口呢? go 服务监听其他端口, 然后 nginx 将请求进行端口的转发不就可以了么. 真是个傻子. 搜了一下 nginx 的配置, 确实有端口转发的配置. 如下:
#配置负载均衡池
upstream test_go_pool{
server 127.0.0.1:8090;
}
#Demo2端口转发
server {
listen 80;
server_name test.go.com;
error_log /var/log/nginx/test_go.error.log;
access_log /var/log/nginx/test_go.access.log;
#将所有请求转发给均衡池的应用处理
location / {
proxy_set_header Host $host;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_pass http://test_go_pool;
}
}
到这里, 如果 go 服务已经启动了, 应该就没有问题了. (如果是线上, go 服务的端口不用对外开放, 仅供内部 nginx 访问即可)
docker 访问宿主网络
但是我的环境访问失败了. 究其原因, 我的 nginx 是跑在 docker 中的, go 服务是跑在宿主主机上的. 所以在对127.0.0.1
回环地址访问的时候, 没有访问的宿主主机. 解决的方向也很清晰, 只要能够访问到宿主主机的网络就行了.
方案一
最先想到的方案, 就是将127.0.0.1更换成宿主主机的 IP. mac 下执行 ifconfig
, 可以找到本机的 IP 地址. 将地址更换之后, 就可以了. 但是如果是移动网络等, IP 经常换的话, 就很难受了.
方案二
更改网络链接方式. 虚拟机网络的链接方式有: bridge(桥接网络), host(共用网络), 以及无网络.
docker 默认使用的是桥接的形式, 如果换成host
, 就没有访问不到的问题了.更改之后, 可以直接通过127.0.0.1
回环地址访问, 就没有 IP 更换的问题了. 但同时的, host 方式会将虚拟机的所有端口全都与主机共用, 隔离性太差.
方案三
经过一番摸索, 找到了宿主主机的名字: docker.for.mac.host.internal
. 将地址更换成这个, docker 会自动映射到宿主主机上.
至此, 我的问题已经解决了. 不过就在我准备收手的时候, 又让我找到了另外一个.
方案四
方案三的方式, 看名字也知道, 这个映射是针对 mac 的. 如果不是在 mac 上装的 docker 怎么办呢?
host.docker.internal
这个DNS 名称, 可以直接映射到宿主主机. 完美解决. 同时: gateway.docker.internal
是网关的名称.
方案五
上方的方法, 最终发现在Linux
下不好使. 这时, 可以就可以祭出这个方法了. 通过修改hosts文件的方式.
docker-compose方式
# image 统计
extra_hosts:
# 前面的 host.docker.internal 可修改为任何你想要的域名
- "host.docker.internal:host-gateway"
docker命令行方式
--add-host= "host.docker.internal:host-gateway"
如此处理之后, 你就会发现, 在容器的/etc/hosts
文件中多了这样的内容:
172.17.0.1 host.docker.internal
结束…. 简单记录以下.