docker-compose有两个容器:Web无法连接到数据库

时间:2019-01-24 14:32:54

标签: django postgresql docker docker-compose containers

docker-compose无法构建Web组件,因为它无法连接到先前创建的数据库组件

Mac OSX 10.13.6,conda 4.5.11,Python 3.6.8,Docker版本18.09.1,docker-compose版本1.23.2

django 1.8.3与Dockerfile中的requirements.txt一起安装。不能随意升级。

关于SO的几次非常相似的讨论都没有帮助(例如Docker-compose with django could not translate host name "db" to address: Name or service not known)。

我有一个拥有网络和两个组件的docker-compose.yml

version: '3'
networks:
  bridge:
   driver: bridge
services:
  db:
    image: postgres:10
    container_name: myapp-db
    volumes:
      - ./postgres_data:/var/lib/postgresql/data/
    ports:
      - "5432:5432"
    environment:
     POSTGRES_DB: actionability-master
     POSTGRES_PASSWORD: postgres
     POSTGRES_USER: postgres
    networks:
      - bridge


  web:
    restart: unless-stopped
    container_name: myapp-web
    build: .
    command: /start_gunicorn.sh
    ports:
      - "8080:8080"
    environment:
      PRODUCTION: 'true'
    networks:
      - bridge

在我的settings.py中,有“数据库”部分:

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.postgresql_psycopg2',
        'NAME': INSTANCE_NAME,
        'USER': 'postgres',
        'PASSWORD': 'postgres',
        'HOST': 'db',
        'PORT': '5432'
    },

}

当我运行$ docker-compose up -d时, 创建第一个映像(db),并启动其容器。 可以看到它正在运行,并使用docker pslsof在端口5432上监听。如果我从web:文件中删除docker-compose.yml组件,也会发生同样的事情

现在,第二个组件(网络)具有Dockerfile,其中包含以下两行(还有许多其他行):

RUN python manage.py makemigrations myapp
RUN python manage.py migrate

像这样的“迁移者”死于此错误:

Traceback (most recent call last):
File "/usr/local/lib/python3.6/site-packages/django/db/backends/base/base.py", line 130, in ensure_connection
    self.connect()
File "/usr/local/lib/python3.6/site-packages/django/db/backends/base/base.py", line 119, in connect
    self.connection = self.get_new_connection(conn_params)
File "/usr/local/lib/python3.6/site-packages/django/db/backends/postgresql_psycopg2/base.py", line 176, in get_new_connection
    connection = Database.connect(**conn_params)
File "/usr/local/lib/python3.6/site-packages/psycopg2/__init__.py", line 130, in connect
    conn = _connect(dsn, connection_factory=connection_factory, **kwasync)
psycopg2.OperationalError: could not connect to server: Connection refused
Is the server running on host "db" (37.34.32.51) and accepting TCP/IP connections on port 5432?

我累了几次调整。  *更改了version '3.7'  *添加到数据库部分

expose:
    - "5432"
  • 添加到Web组件:

depends_on: - "db"

  • 添加到Web组件:

links: - "db"

  • 将PRODUCTION设置为false

environment: PRODUCTION: 'false'

  • settings.py中的HOST更改为容器名称,图像名称,标签,容器ID,“本地主机”,“ 127.0.0.1”等。错误是相同的,只是提到新的主机名,而不是'db'
  • 在conda env之外
  • 使用--build开关(docker-compose up -d --build
  • 完成docker system prune并再次运行

所有相同的错误。


更新: 在建议它docker-compose无法以这种方式工作之后,我尝试将其拆分为两个单独的任务。首先,我建立myapp-db容器并确保它在正确的端口上运行:

$ docker container ps


CONTAINER ID        IMAGE               COMMAND                  CREATED             STATUS              PORTS                      NAMES
c110e8361cda        postgres:10         "docker-entrypoint.s…"   4 hours ago         Up 4 hours          0.0.0.0:5432->5432/tcp     myapp-db

然后我构建myapp-web:

docker build -t myapp-web .

同样的错误仍然发生。那么,为什么现在不找到数据库容器呢?

3 个答案:

答案 0 :(得分:1)

我使用depends_on列表在Web容器之前启动db容器,并使用links来确保主机名可以解析。

我将以下内容添加到网络上:

services:
  db:
    # ...
  web:
    links:
    - "db:db" # resolve the hostname "db" with the ip of the db container
    depends_on:
    - db # start db before web

Example

答案 1 :(得分:0)

我知道这看起来确实很基础,但是为什么不考虑一个稍微复杂一些的命名约定,然后将那些名称简单地添加到主机上的DNS服务或至少A记录中呢?

然后,系统可以通过名称或IP到达数据库。只是一个想法。

很棒的选择以及平台和语言的结合,顺便说一句,Django + PostgreSQL是我的最爱。我还建议使用Bootstrap,尤其是在您需要快速工具部署功能的情况下。

答案 2 :(得分:0)

您可以尝试使用网络别名进行此配置。

version: '3.5'
services:

  db:
    image: postgres:10
    container_name: myapp-db
    volumes:
      - ./postgres_data:/var/lib/postgresql/data/
    expose:
      - "5432"
    environment:
       POSTGRES_DB: actionability-master
       POSTGRES_PASSWORD: postgres
       POSTGRES_USER: postgres
    networks:
      services-network:
        aliases:
         - db

  web:
    restart: unless-stopped
    container_name: myapp-web
    build: .
    command: /start_gunicorn.sh
    ports:
      - "8080:8080"
    environment:
      PRODUCTION: 'true'
    depends_on:
      - db
    networks:
      services-network:
        aliases:
         - web

networks:
   services-network:
     name: services-network
     driver: bridge