我最近一直在使用Docker和QGIS,并按照本教程中的说明安装了一个容器。
一切都很好,尽管我无法连接到包含我所有GIS数据的本地主机postgres数据库。我想这是因为我的postgres数据库没有配置为接受远程连接,并且一直在编辑postgres conf文件以允许使用本文中的说明进行远程连接。
I'm still getting an error message when I try and connect to my database running QGIS in Docker: could not connect to server: Connection refused Is the server running on host "localhost" (::1) and accepting TCP/IP connections to port 5433?
The postgres server is running, and I've edited my pg_hba.conf file to allow connections from a range of IP addresses (172.17.0.0/32). I had previously queried the IP address of the docker container using docker ps and although the IP address changes, it has so far always been in the range 172.17.0.x
知道为什么我无法连接到这个数据库吗?我想可能是非常简单的事情!
我运行的是Ubuntu 14.04;Postgres 9.3
你可以在docker运行命令时通过——network=host来访问容器内的localhost。
Ex:
docker run --network=host docker-image-name:latest
如果你想传递环境变量与localhost使用——env-file参数来访问容器内的环境变量。
Ex:
docker run --network=host --env-file .env-file-name docker-image-name:latest
注意:在docker镜像名称之前传递参数,否则参数将无效。(我面对过这个问题,所以要小心!)
3步解决方案
1. 更新docker-compose文件
首先,由于数据库是本地的,我们需要通过向容器服务添加以下配置来将主机网络绑定到容器
services:
...
my-web-app:
...
extra_hosts:
- "host.docker.internal:host-gateway"
...
...
2. 更新/etc/postgresql/12/main/pg_hba.conf
如果这个目录下不存在这个文件,使用find / -name 'pg_hba.conf'来找到它。
在注释标签# IPv4本地连接下添加以下一行:
host all all 172.17.0.1/16 md5
3.更新/etc/postgresql/12/main/postgresql.conf
如果这个目录下不存在这个文件,使用find / -name 'postgresql.conf'找到它。
找到下面的行(这行可能有注释)
#listen_addresses = 'localhost'
并将其更改为以下一行,以便能够从localhost和172.17.0.1连接到postgres
listen_addresses = 'localhost,172.17.0.1'
您还可以将其更改为如下所示,但不建议在生产环境中这样做,因为它将向世界公开数据库(意味着任何IP地址都可以连接到数据库)
listen_addresses = '*'
最后别忘了:
使用sudo systemctl Restart postgresql重新启动postgress服务
将连接字符串host更新为host.docker.internal
让我解释一下我做了什么。
Postgresql
首先,我做了必要的配置,以确保我的Postgres数据库接受来自外部的连接。
打开pg_hba.conf,在最后添加如下一行:
host all all 0.0.0.0/0 md5
打开postgresql.conf,查找listen_addresses并像这样修改:
listen_addresses = '*'
确保上面的行没有使用#注释
->重启数据库
OBS:这不是生产环境的推荐配置
接下来,我寻找我的主机的ip。我使用的是localhosts ip 127.0.0.1,但是容器看不到它,所以在运行容器时出现了有问题的Connection Refused消息。在网上搜索了很长一段时间后,我读到容器看到了来自本地网络的内部ip(你的路由器将它属性为连接到它的每个设备,我不是在谈论让你访问互联网的ip)。也就是说,我打开了一个终端,做了以下事情:
查找本地网络ip
打开终端或CMD
(MacOS / Linux)
$ ifconfig
(窗口)
$ ipconfig
这个命令将显示您的网络配置信息。
看起来是这样的:
en4:
ether d0:37:45:da:1b:6e
inet6 fe80::188d:ddbe:9796:8411%en4 prefixlen 64 secured scopeid 0x7
inet 192.168.0.103 netmask 0xffffff00 broadcast 192.168.0.255
nd6 options=201<PERFORMNUD,DAD>
media: autoselect (100baseTX <full-duplex>)
status: active
寻找一个活跃的。
在我的例子中,我的本地网络ip是192.168.0.103
完成这些操作后,我运行容器。
码头工人
使用——add-host参数运行容器,如下所示:
$ docker run --add-host=aNameForYourDataBaseHost:yourLocalNetWorkIp --name containerName -di -p HostsportToBind:containerPort imageNameOrId
就我而言,我做到了:
$ docker run --add-host=db:192.168.0.103 --name myCon -di -p 8000:8000 myImage
我使用的是Django,所以默认端口是8000。
Django应用程序
访问数据库的配置如下:
在settings.py
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.postgresql',
'NAME': ‘myDataBaseName',
'USER': ‘username',
'PASSWORD': '123456',
'HOST': '192.168.0.103',
'PORT': 5432,
}
}
参考文献
关于-p标志:
使用网口映射进行连接
关于docker run:
Docker运行文档
有趣的文章:
Docker提示#35:连接到运行在Docker主机上的数据库