Рефакторинг инфраструктуры и конфигурации проекта
- Перенесена структура Django-конфига в src/core и src/settings - Унифицирована Docker-сборка и docker-compose для dev/prod - Добавлены startup-checks (DB/Redis) и обновлены env-шаблоны - Расширена OpenAPI-документация и ответы API - Удалены устаревшие deploy/requirements/служебные скрипты - Обновлены CI/CD, README и тесты
This commit is contained in:
9
src/settings/__init__.py
Normal file
9
src/settings/__init__.py
Normal file
@@ -0,0 +1,9 @@
|
||||
"""
|
||||
Django settings module.
|
||||
"""
|
||||
|
||||
# This will be overridden by the specific settings file
|
||||
try:
|
||||
from .dev import *
|
||||
except ImportError:
|
||||
from .base import *
|
||||
371
src/settings/base.py
Normal file
371
src/settings/base.py
Normal file
@@ -0,0 +1,371 @@
|
||||
"""
|
||||
Base settings for Django project.
|
||||
"""
|
||||
|
||||
import os
|
||||
import warnings
|
||||
from datetime import timedelta
|
||||
from pathlib import Path
|
||||
|
||||
BASE_DIR = Path(__file__).resolve().parent.parent
|
||||
# Repository root (one level above src/)
|
||||
PROJECT_ROOT = BASE_DIR.parent
|
||||
|
||||
APP_VERSION = "1.0.0"
|
||||
OPENAPI_USE_ENGLISH_TAGS = False
|
||||
STARTUP_CHECKS_ENABLED = True
|
||||
STARTUP_DB_TIMEOUT_SECONDS = 3
|
||||
STARTUP_REDIS_TIMEOUT_SECONDS = 3
|
||||
|
||||
# Suppress noisy warning from legacy coreapi used by drf-yasg.
|
||||
warnings.filterwarnings(
|
||||
"ignore",
|
||||
message="pkg_resources is deprecated as an API.*",
|
||||
category=UserWarning,
|
||||
module="coreapi.utils",
|
||||
)
|
||||
|
||||
# SECRET_KEY, DEBUG, ALLOWED_HOSTS определяются в dev.py / production.py
|
||||
|
||||
# Application definition
|
||||
INSTALLED_APPS = [
|
||||
"jazzmin", # Django Jazzmin - modern admin theme (must be before admin)
|
||||
"django.contrib.admin",
|
||||
"django.contrib.auth",
|
||||
"django.contrib.contenttypes",
|
||||
"django.contrib.sessions",
|
||||
"django.contrib.messages",
|
||||
"django.contrib.staticfiles",
|
||||
# Third-party apps
|
||||
"rest_framework",
|
||||
"django_filters",
|
||||
"corsheaders",
|
||||
"django_celery_beat",
|
||||
"django_celery_results",
|
||||
"drf_yasg",
|
||||
# Local apps
|
||||
"apps.core",
|
||||
"apps.user",
|
||||
"apps.parsers",
|
||||
]
|
||||
|
||||
# Jazzmin Admin Configuration
|
||||
JAZZMIN_SETTINGS = {
|
||||
# Title
|
||||
"site_title": "Mostovik Admin",
|
||||
"site_header": "Mostovik",
|
||||
"site_brand": "Mostovik",
|
||||
"site_logo": None,
|
||||
"login_logo": None,
|
||||
"login_logo_dark": None,
|
||||
"site_logo_classes": "img-circle",
|
||||
"site_icon": None,
|
||||
"welcome_sign": "Добро пожаловать в панель управления",
|
||||
"copyright": "Mostovik Backend",
|
||||
# Search
|
||||
"search_model": ["user.User", "parsers.IndustrialCertificateRecord"],
|
||||
# User menu
|
||||
"topmenu_links": [
|
||||
{"name": "Главная", "url": "admin:index", "permissions": ["auth.view_user"]},
|
||||
{"name": "API Docs", "url": "/api/docs/", "new_window": True},
|
||||
{"model": "user.User"},
|
||||
],
|
||||
# Side menu
|
||||
"show_sidebar": True,
|
||||
"navigation_expanded": True,
|
||||
"hide_apps": ["django_celery_results"],
|
||||
"hide_models": [],
|
||||
"order_with_respect_to": [
|
||||
"user",
|
||||
"parsers",
|
||||
"core",
|
||||
"django_celery_beat",
|
||||
],
|
||||
# Icons (Font Awesome)
|
||||
"icons": {
|
||||
"auth": "fas fa-users-cog",
|
||||
"auth.Group": "fas fa-users",
|
||||
"user.User": "fas fa-user",
|
||||
"user.Profile": "fas fa-id-card",
|
||||
"parsers.Proxy": "fas fa-shield-alt",
|
||||
"parsers.ParserLoadLog": "fas fa-history",
|
||||
"parsers.IndustrialCertificateRecord": "fas fa-certificate",
|
||||
"parsers.ManufacturerRecord": "fas fa-industry",
|
||||
"core.BackgroundJob": "fas fa-tasks",
|
||||
"django_celery_beat.PeriodicTask": "fas fa-clock",
|
||||
"django_celery_beat.CrontabSchedule": "fas fa-calendar-alt",
|
||||
"django_celery_beat.IntervalSchedule": "fas fa-stopwatch",
|
||||
"django_celery_results.TaskResult": "fas fa-clipboard-check",
|
||||
},
|
||||
"default_icon_parents": "fas fa-chevron-circle-right",
|
||||
"default_icon_children": "fas fa-circle",
|
||||
# Related modal
|
||||
"related_modal_active": True,
|
||||
# UI Tweaks
|
||||
"custom_css": None,
|
||||
"custom_js": None,
|
||||
"use_google_fonts_cdn": True,
|
||||
"show_ui_builder": False,
|
||||
# Change view
|
||||
"changeform_format": "horizontal_tabs",
|
||||
"changeform_format_overrides": {
|
||||
"user.User": "collapsible",
|
||||
"parsers.IndustrialCertificateRecord": "vertical_tabs",
|
||||
},
|
||||
}
|
||||
|
||||
JAZZMIN_UI_TWEAKS = {
|
||||
"navbar_small_text": False,
|
||||
"footer_small_text": False,
|
||||
"body_small_text": False,
|
||||
"brand_small_text": False,
|
||||
"brand_colour": "navbar-primary",
|
||||
"accent": "accent-primary",
|
||||
"navbar": "navbar-dark",
|
||||
"no_navbar_border": False,
|
||||
"navbar_fixed": True,
|
||||
"layout_boxed": False,
|
||||
"footer_fixed": False,
|
||||
"sidebar_fixed": True,
|
||||
"sidebar": "sidebar-dark-primary",
|
||||
"sidebar_nav_small_text": False,
|
||||
"sidebar_disable_expand": False,
|
||||
"sidebar_nav_child_indent": False,
|
||||
"sidebar_nav_compact_style": False,
|
||||
"sidebar_nav_legacy_style": False,
|
||||
"sidebar_nav_flat_style": False,
|
||||
"theme": "default",
|
||||
"dark_mode_theme": "darkly",
|
||||
"button_classes": {
|
||||
"primary": "btn-primary",
|
||||
"secondary": "btn-secondary",
|
||||
"info": "btn-info",
|
||||
"warning": "btn-warning",
|
||||
"danger": "btn-danger",
|
||||
"success": "btn-success",
|
||||
},
|
||||
}
|
||||
|
||||
MIDDLEWARE = [
|
||||
"apps.core.middleware.RequestIDMiddleware",
|
||||
"corsheaders.middleware.CorsMiddleware",
|
||||
"django.middleware.security.SecurityMiddleware",
|
||||
"whitenoise.middleware.WhiteNoiseMiddleware",
|
||||
"django.contrib.sessions.middleware.SessionMiddleware",
|
||||
"django.middleware.common.CommonMiddleware",
|
||||
"django.middleware.csrf.CsrfViewMiddleware",
|
||||
"django.contrib.auth.middleware.AuthenticationMiddleware",
|
||||
"django.contrib.messages.middleware.MessageMiddleware",
|
||||
"django.middleware.clickjacking.XFrameOptionsMiddleware",
|
||||
]
|
||||
|
||||
ROOT_URLCONF = "core.urls"
|
||||
|
||||
TEMPLATES = [
|
||||
{
|
||||
"BACKEND": "django.template.backends.django.DjangoTemplates",
|
||||
"DIRS": [BASE_DIR / "templates"],
|
||||
"APP_DIRS": True,
|
||||
"OPTIONS": {
|
||||
"context_processors": [
|
||||
"django.template.context_processors.debug",
|
||||
"django.template.context_processors.request",
|
||||
"django.contrib.auth.context_processors.auth",
|
||||
"django.contrib.messages.context_processors.messages",
|
||||
],
|
||||
},
|
||||
},
|
||||
]
|
||||
|
||||
WSGI_APPLICATION = "core.wsgi.application"
|
||||
|
||||
# Database и Cache определяются в dev.py / production.py
|
||||
|
||||
# =============================================================================
|
||||
# PARSERS SETTINGS
|
||||
# =============================================================================
|
||||
|
||||
ZAKUPKI_TOKEN = os.getenv("ZAKUPKI_TOKEN", "")
|
||||
FNS_LOCK_TTL_SECONDS = 3600
|
||||
PARSER_PROXIES = []
|
||||
|
||||
|
||||
# Password validation
|
||||
AUTH_PASSWORD_VALIDATORS = [
|
||||
{
|
||||
"NAME": "django.contrib.auth.password_validation.UserAttributeSimilarityValidator",
|
||||
},
|
||||
{
|
||||
"NAME": "django.contrib.auth.password_validation.MinimumLengthValidator",
|
||||
},
|
||||
{
|
||||
"NAME": "django.contrib.auth.password_validation.CommonPasswordValidator",
|
||||
},
|
||||
{
|
||||
"NAME": "django.contrib.auth.password_validation.NumericPasswordValidator",
|
||||
},
|
||||
]
|
||||
|
||||
# Internationalization
|
||||
LANGUAGE_CODE = "ru-RU"
|
||||
TIME_ZONE = "UTC"
|
||||
USE_I18N = True
|
||||
USE_L10N = True
|
||||
USE_TZ = True
|
||||
|
||||
# Static files (CSS, JavaScript, Images)
|
||||
STATIC_URL = "/static/"
|
||||
STATIC_ROOT = PROJECT_ROOT / "staticfiles"
|
||||
STATICFILES_DIRS = [BASE_DIR / "static"]
|
||||
|
||||
# Media files
|
||||
MEDIA_URL = "/media/"
|
||||
MEDIA_ROOT = PROJECT_ROOT / "media"
|
||||
|
||||
# Default primary key field type
|
||||
DEFAULT_AUTO_FIELD = "django.db.models.BigAutoField"
|
||||
|
||||
# Custom user model
|
||||
AUTH_USER_MODEL = "user.User"
|
||||
|
||||
# Login URL for drf-yasg and DRF browsable API
|
||||
LOGIN_URL = "/auth/login/"
|
||||
|
||||
# REST Framework settings
|
||||
REST_FRAMEWORK = {
|
||||
"DEFAULT_AUTHENTICATION_CLASSES": [
|
||||
"rest_framework_simplejwt.authentication.JWTAuthentication",
|
||||
],
|
||||
"DEFAULT_PERMISSION_CLASSES": [
|
||||
"rest_framework.permissions.IsAuthenticatedOrReadOnly",
|
||||
],
|
||||
"DEFAULT_FILTER_BACKENDS": [
|
||||
"django_filters.rest_framework.DjangoFilterBackend",
|
||||
"rest_framework.filters.SearchFilter",
|
||||
"rest_framework.filters.OrderingFilter",
|
||||
],
|
||||
"DEFAULT_PAGINATION_CLASS": "apps.core.pagination.StandardPagination",
|
||||
"PAGE_SIZE": 20,
|
||||
"DEFAULT_RENDERER_CLASSES": [
|
||||
"rest_framework.renderers.JSONRenderer",
|
||||
"rest_framework.renderers.BrowsableAPIRenderer",
|
||||
],
|
||||
"EXCEPTION_HANDLER": "apps.core.exception_handler.custom_exception_handler",
|
||||
# Rate limiting
|
||||
"DEFAULT_THROTTLE_CLASSES": [
|
||||
"rest_framework.throttling.AnonRateThrottle",
|
||||
"rest_framework.throttling.UserRateThrottle",
|
||||
],
|
||||
"DEFAULT_THROTTLE_RATES": {
|
||||
"anon": "100/hour",
|
||||
"user": "1000/hour",
|
||||
},
|
||||
}
|
||||
|
||||
# JWT settings (SIGNING_KEY определяется в dev.py / production.py)
|
||||
SIMPLE_JWT = {
|
||||
"ACCESS_TOKEN_LIFETIME": timedelta(minutes=60),
|
||||
"REFRESH_TOKEN_LIFETIME": timedelta(days=7),
|
||||
"ROTATE_REFRESH_TOKENS": True,
|
||||
"BLACKLIST_AFTER_ROTATION": True,
|
||||
"UPDATE_LAST_LOGIN": True,
|
||||
"ALGORITHM": "HS256",
|
||||
"VERIFYING_KEY": None,
|
||||
"AUDIENCE": None,
|
||||
"ISSUER": None,
|
||||
"JWK_URL": None,
|
||||
"LEEWAY": 0,
|
||||
"AUTH_HEADER_TYPES": ("Bearer",),
|
||||
"AUTH_HEADER_NAME": "HTTP_AUTHORIZATION",
|
||||
"USER_ID_FIELD": "id",
|
||||
"USER_ID_CLAIM": "user_id",
|
||||
"USER_AUTHENTICATION_RULE": "rest_framework_simplejwt.authentication.default_user_authentication_rule",
|
||||
"AUTH_TOKEN_CLASSES": ("rest_framework_simplejwt.tokens.AccessToken",),
|
||||
"TOKEN_TYPE_CLAIM": "token_type",
|
||||
"TOKEN_USER_CLASS": "rest_framework_simplejwt.models.TokenUser",
|
||||
"JTI_CLAIM": "jti",
|
||||
"SLIDING_TOKEN_REFRESH_EXP_CLAIM": "refresh_exp",
|
||||
"SLIDING_TOKEN_LIFETIME": timedelta(minutes=5),
|
||||
"SLIDING_TOKEN_REFRESH_LIFETIME": timedelta(days=1),
|
||||
}
|
||||
|
||||
# CORS settings
|
||||
CORS_ALLOWED_ORIGINS = ["http://localhost:3000", "http://127.0.0.1:3000"]
|
||||
CORS_ALLOW_CREDENTIALS = True
|
||||
|
||||
# =============================================================================
|
||||
# SWAGGER SETTINGS (drf-yasg)
|
||||
# =============================================================================
|
||||
SWAGGER_SETTINGS = {
|
||||
"SECURITY_DEFINITIONS": {
|
||||
"Bearer": {
|
||||
"type": "apiKey",
|
||||
"name": "Authorization",
|
||||
"in": "header",
|
||||
"description": "JWT авторизация. Формат: Bearer <access_token>",
|
||||
}
|
||||
},
|
||||
"USE_SESSION_AUTH": True,
|
||||
"PERSIST_AUTH": True,
|
||||
"REFETCH_SCHEMA_WITH_AUTH": True,
|
||||
"REFETCH_SCHEMA_ON_LOGOUT": True,
|
||||
}
|
||||
|
||||
# Logging configuration
|
||||
LOGGING = {
|
||||
"version": 1,
|
||||
"disable_existing_loggers": False,
|
||||
"formatters": {
|
||||
"verbose": {
|
||||
"format": "{levelname} {asctime} {module} {process:d} {thread:d} {message}",
|
||||
"style": "{",
|
||||
},
|
||||
"simple": {
|
||||
"format": "{levelname} {message}",
|
||||
"style": "{",
|
||||
},
|
||||
},
|
||||
"handlers": {
|
||||
"file": {
|
||||
"level": "INFO",
|
||||
"class": "logging.FileHandler",
|
||||
"filename": PROJECT_ROOT / "logs/django.log",
|
||||
"formatter": "verbose",
|
||||
},
|
||||
"console": {
|
||||
"level": "INFO",
|
||||
"class": "logging.StreamHandler",
|
||||
"formatter": "simple",
|
||||
},
|
||||
},
|
||||
"root": {
|
||||
"handlers": ["console", "file"],
|
||||
"level": "INFO",
|
||||
},
|
||||
"loggers": {
|
||||
"django": {
|
||||
"handlers": ["console", "file"],
|
||||
"level": "INFO",
|
||||
"propagate": False,
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
# =============================================================================
|
||||
# FNS Parser Settings
|
||||
# =============================================================================
|
||||
|
||||
# Directory for watching incoming FNS files
|
||||
FNS_WATCH_DIRECTORY = PROJECT_ROOT / "input" / "fns"
|
||||
|
||||
# Directory for processed files (moved after successful processing)
|
||||
FNS_PROCESSED_DIRECTORY = PROJECT_ROOT / "input" / "fns" / "processed"
|
||||
|
||||
# Directory for failed files (moved after failed processing)
|
||||
FNS_FAILED_DIRECTORY = PROJECT_ROOT / "input" / "fns" / "failed"
|
||||
|
||||
# =============================================================================
|
||||
# Checko API Settings (checko.ru)
|
||||
# =============================================================================
|
||||
|
||||
CHECKO_API_KEY = os.getenv("CHECKO_API_KEY", "")
|
||||
71
src/settings/dev.py
Normal file
71
src/settings/dev.py
Normal file
@@ -0,0 +1,71 @@
|
||||
"""
|
||||
Development settings - закрытый контур, без переменных окружения.
|
||||
"""
|
||||
|
||||
import os
|
||||
|
||||
from .base import *
|
||||
|
||||
SECRET_KEY = "django-insecure-development-key-mostovik-2024"
|
||||
DEBUG = True
|
||||
ALLOWED_HOSTS = ["*"]
|
||||
OPENAPI_USE_ENGLISH_TAGS = True
|
||||
|
||||
# JWT
|
||||
SIMPLE_JWT["SIGNING_KEY"] = SECRET_KEY
|
||||
|
||||
|
||||
def _normalize_local_host(host: str) -> str:
|
||||
"""Нормализует host для локального запуска backend на хосте."""
|
||||
if host == "host.docker.internal":
|
||||
return "127.0.0.1"
|
||||
return host
|
||||
|
||||
|
||||
_postgres_host = _normalize_local_host(os.getenv("POSTGRES_HOST", "127.0.0.1"))
|
||||
_redis_host = _normalize_local_host(os.getenv("REDIS_HOST", "127.0.0.1"))
|
||||
|
||||
# Database
|
||||
DATABASES = {
|
||||
"default": {
|
||||
"ENGINE": "django.db.backends.postgresql",
|
||||
"NAME": os.getenv("POSTGRES_DB", "mostovik"),
|
||||
"USER": os.getenv("POSTGRES_USER", "postgres"),
|
||||
"PASSWORD": os.getenv("POSTGRES_PASSWORD", "postgres"),
|
||||
# При локальном запуске backend (не в контейнере) подключаемся к портам,
|
||||
# проброшенным docker-compose.service.yml
|
||||
"HOST": _postgres_host,
|
||||
"PORT": os.getenv("POSTGRES_PORT", "5432"),
|
||||
}
|
||||
}
|
||||
|
||||
# Celery
|
||||
_default_redis_broker = f"redis://{_redis_host}:6379/0"
|
||||
_default_redis_cache = f"redis://{_redis_host}:6379/1"
|
||||
|
||||
CELERY_BROKER_URL = os.getenv("CELERY_BROKER_URL", _default_redis_broker).replace(
|
||||
"host.docker.internal", "127.0.0.1"
|
||||
)
|
||||
CELERY_RESULT_BACKEND = os.getenv(
|
||||
"CELERY_RESULT_BACKEND", _default_redis_broker
|
||||
).replace("host.docker.internal", "127.0.0.1")
|
||||
CELERY_ACCEPT_CONTENT = ["json"]
|
||||
CELERY_TASK_SERIALIZER = "json"
|
||||
CELERY_RESULT_SERIALIZER = "json"
|
||||
CELERY_TIMEZONE = "Europe/Moscow"
|
||||
|
||||
# Cache
|
||||
CACHES = {
|
||||
"default": {
|
||||
"BACKEND": "django_redis.cache.RedisCache",
|
||||
"LOCATION": os.getenv("REDIS_CACHE_URL", _default_redis_cache).replace(
|
||||
"host.docker.internal", "127.0.0.1"
|
||||
),
|
||||
"OPTIONS": {
|
||||
"CLIENT_CLASS": "django_redis.client.DefaultClient",
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
# Email
|
||||
EMAIL_BACKEND = "django.core.mail.backends.console.EmailBackend"
|
||||
94
src/settings/production.py
Normal file
94
src/settings/production.py
Normal file
@@ -0,0 +1,94 @@
|
||||
"""
|
||||
Production settings - закрытый контур, все настройки захардкожены.
|
||||
Docker Compose сеть - используются имена сервисов (db, redis).
|
||||
"""
|
||||
|
||||
import os
|
||||
|
||||
from .base import *
|
||||
|
||||
SECRET_KEY = os.getenv("SECRET_KEY", "production-secret-key-mostovik-change-me-2024")
|
||||
DEBUG = os.getenv("DEBUG", "False").lower() == "true"
|
||||
ALLOWED_HOSTS = os.getenv("ALLOWED_HOSTS", "*").split(",")
|
||||
|
||||
# JWT
|
||||
SIMPLE_JWT["SIGNING_KEY"] = SECRET_KEY
|
||||
|
||||
# HTTPS settings (раскомментировать если используется HTTPS)
|
||||
# SECURE_SSL_REDIRECT = True
|
||||
# SECURE_PROXY_SSL_HEADER = ("HTTP_X_FORWARDED_PROTO", "https")
|
||||
# SECURE_HSTS_SECONDS = 31536000
|
||||
# SECURE_HSTS_INCLUDE_SUBDOMAINS = True
|
||||
# SECURE_HSTS_PRELOAD = True
|
||||
# SESSION_COOKIE_SECURE = True
|
||||
# CSRF_COOKIE_SECURE = True
|
||||
|
||||
# Database
|
||||
DATABASES = {
|
||||
"default": {
|
||||
"ENGINE": "django.db.backends.postgresql",
|
||||
"NAME": os.getenv("POSTGRES_DB", "mostovik"),
|
||||
"USER": os.getenv("POSTGRES_USER", "postgres"),
|
||||
"PASSWORD": os.getenv("POSTGRES_PASSWORD", "postgres"),
|
||||
"HOST": os.getenv("POSTGRES_HOST", "db"),
|
||||
"PORT": os.getenv("POSTGRES_PORT", "5432"),
|
||||
"OPTIONS": {
|
||||
"sslmode": os.getenv("POSTGRES_SSLMODE", "disable"),
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
# Celery
|
||||
CELERY_BROKER_URL = os.getenv("CELERY_BROKER_URL", "redis://redis:6379/0")
|
||||
CELERY_RESULT_BACKEND = os.getenv("CELERY_RESULT_BACKEND", "redis://redis:6379/0")
|
||||
CELERY_ACCEPT_CONTENT = ["json"]
|
||||
CELERY_TASK_SERIALIZER = "json"
|
||||
CELERY_RESULT_SERIALIZER = "json"
|
||||
CELERY_TIMEZONE = "Europe/Moscow"
|
||||
CELERY_TASK_ALWAYS_EAGER = False
|
||||
CELERY_WORKER_PREFETCH_MULTIPLIER = 1
|
||||
|
||||
# Cache
|
||||
CACHES = {
|
||||
"default": {
|
||||
"BACKEND": "django_redis.cache.RedisCache",
|
||||
"LOCATION": os.getenv("REDIS_CACHE_URL", "redis://redis:6379/1"),
|
||||
"OPTIONS": {
|
||||
"CLIENT_CLASS": "django_redis.client.DefaultClient",
|
||||
"CONNECTION_POOL_KWARGS": {
|
||||
"max_connections": 20,
|
||||
"retry_on_timeout": True,
|
||||
},
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
# Logging (stdout для Docker)
|
||||
LOGGING = {
|
||||
"version": 1,
|
||||
"disable_existing_loggers": False,
|
||||
"formatters": {
|
||||
"verbose": {
|
||||
"format": "{levelname} {asctime} {module} {process:d} {thread:d} {message}",
|
||||
"style": "{",
|
||||
},
|
||||
},
|
||||
"handlers": {
|
||||
"console": {
|
||||
"level": "INFO",
|
||||
"class": "logging.StreamHandler",
|
||||
"formatter": "verbose",
|
||||
},
|
||||
},
|
||||
"root": {
|
||||
"handlers": ["console"],
|
||||
"level": "INFO",
|
||||
},
|
||||
"loggers": {
|
||||
"django": {
|
||||
"handlers": ["console"],
|
||||
"level": "INFO",
|
||||
"propagate": False,
|
||||
},
|
||||
},
|
||||
}
|
||||
110
src/settings/test.py
Normal file
110
src/settings/test.py
Normal file
@@ -0,0 +1,110 @@
|
||||
"""
|
||||
Test settings - для запуска тестов.
|
||||
"""
|
||||
|
||||
from .base import *
|
||||
|
||||
SECRET_KEY = "django-insecure-test-key-only-for-testing"
|
||||
DEBUG = True
|
||||
STARTUP_CHECKS_ENABLED = False
|
||||
|
||||
# JWT
|
||||
SIMPLE_JWT["SIGNING_KEY"] = SECRET_KEY
|
||||
|
||||
ALLOWED_HOSTS = ["localhost", "127.0.0.1", "0.0.0.0", "testserver"] # noqa: S104
|
||||
|
||||
# Use in-memory SQLite database for faster tests
|
||||
DATABASES = {
|
||||
"default": {
|
||||
"ENGINE": "django.db.backends.sqlite3",
|
||||
"NAME": ":memory:",
|
||||
"TEST": {
|
||||
"NAME": ":memory:",
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
# Disable migrations for faster tests
|
||||
class DisableMigrations:
|
||||
def __contains__(self, item):
|
||||
return True
|
||||
|
||||
def __getitem__(self, item):
|
||||
return None
|
||||
|
||||
|
||||
MIGRATION_MODULES = DisableMigrations()
|
||||
|
||||
# Cache configuration for tests (use local memory)
|
||||
CACHES = {
|
||||
"default": {
|
||||
"BACKEND": "django.core.cache.backends.locmem.LocMemCache",
|
||||
}
|
||||
}
|
||||
|
||||
# Email backend for tests
|
||||
EMAIL_BACKEND = "django.core.mail.backends.locmem.EmailBackend"
|
||||
|
||||
# Celery Configuration for Tests (use eager execution)
|
||||
CELERY_TASK_ALWAYS_EAGER = True
|
||||
CELERY_TASK_EAGER_PROPAGATES = True
|
||||
CELERY_BROKER_URL = "memory://"
|
||||
CELERY_RESULT_BACKEND = "cache+memory://"
|
||||
|
||||
# Password hashers - use fast hasher for tests
|
||||
PASSWORD_HASHERS = [
|
||||
"django.contrib.auth.hashers.MD5PasswordHasher",
|
||||
]
|
||||
|
||||
# Disable logging during tests
|
||||
LOGGING = {
|
||||
"version": 1,
|
||||
"disable_existing_loggers": False,
|
||||
"handlers": {
|
||||
"null": {
|
||||
"class": "logging.NullHandler",
|
||||
},
|
||||
},
|
||||
"root": {
|
||||
"handlers": ["null"],
|
||||
},
|
||||
"loggers": {
|
||||
"django": {
|
||||
"handlers": ["null"],
|
||||
"propagate": False,
|
||||
},
|
||||
"django.request": {
|
||||
"handlers": ["null"],
|
||||
"propagate": False,
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
# Media files for tests
|
||||
MEDIA_ROOT = "/tmp/test_media" # noqa: S108
|
||||
|
||||
# Static files for tests
|
||||
STATICFILES_STORAGE = "django.contrib.staticfiles.storage.StaticFilesStorage"
|
||||
|
||||
# Disable CSRF for API tests and disable throttling
|
||||
REST_FRAMEWORK = {
|
||||
**globals().get("REST_FRAMEWORK", {}),
|
||||
"DEFAULT_AUTHENTICATION_CLASSES": [
|
||||
"rest_framework_simplejwt.authentication.JWTAuthentication",
|
||||
],
|
||||
"TEST_REQUEST_DEFAULT_FORMAT": "json",
|
||||
# Disable throttling for tests
|
||||
"DEFAULT_THROTTLE_CLASSES": [],
|
||||
"DEFAULT_THROTTLE_RATES": {},
|
||||
}
|
||||
|
||||
# JWT settings for tests
|
||||
from datetime import timedelta
|
||||
|
||||
SIMPLE_JWT = {
|
||||
**globals().get("SIMPLE_JWT", {}),
|
||||
"ACCESS_TOKEN_LIFETIME": timedelta(minutes=5),
|
||||
"REFRESH_TOKEN_LIFETIME": timedelta(days=1),
|
||||
"ROTATE_REFRESH_TOKENS": True,
|
||||
}
|
||||
Reference in New Issue
Block a user