django-cfg 1.2.29__py3-none-any.whl → 1.3.1__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- django_cfg/__init__.py +1 -1
- django_cfg/apps/api/health/views.py +4 -2
- django_cfg/apps/knowbase/config/settings.py +16 -15
- django_cfg/apps/payments/README.md +326 -0
- django_cfg/apps/payments/admin/__init__.py +20 -9
- django_cfg/apps/payments/admin/api_keys_admin.py +521 -237
- django_cfg/apps/payments/admin/balance_admin.py +592 -297
- django_cfg/apps/payments/admin/currencies_admin.py +600 -108
- django_cfg/apps/payments/admin/filters.py +306 -199
- django_cfg/apps/payments/admin/payments_admin.py +470 -64
- django_cfg/apps/payments/admin/subscriptions_admin.py +578 -128
- django_cfg/apps/payments/admin_interface/__init__.py +18 -0
- django_cfg/apps/payments/admin_interface/templates/payments/base.html +162 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/dev_tool_card.html +38 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/loading_spinner.html +16 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/notification.html +27 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/provider_card.html +86 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/status_card.html +39 -0
- django_cfg/apps/payments/admin_interface/templates/payments/currency_converter.html +382 -0
- django_cfg/apps/payments/admin_interface/templates/payments/payment_dashboard.html +300 -0
- django_cfg/apps/payments/admin_interface/templates/payments/payment_form.html +303 -0
- django_cfg/apps/payments/admin_interface/templates/payments/payment_list.html +382 -0
- django_cfg/apps/payments/admin_interface/templates/payments/payment_status.html +500 -0
- django_cfg/apps/payments/admin_interface/templates/payments/webhook_dashboard.html +594 -0
- django_cfg/apps/payments/admin_interface/views/__init__.py +23 -0
- django_cfg/apps/payments/admin_interface/views/payment_views.py +259 -0
- django_cfg/apps/payments/admin_interface/views/webhook_dashboard.py +37 -0
- django_cfg/apps/payments/apps.py +34 -9
- django_cfg/apps/payments/config/__init__.py +28 -51
- django_cfg/apps/payments/config/constance/__init__.py +22 -0
- django_cfg/apps/payments/config/constance/config_service.py +123 -0
- django_cfg/apps/payments/config/constance/fields.py +69 -0
- django_cfg/apps/payments/config/constance/settings.py +160 -0
- django_cfg/apps/payments/config/django_cfg_integration.py +202 -0
- django_cfg/apps/payments/config/helpers.py +130 -0
- django_cfg/apps/payments/management/__init__.py +1 -3
- django_cfg/apps/payments/management/commands/__init__.py +1 -3
- django_cfg/apps/payments/management/commands/manage_currencies.py +381 -0
- django_cfg/apps/payments/management/commands/manage_providers.py +408 -0
- django_cfg/apps/payments/middleware/__init__.py +3 -1
- django_cfg/apps/payments/middleware/api_access.py +329 -222
- django_cfg/apps/payments/middleware/rate_limiting.py +343 -163
- django_cfg/apps/payments/middleware/usage_tracking.py +250 -238
- django_cfg/apps/payments/migrations/0001_initial.py +708 -536
- django_cfg/apps/payments/models/__init__.py +16 -20
- django_cfg/apps/payments/models/api_keys.py +121 -43
- django_cfg/apps/payments/models/balance.py +150 -115
- django_cfg/apps/payments/models/base.py +68 -15
- django_cfg/apps/payments/models/currencies.py +207 -67
- django_cfg/apps/payments/models/managers/__init__.py +44 -0
- django_cfg/apps/payments/models/managers/api_key_managers.py +329 -0
- django_cfg/apps/payments/models/managers/balance_managers.py +599 -0
- django_cfg/apps/payments/models/managers/currency_managers.py +385 -0
- django_cfg/apps/payments/models/managers/payment_managers.py +511 -0
- django_cfg/apps/payments/models/managers/subscription_managers.py +641 -0
- django_cfg/apps/payments/models/payments.py +235 -284
- django_cfg/apps/payments/models/subscriptions.py +257 -177
- django_cfg/apps/payments/models/tariffs.py +147 -40
- django_cfg/apps/payments/services/__init__.py +209 -56
- django_cfg/apps/payments/services/cache/__init__.py +6 -6
- django_cfg/apps/payments/services/cache/{simple_cache.py → cache_service.py} +112 -12
- django_cfg/apps/payments/services/core/__init__.py +10 -6
- django_cfg/apps/payments/services/core/balance_service.py +435 -360
- django_cfg/apps/payments/services/core/base.py +166 -0
- django_cfg/apps/payments/services/core/currency_service.py +478 -0
- django_cfg/apps/payments/services/core/payment_service.py +344 -468
- django_cfg/apps/payments/services/core/subscription_service.py +425 -484
- django_cfg/apps/payments/services/core/webhook_service.py +410 -0
- django_cfg/apps/payments/services/integrations/__init__.py +29 -0
- django_cfg/apps/payments/services/integrations/ngrok_service.py +47 -0
- django_cfg/apps/payments/services/integrations/providers_config.py +107 -0
- django_cfg/apps/payments/services/providers/__init__.py +9 -14
- django_cfg/apps/payments/services/providers/base.py +232 -71
- django_cfg/apps/payments/services/providers/nowpayments.py +404 -219
- django_cfg/apps/payments/services/providers/registry.py +429 -80
- django_cfg/apps/payments/services/types/__init__.py +78 -0
- django_cfg/apps/payments/services/types/data.py +177 -0
- django_cfg/apps/payments/services/types/requests.py +150 -0
- django_cfg/apps/payments/services/types/responses.py +156 -0
- django_cfg/apps/payments/services/types/webhooks.py +232 -0
- django_cfg/apps/payments/signals/__init__.py +33 -8
- django_cfg/apps/payments/signals/api_key_signals.py +211 -130
- django_cfg/apps/payments/signals/balance_signals.py +174 -0
- django_cfg/apps/payments/signals/payment_signals.py +129 -98
- django_cfg/apps/payments/signals/subscription_signals.py +195 -143
- django_cfg/apps/payments/static/payments/css/components.css +380 -0
- django_cfg/apps/payments/static/payments/css/dashboard.css +188 -0
- django_cfg/apps/payments/static/payments/js/components.js +545 -0
- django_cfg/apps/payments/static/payments/js/utils.js +412 -0
- django_cfg/apps/payments/templatetags/__init__.py +1 -1
- django_cfg/apps/payments/templatetags/payment_tags.py +466 -0
- django_cfg/apps/payments/urls.py +46 -47
- django_cfg/apps/payments/urls_admin.py +49 -0
- django_cfg/apps/payments/views/api/__init__.py +101 -0
- django_cfg/apps/payments/views/api/api_keys.py +387 -0
- django_cfg/apps/payments/views/api/balances.py +381 -0
- django_cfg/apps/payments/views/api/base.py +298 -0
- django_cfg/apps/payments/views/api/currencies.py +402 -0
- django_cfg/apps/payments/views/api/payments.py +415 -0
- django_cfg/apps/payments/views/api/subscriptions.py +475 -0
- django_cfg/apps/payments/views/api/webhooks.py +476 -0
- django_cfg/apps/payments/views/serializers/__init__.py +99 -0
- django_cfg/apps/payments/views/serializers/api_keys.py +424 -0
- django_cfg/apps/payments/views/serializers/balances.py +300 -0
- django_cfg/apps/payments/views/serializers/currencies.py +335 -0
- django_cfg/apps/payments/views/serializers/payments.py +387 -0
- django_cfg/apps/payments/views/serializers/subscriptions.py +429 -0
- django_cfg/apps/payments/views/serializers/webhooks.py +137 -0
- django_cfg/apps/tasks/urls.py +0 -2
- django_cfg/apps/tasks/urls_admin.py +14 -0
- django_cfg/apps/urls.py +4 -4
- django_cfg/config.py +1 -1
- django_cfg/core/config.py +75 -4
- django_cfg/core/generation.py +25 -4
- django_cfg/core/integration/README.md +363 -0
- django_cfg/core/integration/__init__.py +47 -0
- django_cfg/core/integration/commands_collector.py +239 -0
- django_cfg/core/integration/display/__init__.py +15 -0
- django_cfg/core/integration/display/base.py +157 -0
- django_cfg/core/integration/display/ngrok.py +164 -0
- django_cfg/core/integration/display/startup.py +815 -0
- django_cfg/core/integration/url_integration.py +123 -0
- django_cfg/core/integration/version_checker.py +160 -0
- django_cfg/management/commands/auto_generate.py +4 -0
- django_cfg/management/commands/check_settings.py +6 -0
- django_cfg/management/commands/clear_constance.py +5 -2
- django_cfg/management/commands/create_token.py +6 -0
- django_cfg/management/commands/list_urls.py +6 -0
- django_cfg/management/commands/migrate_all.py +6 -0
- django_cfg/management/commands/migrator.py +3 -0
- django_cfg/management/commands/rundramatiq.py +6 -0
- django_cfg/management/commands/runserver_ngrok.py +51 -29
- django_cfg/management/commands/script.py +6 -0
- django_cfg/management/commands/show_config.py +12 -2
- django_cfg/management/commands/show_urls.py +4 -0
- django_cfg/management/commands/superuser.py +6 -0
- django_cfg/management/commands/task_clear.py +4 -1
- django_cfg/management/commands/task_status.py +3 -1
- django_cfg/management/commands/test_email.py +3 -0
- django_cfg/management/commands/test_telegram.py +6 -0
- django_cfg/management/commands/test_twilio.py +6 -0
- django_cfg/management/commands/tree.py +6 -0
- django_cfg/management/commands/validate_config.py +155 -149
- django_cfg/models/constance.py +31 -11
- django_cfg/models/payments.py +175 -498
- django_cfg/modules/django_currency/__init__.py +16 -11
- django_cfg/modules/django_currency/clients/__init__.py +4 -4
- django_cfg/modules/django_currency/clients/coinpaprika_client.py +289 -0
- django_cfg/modules/django_currency/clients/yahoo_client.py +157 -0
- django_cfg/modules/django_currency/core/__init__.py +1 -7
- django_cfg/modules/django_currency/core/converter.py +18 -23
- django_cfg/modules/django_currency/core/models.py +122 -11
- django_cfg/modules/django_currency/database/__init__.py +4 -4
- django_cfg/modules/django_currency/database/database_loader.py +190 -309
- django_cfg/modules/django_logger.py +160 -146
- django_cfg/modules/django_unfold/dashboard.py +65 -12
- django_cfg/registry/core.py +1 -0
- django_cfg/template_archive/django_sample.zip +0 -0
- django_cfg/templates/admin/components/action_grid.html +9 -9
- django_cfg/templates/admin/components/metric_card.html +5 -5
- django_cfg/templates/admin/components/status_badge.html +2 -2
- django_cfg/templates/admin/layouts/dashboard_with_tabs.html +152 -24
- django_cfg/templates/admin/snippets/components/quick_actions.html +3 -3
- django_cfg/templates/admin/snippets/components/system_health.html +1 -1
- django_cfg/templates/admin/snippets/tabs/overview_tab.html +49 -52
- django_cfg/utils/smart_defaults.py +222 -571
- django_cfg/utils/toolkit.py +51 -11
- {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/METADATA +5 -4
- {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/RECORD +172 -182
- django_cfg/apps/payments/__init__.py +0 -8
- django_cfg/apps/payments/admin/tariffs_admin.py +0 -199
- django_cfg/apps/payments/config/module.py +0 -70
- django_cfg/apps/payments/config/providers.py +0 -105
- django_cfg/apps/payments/config/settings.py +0 -96
- django_cfg/apps/payments/config/utils.py +0 -52
- django_cfg/apps/payments/decorators.py +0 -291
- django_cfg/apps/payments/management/commands/README.md +0 -178
- django_cfg/apps/payments/management/commands/currency_stats.py +0 -323
- django_cfg/apps/payments/management/commands/populate_currencies.py +0 -246
- django_cfg/apps/payments/management/commands/update_currencies.py +0 -336
- django_cfg/apps/payments/managers/__init__.py +0 -22
- django_cfg/apps/payments/managers/api_key_manager.py +0 -35
- django_cfg/apps/payments/managers/balance_manager.py +0 -361
- django_cfg/apps/payments/managers/currency_manager.py +0 -83
- django_cfg/apps/payments/managers/payment_manager.py +0 -44
- django_cfg/apps/payments/managers/subscription_manager.py +0 -37
- django_cfg/apps/payments/managers/tariff_manager.py +0 -29
- django_cfg/apps/payments/models/events.py +0 -73
- django_cfg/apps/payments/serializers/__init__.py +0 -56
- django_cfg/apps/payments/serializers/api_keys.py +0 -51
- django_cfg/apps/payments/serializers/balance.py +0 -59
- django_cfg/apps/payments/serializers/currencies.py +0 -55
- django_cfg/apps/payments/serializers/payments.py +0 -62
- django_cfg/apps/payments/serializers/subscriptions.py +0 -71
- django_cfg/apps/payments/serializers/tariffs.py +0 -56
- django_cfg/apps/payments/services/billing/__init__.py +0 -8
- django_cfg/apps/payments/services/cache/base.py +0 -30
- django_cfg/apps/payments/services/core/fallback_service.py +0 -432
- django_cfg/apps/payments/services/internal_types.py +0 -297
- django_cfg/apps/payments/services/middleware/__init__.py +0 -8
- django_cfg/apps/payments/services/monitoring/__init__.py +0 -22
- django_cfg/apps/payments/services/monitoring/api_schemas.py +0 -222
- django_cfg/apps/payments/services/monitoring/provider_health.py +0 -372
- django_cfg/apps/payments/services/providers/cryptapi.py +0 -273
- django_cfg/apps/payments/services/providers/cryptomus.py +0 -311
- django_cfg/apps/payments/services/security/__init__.py +0 -34
- django_cfg/apps/payments/services/security/error_handler.py +0 -637
- django_cfg/apps/payments/services/security/payment_notifications.py +0 -342
- django_cfg/apps/payments/services/security/webhook_validator.py +0 -475
- django_cfg/apps/payments/services/validators/__init__.py +0 -8
- django_cfg/apps/payments/static/payments/css/payments.css +0 -340
- django_cfg/apps/payments/static/payments/js/notifications.js +0 -202
- django_cfg/apps/payments/static/payments/js/payment-utils.js +0 -318
- django_cfg/apps/payments/static/payments/js/theme.js +0 -86
- django_cfg/apps/payments/tasks/__init__.py +0 -12
- django_cfg/apps/payments/tasks/webhook_processing.py +0 -177
- django_cfg/apps/payments/templates/payments/base.html +0 -182
- django_cfg/apps/payments/templates/payments/components/payment_card.html +0 -201
- django_cfg/apps/payments/templates/payments/components/payment_qr_code.html +0 -109
- django_cfg/apps/payments/templates/payments/components/progress_bar.html +0 -36
- django_cfg/apps/payments/templates/payments/components/provider_stats.html +0 -40
- django_cfg/apps/payments/templates/payments/components/status_badge.html +0 -27
- django_cfg/apps/payments/templates/payments/components/status_overview.html +0 -144
- django_cfg/apps/payments/templates/payments/dashboard.html +0 -346
- django_cfg/apps/payments/templatetags/payments_tags.py +0 -315
- django_cfg/apps/payments/urls_templates.py +0 -52
- django_cfg/apps/payments/utils/__init__.py +0 -45
- django_cfg/apps/payments/utils/billing_utils.py +0 -342
- django_cfg/apps/payments/utils/config_utils.py +0 -245
- django_cfg/apps/payments/utils/middleware_utils.py +0 -228
- django_cfg/apps/payments/utils/validation_utils.py +0 -94
- django_cfg/apps/payments/views/__init__.py +0 -62
- django_cfg/apps/payments/views/api_key_views.py +0 -164
- django_cfg/apps/payments/views/balance_views.py +0 -75
- django_cfg/apps/payments/views/currency_views.py +0 -111
- django_cfg/apps/payments/views/payment_views.py +0 -149
- django_cfg/apps/payments/views/subscription_views.py +0 -135
- django_cfg/apps/payments/views/tariff_views.py +0 -131
- django_cfg/apps/payments/views/templates/__init__.py +0 -25
- django_cfg/apps/payments/views/templates/ajax.py +0 -312
- django_cfg/apps/payments/views/templates/base.py +0 -204
- django_cfg/apps/payments/views/templates/dashboard.py +0 -60
- django_cfg/apps/payments/views/templates/payment_detail.py +0 -102
- django_cfg/apps/payments/views/templates/payment_management.py +0 -164
- django_cfg/apps/payments/views/templates/qr_code.py +0 -174
- django_cfg/apps/payments/views/templates/stats.py +0 -240
- django_cfg/apps/payments/views/templates/utils.py +0 -181
- django_cfg/apps/payments/views/webhook_views.py +0 -266
- django_cfg/apps/payments/viewsets.py +0 -65
- django_cfg/core/integration.py +0 -160
- django_cfg/modules/django_currency/clients/coingecko_client.py +0 -257
- django_cfg/modules/django_currency/clients/yfinance_client.py +0 -246
- django_cfg/template_archive/.gitignore +0 -1
- django_cfg/template_archive/__init__.py +0 -0
- django_cfg/urls.py +0 -33
- {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/WHEEL +0 -0
- {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/entry_points.txt +0 -0
- {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/licenses/LICENSE +0 -0
@@ -14,13 +14,18 @@ from django.core.exceptions import ValidationError
|
|
14
14
|
from django.conf import settings
|
15
15
|
import questionary
|
16
16
|
from datetime import datetime
|
17
|
+
from django_cfg.modules.django_logger import get_logger
|
18
|
+
|
17
19
|
|
18
20
|
from django_cfg import ConfigToolkit
|
19
21
|
|
20
22
|
User = get_user_model()
|
21
23
|
|
22
24
|
|
25
|
+
logger = get_logger('superuser')
|
26
|
+
|
23
27
|
class Command(BaseCommand):
|
28
|
+
|
24
29
|
help = 'Create a superuser with enhanced validation and configuration'
|
25
30
|
|
26
31
|
def add_arguments(self, parser):
|
@@ -56,6 +61,7 @@ class Command(BaseCommand):
|
|
56
61
|
)
|
57
62
|
|
58
63
|
def handle(self, *args, **options):
|
64
|
+
logger.info("Starting superuser command")
|
59
65
|
if options['interactive'] or not any([options['username'], options['email'], options['password']]):
|
60
66
|
self.create_superuser_interactive()
|
61
67
|
else:
|
@@ -9,8 +9,9 @@ from django.core.management.base import BaseCommand, CommandError
|
|
9
9
|
from django.conf import settings
|
10
10
|
from typing import Any, Optional, List
|
11
11
|
import logging
|
12
|
+
from django_cfg.modules.django_logger import get_logger
|
12
13
|
|
13
|
-
logger =
|
14
|
+
logger = get_logger('task_clear')
|
14
15
|
|
15
16
|
|
16
17
|
class Command(BaseCommand):
|
@@ -50,6 +51,8 @@ class Command(BaseCommand):
|
|
50
51
|
|
51
52
|
def handle(self, *args, **options):
|
52
53
|
"""Handle the command execution."""
|
54
|
+
logger.info("Starting task_clear command")
|
55
|
+
|
53
56
|
try:
|
54
57
|
# Import here to avoid issues if dramatiq is not installed
|
55
58
|
from django_cfg.modules.django_tasks import get_task_service
|
@@ -11,8 +11,9 @@ from django.conf import settings
|
|
11
11
|
from typing import Any, Dict, List
|
12
12
|
import json
|
13
13
|
import logging
|
14
|
+
from django_cfg.modules.django_logger import get_logger
|
14
15
|
|
15
|
-
logger =
|
16
|
+
logger = get_logger('task_status')
|
16
17
|
|
17
18
|
|
18
19
|
class Command(BaseCommand):
|
@@ -45,6 +46,7 @@ class Command(BaseCommand):
|
|
45
46
|
|
46
47
|
def handle(self, *args, **options):
|
47
48
|
"""Handle the command execution."""
|
49
|
+
logger.info("Starting task_status command")
|
48
50
|
try:
|
49
51
|
# Import here to avoid issues if dramatiq is not installed
|
50
52
|
from django_cfg.modules.django_tasks import get_task_service
|
@@ -6,8 +6,10 @@ Tests email sending functionality using django_cfg configuration.
|
|
6
6
|
|
7
7
|
from django.core.management.base import BaseCommand
|
8
8
|
from django.contrib.auth import get_user_model
|
9
|
+
from django_cfg.modules.django_logger import get_logger
|
9
10
|
|
10
11
|
User = get_user_model()
|
12
|
+
logger = get_logger('test_email')
|
11
13
|
|
12
14
|
|
13
15
|
class Command(BaseCommand):
|
@@ -40,6 +42,7 @@ class Command(BaseCommand):
|
|
40
42
|
subject = options["subject"]
|
41
43
|
message = options["message"]
|
42
44
|
|
45
|
+
logger.info(f"Starting email test for {email}")
|
43
46
|
self.stdout.write(f"🚀 Testing email service for {email}")
|
44
47
|
|
45
48
|
# Create test user if not exists
|
@@ -5,10 +5,15 @@ Tests Telegram notification functionality using django_cfg configuration.
|
|
5
5
|
"""
|
6
6
|
|
7
7
|
from django.core.management.base import BaseCommand
|
8
|
+
from django_cfg.modules.django_logger import get_logger
|
8
9
|
|
9
10
|
|
11
|
+
|
12
|
+
logger = get_logger('test_telegram')
|
13
|
+
|
10
14
|
class Command(BaseCommand):
|
11
15
|
"""Command to test Telegram functionality."""
|
16
|
+
|
12
17
|
help = "Test Telegram notification functionality"
|
13
18
|
|
14
19
|
def add_arguments(self, parser):
|
@@ -20,6 +25,7 @@ class Command(BaseCommand):
|
|
20
25
|
)
|
21
26
|
|
22
27
|
def handle(self, *args, **options):
|
28
|
+
logger.info("Starting test_telegram command")
|
23
29
|
message = options["message"]
|
24
30
|
|
25
31
|
self.stdout.write("🚀 Testing Telegram notification service")
|
@@ -5,10 +5,15 @@ Tests Twilio messaging functionality using django_cfg configuration.
|
|
5
5
|
"""
|
6
6
|
|
7
7
|
from django.core.management.base import BaseCommand
|
8
|
+
from django_cfg.modules.django_logger import get_logger
|
8
9
|
|
9
10
|
|
11
|
+
|
12
|
+
logger = get_logger('test_twilio')
|
13
|
+
|
10
14
|
class Command(BaseCommand):
|
11
15
|
"""Command to test Twilio functionality."""
|
16
|
+
|
12
17
|
help = "Test Twilio messaging functionality"
|
13
18
|
|
14
19
|
def add_arguments(self, parser):
|
@@ -36,6 +41,7 @@ class Command(BaseCommand):
|
|
36
41
|
)
|
37
42
|
|
38
43
|
def handle(self, *args, **options):
|
44
|
+
logger.info("Starting test_twilio command")
|
39
45
|
to_number = options["to"]
|
40
46
|
message = options["message"]
|
41
47
|
is_whatsapp = options["whatsapp"]
|
@@ -11,14 +11,19 @@ from typing import List, Optional
|
|
11
11
|
|
12
12
|
from django.core.management.base import BaseCommand, CommandError
|
13
13
|
from django.conf import settings
|
14
|
+
from django_cfg.modules.django_logger import get_logger
|
15
|
+
|
14
16
|
|
15
17
|
from django_cfg.core.config import get_current_config
|
16
18
|
from django_cfg.utils.path_resolution import PathResolver
|
17
19
|
|
18
20
|
|
21
|
+
logger = get_logger('tree')
|
22
|
+
|
19
23
|
class Command(BaseCommand):
|
20
24
|
"""Display Django project structure in tree format."""
|
21
25
|
|
26
|
+
|
22
27
|
help = "Display Django project structure based on django-cfg configuration"
|
23
28
|
|
24
29
|
def add_arguments(self, parser):
|
@@ -78,6 +83,7 @@ class Command(BaseCommand):
|
|
78
83
|
|
79
84
|
def handle(self, *args, **options):
|
80
85
|
"""Execute the command."""
|
86
|
+
logger.info("Starting tree command")
|
81
87
|
try:
|
82
88
|
# Get django-cfg configuration
|
83
89
|
config = get_current_config()
|
@@ -1,183 +1,189 @@
|
|
1
|
-
"""
|
2
|
-
Django management command to validate configuration.
|
1
|
+
# """
|
2
|
+
# Django management command to validate configuration.
|
3
3
|
|
4
|
-
Usage:
|
5
|
-
|
6
|
-
"""
|
4
|
+
# Usage:
|
5
|
+
# python manage.py validate_config
|
6
|
+
# """
|
7
7
|
|
8
|
-
from django.core.management.base import BaseCommand, CommandError
|
9
|
-
from django.conf import settings
|
10
|
-
from
|
8
|
+
# from django.core.management.base import BaseCommand, CommandError
|
9
|
+
# from django.conf import settings
|
10
|
+
# from django.core.cache import cache
|
11
11
|
|
12
12
|
|
13
|
-
class Command(BaseCommand):
|
14
|
-
help = 'Validate Django Config Toolkit configuration'
|
15
13
|
|
16
|
-
|
17
|
-
|
18
|
-
'--show-details',
|
19
|
-
action='store_true',
|
20
|
-
help='Show detailed configuration information',
|
21
|
-
)
|
22
|
-
parser.add_argument(
|
23
|
-
'--check-connections',
|
24
|
-
action='store_true',
|
25
|
-
help='Test database and cache connections',
|
26
|
-
)
|
14
|
+
# class Command(BaseCommand):
|
15
|
+
# help = 'Validate Django Config Toolkit configuration'
|
27
16
|
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
17
|
+
# def add_arguments(self, parser):
|
18
|
+
# parser.add_argument(
|
19
|
+
# '--show-details',
|
20
|
+
# action='store_true',
|
21
|
+
# help='Show detailed configuration information',
|
22
|
+
# )
|
23
|
+
# parser.add_argument(
|
24
|
+
# '--check-connections',
|
25
|
+
# action='store_true',
|
26
|
+
# help='Test database and cache connections',
|
27
|
+
# )
|
28
|
+
|
29
|
+
# def handle(self, *args, **options):
|
30
|
+
# """Validate configuration and optionally show details."""
|
31
|
+
# self.stdout.write(
|
32
|
+
# self.style.HTTP_INFO('🚀 Django Config Toolkit - Configuration Validation')
|
33
|
+
# )
|
34
|
+
# self.stdout.write('=' * 60)
|
34
35
|
|
35
|
-
|
36
|
-
|
37
|
-
|
36
|
+
# try:
|
37
|
+
# # Initialize toolkit
|
38
|
+
# toolkit = ConfigToolkit()
|
38
39
|
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
40
|
+
# # Basic validation
|
41
|
+
# self.stdout.write(
|
42
|
+
# self.style.SUCCESS(f'✅ Configuration loaded successfully')
|
43
|
+
# )
|
44
|
+
# self.stdout.write(f' Environment: {toolkit.environment}')
|
45
|
+
# self.stdout.write(f' Debug: {toolkit.debug}')
|
46
|
+
# self.stdout.write(f' Configs loaded: {toolkit._config_count}')
|
47
|
+
# self.stdout.write(f' Init time: {toolkit._init_time_ms:.2f}ms')
|
47
48
|
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
49
|
+
# # Check secret key
|
50
|
+
# if len(toolkit.secret_key) >= 50:
|
51
|
+
# self.stdout.write(self.style.SUCCESS('✅ Secret key is secure'))
|
52
|
+
# else:
|
53
|
+
# self.stdout.write(
|
54
|
+
# self.style.WARNING('⚠️ Secret key is too short (< 50 chars)')
|
55
|
+
# )
|
55
56
|
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
57
|
+
# # Check extended features
|
58
|
+
# self.stdout.write('\n🎨 Extended Features:')
|
59
|
+
# features = [
|
60
|
+
# ('Unfold Admin', toolkit.unfold_enabled),
|
61
|
+
# ('Revolution API', toolkit.revolution_enabled),
|
62
|
+
# ('Constance Settings', toolkit.constance_enabled),
|
63
|
+
# ('Advanced Logging', toolkit.logging_enabled),
|
64
|
+
# ]
|
64
65
|
|
65
|
-
|
66
|
-
|
67
|
-
|
66
|
+
# for name, enabled in features:
|
67
|
+
# status = '✅' if enabled else '❌'
|
68
|
+
# self.stdout.write(f' {status} {name}')
|
68
69
|
|
69
|
-
|
70
|
-
|
71
|
-
|
70
|
+
# # Detailed information
|
71
|
+
# if options['show_details']:
|
72
|
+
# self._show_detailed_config(toolkit)
|
72
73
|
|
73
|
-
|
74
|
-
|
75
|
-
|
74
|
+
# # Connection tests
|
75
|
+
# if options['check_connections']:
|
76
|
+
# self._test_connections(toolkit)
|
76
77
|
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
78
|
+
# except Exception as e:
|
79
|
+
# self.stdout.write(
|
80
|
+
# self.style.ERROR(f'❌ Configuration validation failed: {e}')
|
81
|
+
# )
|
82
|
+
# raise CommandError(f'Configuration error: {e}')
|
82
83
|
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
84
|
+
# self.stdout.write('=' * 60)
|
85
|
+
# self.stdout.write(
|
86
|
+
# self.style.SUCCESS('✨ Configuration validation completed!')
|
87
|
+
# )
|
87
88
|
|
88
|
-
|
89
|
-
|
90
|
-
|
89
|
+
# def _show_detailed_config(self, toolkit):
|
90
|
+
# """Show detailed configuration information."""
|
91
|
+
# self.stdout.write('\n📋 Detailed Configuration:')
|
92
|
+
|
93
|
+
# # Environment configuration
|
94
|
+
# self.stdout.write('\n🌍 Environment:')
|
95
|
+
# if hasattr(toolkit._config, 'env_mode'):
|
96
|
+
# self.stdout.write(f' Mode: {toolkit._config.env_mode}')
|
97
|
+
# self.stdout.write(f' Debug: {toolkit._config.debug}')
|
91
98
|
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
99
|
+
# # Database configuration
|
100
|
+
# self.stdout.write('\n🗄️ Database:')
|
101
|
+
# self.stdout.write(f' Engine: {toolkit.database_engine}')
|
102
|
+
# self.stdout.write(f' Name: {toolkit.database_name}')
|
103
|
+
# if hasattr(toolkit._db_config, 'has_multiple_databases'):
|
104
|
+
# self.stdout.write(f' Multiple DBs: {toolkit._db_config.has_multiple_databases}')
|
98
105
|
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
106
|
+
# # Security configuration
|
107
|
+
# self.stdout.write('\n🔒 Security:')
|
108
|
+
# self.stdout.write(f' CORS Enabled: {toolkit.cors_enabled}')
|
109
|
+
# self.stdout.write(f' CSRF Enabled: {toolkit.csrf_enabled}')
|
110
|
+
# self.stdout.write(f' SSL Redirect: {toolkit.ssl_enabled}')
|
104
111
|
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
|
112
|
+
# # API configuration
|
113
|
+
# self.stdout.write('\n🌐 API:')
|
114
|
+
# self.stdout.write(f' Page Size: {toolkit.api_page_size}')
|
115
|
+
# self.stdout.write(f' Rate Limiting: {toolkit.api_rate_limit_enabled}')
|
109
116
|
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
117
|
+
# # Cache configuration
|
118
|
+
# self.stdout.write('\n💾 Cache:')
|
119
|
+
# self.stdout.write(f' Backend: {toolkit.cache_backend}')
|
120
|
+
# self.stdout.write(f' Timeout: {toolkit.cache_timeout}s')
|
114
121
|
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
122
|
+
# # Extended features details
|
123
|
+
# if toolkit.unfold_enabled:
|
124
|
+
# self.stdout.write('\n🎨 Unfold:')
|
125
|
+
# self.stdout.write(f' Site Title: {toolkit.site_title}')
|
119
126
|
|
120
|
-
|
121
|
-
|
122
|
-
|
127
|
+
# if toolkit.revolution_enabled:
|
128
|
+
# self.stdout.write('\n🚀 Revolution:')
|
129
|
+
# self.stdout.write(f' API Prefix: {toolkit.api_prefix}')
|
123
130
|
|
124
|
-
|
125
|
-
|
126
|
-
|
131
|
+
# if toolkit.constance_enabled:
|
132
|
+
# self.stdout.write('\n⚙️ Constance:')
|
133
|
+
# self.stdout.write(f' Backend: {toolkit.constance_backend}')
|
127
134
|
|
128
|
-
|
129
|
-
|
135
|
+
# self.stdout.write('\n📝 Logging:')
|
136
|
+
# self.stdout.write(f' Log Level: {toolkit.log_level}')
|
130
137
|
|
131
|
-
|
132
|
-
|
133
|
-
|
138
|
+
# def _test_connections(self, toolkit):
|
139
|
+
# """Test database and cache connections."""
|
140
|
+
# self.stdout.write('\n🔗 Connection Tests:')
|
134
141
|
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
142
|
+
# # Test database connections
|
143
|
+
# self.stdout.write('\n🗄️ Database connections:')
|
144
|
+
# try:
|
145
|
+
# from django.db import connections
|
139
146
|
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
147
|
+
# for db_name in connections:
|
148
|
+
# try:
|
149
|
+
# connection = connections[db_name]
|
150
|
+
# with connection.cursor() as cursor:
|
151
|
+
# cursor.execute("SELECT 1")
|
152
|
+
# cursor.fetchone()
|
146
153
|
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
+
# self.stdout.write(
|
155
|
+
# self.style.SUCCESS(f' ✅ {db_name}: Connected')
|
156
|
+
# )
|
157
|
+
# except Exception as e:
|
158
|
+
# self.stdout.write(
|
159
|
+
# self.style.ERROR(f' ❌ {db_name}: {str(e)}')
|
160
|
+
# )
|
154
161
|
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
162
|
+
# except Exception as e:
|
163
|
+
# self.stdout.write(
|
164
|
+
# self.style.ERROR(f' ❌ Database test failed: {e}')
|
165
|
+
# )
|
159
166
|
|
160
|
-
|
161
|
-
|
162
|
-
|
163
|
-
from django.core.cache import cache
|
167
|
+
# # Test cache connection
|
168
|
+
# self.stdout.write('\n💾 Cache connection:')
|
169
|
+
# try:
|
164
170
|
|
165
|
-
|
166
|
-
|
171
|
+
# test_key = 'config_validation_test'
|
172
|
+
# test_value = 'test_value'
|
167
173
|
|
168
|
-
|
169
|
-
|
174
|
+
# cache.set(test_key, test_value, 30)
|
175
|
+
# retrieved_value = cache.get(test_key)
|
170
176
|
|
171
|
-
|
172
|
-
|
173
|
-
|
174
|
-
|
175
|
-
|
176
|
-
|
177
|
-
|
178
|
-
|
177
|
+
# if retrieved_value == test_value:
|
178
|
+
# self.stdout.write(
|
179
|
+
# self.style.SUCCESS(' ✅ Cache: Working')
|
180
|
+
# )
|
181
|
+
# else:
|
182
|
+
# self.stdout.write(
|
183
|
+
# self.style.WARNING(' ⚠️ Cache: Read/write test failed')
|
184
|
+
# )
|
179
185
|
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
186
|
+
# except Exception as e:
|
187
|
+
# self.stdout.write(
|
188
|
+
# self.style.ERROR(f' ❌ Cache test failed: {e}')
|
189
|
+
# )
|
django_cfg/models/constance.py
CHANGED
@@ -7,7 +7,7 @@ Unfold admin integration and smart field grouping.
|
|
7
7
|
|
8
8
|
from typing import Dict, List, Optional, Any, Union, Literal
|
9
9
|
from pydantic import BaseModel, Field, field_validator
|
10
|
-
|
10
|
+
import traceback
|
11
11
|
from django_cfg.models.cfg import BaseCfgAutoModule
|
12
12
|
|
13
13
|
|
@@ -150,14 +150,30 @@ class ConstanceConfig(BaseModel, BaseCfgAutoModule):
|
|
150
150
|
default_factory=list,
|
151
151
|
description="List of Constance fields",
|
152
152
|
)
|
153
|
-
|
153
|
+
|
154
|
+
# Cache for app fields to avoid multiple calls
|
155
|
+
_app_fields_cache: Optional[List[ConstanceField]] = None
|
156
|
+
|
154
157
|
def _get_app_constance_fields(self) -> List[ConstanceField]:
|
155
158
|
"""Automatically collect constance fields from django-cfg apps."""
|
159
|
+
# Return cached result if available
|
160
|
+
if self._app_fields_cache is not None:
|
161
|
+
return self._app_fields_cache
|
162
|
+
|
156
163
|
app_fields = []
|
157
164
|
config = self.get_config()
|
165
|
+
|
166
|
+
# Get fields from tasks app (only if knowbase or agents are enabled)
|
167
|
+
if config and config.should_enable_tasks():
|
168
|
+
try:
|
169
|
+
from django_cfg.modules.django_tasks import extend_constance_config_with_tasks
|
170
|
+
tasks_fields = extend_constance_config_with_tasks()
|
171
|
+
app_fields.extend(tasks_fields)
|
172
|
+
except (ImportError, Exception):
|
173
|
+
pass
|
158
174
|
|
159
175
|
# Get fields from knowbase app (only if enabled)
|
160
|
-
if config and
|
176
|
+
if config and config.enable_knowbase:
|
161
177
|
try:
|
162
178
|
from django_cfg.apps.knowbase.config import get_django_cfg_knowbase_constance_fields
|
163
179
|
knowbase_fields = get_django_cfg_knowbase_constance_fields()
|
@@ -165,15 +181,19 @@ class ConstanceConfig(BaseModel, BaseCfgAutoModule):
|
|
165
181
|
except (ImportError, Exception):
|
166
182
|
pass
|
167
183
|
|
168
|
-
# Get fields from
|
169
|
-
if config and
|
184
|
+
# Get fields from payments app (only if enabled)
|
185
|
+
if config and config.payments and config.payments.enabled:
|
170
186
|
try:
|
171
|
-
from django_cfg.
|
172
|
-
|
173
|
-
app_fields.extend(
|
174
|
-
|
175
|
-
|
176
|
-
|
187
|
+
from django_cfg.apps.payments.config import get_django_cfg_payments_constance_fields
|
188
|
+
payments_fields = get_django_cfg_payments_constance_fields()
|
189
|
+
app_fields.extend(payments_fields)
|
190
|
+
print(f"✅ Added {len(payments_fields)} payments fields to Constance")
|
191
|
+
except (ImportError, Exception) as e:
|
192
|
+
print(f"❌ Failed to load payments constance fields: {e}")
|
193
|
+
traceback.print_exc()
|
194
|
+
|
195
|
+
# Cache the result
|
196
|
+
self._app_fields_cache = app_fields
|
177
197
|
return app_fields
|
178
198
|
|
179
199
|
def get_all_fields(self) -> List[ConstanceField]:
|