django-cfg 1.4.20__py3-none-any.whl → 1.4.23__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 +4 -4
- django_cfg/apps/accounts/__init__.py +1 -1
- django_cfg/apps/accounts/__models.py +30 -29
- django_cfg/apps/accounts/admin/__init__.py +14 -3
- django_cfg/apps/accounts/admin/activity_admin.py +19 -19
- django_cfg/apps/accounts/admin/filters.py +7 -6
- django_cfg/apps/accounts/admin/group_admin.py +10 -16
- django_cfg/apps/accounts/admin/inlines.py +21 -19
- django_cfg/apps/accounts/admin/otp_admin.py +12 -12
- django_cfg/apps/accounts/admin/registration_admin.py +24 -24
- django_cfg/apps/accounts/admin/resources.py +49 -48
- django_cfg/apps/accounts/admin/twilio_admin.py +37 -37
- django_cfg/apps/accounts/admin/user_admin.py +45 -41
- django_cfg/apps/accounts/management/commands/otp_test.py +11 -11
- django_cfg/apps/accounts/managers/__init__.py +1 -1
- django_cfg/apps/accounts/managers/user_manager.py +6 -7
- django_cfg/apps/accounts/migrations/0001_initial.py +3 -2
- django_cfg/apps/accounts/models/__init__.py +15 -16
- django_cfg/apps/accounts/models/activity.py +5 -5
- django_cfg/apps/accounts/models/integrations.py +15 -15
- django_cfg/apps/accounts/models/registration.py +3 -2
- django_cfg/apps/accounts/models/user.py +3 -2
- django_cfg/apps/accounts/serializers/__init__.py +10 -3
- django_cfg/apps/accounts/serializers/otp.py +5 -4
- django_cfg/apps/accounts/serializers/profile.py +4 -5
- django_cfg/apps/accounts/serializers/webhook.py +13 -13
- django_cfg/apps/accounts/services/activity_service.py +13 -12
- django_cfg/apps/accounts/services/otp_service.py +16 -14
- django_cfg/apps/accounts/signals.py +18 -16
- django_cfg/apps/accounts/urls.py +10 -5
- django_cfg/apps/accounts/utils/notifications.py +22 -22
- django_cfg/apps/accounts/views/__init__.py +1 -1
- django_cfg/apps/accounts/views/otp.py +15 -14
- django_cfg/apps/accounts/views/profile.py +14 -10
- django_cfg/apps/accounts/views/webhook.py +46 -48
- django_cfg/apps/agents/__init__.py +3 -3
- django_cfg/apps/agents/admin/__init__.py +1 -1
- django_cfg/apps/agents/admin/execution_admin.py +68 -72
- django_cfg/apps/agents/admin/registry_admin.py +50 -54
- django_cfg/apps/agents/admin/toolsets_admin.py +63 -67
- django_cfg/apps/agents/apps.py +3 -3
- django_cfg/apps/agents/core/__init__.py +4 -4
- django_cfg/apps/agents/core/dependencies.py +24 -23
- django_cfg/apps/agents/core/django_agent.py +41 -39
- django_cfg/apps/agents/core/exceptions.py +12 -12
- django_cfg/apps/agents/core/models.py +16 -15
- django_cfg/apps/agents/core/orchestrator.py +58 -58
- django_cfg/apps/agents/examples/simple_example.py +24 -19
- django_cfg/apps/agents/integration/__init__.py +1 -1
- django_cfg/apps/agents/integration/middleware.py +12 -11
- django_cfg/apps/agents/integration/registry.py +52 -49
- django_cfg/apps/agents/integration/signals.py +5 -5
- django_cfg/apps/agents/management/commands/create_agent.py +39 -38
- django_cfg/apps/agents/management/commands/orchestrator_status.py +40 -39
- django_cfg/apps/agents/managers/__init__.py +8 -3
- django_cfg/apps/agents/managers/execution.py +42 -41
- django_cfg/apps/agents/managers/registry.py +40 -39
- django_cfg/apps/agents/managers/toolsets.py +87 -86
- django_cfg/apps/agents/migrations/0001_initial.py +2 -1
- django_cfg/apps/agents/models/__init__.py +2 -2
- django_cfg/apps/agents/models/execution.py +43 -42
- django_cfg/apps/agents/models/registry.py +46 -45
- django_cfg/apps/agents/models/toolsets.py +63 -62
- django_cfg/apps/agents/patterns/__init__.py +5 -5
- django_cfg/apps/agents/patterns/content_agents.py +17 -16
- django_cfg/apps/agents/toolsets/__init__.py +3 -3
- django_cfg/apps/agents/toolsets/cache_toolset.py +56 -55
- django_cfg/apps/agents/toolsets/django_toolset.py +43 -42
- django_cfg/apps/agents/toolsets/file_toolset.py +64 -63
- django_cfg/apps/agents/toolsets/orm_toolset.py +75 -74
- django_cfg/apps/agents/urls.py +3 -2
- django_cfg/apps/api/commands/urls.py +1 -0
- django_cfg/apps/api/commands/views.py +23 -26
- django_cfg/apps/api/endpoints/checker.py +5 -4
- django_cfg/apps/api/endpoints/drf_views.py +2 -2
- django_cfg/apps/api/endpoints/tests.py +6 -5
- django_cfg/apps/api/endpoints/urls.py +2 -1
- django_cfg/apps/api/endpoints/views.py +1 -0
- django_cfg/apps/api/health/drf_views.py +6 -6
- django_cfg/apps/api/health/urls.py +2 -1
- django_cfg/apps/api/health/views.py +41 -41
- django_cfg/{modules/django_ipc_client → apps/ipc}/__init__.py +6 -6
- django_cfg/apps/ipc/apps.py +28 -0
- django_cfg/apps/ipc/serializers/__init__.py +19 -0
- django_cfg/apps/ipc/serializers/serializers.py +229 -0
- django_cfg/apps/ipc/services/__init__.py +7 -0
- django_cfg/apps/ipc/services/client/__init__.py +23 -0
- django_cfg/{modules/django_ipc_client → apps/ipc/services/client}/client.py +7 -6
- django_cfg/{modules/django_ipc_client → apps/ipc/services/client}/exceptions.py +1 -1
- django_cfg/{modules/django_ipc_client/dashboard → apps/ipc/services}/monitor.py +23 -5
- django_cfg/{modules/django_ipc_client/dashboard/static/django_ipc_dashboard/js/dashboard.js → apps/ipc/static/django_cfg_ipc/js/dashboard.mjs} +131 -63
- django_cfg/{modules/django_ipc_client/dashboard/templates/django_ipc_dashboard → apps/ipc/templates/django_cfg_ipc}/base.html +5 -10
- django_cfg/apps/ipc/templates/django_cfg_ipc/dashboard.html +202 -0
- django_cfg/apps/ipc/urls.py +21 -0
- django_cfg/apps/ipc/urls_admin.py +20 -0
- django_cfg/apps/ipc/views/__init__.py +8 -0
- django_cfg/apps/ipc/views/dashboard.py +15 -0
- django_cfg/apps/ipc/views/viewsets.py +245 -0
- django_cfg/apps/knowbase/admin/__init__.py +2 -2
- django_cfg/apps/knowbase/admin/actions/__init__.py +1 -1
- django_cfg/apps/knowbase/admin/actions/visibility_actions.py +2 -1
- django_cfg/apps/knowbase/admin/archive_admin.py +81 -84
- django_cfg/apps/knowbase/admin/chat_admin.py +70 -72
- django_cfg/apps/knowbase/admin/document_admin.py +10 -11
- django_cfg/apps/knowbase/admin/external_data_admin.py +69 -71
- django_cfg/apps/knowbase/admin/helpers/__init__.py +1 -1
- django_cfg/apps/knowbase/admin/helpers/configs.py +2 -2
- django_cfg/apps/knowbase/admin/helpers/statistics.py +1 -1
- django_cfg/apps/knowbase/apps.py +13 -13
- django_cfg/apps/knowbase/config/__init__.py +7 -6
- django_cfg/apps/knowbase/config/constance_fields.py +14 -12
- django_cfg/apps/knowbase/config/constance_settings.py +32 -31
- django_cfg/apps/knowbase/config/settings.py +28 -28
- django_cfg/apps/knowbase/examples/external_data_usage.py +35 -32
- django_cfg/apps/knowbase/management/commands/knowbase_stats.py +33 -32
- django_cfg/apps/knowbase/management/commands/setup_knowbase.py +11 -13
- django_cfg/apps/knowbase/managers/__init__.py +2 -2
- django_cfg/apps/knowbase/managers/archive.py +86 -85
- django_cfg/apps/knowbase/managers/base.py +5 -5
- django_cfg/apps/knowbase/managers/chat.py +29 -28
- django_cfg/apps/knowbase/managers/document.py +39 -39
- django_cfg/apps/knowbase/managers/external_data.py +74 -73
- django_cfg/apps/knowbase/migrations/0001_initial.py +2 -1
- django_cfg/apps/knowbase/migrations/0002_archiveitem_archiveitemchunk_documentarchive_and_more.py +2 -1
- django_cfg/apps/knowbase/migrations/0003_alter_documentarchive_archive_type.py +29 -0
- django_cfg/apps/knowbase/mixins/__init__.py +4 -4
- django_cfg/apps/knowbase/mixins/config/__init__.py +1 -1
- django_cfg/apps/knowbase/mixins/config/meta_config.py +1 -1
- django_cfg/apps/knowbase/mixins/config.py +19 -18
- django_cfg/apps/knowbase/mixins/creator.py +7 -7
- django_cfg/apps/knowbase/mixins/examples/vehicle_model_example.py +29 -28
- django_cfg/apps/knowbase/mixins/external_data_mixin.py +6 -5
- django_cfg/apps/knowbase/mixins/generators/__init__.py +1 -1
- django_cfg/apps/knowbase/mixins/generators/content_generator.py +2 -2
- django_cfg/apps/knowbase/mixins/service.py +47 -45
- django_cfg/apps/knowbase/models/__init__.py +7 -7
- django_cfg/apps/knowbase/models/archive.py +72 -72
- django_cfg/apps/knowbase/models/base.py +12 -13
- django_cfg/apps/knowbase/models/chat.py +20 -19
- django_cfg/apps/knowbase/models/document.py +37 -35
- django_cfg/apps/knowbase/models/external_data.py +41 -42
- django_cfg/apps/knowbase/serializers/__init__.py +7 -7
- django_cfg/apps/knowbase/serializers/archive_serializers.py +50 -42
- django_cfg/apps/knowbase/serializers/chat_serializers.py +16 -15
- django_cfg/apps/knowbase/serializers/document_serializers.py +13 -12
- django_cfg/apps/knowbase/serializers/external_data_serializers.py +31 -31
- django_cfg/apps/knowbase/serializers/public_serializers.py +10 -9
- django_cfg/apps/knowbase/services/__init__.py +7 -7
- django_cfg/apps/knowbase/services/archive/__init__.py +7 -7
- django_cfg/apps/knowbase/services/archive/analyzers/__init__.py +1 -1
- django_cfg/apps/knowbase/services/archive/analyzers/tag_generator.py +1 -1
- django_cfg/apps/knowbase/services/archive/archive_service.py +109 -112
- django_cfg/apps/knowbase/services/archive/chunking/__init__.py +3 -3
- django_cfg/apps/knowbase/services/archive/chunking/base.py +1 -0
- django_cfg/apps/knowbase/services/archive/chunking/json_chunker.py +4 -3
- django_cfg/apps/knowbase/services/archive/chunking/markdown_chunker.py +5 -4
- django_cfg/apps/knowbase/services/archive/chunking/python_chunker.py +6 -5
- django_cfg/apps/knowbase/services/archive/chunking/text_chunker.py +4 -3
- django_cfg/apps/knowbase/services/archive/chunking_service.py +3 -7
- django_cfg/apps/knowbase/services/archive/context/__init__.py +1 -1
- django_cfg/apps/knowbase/services/archive/context/builders.py +2 -1
- django_cfg/apps/knowbase/services/archive/context/models.py +2 -1
- django_cfg/apps/knowbase/services/archive/exceptions.py +5 -5
- django_cfg/apps/knowbase/services/archive/extraction_service.py +111 -110
- django_cfg/apps/knowbase/services/archive/vectorization_service.py +80 -77
- django_cfg/apps/knowbase/services/base.py +11 -9
- django_cfg/apps/knowbase/services/chat_service.py +40 -39
- django_cfg/apps/knowbase/services/document_service.py +28 -27
- django_cfg/apps/knowbase/services/embedding/__init__.py +9 -9
- django_cfg/apps/knowbase/services/embedding/async_processor.py +38 -40
- django_cfg/apps/knowbase/services/embedding/batch_processor.py +45 -42
- django_cfg/apps/knowbase/services/embedding/batch_result.py +7 -6
- django_cfg/apps/knowbase/services/embedding/models.py +52 -51
- django_cfg/apps/knowbase/services/embedding/processors.py +24 -23
- django_cfg/apps/knowbase/services/embedding/utils.py +17 -17
- django_cfg/apps/knowbase/services/prompt_builder.py +24 -23
- django_cfg/apps/knowbase/services/search_service.py +52 -49
- django_cfg/apps/knowbase/signals/__init__.py +2 -5
- django_cfg/apps/knowbase/signals/archive_signals.py +35 -34
- django_cfg/apps/knowbase/signals/chat_signals.py +6 -5
- django_cfg/apps/knowbase/signals/document_signals.py +22 -22
- django_cfg/apps/knowbase/signals/external_data_signals.py +22 -22
- django_cfg/apps/knowbase/tasks/__init__.py +6 -7
- django_cfg/apps/knowbase/tasks/archive_tasks.py +41 -41
- django_cfg/apps/knowbase/tasks/document_processing.py +49 -44
- django_cfg/apps/knowbase/tasks/external_data_tasks.py +46 -44
- django_cfg/apps/knowbase/tasks/maintenance.py +26 -24
- django_cfg/apps/knowbase/urls.py +3 -2
- django_cfg/apps/knowbase/urls_admin.py +6 -3
- django_cfg/apps/knowbase/urls_system.py +4 -5
- django_cfg/apps/knowbase/utils/chunk_settings.py +22 -20
- django_cfg/apps/knowbase/utils/text_processing.py +76 -75
- django_cfg/apps/knowbase/utils/validation.py +9 -9
- django_cfg/apps/knowbase/views/__init__.py +5 -5
- django_cfg/apps/knowbase/views/archive_views.py +90 -87
- django_cfg/apps/knowbase/views/base.py +9 -12
- django_cfg/apps/knowbase/views/chat_views.py +35 -32
- django_cfg/apps/knowbase/views/document_views.py +27 -28
- django_cfg/apps/knowbase/views/public_views.py +19 -19
- django_cfg/apps/leads/admin/leads_admin.py +41 -45
- django_cfg/apps/leads/admin/resources.py +14 -14
- django_cfg/apps/leads/admin.py +7 -9
- django_cfg/apps/leads/apps.py +1 -1
- django_cfg/apps/leads/models.py +17 -17
- django_cfg/apps/leads/serializers.py +5 -4
- django_cfg/apps/leads/signals.py +7 -7
- django_cfg/apps/leads/tests.py +47 -47
- django_cfg/apps/leads/urls.py +2 -2
- django_cfg/apps/leads/views.py +11 -11
- django_cfg/apps/maintenance/__init__.py +3 -3
- django_cfg/apps/maintenance/admin/__init__.py +1 -1
- django_cfg/apps/maintenance/admin/api_key_admin.py +36 -36
- django_cfg/apps/maintenance/admin/log_admin.py +35 -37
- django_cfg/apps/maintenance/admin/scheduled_admin.py +47 -51
- django_cfg/apps/maintenance/admin/site_admin.py +49 -52
- django_cfg/apps/maintenance/apps.py +2 -2
- django_cfg/apps/maintenance/management/commands/maintenance.py +53 -52
- django_cfg/apps/maintenance/management/commands/process_scheduled_maintenance.py +44 -44
- django_cfg/apps/maintenance/management/commands/sync_cloudflare.py +32 -32
- django_cfg/apps/maintenance/managers/__init__.py +1 -1
- django_cfg/apps/maintenance/managers/cloudflare_site_manager.py +39 -38
- django_cfg/apps/maintenance/managers/maintenance_log_manager.py +32 -31
- django_cfg/apps/maintenance/migrations/0001_initial.py +1 -0
- django_cfg/apps/maintenance/models/__init__.py +2 -2
- django_cfg/apps/maintenance/models/cloudflare_api_key.py +15 -15
- django_cfg/apps/maintenance/models/cloudflare_site.py +24 -22
- django_cfg/apps/maintenance/models/maintenance_log.py +15 -14
- django_cfg/apps/maintenance/models/scheduled_maintenance.py +53 -52
- django_cfg/apps/maintenance/services/__init__.py +3 -3
- django_cfg/apps/maintenance/services/bulk_operations_service.py +68 -68
- django_cfg/apps/maintenance/services/maintenance_service.py +59 -58
- django_cfg/apps/maintenance/services/scheduled_maintenance_service.py +52 -52
- django_cfg/apps/maintenance/services/site_sync_service.py +64 -65
- django_cfg/apps/maintenance/utils/__init__.py +1 -1
- django_cfg/apps/maintenance/utils/retry_utils.py +17 -17
- django_cfg/apps/newsletter/admin/__init__.py +12 -16
- django_cfg/apps/newsletter/admin/newsletter_admin.py +90 -87
- django_cfg/apps/newsletter/admin/resources.py +29 -29
- django_cfg/apps/newsletter/admin.py +39 -32
- django_cfg/apps/newsletter/management/commands/test_newsletter.py +3 -3
- django_cfg/apps/newsletter/managers/__init__.py +1 -1
- django_cfg/apps/newsletter/migrations/0001_initial.py +2 -1
- django_cfg/apps/newsletter/models.py +34 -33
- django_cfg/apps/newsletter/serializers.py +13 -13
- django_cfg/apps/newsletter/services/email_service.py +42 -40
- django_cfg/apps/newsletter/signals.py +4 -3
- django_cfg/apps/newsletter/urls.py +7 -7
- django_cfg/apps/newsletter/views/__init__.py +14 -19
- django_cfg/apps/newsletter/views/campaigns.py +19 -19
- django_cfg/apps/newsletter/views/emails.py +20 -20
- django_cfg/apps/newsletter/views/newsletters.py +5 -5
- django_cfg/apps/newsletter/views/subscriptions.py +24 -24
- django_cfg/apps/newsletter/views/tracking.py +6 -5
- django_cfg/apps/payments/admin/__init__.py +6 -5
- django_cfg/apps/payments/admin/api_keys_admin.py +43 -45
- django_cfg/apps/payments/admin/balance_admin.py +41 -47
- django_cfg/apps/payments/admin/currencies_admin.py +60 -62
- django_cfg/apps/payments/admin/endpoint_groups_admin.py +14 -28
- django_cfg/apps/payments/admin/filters.py +59 -59
- django_cfg/apps/payments/admin/networks_admin.py +36 -50
- django_cfg/apps/payments/admin/payments_admin.py +47 -49
- django_cfg/apps/payments/admin/subscriptions_admin.py +30 -32
- django_cfg/apps/payments/admin/tariffs_admin.py +37 -42
- django_cfg/apps/payments/admin_interface/serializers/__init__.py +12 -13
- django_cfg/apps/payments/admin_interface/serializers/payment_serializers.py +28 -27
- django_cfg/apps/payments/admin_interface/serializers/webhook_serializers.py +4 -5
- django_cfg/apps/payments/admin_interface/templates/payments/base.html +407 -3
- django_cfg/apps/payments/admin_interface/templates/payments/components/ngrok_status.html +13 -13
- django_cfg/apps/payments/admin_interface/templates/payments/payment_dashboard.html +5 -1
- django_cfg/apps/payments/admin_interface/templates/payments/payment_detail.html +105 -49
- django_cfg/apps/payments/admin_interface/templates/payments/payment_form.html +4 -1
- django_cfg/apps/payments/admin_interface/templates/payments/payment_list.html +16 -8
- django_cfg/apps/payments/admin_interface/templates/payments/webhook_dashboard.html +16 -8
- django_cfg/apps/payments/admin_interface/views/__init__.py +8 -9
- django_cfg/apps/payments/admin_interface/views/api/__init__.py +3 -3
- django_cfg/apps/payments/admin_interface/views/api/payments.py +52 -54
- django_cfg/apps/payments/admin_interface/views/api/stats.py +33 -30
- django_cfg/apps/payments/admin_interface/views/api/users.py +10 -11
- django_cfg/apps/payments/admin_interface/views/api/webhook_admin.py +46 -51
- django_cfg/apps/payments/admin_interface/views/api/webhook_public.py +9 -9
- django_cfg/apps/payments/admin_interface/views/base.py +17 -19
- django_cfg/apps/payments/admin_interface/views/dashboard.py +10 -10
- django_cfg/apps/payments/admin_interface/views/forms.py +21 -20
- django_cfg/apps/payments/apps.py +7 -6
- django_cfg/apps/payments/config/__init__.py +4 -4
- django_cfg/apps/payments/config/django_cfg_integration.py +22 -21
- django_cfg/apps/payments/config/helpers.py +14 -12
- django_cfg/apps/payments/management/commands/cleanup_expired_data.py +85 -86
- django_cfg/apps/payments/management/commands/currency_stats.py +84 -87
- django_cfg/apps/payments/management/commands/manage_currencies.py +59 -58
- django_cfg/apps/payments/management/commands/manage_providers.py +103 -105
- django_cfg/apps/payments/management/commands/process_pending_payments.py +67 -70
- django_cfg/apps/payments/management/commands/test_providers.py +83 -84
- django_cfg/apps/payments/middleware/__init__.py +1 -1
- django_cfg/apps/payments/middleware/api_access.py +77 -78
- django_cfg/apps/payments/middleware/rate_limiting.py +72 -72
- django_cfg/apps/payments/middleware/usage_tracking.py +66 -64
- django_cfg/apps/payments/migrations/0001_initial.py +2 -1
- django_cfg/apps/payments/models/__init__.py +11 -12
- django_cfg/apps/payments/models/api_keys.py +29 -27
- django_cfg/apps/payments/models/balance.py +38 -38
- django_cfg/apps/payments/models/base.py +12 -11
- django_cfg/apps/payments/models/currencies.py +53 -52
- django_cfg/apps/payments/models/managers/__init__.py +13 -14
- django_cfg/apps/payments/models/managers/api_key_managers.py +55 -53
- django_cfg/apps/payments/models/managers/balance_managers.py +98 -97
- django_cfg/apps/payments/models/managers/currency_managers.py +70 -69
- django_cfg/apps/payments/models/managers/payment_managers.py +113 -111
- django_cfg/apps/payments/models/managers/subscription_managers.py +99 -97
- django_cfg/apps/payments/models/payments.py +167 -73
- django_cfg/apps/payments/models/subscriptions.py +56 -54
- django_cfg/apps/payments/models/tariffs.py +35 -34
- django_cfg/apps/payments/services/__init__.py +33 -36
- django_cfg/apps/payments/services/cache/__init__.py +7 -1
- django_cfg/apps/payments/services/cache_service/__init__.py +22 -20
- django_cfg/apps/payments/services/cache_service/api_key_cache.py +6 -5
- django_cfg/apps/payments/services/cache_service/interfaces.py +5 -5
- django_cfg/apps/payments/services/cache_service/keys.py +8 -8
- django_cfg/apps/payments/services/cache_service/rate_limit_cache.py +8 -7
- django_cfg/apps/payments/services/cache_service/simple_cache.py +17 -14
- django_cfg/apps/payments/services/core/__init__.py +3 -3
- django_cfg/apps/payments/services/core/balance_service.py +69 -65
- django_cfg/apps/payments/services/core/base.py +25 -22
- django_cfg/apps/payments/services/core/currency/__init__.py +1 -1
- django_cfg/apps/payments/services/core/currency/currency_converter.py +2 -0
- django_cfg/apps/payments/services/core/currency_service.py +68 -66
- django_cfg/apps/payments/services/core/operations/__init__.py +1 -1
- django_cfg/apps/payments/services/core/operations/payment_canceller.py +2 -1
- django_cfg/apps/payments/services/core/operations/payment_creator.py +2 -1
- django_cfg/apps/payments/services/core/operations/status_checker.py +2 -1
- django_cfg/apps/payments/services/core/payment_service.py +10 -7
- django_cfg/apps/payments/services/core/providers/provider_client.py +2 -2
- django_cfg/apps/payments/services/core/subscription_service.py +77 -74
- django_cfg/apps/payments/services/core/utils/data_converter.py +1 -0
- django_cfg/apps/payments/services/core/utils/statistics_calculator.py +1 -0
- django_cfg/apps/payments/services/core/webhook_service.py +67 -63
- django_cfg/apps/payments/services/integrations/__init__.py +3 -3
- django_cfg/apps/payments/services/integrations/ngrok_service.py +1 -0
- django_cfg/apps/payments/services/integrations/providers_config.py +3 -2
- django_cfg/apps/payments/services/providers/base.py +59 -57
- django_cfg/apps/payments/services/providers/models/__init__.py +7 -14
- django_cfg/apps/payments/services/providers/models/base.py +15 -15
- django_cfg/apps/payments/services/providers/models/providers.py +13 -12
- django_cfg/apps/payments/services/providers/models/universal.py +6 -5
- django_cfg/apps/payments/services/providers/nowpayments/__init__.py +4 -4
- django_cfg/apps/payments/services/providers/nowpayments/config.py +9 -8
- django_cfg/apps/payments/services/providers/nowpayments/models.py +15 -15
- django_cfg/apps/payments/services/providers/nowpayments/parsers/data/__init__.py +2 -2
- django_cfg/apps/payments/services/providers/nowpayments/parsers/data/patterns.py +26 -26
- django_cfg/apps/payments/services/providers/nowpayments/parsers/parser.py +3 -2
- django_cfg/apps/payments/services/providers/nowpayments/provider.py +95 -99
- django_cfg/apps/payments/services/providers/nowpayments/sync.py +41 -40
- django_cfg/apps/payments/services/providers/registry.py +65 -63
- django_cfg/apps/payments/services/providers/sync_service.py +50 -50
- django_cfg/apps/payments/services/types/__init__.py +21 -22
- django_cfg/apps/payments/services/types/data.py +14 -13
- django_cfg/apps/payments/services/types/requests.py +21 -22
- django_cfg/apps/payments/services/types/responses.py +16 -15
- django_cfg/apps/payments/services/types/webhooks.py +30 -30
- django_cfg/apps/payments/signals/__init__.py +4 -6
- django_cfg/apps/payments/signals/api_key_signals.py +36 -35
- django_cfg/apps/payments/signals/balance_signals.py +28 -26
- django_cfg/apps/payments/signals/payment_signals.py +29 -28
- django_cfg/apps/payments/signals/subscription_signals.py +39 -38
- django_cfg/apps/payments/static/payments/js/ngrok-status.js +12 -8
- django_cfg/apps/payments/static/payments/js/payment-detail.js +1 -1
- django_cfg/apps/payments/static/payments/js/payment-form.js +3 -3
- django_cfg/apps/payments/static/payments/js/payment-list.js +13 -6
- django_cfg/apps/payments/static/payments/js/webhook-dashboard-mjs.js +241 -0
- django_cfg/apps/payments/tasks/__init__.py +11 -12
- django_cfg/apps/payments/tasks/types.py +10 -9
- django_cfg/apps/payments/tasks/usage_tracking.py +44 -46
- django_cfg/apps/payments/templatetags/payment_tags.py +27 -27
- django_cfg/apps/payments/urls.py +31 -14
- django_cfg/apps/payments/urls_admin.py +10 -10
- django_cfg/apps/payments/views/api/__init__.py +32 -33
- django_cfg/apps/payments/views/api/api_keys.py +62 -62
- django_cfg/apps/payments/views/api/balances.py +63 -64
- django_cfg/apps/payments/views/api/base.py +52 -52
- django_cfg/apps/payments/views/api/currencies.py +75 -63
- django_cfg/apps/payments/views/api/payments.py +73 -74
- django_cfg/apps/payments/views/api/subscriptions.py +71 -72
- django_cfg/apps/payments/views/api/webhooks.py +85 -84
- django_cfg/apps/payments/views/overview/__init__.py +7 -7
- django_cfg/apps/payments/views/overview/serializers.py +13 -14
- django_cfg/apps/payments/views/overview/services.py +66 -67
- django_cfg/apps/payments/views/overview/urls.py +2 -1
- django_cfg/apps/payments/views/overview/views.py +31 -31
- django_cfg/apps/payments/views/serializers/__init__.py +35 -36
- django_cfg/apps/payments/views/serializers/api_keys.py +59 -57
- django_cfg/apps/payments/views/serializers/balances.py +34 -33
- django_cfg/apps/payments/views/serializers/currencies.py +36 -34
- django_cfg/apps/payments/views/serializers/payments.py +48 -47
- django_cfg/apps/payments/views/serializers/subscriptions.py +50 -45
- django_cfg/apps/payments/views/serializers/webhooks.py +17 -16
- django_cfg/apps/support/admin/__init__.py +3 -3
- django_cfg/apps/support/admin/resources.py +26 -26
- django_cfg/apps/support/admin/support_admin.py +44 -48
- django_cfg/apps/support/admin.py +16 -15
- django_cfg/apps/support/apps.py +1 -1
- django_cfg/apps/support/managers/message_manager.py +4 -4
- django_cfg/apps/support/managers/ticket_manager.py +13 -12
- django_cfg/apps/support/migrations/0001_initial.py +2 -1
- django_cfg/apps/support/models.py +3 -1
- django_cfg/apps/support/serializers.py +4 -2
- django_cfg/apps/support/signals.py +12 -10
- django_cfg/apps/support/urls.py +4 -3
- django_cfg/apps/support/utils/support_email_service.py +11 -9
- django_cfg/apps/support/views/__init__.py +3 -3
- django_cfg/apps/support/views/admin.py +9 -9
- django_cfg/apps/support/views/api.py +12 -13
- django_cfg/apps/support/views/chat.py +14 -14
- django_cfg/apps/tasks/admin/tasks_admin.py +65 -74
- django_cfg/apps/tasks/apps.py +2 -2
- django_cfg/apps/tasks/serializers.py +6 -6
- django_cfg/apps/tasks/static/tasks/js/dashboard/main.mjs +44 -20
- django_cfg/apps/tasks/static/tasks/js/dashboard/overview.mjs +7 -5
- django_cfg/apps/tasks/static/tasks/js/dashboard/queues.mjs +5 -3
- django_cfg/apps/tasks/static/tasks/js/dashboard/tasks.mjs +5 -3
- django_cfg/apps/tasks/static/tasks/js/dashboard/workers.mjs +5 -3
- django_cfg/apps/tasks/tasks/demo_tasks.py +12 -11
- django_cfg/apps/tasks/templates/tasks/components/tasks_mjs_integration.html +269 -0
- django_cfg/apps/tasks/templates/tasks/pages/dashboard-improved.html +168 -0
- django_cfg/apps/tasks/templates/tasks/pages/dashboard.html +21 -2
- django_cfg/apps/tasks/urls.py +3 -2
- django_cfg/apps/tasks/urls_admin.py +1 -0
- django_cfg/apps/tasks/utils/simulator.py +49 -52
- django_cfg/apps/tasks/views/api.py +75 -73
- django_cfg/apps/tasks/views/dashboard.py +5 -4
- django_cfg/apps/urls.py +20 -11
- django_cfg/apps.py +6 -5
- django_cfg/cli/commands/create_project.py +7 -6
- django_cfg/cli/commands/info.py +25 -25
- django_cfg/cli/utils.py +27 -27
- django_cfg/config.py +1 -1
- django_cfg/core/__init__.py +8 -8
- django_cfg/core/base/config_model.py +13 -12
- django_cfg/core/builders/apps_builder.py +2 -2
- django_cfg/core/builders/middleware_builder.py +1 -1
- django_cfg/core/builders/security_builder.py +1 -1
- django_cfg/core/config.py +2 -2
- django_cfg/core/environment/detector.py +27 -28
- django_cfg/core/exceptions.py +1 -1
- django_cfg/core/generation/core_generators/settings.py +1 -1
- django_cfg/core/generation/core_generators/static.py +11 -5
- django_cfg/core/generation/core_generators/templates.py +1 -1
- django_cfg/core/generation/data_generators/__init__.py +1 -1
- django_cfg/core/generation/data_generators/cache.py +1 -1
- django_cfg/core/generation/data_generators/database.py +1 -1
- django_cfg/core/generation/generation.py +1 -3
- django_cfg/core/generation/integration_generators/__init__.py +2 -2
- django_cfg/core/generation/integration_generators/api.py +12 -2
- django_cfg/core/generation/integration_generators/sessions.py +1 -1
- django_cfg/core/generation/integration_generators/tailwind.py +1 -1
- django_cfg/core/generation/integration_generators/tasks.py +1 -1
- django_cfg/core/generation/integration_generators/third_party.py +1 -1
- django_cfg/core/generation/orchestrator.py +1 -1
- django_cfg/core/generation/protocols.py +1 -1
- django_cfg/core/generation/utility_generators/__init__.py +1 -1
- django_cfg/core/generation/utility_generators/email.py +1 -1
- django_cfg/core/generation/utility_generators/i18n.py +1 -1
- django_cfg/core/generation/utility_generators/limits.py +1 -1
- django_cfg/core/generation/utility_generators/logging.py +1 -1
- django_cfg/core/generation/utility_generators/security.py +1 -1
- django_cfg/core/generation/utils/helpers.py +1 -1
- django_cfg/core/integration/__init__.py +5 -5
- django_cfg/core/integration/commands_collector.py +38 -39
- django_cfg/core/integration/display/__init__.py +2 -2
- django_cfg/core/integration/display/base.py +30 -30
- django_cfg/core/integration/display/ngrok.py +35 -36
- django_cfg/core/integration/display/startup.py +149 -139
- django_cfg/core/integration/url_integration.py +10 -10
- django_cfg/core/integration/version_checker.py +20 -19
- django_cfg/core/services/config_service.py +4 -4
- django_cfg/core/state/__init__.py +1 -1
- django_cfg/core/state/registry.py +1 -1
- django_cfg/core/types/__init__.py +8 -1
- django_cfg/core/validation.py +36 -39
- django_cfg/management/commands/check_endpoints.py +3 -1
- django_cfg/management/commands/check_settings.py +3 -1
- django_cfg/management/commands/clear_constance.py +3 -1
- django_cfg/management/commands/create_token.py +3 -1
- django_cfg/management/commands/generate_clients.py +3 -1
- django_cfg/management/commands/migrate_all.py +3 -1
- django_cfg/management/commands/rundramatiq.py +3 -1
- django_cfg/management/commands/rundramatiq_simulator.py +3 -1
- django_cfg/management/commands/runserver_ngrok.py +3 -1
- django_cfg/management/commands/show_config.py +3 -1
- django_cfg/management/commands/superuser.py +3 -1
- django_cfg/management/commands/task_clear.py +3 -1
- django_cfg/management/commands/task_status.py +3 -1
- django_cfg/management/commands/test_email.py +3 -1
- django_cfg/management/commands/test_telegram.py +3 -1
- django_cfg/management/commands/test_twilio.py +3 -1
- django_cfg/management/commands/validate_openapi.py +3 -1
- django_cfg/middleware/pagination.py +8 -8
- django_cfg/middleware/public_endpoints.py +24 -23
- django_cfg/middleware/user_activity.py +27 -25
- django_cfg/models/__init__.py +19 -20
- django_cfg/models/api/__init__.py +4 -4
- django_cfg/models/api/config.py +23 -21
- django_cfg/models/api/cors.py +17 -16
- django_cfg/models/api/drf/__init__.py +1 -1
- django_cfg/models/api/drf/config.py +2 -1
- django_cfg/models/api/drf/redoc.py +2 -1
- django_cfg/models/api/drf/spectacular.py +4 -2
- django_cfg/models/api/drf/swagger.py +2 -1
- django_cfg/models/api/jwt.py +37 -36
- django_cfg/models/api/keys.py +13 -12
- django_cfg/models/api/limits.py +31 -30
- django_cfg/models/base/config.py +40 -41
- django_cfg/models/base/module.py +8 -8
- django_cfg/models/django/__init__.py +1 -1
- django_cfg/models/django/constance.py +8 -7
- django_cfg/models/django/environment.py +5 -3
- django_cfg/models/django/openapi.py +6 -16
- django_cfg/models/django/revolution_legacy.py +17 -16
- django_cfg/models/infrastructure/__init__.py +1 -1
- django_cfg/models/infrastructure/cache.py +46 -45
- django_cfg/models/infrastructure/database/config.py +4 -6
- django_cfg/models/infrastructure/database/converters.py +1 -1
- django_cfg/models/infrastructure/database/parsers.py +1 -1
- django_cfg/models/infrastructure/database/validators.py +1 -1
- django_cfg/models/infrastructure/logging.py +59 -57
- django_cfg/models/infrastructure/security.py +26 -24
- django_cfg/models/ngrok/auth.py +2 -1
- django_cfg/models/ngrok/config.py +3 -2
- django_cfg/models/ngrok/tunnel.py +2 -1
- django_cfg/models/payments/__init__.py +1 -1
- django_cfg/models/payments/api_keys.py +3 -1
- django_cfg/models/payments/config.py +4 -1
- django_cfg/models/payments/providers/base.py +2 -1
- django_cfg/models/payments/providers/nowpayments.py +3 -1
- django_cfg/models/services/__init__.py +1 -1
- django_cfg/models/services/base.py +2 -1
- django_cfg/models/services/email.py +28 -26
- django_cfg/models/services/telegram.py +2 -1
- django_cfg/models/tasks/__init__.py +2 -2
- django_cfg/models/tasks/backends.py +4 -3
- django_cfg/models/tasks/config.py +6 -4
- django_cfg/models/tasks/utils.py +3 -3
- django_cfg/modules/base.py +18 -17
- django_cfg/modules/django_admin/__init__.py +14 -15
- django_cfg/modules/django_admin/decorators/__init__.py +1 -1
- django_cfg/modules/django_admin/decorators/actions.py +8 -7
- django_cfg/modules/django_admin/decorators/display.py +9 -7
- django_cfg/modules/django_admin/icons/__init__.py +1 -1
- django_cfg/modules/django_admin/icons/constants.py +27 -27
- django_cfg/modules/django_admin/icons/generate_icons.py +54 -54
- django_cfg/modules/django_admin/management/commands/check_endpoints.py +5 -3
- django_cfg/modules/django_admin/management/commands/check_settings.py +40 -44
- django_cfg/modules/django_admin/management/commands/clear_constance.py +29 -30
- django_cfg/modules/django_admin/management/commands/create_token.py +42 -42
- django_cfg/modules/django_admin/management/commands/list_urls.py +37 -38
- django_cfg/modules/django_admin/management/commands/migrate_all.py +13 -15
- django_cfg/modules/django_admin/management/commands/migrator.py +17 -17
- django_cfg/modules/django_admin/management/commands/script.py +58 -60
- django_cfg/modules/django_admin/management/commands/show_config.py +32 -30
- django_cfg/modules/django_admin/management/commands/show_urls.py +46 -44
- django_cfg/modules/django_admin/management/commands/superuser.py +47 -48
- django_cfg/modules/django_admin/management/commands/tree.py +50 -54
- django_cfg/modules/django_admin/mixins/display_mixin.py +16 -15
- django_cfg/modules/django_admin/mixins/optimization_mixin.py +9 -8
- django_cfg/modules/django_admin/mixins/standalone_actions_mixin.py +25 -24
- django_cfg/modules/django_admin/models/__init__.py +4 -4
- django_cfg/modules/django_admin/models/action_models.py +3 -1
- django_cfg/modules/django_admin/models/badge_models.py +4 -2
- django_cfg/modules/django_admin/models/base.py +3 -3
- django_cfg/modules/django_admin/models/display_models.py +1 -0
- django_cfg/modules/django_admin/utils/badges.py +27 -26
- django_cfg/modules/django_admin/utils/displays.py +49 -49
- django_cfg/modules/django_client/apps.py +21 -3
- django_cfg/modules/django_client/core/__init__.py +9 -10
- django_cfg/modules/django_client/core/archive/manager.py +2 -2
- django_cfg/modules/django_client/core/cli/main.py +1 -3
- django_cfg/modules/django_client/core/config/config.py +3 -1
- django_cfg/modules/django_client/core/config/group.py +1 -0
- django_cfg/modules/django_client/core/config/service.py +2 -1
- django_cfg/modules/django_client/core/generator/__init__.py +1 -1
- django_cfg/modules/django_client/core/generator/base.py +2 -2
- django_cfg/modules/django_client/core/generator/python/async_client_gen.py +1 -1
- django_cfg/modules/django_client/core/generator/python/files_generator.py +1 -1
- django_cfg/modules/django_client/core/generator/python/generator.py +4 -4
- django_cfg/modules/django_client/core/generator/python/models_generator.py +1 -1
- django_cfg/modules/django_client/core/generator/python/operations_generator.py +2 -2
- django_cfg/modules/django_client/core/generator/python/sync_client_gen.py +1 -1
- django_cfg/modules/django_client/core/generator/typescript/client_generator.py +3 -1
- django_cfg/modules/django_client/core/generator/typescript/fetchers_generator.py +14 -13
- django_cfg/modules/django_client/core/generator/typescript/files_generator.py +1 -0
- django_cfg/modules/django_client/core/generator/typescript/generator.py +6 -6
- django_cfg/modules/django_client/core/generator/typescript/hooks_generator.py +12 -10
- django_cfg/modules/django_client/core/generator/typescript/models_generator.py +2 -1
- django_cfg/modules/django_client/core/generator/typescript/naming.py +2 -3
- django_cfg/modules/django_client/core/generator/typescript/operations_generator.py +12 -10
- django_cfg/modules/django_client/core/generator/typescript/schemas_generator.py +3 -2
- django_cfg/modules/django_client/core/generator/typescript/templates/client/client.ts.jinja +14 -6
- django_cfg/modules/django_client/core/generator/typescript/templates/main_index.ts.jinja +4 -10
- django_cfg/modules/django_client/core/groups/__init__.py +1 -1
- django_cfg/modules/django_client/core/groups/detector.py +2 -1
- django_cfg/modules/django_client/core/groups/manager.py +2 -1
- django_cfg/modules/django_client/core/ir/schema.py +1 -1
- django_cfg/modules/django_client/core/parser/base.py +0 -2
- django_cfg/modules/django_client/core/parser/models/schema.py +1 -1
- django_cfg/modules/django_client/core/validation/__init__.py +1 -1
- django_cfg/modules/django_client/core/validation/fixer.py +1 -2
- django_cfg/modules/django_client/core/validation/reporter.py +2 -2
- django_cfg/modules/django_client/core/validation/safety.py +1 -1
- django_cfg/modules/django_client/management/commands/generate_client.py +14 -11
- django_cfg/modules/django_client/management/commands/validate_openapi.py +4 -6
- django_cfg/modules/django_client/spectacular/__init__.py +1 -1
- django_cfg/modules/django_client/spectacular/async_detection.py +3 -2
- django_cfg/modules/django_client/spectacular/enum_naming.py +1 -1
- django_cfg/modules/django_client/spectacular/schema.py +5 -5
- django_cfg/modules/django_client/system/__init__.py +24 -0
- django_cfg/modules/django_client/system/base_generator.py +123 -0
- django_cfg/modules/django_client/system/generate_mjs_clients.py +174 -0
- django_cfg/modules/django_client/system/mjs_generator.py +219 -0
- django_cfg/modules/django_client/system/schema_parser.py +195 -0
- django_cfg/modules/django_client/system/templates/api_client.js.j2 +87 -0
- django_cfg/modules/django_client/system/templates/app_index.js.j2 +13 -0
- django_cfg/modules/django_client/system/templates/base_client.js.j2 +166 -0
- django_cfg/modules/django_client/system/templates/main_index.js.j2 +80 -0
- django_cfg/modules/django_client/system/templates/types.js.j2 +24 -0
- django_cfg/modules/django_client/urls.py +3 -2
- django_cfg/modules/django_currency/__init__.py +17 -18
- django_cfg/modules/django_currency/clients/__init__.py +2 -2
- django_cfg/modules/django_currency/clients/coinpaprika_client.py +48 -48
- django_cfg/modules/django_currency/clients/hybrid_client.py +76 -75
- django_cfg/modules/django_currency/core/__init__.py +7 -13
- django_cfg/modules/django_currency/core/converter.py +25 -24
- django_cfg/modules/django_currency/core/models.py +9 -8
- django_cfg/modules/django_currency/database/__init__.py +4 -4
- django_cfg/modules/django_currency/database/database_loader.py +65 -66
- django_cfg/modules/django_currency/examples/example_database_usage.py +29 -28
- django_cfg/modules/django_currency/utils/cache.py +10 -11
- django_cfg/modules/django_dashboard/__init__.py +4 -5
- django_cfg/modules/django_dashboard/components.py +11 -7
- django_cfg/modules/django_dashboard/debug.py +1 -3
- django_cfg/modules/django_dashboard/management/commands/debug_dashboard.py +3 -3
- django_cfg/modules/django_dashboard/sections/base.py +2 -1
- django_cfg/modules/django_dashboard/sections/commands.py +3 -2
- django_cfg/modules/django_dashboard/sections/documentation.py +8 -6
- django_cfg/modules/django_dashboard/sections/overview.py +13 -9
- django_cfg/modules/django_dashboard/sections/stats.py +2 -2
- django_cfg/modules/django_dashboard/sections/system.py +2 -1
- django_cfg/modules/django_drf_theme/templatetags/tailwind_tags.py +12 -4
- django_cfg/modules/django_email/management/commands/test_email.py +8 -7
- django_cfg/modules/django_email/service.py +5 -4
- django_cfg/modules/django_health/service.py +46 -44
- django_cfg/modules/django_import_export/__init__.py +7 -3
- django_cfg/modules/django_llm/__init__.py +3 -2
- django_cfg/modules/django_llm/example.py +58 -56
- django_cfg/modules/django_llm/llm/__init__.py +1 -1
- django_cfg/modules/django_llm/llm/cache.py +21 -20
- django_cfg/modules/django_llm/llm/client.py +9 -9
- django_cfg/modules/django_llm/llm/costs.py +14 -14
- django_cfg/modules/django_llm/llm/embeddings/__init__.py +1 -1
- django_cfg/modules/django_llm/llm/embeddings/mock_embedder.py +1 -2
- django_cfg/modules/django_llm/llm/embeddings/openai_embedder.py +1 -2
- django_cfg/modules/django_llm/llm/extractor.py +8 -8
- django_cfg/modules/django_llm/llm/models.py +5 -5
- django_cfg/modules/django_llm/llm/models_api/models_query.py +2 -2
- django_cfg/modules/django_llm/llm/models_cache.py +91 -92
- django_cfg/modules/django_llm/llm/providers/config_builder.py +1 -1
- django_cfg/modules/django_llm/llm/providers/provider_manager.py +2 -1
- django_cfg/modules/django_llm/llm/requests/cache_manager.py +1 -1
- django_cfg/modules/django_llm/llm/requests/chat_handler.py +2 -2
- django_cfg/modules/django_llm/llm/requests/embedding_handler.py +1 -1
- django_cfg/modules/django_llm/llm/responses/response_builder.py +2 -2
- django_cfg/modules/django_llm/llm/stats/stats_manager.py +1 -1
- django_cfg/modules/django_llm/llm/tokenizer.py +10 -9
- django_cfg/modules/django_llm/translator/__init__.py +1 -1
- django_cfg/modules/django_llm/translator/cache.py +36 -35
- django_cfg/modules/django_llm/translator/detectors/__init__.py +1 -1
- django_cfg/modules/django_llm/translator/detectors/script_detector.py +0 -1
- django_cfg/modules/django_llm/translator/stats/stats_tracker.py +1 -1
- django_cfg/modules/django_llm/translator/translator.py +5 -4
- django_cfg/modules/django_llm/translator/translators/__init__.py +1 -1
- django_cfg/modules/django_llm/translator/translators/json_translator.py +1 -1
- django_cfg/modules/django_llm/translator/utils/__init__.py +1 -1
- django_cfg/modules/django_llm/translator/utils/prompt_builder.py +0 -1
- django_cfg/modules/django_logging/__init__.py +1 -1
- django_cfg/modules/django_logging/django_logger.py +33 -34
- django_cfg/modules/django_logging/logger.py +3 -7
- django_cfg/modules/django_ngrok/__init__.py +7 -7
- django_cfg/modules/django_ngrok/management/commands/runserver_ngrok.py +33 -30
- django_cfg/modules/django_ngrok/service.py +33 -32
- django_cfg/modules/django_tailwind/templates/django_tailwind/base.html +4 -36
- django_cfg/modules/django_tailwind/templates/django_tailwind/components/navbar.html +1 -1
- django_cfg/modules/django_tasks/__init__.py +5 -5
- django_cfg/modules/django_tasks/dramatiq_setup.py +1 -1
- django_cfg/modules/django_tasks/factory.py +1 -1
- django_cfg/modules/django_tasks/management/commands/rundramatiq.py +39 -40
- django_cfg/modules/django_tasks/management/commands/rundramatiq_simulator.py +79 -80
- django_cfg/modules/django_tasks/management/commands/task_clear.py +34 -34
- django_cfg/modules/django_tasks/management/commands/task_status.py +34 -34
- django_cfg/modules/django_tasks/service.py +4 -3
- django_cfg/modules/django_tasks/settings.py +1 -1
- django_cfg/modules/django_telegram/__init__.py +4 -4
- django_cfg/modules/django_telegram/management/commands/test_telegram.py +4 -5
- django_cfg/modules/django_telegram/service.py +4 -3
- django_cfg/modules/django_telegram/utils.py +1 -1
- django_cfg/modules/django_twilio/__init__.py +15 -16
- django_cfg/modules/django_twilio/_imports.py +1 -1
- django_cfg/modules/django_twilio/base.py +9 -5
- django_cfg/modules/django_twilio/email_otp.py +4 -3
- django_cfg/modules/django_twilio/exceptions.py +36 -36
- django_cfg/modules/django_twilio/management/commands/test_twilio.py +6 -7
- django_cfg/modules/django_twilio/models.py +54 -53
- django_cfg/modules/django_twilio/sendgrid_service.py +70 -72
- django_cfg/modules/django_twilio/simple_service.py +42 -41
- django_cfg/modules/django_twilio/sms.py +1 -0
- django_cfg/modules/django_twilio/twilio_service.py +79 -83
- django_cfg/modules/django_twilio/unified.py +6 -5
- django_cfg/modules/django_twilio/utils.py +2 -3
- django_cfg/modules/django_twilio/whatsapp.py +3 -2
- django_cfg/modules/django_unfold/__init__.py +7 -6
- django_cfg/modules/django_unfold/callbacks/actions.py +6 -5
- django_cfg/modules/django_unfold/callbacks/apizones.py +122 -0
- django_cfg/modules/django_unfold/callbacks/base.py +9 -8
- django_cfg/modules/django_unfold/callbacks/charts.py +36 -37
- django_cfg/modules/django_unfold/callbacks/commands.py +2 -2
- django_cfg/modules/django_unfold/callbacks/main.py +27 -27
- django_cfg/modules/django_unfold/callbacks/statistics.py +12 -12
- django_cfg/modules/django_unfold/callbacks/system.py +5 -5
- django_cfg/modules/django_unfold/callbacks/users.py +4 -4
- django_cfg/modules/django_unfold/dashboard.py +29 -29
- django_cfg/modules/django_unfold/models/__init__.py +23 -8
- django_cfg/modules/django_unfold/models/config.py +84 -81
- django_cfg/modules/django_unfold/models/dashboard.py +20 -19
- django_cfg/modules/django_unfold/models/dropdown.py +6 -4
- django_cfg/modules/django_unfold/models/navigation.py +6 -4
- django_cfg/modules/django_unfold/models/tabs.py +4 -3
- django_cfg/modules/django_unfold/models.py +2 -3
- django_cfg/modules/django_unfold/system_monitor.py +27 -25
- django_cfg/modules/django_unfold/tailwind.py +12 -14
- django_cfg/modules/django_unfold/utils.py +7 -6
- django_cfg/pyproject.toml +1 -1
- django_cfg/registry/__init__.py +3 -3
- django_cfg/registry/core.py +8 -8
- django_cfg/registry/modules.py +2 -2
- django_cfg/registry/services.py +2 -2
- django_cfg/registry/third_party.py +3 -3
- django_cfg/routing/__init__.py +3 -3
- django_cfg/routing/callbacks.py +27 -26
- django_cfg/routing/routers.py +2 -2
- django_cfg/static/js/api/accounts/client.mjs +69 -0
- django_cfg/static/js/api/accounts/index.mjs +13 -0
- django_cfg/static/js/api/base.mjs +166 -0
- django_cfg/static/js/api/index.mjs +100 -0
- django_cfg/static/js/api/ipc/client.mjs +74 -0
- django_cfg/static/js/api/ipc/index.mjs +13 -0
- django_cfg/static/js/api/leads/client.mjs +71 -0
- django_cfg/static/js/api/leads/index.mjs +13 -0
- django_cfg/static/js/api/newsletter/client.mjs +109 -0
- django_cfg/static/js/api/newsletter/index.mjs +13 -0
- django_cfg/static/js/api/payments/client.mjs +1264 -0
- django_cfg/static/js/api/payments/index.mjs +13 -0
- django_cfg/static/js/api/support/client.mjs +84 -0
- django_cfg/static/js/api/support/index.mjs +13 -0
- django_cfg/static/js/api/tasks/client.mjs +74 -0
- django_cfg/static/js/api/tasks/index.mjs +13 -0
- django_cfg/static/js/api/types.mjs +729 -0
- django_cfg/static/js/api-loader.mjs +169 -0
- django_cfg/templates/admin/snippets/zones/zones_table.html +4 -3
- django_cfg/templatetags/django_cfg.py +4 -4
- django_cfg/utils/path_resolution.py +49 -50
- django_cfg/utils/smart_defaults.py +27 -29
- django_cfg/utils/version_check.py +14 -14
- {django_cfg-1.4.20.dist-info → django_cfg-1.4.23.dist-info}/METADATA +1 -1
- django_cfg-1.4.23.dist-info/RECORD +1137 -0
- django_cfg/apps/payments/static/payments/js/api-client.js +0 -408
- django_cfg/modules/django_ipc_client/dashboard/README.md +0 -517
- django_cfg/modules/django_ipc_client/dashboard/UNFOLD_INTEGRATION.md +0 -439
- django_cfg/modules/django_ipc_client/dashboard/__init__.py +0 -11
- django_cfg/modules/django_ipc_client/dashboard/apps.py +0 -22
- django_cfg/modules/django_ipc_client/dashboard/templates/django_ipc_dashboard/dashboard.html +0 -200
- django_cfg/modules/django_ipc_client/dashboard/urls.py +0 -22
- django_cfg/modules/django_ipc_client/dashboard/urls_admin.py +0 -9
- django_cfg/modules/django_ipc_client/dashboard/views.py +0 -251
- django_cfg/modules/django_rpc_old/POETRY.md +0 -344
- django_cfg/modules/django_rpc_old/README.md +0 -397
- django_cfg/modules/django_rpc_old/TESTING.md +0 -358
- django_cfg/modules/django_rpc_old/__init__.py +0 -39
- django_cfg/modules/django_rpc_old/client.py +0 -531
- django_cfg/modules/django_rpc_old/config.py +0 -279
- django_cfg/modules/django_rpc_old/exceptions.py +0 -172
- django_cfg/modules/django_unfold/callbacks/revolution.py +0 -81
- django_cfg-1.4.20.dist-info/RECORD +0 -1110
- /django_cfg/{modules/django_ipc_client → apps/ipc}/README.md +0 -0
- /django_cfg/{modules/django_ipc_client → apps/ipc/services/client}/config.py +0 -0
- {django_cfg-1.4.20.dist-info → django_cfg-1.4.23.dist-info}/WHEEL +0 -0
- {django_cfg-1.4.20.dist-info → django_cfg-1.4.23.dist-info}/entry_points.txt +0 -0
- {django_cfg-1.4.20.dist-info → django_cfg-1.4.23.dist-info}/licenses/LICENSE +0 -0
@@ -2,42 +2,42 @@
|
|
2
2
|
Import/Export resources for Newsletter app.
|
3
3
|
"""
|
4
4
|
|
5
|
-
from import_export import resources, fields
|
6
|
-
from import_export.widgets import ForeignKeyWidget, DateTimeWidget, BooleanWidget
|
7
5
|
from django.contrib.auth import get_user_model
|
6
|
+
from import_export import fields, resources
|
7
|
+
from import_export.widgets import BooleanWidget, DateTimeWidget, ForeignKeyWidget
|
8
8
|
|
9
|
-
from ..models import Newsletter, NewsletterSubscription
|
9
|
+
from ..models import EmailLog, Newsletter, NewsletterSubscription
|
10
10
|
|
11
11
|
User = get_user_model()
|
12
12
|
|
13
13
|
|
14
14
|
class NewsletterResource(resources.ModelResource):
|
15
15
|
"""Resource for importing/exporting newsletters."""
|
16
|
-
|
16
|
+
|
17
17
|
subscribers_count = fields.Field(
|
18
18
|
column_name='subscribers_count',
|
19
19
|
attribute='subscribers_count',
|
20
20
|
readonly=True
|
21
21
|
)
|
22
|
-
|
22
|
+
|
23
23
|
is_active = fields.Field(
|
24
24
|
column_name='is_active',
|
25
25
|
attribute='is_active',
|
26
26
|
widget=BooleanWidget()
|
27
27
|
)
|
28
|
-
|
28
|
+
|
29
29
|
auto_subscribe = fields.Field(
|
30
30
|
column_name='auto_subscribe',
|
31
31
|
attribute='auto_subscribe',
|
32
32
|
widget=BooleanWidget()
|
33
33
|
)
|
34
|
-
|
34
|
+
|
35
35
|
created_at = fields.Field(
|
36
36
|
column_name='created_at',
|
37
37
|
attribute='created_at',
|
38
38
|
widget=DateTimeWidget(format='%Y-%m-%d %H:%M:%S')
|
39
39
|
)
|
40
|
-
|
40
|
+
|
41
41
|
updated_at = fields.Field(
|
42
42
|
column_name='updated_at',
|
43
43
|
attribute='updated_at',
|
@@ -60,7 +60,7 @@ class NewsletterResource(resources.ModelResource):
|
|
60
60
|
import_id_fields = ('title',) # Use title as unique identifier
|
61
61
|
skip_unchanged = True
|
62
62
|
report_skipped = True
|
63
|
-
|
63
|
+
|
64
64
|
def before_import_row(self, row, **kwargs):
|
65
65
|
"""Process row before import."""
|
66
66
|
# Ensure title is not empty
|
@@ -72,33 +72,33 @@ class NewsletterResource(resources.ModelResource):
|
|
72
72
|
|
73
73
|
class NewsletterSubscriptionResource(resources.ModelResource):
|
74
74
|
"""Resource for importing/exporting newsletter subscriptions."""
|
75
|
-
|
75
|
+
|
76
76
|
newsletter_title = fields.Field(
|
77
77
|
column_name='newsletter_title',
|
78
78
|
attribute='newsletter__title',
|
79
79
|
widget=ForeignKeyWidget(Newsletter, field='title'),
|
80
80
|
readonly=False
|
81
81
|
)
|
82
|
-
|
82
|
+
|
83
83
|
user_email = fields.Field(
|
84
84
|
column_name='user_email',
|
85
85
|
attribute='user__email',
|
86
86
|
widget=ForeignKeyWidget(User, field='email'),
|
87
87
|
readonly=False
|
88
88
|
)
|
89
|
-
|
89
|
+
|
90
90
|
is_active = fields.Field(
|
91
91
|
column_name='is_active',
|
92
92
|
attribute='is_active',
|
93
93
|
widget=BooleanWidget()
|
94
94
|
)
|
95
|
-
|
95
|
+
|
96
96
|
subscribed_at = fields.Field(
|
97
97
|
column_name='subscribed_at',
|
98
98
|
attribute='subscribed_at',
|
99
99
|
widget=DateTimeWidget(format='%Y-%m-%d %H:%M:%S')
|
100
100
|
)
|
101
|
-
|
101
|
+
|
102
102
|
unsubscribed_at = fields.Field(
|
103
103
|
column_name='unsubscribed_at',
|
104
104
|
attribute='unsubscribed_at',
|
@@ -120,13 +120,13 @@ class NewsletterSubscriptionResource(resources.ModelResource):
|
|
120
120
|
import_id_fields = ('newsletter_title', 'email') # Composite unique identifier
|
121
121
|
skip_unchanged = True
|
122
122
|
report_skipped = True
|
123
|
-
|
123
|
+
|
124
124
|
def before_import_row(self, row, **kwargs):
|
125
125
|
"""Process row before import."""
|
126
126
|
# Ensure email is lowercase
|
127
127
|
if 'email' in row:
|
128
128
|
row['email'] = row['email'].lower().strip()
|
129
|
-
|
129
|
+
|
130
130
|
# Handle newsletter assignment by title
|
131
131
|
if 'newsletter_title' in row and row['newsletter_title']:
|
132
132
|
try:
|
@@ -134,7 +134,7 @@ class NewsletterSubscriptionResource(resources.ModelResource):
|
|
134
134
|
row['newsletter'] = newsletter.pk
|
135
135
|
except Newsletter.DoesNotExist:
|
136
136
|
raise ValueError(f"Newsletter '{row['newsletter_title']}' not found")
|
137
|
-
|
137
|
+
|
138
138
|
# Handle user assignment by email (optional)
|
139
139
|
if 'user_email' in row and row['user_email']:
|
140
140
|
try:
|
@@ -143,7 +143,7 @@ class NewsletterSubscriptionResource(resources.ModelResource):
|
|
143
143
|
except User.DoesNotExist:
|
144
144
|
# Clear user field if email not found
|
145
145
|
row['user'] = None
|
146
|
-
|
146
|
+
|
147
147
|
def get_queryset(self):
|
148
148
|
"""Optimize queryset for export."""
|
149
149
|
return super().get_queryset().select_related('newsletter', 'user')
|
@@ -151,63 +151,63 @@ class NewsletterSubscriptionResource(resources.ModelResource):
|
|
151
151
|
|
152
152
|
class EmailLogResource(resources.ModelResource):
|
153
153
|
"""Resource for exporting email logs (export only)."""
|
154
|
-
|
154
|
+
|
155
155
|
user_email = fields.Field(
|
156
156
|
column_name='user_email',
|
157
157
|
attribute='user__email',
|
158
158
|
readonly=True
|
159
159
|
)
|
160
|
-
|
160
|
+
|
161
161
|
newsletter_title = fields.Field(
|
162
162
|
column_name='newsletter_title',
|
163
163
|
attribute='newsletter__title',
|
164
164
|
readonly=True
|
165
165
|
)
|
166
|
-
|
166
|
+
|
167
167
|
campaign_subject = fields.Field(
|
168
168
|
column_name='campaign_subject',
|
169
169
|
attribute='campaign__subject',
|
170
170
|
readonly=True
|
171
171
|
)
|
172
|
-
|
172
|
+
|
173
173
|
status_display = fields.Field(
|
174
174
|
column_name='status_display',
|
175
175
|
attribute='get_status_display',
|
176
176
|
readonly=True
|
177
177
|
)
|
178
|
-
|
178
|
+
|
179
179
|
is_opened = fields.Field(
|
180
180
|
column_name='is_opened',
|
181
181
|
attribute='is_opened',
|
182
182
|
widget=BooleanWidget(),
|
183
183
|
readonly=True
|
184
184
|
)
|
185
|
-
|
185
|
+
|
186
186
|
is_clicked = fields.Field(
|
187
187
|
column_name='is_clicked',
|
188
188
|
attribute='is_clicked',
|
189
189
|
widget=BooleanWidget(),
|
190
190
|
readonly=True
|
191
191
|
)
|
192
|
-
|
192
|
+
|
193
193
|
created_at = fields.Field(
|
194
194
|
column_name='created_at',
|
195
195
|
attribute='created_at',
|
196
196
|
widget=DateTimeWidget(format='%Y-%m-%d %H:%M:%S')
|
197
197
|
)
|
198
|
-
|
198
|
+
|
199
199
|
sent_at = fields.Field(
|
200
200
|
column_name='sent_at',
|
201
201
|
attribute='sent_at',
|
202
202
|
widget=DateTimeWidget(format='%Y-%m-%d %H:%M:%S')
|
203
203
|
)
|
204
|
-
|
204
|
+
|
205
205
|
opened_at = fields.Field(
|
206
206
|
column_name='opened_at',
|
207
207
|
attribute='opened_at',
|
208
208
|
widget=DateTimeWidget(format='%Y-%m-%d %H:%M:%S')
|
209
209
|
)
|
210
|
-
|
210
|
+
|
211
211
|
clicked_at = fields.Field(
|
212
212
|
column_name='clicked_at',
|
213
213
|
attribute='clicked_at',
|
@@ -235,7 +235,7 @@ class EmailLogResource(resources.ModelResource):
|
|
235
235
|
)
|
236
236
|
export_order = fields
|
237
237
|
# No import - this is export only
|
238
|
-
|
238
|
+
|
239
239
|
def get_queryset(self):
|
240
240
|
"""Optimize queryset for export."""
|
241
241
|
return super().get_queryset().select_related('user', 'newsletter', 'campaign')
|
@@ -1,14 +1,21 @@
|
|
1
1
|
from django import forms
|
2
2
|
from django.contrib import admin, messages
|
3
|
+
from django.http import HttpResponseRedirect
|
3
4
|
from django.urls import reverse
|
4
5
|
from django.utils.html import format_html
|
5
|
-
from django.http import HttpResponseRedirect
|
6
6
|
from unfold.admin import ModelAdmin
|
7
|
-
from unfold.decorators import action
|
8
7
|
from unfold.contrib.forms.widgets import WysiwygWidget
|
8
|
+
from unfold.decorators import action
|
9
9
|
from unfold.enums import ActionVariant
|
10
|
-
|
11
|
-
from .admin_filters import
|
10
|
+
|
11
|
+
from .admin_filters import (
|
12
|
+
EmailClickedFilter,
|
13
|
+
EmailOpenedFilter,
|
14
|
+
HasUserFilter,
|
15
|
+
UserEmailFilter,
|
16
|
+
UserNameFilter,
|
17
|
+
)
|
18
|
+
from .models import EmailLog, Newsletter, NewsletterCampaign, NewsletterSubscription
|
12
19
|
|
13
20
|
|
14
21
|
@admin.register(EmailLog)
|
@@ -34,15 +41,15 @@ class EmailLogAdmin(ModelAdmin):
|
|
34
41
|
return format_html('<a href="{}">{}</a>', link, obj.newsletter.title)
|
35
42
|
return "-"
|
36
43
|
newsletter_link.short_description = 'Newsletter'
|
37
|
-
|
44
|
+
|
38
45
|
def tracking_status(self, obj):
|
39
46
|
"""Show clean tracking status."""
|
40
47
|
opened_status = "Opened" if obj.is_opened else "Not opened"
|
41
48
|
clicked_status = "Clicked" if obj.is_clicked else "Not clicked"
|
42
|
-
|
49
|
+
|
43
50
|
opened_color = "#28a745" if obj.is_opened else "#dc3545"
|
44
51
|
clicked_color = "#007bff" if obj.is_clicked else "#6c757d"
|
45
|
-
|
52
|
+
|
46
53
|
return format_html(
|
47
54
|
'<div style="display: flex; gap: 8px;">'
|
48
55
|
'<span style="background: {}; color: white; padding: 2px 6px; border-radius: 3px; font-size: 11px;">{}</span>'
|
@@ -119,10 +126,10 @@ class NewsletterCampaignAdmin(ModelAdmin):
|
|
119
126
|
readonly_fields = ('status', 'created_at', 'sent_at', 'recipient_count')
|
120
127
|
search_fields = ('subject', 'email_title', 'main_text')
|
121
128
|
autocomplete_fields = ('newsletter',)
|
122
|
-
|
129
|
+
|
123
130
|
# Django admin actions
|
124
131
|
actions = ["send_selected_campaigns"]
|
125
|
-
|
132
|
+
|
126
133
|
# Unfold actions configuration
|
127
134
|
actions_list = [] # Changelist actions (removed send_selected_campaigns)
|
128
135
|
actions_detail = ["send_campaign"] # Detail page actions
|
@@ -141,29 +148,29 @@ class NewsletterCampaignAdmin(ModelAdmin):
|
|
141
148
|
if not campaign:
|
142
149
|
self.message_user(request, "Campaign not found.", messages.ERROR)
|
143
150
|
return HttpResponseRedirect(request.META.get('HTTP_REFERER'))
|
144
|
-
|
151
|
+
|
145
152
|
if campaign.status != NewsletterCampaign.CampaignStatus.DRAFT:
|
146
153
|
self.message_user(
|
147
|
-
request,
|
148
|
-
f"Campaign '{campaign.subject}' is not in draft status.",
|
154
|
+
request,
|
155
|
+
f"Campaign '{campaign.subject}' is not in draft status.",
|
149
156
|
messages.WARNING
|
150
157
|
)
|
151
158
|
return HttpResponseRedirect(request.META.get('HTTP_REFERER'))
|
152
|
-
|
159
|
+
|
153
160
|
success = campaign.send_campaign()
|
154
161
|
if success:
|
155
162
|
self.message_user(
|
156
|
-
request,
|
157
|
-
f"Campaign '{campaign.subject}' sent successfully.",
|
163
|
+
request,
|
164
|
+
f"Campaign '{campaign.subject}' sent successfully.",
|
158
165
|
messages.SUCCESS
|
159
166
|
)
|
160
167
|
else:
|
161
168
|
self.message_user(
|
162
|
-
request,
|
163
|
-
f"Campaign '{campaign.subject}' failed to send.",
|
169
|
+
request,
|
170
|
+
f"Campaign '{campaign.subject}' failed to send.",
|
164
171
|
messages.ERROR
|
165
172
|
)
|
166
|
-
|
173
|
+
|
167
174
|
except Exception as e:
|
168
175
|
self.message_user(request, f"An error occurred: {e}", messages.ERROR)
|
169
176
|
|
@@ -173,7 +180,7 @@ class NewsletterCampaignAdmin(ModelAdmin):
|
|
173
180
|
"""Send multiple campaigns (standard Django admin action)."""
|
174
181
|
sent_count = 0
|
175
182
|
skipped_count = 0
|
176
|
-
|
183
|
+
|
177
184
|
for campaign in queryset:
|
178
185
|
if campaign.status == NewsletterCampaign.CampaignStatus.DRAFT:
|
179
186
|
success = campaign.send_campaign()
|
@@ -184,24 +191,24 @@ class NewsletterCampaignAdmin(ModelAdmin):
|
|
184
191
|
else:
|
185
192
|
skipped_count += 1
|
186
193
|
messages.warning(
|
187
|
-
request,
|
194
|
+
request,
|
188
195
|
f"Campaign '{campaign.subject}' skipped (not in Draft status)."
|
189
196
|
)
|
190
197
|
|
191
198
|
if sent_count > 0:
|
192
199
|
self.message_user(
|
193
|
-
request,
|
194
|
-
f"Successfully sent {sent_count} campaigns.",
|
200
|
+
request,
|
201
|
+
f"Successfully sent {sent_count} campaigns.",
|
195
202
|
messages.SUCCESS
|
196
203
|
)
|
197
|
-
|
204
|
+
|
198
205
|
if skipped_count > 0:
|
199
206
|
self.message_user(
|
200
|
-
request,
|
201
|
-
f"{skipped_count} campaigns were skipped.",
|
207
|
+
request,
|
208
|
+
f"{skipped_count} campaigns were skipped.",
|
202
209
|
messages.WARNING
|
203
210
|
)
|
204
|
-
|
211
|
+
|
205
212
|
send_selected_campaigns.short_description = "Send selected campaigns"
|
206
213
|
|
207
214
|
@action(
|
@@ -216,19 +223,19 @@ class NewsletterCampaignAdmin(ModelAdmin):
|
|
216
223
|
success = obj.send_campaign()
|
217
224
|
if success:
|
218
225
|
self.message_user(
|
219
|
-
request,
|
220
|
-
f"Campaign '{obj.subject}' sent successfully.",
|
226
|
+
request,
|
227
|
+
f"Campaign '{obj.subject}' sent successfully.",
|
221
228
|
messages.SUCCESS
|
222
229
|
)
|
223
230
|
else:
|
224
231
|
self.message_user(
|
225
|
-
request,
|
226
|
-
f"Campaign '{obj.subject}' failed to send.",
|
232
|
+
request,
|
233
|
+
f"Campaign '{obj.subject}' failed to send.",
|
227
234
|
messages.ERROR
|
228
235
|
)
|
229
236
|
else:
|
230
237
|
self.message_user(
|
231
|
-
request,
|
232
|
-
f"Campaign '{obj.subject}' is not in draft status.",
|
238
|
+
request,
|
239
|
+
f"Campaign '{obj.subject}' is not in draft status.",
|
233
240
|
messages.WARNING
|
234
241
|
)
|
@@ -2,10 +2,10 @@
|
|
2
2
|
Management command to test newsletter sending functionality.
|
3
3
|
"""
|
4
4
|
|
5
|
-
from django.core.management.base import BaseCommand, CommandError
|
6
5
|
from django.contrib.auth import get_user_model
|
6
|
+
from django.core.management.base import BaseCommand, CommandError
|
7
7
|
|
8
|
-
from django_cfg.apps.newsletter.models import Newsletter,
|
8
|
+
from django_cfg.apps.newsletter.models import Newsletter, NewsletterCampaign, NewsletterSubscription
|
9
9
|
from django_cfg.apps.newsletter.services.email_service import NewsletterEmailService
|
10
10
|
|
11
11
|
User = get_user_model()
|
@@ -107,7 +107,7 @@ class Command(BaseCommand):
|
|
107
107
|
|
108
108
|
# Send test email
|
109
109
|
self.stdout.write("Sending test newsletter...")
|
110
|
-
|
110
|
+
|
111
111
|
email_service = NewsletterEmailService()
|
112
112
|
result = email_service.send_newsletter_email(
|
113
113
|
newsletter=newsletter,
|
@@ -1 +1 @@
|
|
1
|
-
# Managers package
|
1
|
+
# Managers package
|
@@ -1,16 +1,17 @@
|
|
1
1
|
import uuid
|
2
|
-
|
2
|
+
|
3
3
|
from django.conf import settings
|
4
|
+
from django.db import models
|
4
5
|
from django.utils import timezone
|
5
6
|
|
6
7
|
|
7
8
|
class NewsletterManager(models.Manager):
|
8
9
|
"""Custom manager for Newsletter model."""
|
9
|
-
|
10
|
+
|
10
11
|
def active(self):
|
11
12
|
"""Get active newsletters."""
|
12
13
|
return self.filter(is_active=True)
|
13
|
-
|
14
|
+
|
14
15
|
def with_auto_subscribe(self):
|
15
16
|
"""Get newsletters with auto-subscribe enabled."""
|
16
17
|
return self.filter(is_active=True, auto_subscribe=True)
|
@@ -18,29 +19,29 @@ class NewsletterManager(models.Manager):
|
|
18
19
|
|
19
20
|
class Newsletter(models.Model):
|
20
21
|
"""Newsletter model for managing email campaigns."""
|
21
|
-
|
22
|
+
|
22
23
|
title = models.CharField(max_length=255, verbose_name="Newsletter Title")
|
23
24
|
description = models.TextField(blank=True, verbose_name="Description")
|
24
25
|
is_active = models.BooleanField(default=True, verbose_name="Active")
|
25
26
|
auto_subscribe = models.BooleanField(
|
26
|
-
default=False,
|
27
|
+
default=False,
|
27
28
|
verbose_name="Auto Subscribe New Users",
|
28
29
|
help_text="Automatically subscribe new users to this newsletter"
|
29
30
|
)
|
30
31
|
created_at = models.DateTimeField(auto_now_add=True)
|
31
32
|
updated_at = models.DateTimeField(auto_now=True)
|
32
|
-
|
33
|
+
|
33
34
|
objects = NewsletterManager()
|
34
|
-
|
35
|
+
|
35
36
|
class Meta:
|
36
37
|
app_label = 'django_cfg_newsletter'
|
37
38
|
verbose_name = "Newsletter"
|
38
39
|
verbose_name_plural = "Newsletters"
|
39
40
|
ordering = ['-created_at']
|
40
|
-
|
41
|
+
|
41
42
|
def __str__(self):
|
42
43
|
return self.title
|
43
|
-
|
44
|
+
|
44
45
|
@property
|
45
46
|
def subscribers_count(self):
|
46
47
|
"""Get count of active subscribers."""
|
@@ -49,11 +50,11 @@ class Newsletter(models.Model):
|
|
49
50
|
|
50
51
|
class NewsletterSubscriptionManager(models.Manager):
|
51
52
|
"""Custom manager for NewsletterSubscription model."""
|
52
|
-
|
53
|
+
|
53
54
|
def active(self):
|
54
55
|
"""Get active subscriptions."""
|
55
56
|
return self.filter(is_active=True)
|
56
|
-
|
57
|
+
|
57
58
|
def for_newsletter(self, newsletter):
|
58
59
|
"""Get subscriptions for specific newsletter."""
|
59
60
|
return self.filter(newsletter=newsletter, is_active=True)
|
@@ -61,10 +62,10 @@ class NewsletterSubscriptionManager(models.Manager):
|
|
61
62
|
|
62
63
|
class NewsletterSubscription(models.Model):
|
63
64
|
"""Newsletter subscription model."""
|
64
|
-
|
65
|
+
|
65
66
|
newsletter = models.ForeignKey(
|
66
|
-
Newsletter,
|
67
|
-
on_delete=models.CASCADE,
|
67
|
+
Newsletter,
|
68
|
+
on_delete=models.CASCADE,
|
68
69
|
related_name='subscriptions',
|
69
70
|
verbose_name="Newsletter"
|
70
71
|
)
|
@@ -79,19 +80,19 @@ class NewsletterSubscription(models.Model):
|
|
79
80
|
is_active = models.BooleanField(default=True, verbose_name="Active")
|
80
81
|
subscribed_at = models.DateTimeField(auto_now_add=True)
|
81
82
|
unsubscribed_at = models.DateTimeField(null=True, blank=True)
|
82
|
-
|
83
|
+
|
83
84
|
objects = NewsletterSubscriptionManager()
|
84
|
-
|
85
|
+
|
85
86
|
class Meta:
|
86
87
|
app_label = 'django_cfg_newsletter'
|
87
88
|
verbose_name = "Newsletter Subscription"
|
88
89
|
verbose_name_plural = "Newsletter Subscriptions"
|
89
90
|
unique_together = ['newsletter', 'email']
|
90
91
|
ordering = ['-subscribed_at']
|
91
|
-
|
92
|
+
|
92
93
|
def __str__(self):
|
93
94
|
return f"{self.email} -> {self.newsletter.title}"
|
94
|
-
|
95
|
+
|
95
96
|
def unsubscribe(self):
|
96
97
|
"""Unsubscribe from newsletter."""
|
97
98
|
self.is_active = False
|
@@ -141,7 +142,7 @@ class EmailLog(models.Model):
|
|
141
142
|
created_at = models.DateTimeField('Created At', auto_now_add=True)
|
142
143
|
sent_at = models.DateTimeField('Sent At', null=True, blank=True)
|
143
144
|
error_message = models.TextField('Error Message', blank=True, null=True)
|
144
|
-
|
145
|
+
|
145
146
|
# Simple tracking fields
|
146
147
|
opened_at = models.DateTimeField('Opened At', null=True, blank=True, help_text="When email was first opened")
|
147
148
|
clicked_at = models.DateTimeField('Clicked At', null=True, blank=True, help_text="When link was first clicked")
|
@@ -155,26 +156,26 @@ class EmailLog(models.Model):
|
|
155
156
|
def __str__(self):
|
156
157
|
user_info = f"User: {self.user.email}" if self.user else f"Recipient(s): {self.recipient}"
|
157
158
|
return f"{user_info} | Subject: {self.subject} | Status: {self.status}"
|
158
|
-
|
159
|
+
|
159
160
|
def mark_opened(self):
|
160
161
|
"""Mark email as opened (only first time)."""
|
161
162
|
if not self.opened_at:
|
162
163
|
self.opened_at = timezone.now()
|
163
164
|
self.save(update_fields=['opened_at'])
|
164
165
|
return self
|
165
|
-
|
166
|
+
|
166
167
|
def mark_clicked(self):
|
167
168
|
"""Mark email link as clicked (only first time)."""
|
168
169
|
if not self.clicked_at:
|
169
170
|
self.clicked_at = timezone.now()
|
170
171
|
self.save(update_fields=['clicked_at'])
|
171
172
|
return self
|
172
|
-
|
173
|
+
|
173
174
|
@property
|
174
175
|
def is_opened(self):
|
175
176
|
"""Check if email was opened."""
|
176
177
|
return self.opened_at is not None
|
177
|
-
|
178
|
+
|
178
179
|
@property
|
179
180
|
def is_clicked(self):
|
180
181
|
"""Check if link was clicked."""
|
@@ -203,11 +204,11 @@ class NewsletterCampaign(models.Model):
|
|
203
204
|
button_text = models.CharField('Button Text', max_length=100, blank=True)
|
204
205
|
button_url = models.URLField('Button URL', blank=True)
|
205
206
|
secondary_text = models.TextField('Secondary Text', blank=True)
|
206
|
-
|
207
|
+
|
207
208
|
status = models.CharField(
|
208
|
-
'Status',
|
209
|
-
max_length=10,
|
210
|
-
choices=CampaignStatus.choices,
|
209
|
+
'Status',
|
210
|
+
max_length=10,
|
211
|
+
choices=CampaignStatus.choices,
|
211
212
|
default=CampaignStatus.DRAFT
|
212
213
|
)
|
213
214
|
created_at = models.DateTimeField('Created At', auto_now_add=True)
|
@@ -222,17 +223,17 @@ class NewsletterCampaign(models.Model):
|
|
222
223
|
|
223
224
|
def __str__(self):
|
224
225
|
return f"{self.newsletter.title}: {self.subject} ({self.status})"
|
225
|
-
|
226
|
+
|
226
227
|
def send_campaign(self):
|
227
228
|
"""Send this campaign using MailerEmailService."""
|
228
229
|
from .services.email_service import NewsletterEmailService
|
229
|
-
|
230
|
+
|
230
231
|
if self.status != self.CampaignStatus.DRAFT:
|
231
232
|
return False
|
232
|
-
|
233
|
+
|
233
234
|
self.status = self.CampaignStatus.SENDING
|
234
235
|
self.save()
|
235
|
-
|
236
|
+
|
236
237
|
email_service = NewsletterEmailService()
|
237
238
|
result = email_service.send_newsletter_email(
|
238
239
|
newsletter=self.newsletter,
|
@@ -246,14 +247,14 @@ class NewsletterCampaign(models.Model):
|
|
246
247
|
send_to_all=True,
|
247
248
|
campaign=self
|
248
249
|
)
|
249
|
-
|
250
|
+
|
250
251
|
if result['success']:
|
251
252
|
self.status = self.CampaignStatus.SENT
|
252
253
|
self.sent_at = timezone.now()
|
253
254
|
self.recipient_count = result['sent_count']
|
254
255
|
else:
|
255
256
|
self.status = self.CampaignStatus.FAILED
|
256
|
-
|
257
|
+
|
257
258
|
self.save()
|
258
259
|
return result['success']
|
259
260
|
|