Reputation: 151
I am facing problem using docker-compose to link a django container with postgres and mongo containers? I am trying to use "docker-compose up" which starts up the mongo and postgres containers (as I need to link both) but still the django app is not able to connect to mongodb on default settings. My django-compose.yml file contents are copied below:
db1:
image: postgres
db2:
image: mongo
ports:
- "27017:27017"
web:
build: .
command: python manage.py runserver 0.0.0.0:8000
volumes:
- .:/code
ports:
- "8000:8000"
links:
- db1
- db2
It does connect with postgres with default settings. I can also telnet to the mongodb port locally. Still, I get this error on starting the web container:
File "/usr/local/lib/python2.7/site-packages/mongoengine/connection.py", line 124, in get_connection web_1 | raise ConnectionError("Cannot connect to database %s :\n%s" % (alias, e)) web_1 | mongoengine.connection.ConnectionError: Cannot connect to database default : web_1 | [Errno 111] Connection refused
PS: I had successfully started a django-postgres connected app on my localhost, but it failed connecting to db, on an AWS instance. That is another problem I still need to get to root of.
Upvotes: 13
Views: 11796
Reputation: 31
I managed to make this connection with the following configuration:
docker-compose.yml
version: "3.9"
services:
mongodb:
environment:
MONGO_INITDB_DATABASE: mydatabase
MONGO_INITDB_ROOT_USERNAME: root
MONGO_INITDB_ROOT_PASSWORD: password
hostname: mongodb
image: mongo
ports:
- "27017:27017"
volumes:
- ./data-mongodb:/data/db
api:
build: .
command: "python manage.py runserver 0.0.0.0:8000"
volumes:
- .:/code
ports:
- "8000:8000"
depends_on:
- mongodb
Dockerfile
# syntax=docker/dockerfile:1
FROM python:3
ENV PYTHONUNBUFFERED=1
WORKDIR /code
COPY requirements.txt /code/
RUN pip install --upgrade pip
RUN pip install -r requirements.txt
COPY . /code/
requirements.txt
asgiref==3.4.1
Django==3.2.7
django-annoying==0.10.6
djangorestframework==3.12.4
djongo==1.3.6
psycopg2==2.9.1
psycopg2-binary==2.9.1
pymongo==3.12.1
pytz==2021.3
six==1.16.0
sqlparse==0.2.4
settings.py
DATABASES = {
'default': {
'ENGINE': 'djongo',
'NAME': 'mydatabase',
'CLIENT': {
'host': 'mongodb://mongodb:27017',
'username': 'root',
'password': 'password',
'authSource': 'admin',
'authMechanism': 'SCRAM-SHA-1',
}
}
}
In this, the model used is from Djongo instead of from django.db import models
models.py
from djongo import models
After the build, you need to enter the api container and run the ./manage.py migrate
Upvotes: 3
Reputation: 61
I was able to containerize Django and MongoDB, connect both containers. I used Dockerfiles to build both containers and docker run to start the containers and have them connected. Just follow the steps in this repo. It was necessary for me to use Dockerfiles to have more power over the installed versions of the needed libraries because the latest versions of Django and mongoengine are not compatible. The stable working versions are
Django==1.10.0
pymongo==2.7.1
six==1.10.0
mongoengine==0.9.0
Upvotes: 1
Reputation: 1861
I ran into a similar problem but with another service (not MongoDB). I'm not sure of what I'm doing wrong but this is how I could solve it :
import os
import mongoengine
MONGODB_HOST = os.environ.get('DB2_PORT_27017_TCP_ADDR', '127.0.0.1')
mongoengine.connect(host=MONGODB_HOST)
DB2
being the name of your service in docker-compose.yml27017
being the port of the exposed service. UPDATE
Now docker-compose containers are reachable by other services using a hostname similar to their alias. link documentation :
Containers for the linked service will be reachable at a hostname identical to the alias, or the service name if no alias was specified.
And that way you can connect to MongoDB like this:
import mongoengine
mongoengine.connect(host="db2")
Upvotes: 8
Reputation: 2593
You should specify host name like in the docker compose file, instead of IP address.
I've faced similar problems in connecting from a Tornado Web app to Mongo DB. Here is my docker-compose.yml:
web:
build: .
ports:
- "8888:8888"
volumes:
- .:/code
links:
- db
db:
image: mongo:3.0
Here is my connection string:
motorengine.connect("db", host='db', port=27017, io_loop=io_loop)
My error was to specify IP address instead of host name (db) like in the docker compose file.
Upvotes: 3
Reputation: 4924
The problem you face is that the application container and the DB container start independently from each other and most importantly, the app container won't wait until the DB container is started. There is as of docker compose 1.1.0 no feature which would allow to consider such dependencies.
This is a well known problem and has been discussed already. Consensus seems to be that this should not be solved on a docker compose level but in docker itself. There is a proposal already for the groundwork in docker.
In my case, I just built this kind of intelligence in the application itself. I check for port connectivity until successful and then start the rest of the application.
Upvotes: 1