Django_debug_toolbar и Docker

Итак, я получил docker и Django для работы локально, сначала создав изображение из файла Docker, затем используя Fig, чтобы получить изображение postgres, связать его с базовым изображением и запустить локальный сервер. Все работает отлично, за исключением django_debug_toolbar. По какой-то причине он просто не появится. Имейте ip-адрес dockerhost, а также в inner_ips. Может ли кто-нибудь помочь мне с этим? Docker работает на mac через boot2docker.

Спасибо!

Мои настройки:

INIT.py

import os

if 'DEBUG' not in os.environ or not os.environ['DEBUG']:
    from .local import *
else:
    pass

base.py

""" common and global settings """

from sys import path
from os import environ
from os.path import abspath, basename, dirname, join, normpath
from django.core.exceptions import ImproperlyConfigured
import dj_database_url


def get_env_variable(var_name):
    try:
        return environ[var_name]
    except KeyError:
        error_msg = "Set the environment variable" % var_name
        raise ImproperlyConfigured(error_msg)

# Paths
DJANGO_ROOT = dirname(dirname(abspath(__file__)))
SITE_ROOT = dirname(DJANGO_ROOT)
SITE_NAME = basename(DJANGO_ROOT)
# End Paths


# URLs
MEDIA_ROOT = normpath(join(SITE_ROOT, 'media'))
MEDIA_URL = "/media/"

STATIC_ROOT = normpath(join(SITE_ROOT, 'assets'))
STATIC_URL = "/static/"

STATICFILES_DIRS = (
    normpath(join(SITE_ROOT, 'static')),
)

STATICFILES_FINDERS = (
    'django.contrib.staticfiles.finders.FileSystemFinder',
    'django.contrib.staticfiles.finders.AppDirectoriesFinder',
)

ROOT_URLCONF = '%s.urls' % SITE_NAME

path.insert(0, join(DJANGO_ROOT, 'apps'))  # add apps folder to system path
# End URLs


# Database
# example:  postgres://joris:[email protected]/bitbybit
DATABASES = {'default': dj_database_url.config(
   default='postgres://[email protected]:5432/postgres')}
# End Database

# Templates
TEMPLATE_CONTEXT_PROCESSORS = (
    'django.contrib.auth.context_processors.auth',
    'django.core.context_processors.debug',
    'django.core.context_processors.i18n',
    'django.core.context_processors.media',
    'django.core.context_processors.static',
    'django.core.context_processors.tz',
    'django.contrib.messages.context_processors.messages',
    'django.core.context_processors.request',
)

TEMPLATE_LOADERS = (
   'django.template.loaders.filesystem.Loader',
   'django.template.loaders.app_directories.Loader',
)

TEMPLATE_DIRS = (
    normpath(join(SITE_ROOT, 'templates')),
)
# End Templates


# SECURITY WARNING: keep the secret key used in production secret!
# make it unique and store it as an environment variable
SECRET_KEY = r"d%g7_h6cz=xbhs*5-i+e$c7mns*s)^_+#^[email protected]^[email protected]"


# Application
DJANGO_APPS = (
   'django.contrib.admin',
   'django.contrib.auth',
   'django.contrib.contenttypes',
   'django.contrib.sessions',
   'django.contrib.messages',
   'django.contrib.staticfiles',
)

LOCAL_APPS = (
   'home',
)

INSTALLED_APPS = DJANGO_APPS + LOCAL_APPS

MIDDLEWARE_CLASSES = (
    'django.contrib.sessions.middleware.SessionMiddleware',
    'django.middleware.common.CommonMiddleware',
    'django.middleware.csrf.CsrfViewMiddleware',
    'django.contrib.auth.middleware.AuthenticationMiddleware',
    'django.contrib.auth.middleware.SessionAuthenticationMiddleware',
    'django.contrib.messages.middleware.MessageMiddleware',
    'django.middleware.clickjacking.XFrameOptionsMiddleware',
    'debug_toolbar.middleware.DebugToolbarMiddleware',
)

WSGI_APPLICATION = '%s.wsgi.application' % SITE_NAME
# End Application


# Internationalization
LANGAUAGE_CODE = 'en-us'

TIME_ZONE = 'UTC'

USE_I18N = True

USE_L10N = True

USE_TZ = True
# End Internationalization

Local.py

from .base import *

# Debug config
DEBUG = True
TEMPLATE_DEBUG = DEBUG
# End Debug config

# Hosts
ALLOWED_HOSTS = ['127.0.0.1', 'localhost']
# End Hosts


# Django Debug Toolbar config
INSTALLED_APPS += (
    'debug_toolbar', )

INTERNAL_IPS = ('127.0.0.1', 'localhost')
# End Django Debug Toolbar config

Ответ 1

Использование конфигурации SHOW_TOOLBAR_CALLBACK разбудило меня

def show_toolbar(request):
        return True

DEBUG_TOOLBAR_CONFIG = {
    'SHOW_TOOLBAR_CALLBACK': show_toolbar,
}

Я надеюсь, что это помогло :)

Ответ 2

Вы можете просто сделать INTERNAL_IPS объектом, который содержит все. Это то, что я делаю:

if DEBUG:
    # 'debug' is only True in templates if the vistor IP is in INTERNAL_IPS.
    INTERNAL_IPS = type(str('c'), (), {'__contains__': lambda *a: True})()

Конечно, вы не должны никогда делать это на производственном хосте!

Объяснение:

Функция типа (вариант с тремя аргументами: https://docs.python.org/3/library/functions.html#type) создает новый класс, который в этом случае имеет только метод __contains__ (https://docs.python.org/3/reference/datamodel.html#object.содержит) - содержит, используется для реализации тестов членства, что означает, что этот метод вызывается при беге, например "if ip in INTERNAL_IPS". Сам метод содержит, вероятно, будет более понятным, если он будет записан как "def __contains__(self):\n return True". Вновь созданный класс сразу создается (последний "()") и присваивается INTERNAL_IPS

Ответ 3

Если вы хотите сделать это программно, а не копировать/вставлять IP-адрес своего контейнера, я бы посоветовал вам сделать это как django-cookiecutter. В вашем локальном файле настроек:

INTERNAL_IPS = ['127.0.0.1', ]
import socket

# tricks to have debug toolbar when developing with docker
ip = socket.gethostbyname(socket.gethostname())
INTERNAL_IPS += [ip[:-1] + '1']

Для справки это ссылка на файл настроек django-cookiecutter local.py.

Ответ 4

IP-адрес, который позволял мне отображать Django Debug Toolbar, был IP-адрес шлюза, связанного с моим док-контейнером. Чтобы получить IP шлюза, я запускаю эту команду

docker inspect my_container_name | grep -e '"Gateway"'
# "Gateway": "172.18.0.1",

Все мои настройки выглядят так

INSTALLED_APPS = (
    'debug_toolbar',
)
INTERNAL_IPS = ['172.18.0.1']

Ответ 5

решаемая. Проверьте значение REMOTE_ADDR в заголовках запроса и добавьте его в INTERNAL_IPS.

Ответ 6

Используя принятый ответ на https://unix.stackexchange.com/questions/87468/is-there-an-easy-way-to-programmatically-extract-ip-address, я смог заставить это работать, передав адрес моста докере хоста команде docker run в качестве переменная среды:

-e "DOCKER_HOST=$(ip -4 addr show docker0 | grep -Po 'inet \K[\d.]+')"

С этим набором следующие две строки в settings.py обнаружили его и позволили хосту увидеть панель инструментов:

if "DOCKER_HOST" in os.environ:
    INTERNAL_IPS = [os.environ["DOCKER_HOST"]]

Ответ 7

Вот хакерский способ, который не требует ввода адреса в контейнер. Я не уверен, как это будет работать, если ваш контейнер подключен к нескольким сетям и может сломаться, если докер изменит способ назначения адресов (хотя это не кажется вероятным).

import subprocess

route = subprocess.Popen(('ip', 'route'), stdout=subprocess.PIPE)
network = subprocess.check_output(
    ('grep', '-Po', 'src \K[\d.]+\.'), stdin=route.stdout).decode().rstrip()
route.wait()
network_gateway = network + '1'
INTERNAL_IPS = [network_gateway]