2026-02-19 01:19:22 +03:30
|
|
|
import os
|
2026-03-25 16:19:28 +03:30
|
|
|
from datetime import timedelta
|
2026-02-19 01:19:22 +03:30
|
|
|
from pathlib import Path
|
|
|
|
|
|
|
|
|
|
from dotenv import load_dotenv
|
|
|
|
|
|
|
|
|
|
load_dotenv()
|
|
|
|
|
|
|
|
|
|
BASE_DIR = Path(__file__).resolve().parent.parent
|
|
|
|
|
|
2026-04-26 01:15:01 +03:30
|
|
|
|
|
|
|
|
def _get_csv_env(name, default=""):
|
|
|
|
|
return [item.strip() for item in os.environ.get(name, default).split(",") if item.strip()]
|
|
|
|
|
|
2026-02-19 01:19:22 +03:30
|
|
|
SECRET_KEY = os.environ.get("SECRET_KEY", "django-insecure-dev-only")
|
|
|
|
|
DEBUG = os.environ.get("DEBUG", "0") == "1"
|
2026-04-26 01:15:01 +03:30
|
|
|
ALLOWED_HOSTS = list(
|
|
|
|
|
dict.fromkeys(
|
|
|
|
|
_get_csv_env("ALLOWED_HOSTS", "localhost,127.0.0.1,0.0.0.0")
|
|
|
|
|
+ ["web", "backend-web", os.environ.get("HOSTNAME", "")]
|
|
|
|
|
)
|
|
|
|
|
)
|
2026-02-19 01:19:22 +03:30
|
|
|
|
2026-03-20 23:16:53 +03:30
|
|
|
AUTH_USER_MODEL = "account.User"
|
|
|
|
|
|
2026-03-23 22:24:30 +03:30
|
|
|
AUTHENTICATION_BACKENDS = [
|
|
|
|
|
"account.backends.MultiFieldBackend",
|
|
|
|
|
]
|
|
|
|
|
|
2026-02-19 01:19:22 +03:30
|
|
|
INSTALLED_APPS = [
|
|
|
|
|
"django.contrib.admin",
|
|
|
|
|
"django.contrib.auth",
|
|
|
|
|
"django.contrib.contenttypes",
|
|
|
|
|
"django.contrib.sessions",
|
|
|
|
|
"django.contrib.messages",
|
|
|
|
|
"django.contrib.staticfiles",
|
|
|
|
|
"auth.apps.AuthConfig",
|
2026-03-20 23:16:53 +03:30
|
|
|
"account.apps.AccountConfig",
|
2026-04-02 23:25:39 +03:30
|
|
|
"farm_hub.apps.FarmHubConfig",
|
2026-04-03 23:51:00 +03:30
|
|
|
"access_control.apps.AccessControlConfig",
|
2026-04-03 15:15:41 +03:30
|
|
|
"sensor_catalog.apps.SensorCatalogConfig",
|
2026-02-19 16:58:41 +03:30
|
|
|
"dashboard",
|
2026-04-11 03:54:15 +03:30
|
|
|
"crop_health.apps.CropHealthConfig",
|
|
|
|
|
"soil.apps.SoilConfig",
|
2026-02-25 12:21:53 +03:30
|
|
|
"crop_zoning",
|
|
|
|
|
"pest_detection",
|
2026-04-11 03:54:15 +03:30
|
|
|
"sensor_7_in_1.apps.Sensor7In1Config",
|
|
|
|
|
"water.apps.WaterConfig",
|
2026-02-25 12:21:53 +03:30
|
|
|
"irrigation_recommendation",
|
2026-04-10 16:12:51 +03:30
|
|
|
"yield_harvest.apps.YieldHarvestConfig",
|
|
|
|
|
"economic_overview.apps.EconomicOverviewConfig",
|
|
|
|
|
"farm_alerts.apps.FarmAlertsConfig",
|
2026-02-25 12:21:53 +03:30
|
|
|
"fertilization_recommendation",
|
|
|
|
|
"farm_ai_assistant",
|
2026-04-03 23:51:00 +03:30
|
|
|
"notifications.apps.NotificationsConfig",
|
2026-03-25 00:51:04 +03:30
|
|
|
"external_api_adapter.apps.ExternalApiAdapterConfig",
|
2026-04-05 00:57:25 +03:30
|
|
|
"sensor_external_api.apps.SensorExternalApiConfig",
|
2026-02-19 01:19:22 +03:30
|
|
|
"rest_framework",
|
2026-03-24 20:10:48 +03:30
|
|
|
"drf_spectacular",
|
|
|
|
|
"drf_spectacular_sidecar",
|
2026-02-19 01:19:22 +03:30
|
|
|
"corsheaders",
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
MIDDLEWARE = [
|
|
|
|
|
"django.middleware.security.SecurityMiddleware",
|
|
|
|
|
"corsheaders.middleware.CorsMiddleware",
|
|
|
|
|
"django.contrib.sessions.middleware.SessionMiddleware",
|
|
|
|
|
"django.middleware.common.CommonMiddleware",
|
|
|
|
|
"django.middleware.csrf.CsrfViewMiddleware",
|
|
|
|
|
"django.contrib.auth.middleware.AuthenticationMiddleware",
|
2026-04-09 23:43:58 +03:30
|
|
|
"access_control.middleware.RouteFeatureAccessMiddleware",
|
2026-02-19 01:19:22 +03:30
|
|
|
"django.contrib.messages.middleware.MessageMiddleware",
|
|
|
|
|
"django.middleware.clickjacking.XFrameOptionsMiddleware",
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
ROOT_URLCONF = "config.urls"
|
|
|
|
|
WSGI_APPLICATION = "config.wsgi.application"
|
|
|
|
|
|
|
|
|
|
TEMPLATES = [
|
|
|
|
|
{
|
|
|
|
|
"BACKEND": "django.template.backends.django.DjangoTemplates",
|
|
|
|
|
"DIRS": [],
|
|
|
|
|
"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",
|
|
|
|
|
],
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
DATABASES = {
|
|
|
|
|
"default": {
|
|
|
|
|
"ENGINE": os.environ.get("DB_ENGINE", "django.db.backends.mysql"),
|
|
|
|
|
"NAME": os.environ.get("DB_NAME", "croplogic"),
|
|
|
|
|
"USER": os.environ.get("DB_USER", "croplogic"),
|
|
|
|
|
"PASSWORD": os.environ.get("DB_PASSWORD", ""),
|
|
|
|
|
"HOST": os.environ.get("DB_HOST", "127.0.0.1"),
|
|
|
|
|
"PORT": os.environ.get("DB_PORT", "3306"),
|
|
|
|
|
"OPTIONS": {
|
|
|
|
|
"charset": "utf8mb4",
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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"},
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
LANGUAGE_CODE = "en-us"
|
|
|
|
|
TIME_ZONE = "UTC"
|
|
|
|
|
USE_I18N = True
|
|
|
|
|
USE_TZ = True
|
|
|
|
|
|
|
|
|
|
STATIC_URL = "static/"
|
|
|
|
|
STATIC_ROOT = BASE_DIR / "staticfiles"
|
|
|
|
|
|
|
|
|
|
DEFAULT_AUTO_FIELD = "django.db.models.BigAutoField"
|
|
|
|
|
|
|
|
|
|
CACHES = {
|
|
|
|
|
"default": {
|
2026-04-09 23:43:58 +03:30
|
|
|
"BACKEND": "django.core.cache.backends.redis.RedisCache",
|
|
|
|
|
"LOCATION": os.getenv("CACHE_URL", os.getenv("CELERY_BROKER_URL", "redis://redis:6379/0")),
|
|
|
|
|
"KEY_PREFIX": "croplogic",
|
2026-02-19 01:19:22 +03:30
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
REST_FRAMEWORK = {
|
|
|
|
|
"DEFAULT_PERMISSION_CLASSES": [
|
2026-03-25 01:54:01 +03:30
|
|
|
"rest_framework.permissions.IsAuthenticated",
|
2026-04-03 23:51:00 +03:30
|
|
|
"access_control.permissions.FeatureAccessPermission",
|
2026-02-19 01:19:22 +03:30
|
|
|
],
|
|
|
|
|
"DEFAULT_AUTHENTICATION_CLASSES": [
|
|
|
|
|
"rest_framework_simplejwt.authentication.JWTAuthentication",
|
|
|
|
|
],
|
2026-03-24 20:10:48 +03:30
|
|
|
"DEFAULT_SCHEMA_CLASS": "drf_spectacular.openapi.AutoSchema",
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SPECTACULAR_SETTINGS = {
|
|
|
|
|
"TITLE": "CropLogic API",
|
|
|
|
|
"DESCRIPTION": "Swagger/OpenAPI documentation for all CropLogic API endpoints.",
|
|
|
|
|
"VERSION": "1.0.0",
|
|
|
|
|
"SERVE_INCLUDE_SCHEMA": False,
|
|
|
|
|
"SWAGGER_UI_DIST": "SIDECAR",
|
|
|
|
|
"SWAGGER_UI_FAVICON_HREF": "SIDECAR",
|
|
|
|
|
"REDOC_DIST": "SIDECAR",
|
|
|
|
|
"SCHEMA_PATH_PREFIX": r"/api/",
|
2026-03-25 01:54:01 +03:30
|
|
|
"SERVE_PERMISSIONS": ["rest_framework.permissions.AllowAny"],
|
2026-04-05 00:57:25 +03:30
|
|
|
"APPEND_COMPONENTS": {
|
|
|
|
|
"securitySchemes": {
|
|
|
|
|
"SensorExternalApiKey": {
|
|
|
|
|
"type": "apiKey",
|
|
|
|
|
"in": "header",
|
|
|
|
|
"name": "X-API-Key",
|
|
|
|
|
"description": "Use API key 12345 for sensor external API endpoints.",
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
"SWAGGER_UI_SETTINGS": {
|
|
|
|
|
"persistAuthorization": True,
|
|
|
|
|
},
|
2026-02-19 01:19:22 +03:30
|
|
|
}
|
|
|
|
|
|
2026-03-20 23:16:53 +03:30
|
|
|
|
|
|
|
|
SMS_IR_API_KEY = os.environ.get("SMS_IR_API_KEY", "")
|
|
|
|
|
SMS_IR_LINE_NUMBER = int(os.environ.get("SMS_IR_LINE_NUMBER", "300000000000"))
|
|
|
|
|
|
2026-02-19 01:19:22 +03:30
|
|
|
CORS_ALLOW_ALL_ORIGINS = DEBUG
|
2026-03-25 00:51:04 +03:30
|
|
|
|
|
|
|
|
USE_EXTERNAL_API_MOCK = os.getenv("USE_EXTERNAL_API_MOCK", "false").lower() == "true"
|
|
|
|
|
EXTERNAL_API_TIMEOUT = int(os.getenv("EXTERNAL_API_TIMEOUT", "30"))
|
|
|
|
|
|
2026-04-09 22:48:54 +03:30
|
|
|
ACCESS_CONTROL_AUTHZ_ENABLED = os.getenv("ACCESS_CONTROL_AUTHZ_ENABLED", "true").lower() == "true"
|
|
|
|
|
ACCESS_CONTROL_AUTHZ_BASE_URL = os.getenv(
|
|
|
|
|
"ACCESS_CONTROL_AUTHZ_BASE_URL",
|
|
|
|
|
"http://croplogic-accsess-opa:8181",
|
|
|
|
|
)
|
|
|
|
|
ACCESS_CONTROL_AUTHZ_BATCH_PATH = os.getenv("ACCESS_CONTROL_AUTHZ_BATCH_PATH", "/v1/data/croplogic/authz/batch_decision")
|
|
|
|
|
ACCESS_CONTROL_AUTHZ_TIMEOUT = int(os.getenv("ACCESS_CONTROL_AUTHZ_TIMEOUT", str(EXTERNAL_API_TIMEOUT)))
|
2026-04-09 23:43:58 +03:30
|
|
|
ACCESS_CONTROL_AUTHZ_CACHE_TIMEOUT = int(os.getenv("ACCESS_CONTROL_AUTHZ_CACHE_TIMEOUT", "300"))
|
2026-04-09 22:48:54 +03:30
|
|
|
|
2026-03-25 00:51:04 +03:30
|
|
|
EXTERNAL_SERVICES = {
|
|
|
|
|
"ai": {
|
2026-04-26 01:15:01 +03:30
|
|
|
"base_url": os.getenv("AI_SERVICE_BASE_URL", "http://ai-web:8000"),
|
2026-03-25 00:51:04 +03:30
|
|
|
"api_key": os.getenv("AI_SERVICE_API_KEY", ""),
|
2026-04-26 01:15:01 +03:30
|
|
|
"host_header": os.getenv("AI_SERVICE_HOST_HEADER", "localhost"),
|
2026-03-25 00:51:04 +03:30
|
|
|
},
|
2026-04-02 23:25:39 +03:30
|
|
|
"farm_hub": {
|
|
|
|
|
"base_url": os.getenv("FARM_HUB_SERVICE_BASE_URL", ""),
|
|
|
|
|
"api_key": os.getenv("FARM_HUB_SERVICE_API_KEY", ""),
|
2026-04-26 01:15:01 +03:30
|
|
|
"host_header": os.getenv("FARM_HUB_SERVICE_HOST_HEADER", ""),
|
2026-03-25 00:51:04 +03:30
|
|
|
},
|
|
|
|
|
}
|
2026-03-25 16:19:28 +03:30
|
|
|
|
|
|
|
|
SIMPLE_JWT = {
|
|
|
|
|
"ACCESS_TOKEN_LIFETIME": timedelta(days=7),
|
|
|
|
|
"REFRESH_TOKEN_LIFETIME": timedelta(days=7),
|
|
|
|
|
"ROTATE_REFRESH_TOKENS": False,
|
|
|
|
|
"BLACKLIST_AFTER_ROTATION": False,
|
|
|
|
|
}
|
2026-03-29 13:40:23 +03:30
|
|
|
|
|
|
|
|
CROP_ZONE_CHUNK_AREA_SQM = float(os.getenv("CROP_ZONE_CHUNK_AREA_SQM", "10000"))
|
2026-04-01 17:28:24 +03:30
|
|
|
CROP_ZONE_TASK_STALE_SECONDS = int(os.getenv("CROP_ZONE_TASK_STALE_SECONDS", "300"))
|
2026-03-29 15:07:14 +03:30
|
|
|
|
2026-04-01 17:28:24 +03:30
|
|
|
CELERY_BROKER_URL = os.getenv("CELERY_BROKER_URL", "redis://redis:6379/0")
|
2026-03-29 15:07:14 +03:30
|
|
|
CELERY_RESULT_BACKEND = os.getenv("CELERY_RESULT_BACKEND", CELERY_BROKER_URL)
|
2026-04-03 23:51:00 +03:30
|
|
|
NOTIFICATION_REDIS_URL = os.getenv("NOTIFICATION_REDIS_URL", CELERY_BROKER_URL)
|
2026-04-05 00:57:25 +03:30
|
|
|
EXTERNAL_NOTIFICATION_API_KEY = os.getenv("EXTERNAL_NOTIFICATION_API_KEY", "12345")
|
|
|
|
|
SENSOR_EXTERNAL_API_KEY = os.getenv("SENSOR_EXTERNAL_API_KEY", "12345")
|
2026-04-07 01:09:27 +03:30
|
|
|
FARM_DATA_API_HOST = os.getenv("FARM_DATA_API_HOST", "")
|
|
|
|
|
FARM_DATA_API_PORT = os.getenv("FARM_DATA_API_PORT", "")
|
|
|
|
|
FARM_DATA_API_PATH = os.getenv("FARM_DATA_API_PATH", "/api/farm-data/")
|
|
|
|
|
FARM_DATA_API_KEY = os.getenv("FARM_DATA_API_KEY", "")
|
|
|
|
|
FARM_DATA_API_TIMEOUT = int(os.getenv("FARM_DATA_API_TIMEOUT", str(EXTERNAL_API_TIMEOUT)))
|
2026-03-29 15:07:14 +03:30
|
|
|
CELERY_TASK_DEFAULT_QUEUE = os.getenv("CELERY_TASK_DEFAULT_QUEUE", "default")
|
|
|
|
|
CELERY_TASK_ACKS_LATE = True
|
|
|
|
|
CELERY_WORKER_PREFETCH_MULTIPLIER = int(os.getenv("CELERY_WORKER_PREFETCH_MULTIPLIER", "1"))
|
|
|
|
|
CELERY_TASK_TIME_LIMIT = int(os.getenv("CELERY_TASK_TIME_LIMIT", "120"))
|
|
|
|
|
CELERY_TASK_SOFT_TIME_LIMIT = int(os.getenv("CELERY_TASK_SOFT_TIME_LIMIT", "90"))
|
2026-04-01 17:28:24 +03:30
|
|
|
CELERY_BROKER_CONNECTION_RETRY_ON_STARTUP = os.getenv("CELERY_BROKER_CONNECTION_RETRY_ON_STARTUP", "true").lower() == "true"
|