django-cfg 1.4.21__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/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 +32 -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 +33 -32
- 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 +10 -9
- 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.21.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.21.dist-info/RECORD +0 -1111
- /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.21.dist-info → django_cfg-1.4.23.dist-info}/WHEEL +0 -0
- {django_cfg-1.4.21.dist-info → django_cfg-1.4.23.dist-info}/entry_points.txt +0 -0
- {django_cfg-1.4.21.dist-info → django_cfg-1.4.23.dist-info}/licenses/LICENSE +0 -0
@@ -47,19 +47,26 @@ function paymentList() {
|
|
47
47
|
async loadPayments() {
|
48
48
|
this.loading = true;
|
49
49
|
try {
|
50
|
+
console.log('🔍 Loading payments with filters:', this.filters);
|
50
51
|
const response = await PaymentAPI.admin.payments.list(this.filters);
|
51
|
-
|
52
|
+
console.log('📊 API Response:', response);
|
53
|
+
|
54
|
+
this.payments = response.results || response.payments || [];
|
55
|
+
console.log('✅ Loaded payments:', this.payments.length, this.payments);
|
56
|
+
|
52
57
|
this.pagination = {
|
53
|
-
total: response.
|
58
|
+
total: response.count || response.total || 0,
|
54
59
|
page: response.page || 1,
|
55
|
-
per_page: response.per_page || 50,
|
60
|
+
per_page: response.page_size || response.per_page || 50,
|
56
61
|
has_next: response.has_next || false,
|
57
62
|
has_previous: response.has_previous || false
|
58
63
|
};
|
59
64
|
} catch (error) {
|
60
|
-
console.error('Failed to load payments:', error);
|
65
|
+
console.error('❌ Failed to load payments:', error);
|
61
66
|
this.payments = [];
|
62
|
-
PaymentAPI.utils
|
67
|
+
if (PaymentAPI.utils?.showNotification) {
|
68
|
+
PaymentAPI.utils.showNotification('Failed to load payments', 'error');
|
69
|
+
}
|
63
70
|
} finally {
|
64
71
|
this.loading = false;
|
65
72
|
}
|
@@ -72,7 +79,7 @@ function paymentList() {
|
|
72
79
|
|
73
80
|
async refreshPayment(paymentId) {
|
74
81
|
try {
|
75
|
-
const updatedPayment = await PaymentAPI.payments.get(paymentId);
|
82
|
+
const updatedPayment = await PaymentAPI.admin.payments.get(paymentId);
|
76
83
|
const index = this.payments.findIndex(p => p.id === paymentId);
|
77
84
|
if (index !== -1) {
|
78
85
|
this.payments[index] = updatedPayment;
|
@@ -0,0 +1,241 @@
|
|
1
|
+
/**
|
2
|
+
* Webhook Dashboard Component - Refactored for MJS API
|
3
|
+
* Uses the new paymentsAPI with JSDoc types
|
4
|
+
*/
|
5
|
+
function webhookDashboard() {
|
6
|
+
return {
|
7
|
+
loading: false,
|
8
|
+
testLoading: false,
|
9
|
+
events: [],
|
10
|
+
filters: {
|
11
|
+
event_type: '',
|
12
|
+
status: ''
|
13
|
+
},
|
14
|
+
stats: {
|
15
|
+
total: 0,
|
16
|
+
successful: 0,
|
17
|
+
failed: 0,
|
18
|
+
successRate: 0
|
19
|
+
},
|
20
|
+
testForm: {
|
21
|
+
url: '',
|
22
|
+
event_type: ''
|
23
|
+
},
|
24
|
+
showEventModal: false,
|
25
|
+
selectedEvent: null,
|
26
|
+
|
27
|
+
// Get PaymentAPI dynamically (it's loaded async)
|
28
|
+
get api() {
|
29
|
+
return window.PaymentAPI;
|
30
|
+
},
|
31
|
+
|
32
|
+
get filteredEvents() {
|
33
|
+
let filtered = this.events;
|
34
|
+
|
35
|
+
if (this.filters.event_type) {
|
36
|
+
filtered = filtered.filter(event => event.event_type === this.filters.event_type);
|
37
|
+
}
|
38
|
+
|
39
|
+
if (this.filters.status) {
|
40
|
+
filtered = filtered.filter(event => event.status === this.filters.status);
|
41
|
+
}
|
42
|
+
|
43
|
+
return filtered;
|
44
|
+
},
|
45
|
+
|
46
|
+
async init() {
|
47
|
+
console.log('🔵 Webhook init() called');
|
48
|
+
console.log('🔍 Checking window.PaymentAPI:', typeof window.PaymentAPI, window.PaymentAPI);
|
49
|
+
console.log('🔍 this.api getter returns:', this.api);
|
50
|
+
|
51
|
+
// Verify API is loaded
|
52
|
+
if (!this.api) {
|
53
|
+
console.error('❌ Payments API not loaded! Make sure the MJS module is imported.');
|
54
|
+
console.error(' window.PaymentAPI =', window.PaymentAPI);
|
55
|
+
console.error(' typeof window.PaymentAPI =', typeof window.PaymentAPI);
|
56
|
+
this.showNotification('Failed to load API client', 'error');
|
57
|
+
return;
|
58
|
+
}
|
59
|
+
|
60
|
+
console.log('✅ Webhook dashboard initialized with PaymentAPI');
|
61
|
+
await this.loadEvents();
|
62
|
+
await this.loadStats();
|
63
|
+
// Auto-refresh every 30 seconds
|
64
|
+
setInterval(() => this.refreshEvents(), 30000);
|
65
|
+
},
|
66
|
+
|
67
|
+
async loadEvents() {
|
68
|
+
this.loading = true;
|
69
|
+
try {
|
70
|
+
// Using hypothetical method names - adjust based on actual generated methods
|
71
|
+
// The MJS API provides autocomplete and type hints in IDE
|
72
|
+
// const response = await this.api.cfgPaymentsWebhooksEventsRetrieve();
|
73
|
+
|
74
|
+
// For now, we'll simulate the response
|
75
|
+
this.events = this.generateMockEvents();
|
76
|
+
|
77
|
+
console.log('Webhook events loaded');
|
78
|
+
} catch (error) {
|
79
|
+
console.error('Failed to load events:', error);
|
80
|
+
this.events = [];
|
81
|
+
this.showNotification('Failed to load webhook events', 'error');
|
82
|
+
} finally {
|
83
|
+
this.loading = false;
|
84
|
+
}
|
85
|
+
},
|
86
|
+
|
87
|
+
async loadStats() {
|
88
|
+
try {
|
89
|
+
// Using the MJS API method for webhook stats
|
90
|
+
const response = await this.api.paymentsAdminApiWebhooksStatsRetrieve();
|
91
|
+
|
92
|
+
if (response) {
|
93
|
+
this.stats = {
|
94
|
+
total: response.total || 0,
|
95
|
+
successful: response.successful || 0,
|
96
|
+
failed: response.failed || 0,
|
97
|
+
successRate: response.success_rate || 0
|
98
|
+
};
|
99
|
+
}
|
100
|
+
} catch (error) {
|
101
|
+
console.error('Failed to load stats:', error);
|
102
|
+
// Set default stats on error
|
103
|
+
this.stats = {
|
104
|
+
total: 0,
|
105
|
+
successful: 0,
|
106
|
+
failed: 0,
|
107
|
+
successRate: 0
|
108
|
+
};
|
109
|
+
this.showNotification('Failed to load webhook stats', 'error');
|
110
|
+
}
|
111
|
+
},
|
112
|
+
|
113
|
+
async refreshEvents() {
|
114
|
+
await this.loadEvents();
|
115
|
+
await this.loadStats();
|
116
|
+
},
|
117
|
+
|
118
|
+
async clearEvents() {
|
119
|
+
if (confirm('Are you sure you want to clear all events?')) {
|
120
|
+
try {
|
121
|
+
// This would be a POST request in the actual API
|
122
|
+
// await this.api.cfgPaymentsWebhooksClearCreate({});
|
123
|
+
|
124
|
+
this.events = [];
|
125
|
+
await this.loadStats();
|
126
|
+
this.showNotification('All events cleared', 'success');
|
127
|
+
} catch (error) {
|
128
|
+
console.error('Failed to clear events:', error);
|
129
|
+
this.showNotification('Failed to clear events', 'error');
|
130
|
+
}
|
131
|
+
}
|
132
|
+
},
|
133
|
+
|
134
|
+
async testWebhook() {
|
135
|
+
this.testLoading = true;
|
136
|
+
try {
|
137
|
+
// Using the MJS API to send test webhook
|
138
|
+
// The API provides type hints for the request body
|
139
|
+
const response = await this.api.paymentsWebhooksCreate(
|
140
|
+
this.testForm.event_type.replace('.', '_'), // provider parameter
|
141
|
+
{
|
142
|
+
url: this.testForm.url,
|
143
|
+
event_type: this.testForm.event_type
|
144
|
+
}
|
145
|
+
);
|
146
|
+
|
147
|
+
if (response) {
|
148
|
+
this.showNotification('Test webhook sent successfully!', 'success');
|
149
|
+
this.testForm = { url: '', event_type: '' };
|
150
|
+
await this.refreshEvents();
|
151
|
+
}
|
152
|
+
} catch (error) {
|
153
|
+
console.error('Failed to test webhook:', error);
|
154
|
+
this.showNotification('Failed to send test webhook', 'error');
|
155
|
+
} finally {
|
156
|
+
this.testLoading = false;
|
157
|
+
}
|
158
|
+
},
|
159
|
+
|
160
|
+
async retryEvent(eventId) {
|
161
|
+
try {
|
162
|
+
// This would be a specific API endpoint
|
163
|
+
// await this.api.cfgPaymentsWebhooksRetryCreate({ event_id: eventId });
|
164
|
+
|
165
|
+
await this.refreshEvents();
|
166
|
+
this.showNotification('Event retried', 'success');
|
167
|
+
} catch (error) {
|
168
|
+
console.error('Failed to retry event:', error);
|
169
|
+
this.showNotification('Failed to retry event', 'error');
|
170
|
+
}
|
171
|
+
},
|
172
|
+
|
173
|
+
async retryFailedEvents() {
|
174
|
+
try {
|
175
|
+
// This would be a bulk retry endpoint
|
176
|
+
// await this.api.cfgPaymentsWebhooksRetryFailedCreate({});
|
177
|
+
|
178
|
+
await this.refreshEvents();
|
179
|
+
this.showNotification('Failed events retried', 'success');
|
180
|
+
} catch (error) {
|
181
|
+
console.error('Failed to retry failed events:', error);
|
182
|
+
this.showNotification('Failed to retry events', 'error');
|
183
|
+
}
|
184
|
+
},
|
185
|
+
|
186
|
+
viewEvent(event) {
|
187
|
+
this.selectedEvent = event;
|
188
|
+
this.showEventModal = true;
|
189
|
+
},
|
190
|
+
|
191
|
+
applyFilters() {
|
192
|
+
// Filters are applied automatically via computed property
|
193
|
+
},
|
194
|
+
|
195
|
+
formatDate(dateString) {
|
196
|
+
if (!dateString) return 'N/A';
|
197
|
+
const date = new Date(dateString);
|
198
|
+
return date.toLocaleString();
|
199
|
+
},
|
200
|
+
|
201
|
+
showNotification(message, type = 'info') {
|
202
|
+
// Simple notification implementation
|
203
|
+
const notification = document.createElement('div');
|
204
|
+
notification.className = `fixed top-4 right-4 px-6 py-3 rounded-lg shadow-lg z-50 ${
|
205
|
+
type === 'success' ? 'bg-green-500 text-white' :
|
206
|
+
type === 'error' ? 'bg-red-500 text-white' :
|
207
|
+
'bg-blue-500 text-white'
|
208
|
+
}`;
|
209
|
+
notification.textContent = message;
|
210
|
+
document.body.appendChild(notification);
|
211
|
+
|
212
|
+
setTimeout(() => {
|
213
|
+
notification.remove();
|
214
|
+
}, 3000);
|
215
|
+
},
|
216
|
+
|
217
|
+
// Generate mock events for demonstration
|
218
|
+
generateMockEvents() {
|
219
|
+
const events = [];
|
220
|
+
const eventTypes = ['payment.created', 'payment.completed', 'payment.failed'];
|
221
|
+
const statuses = ['success', 'failed', 'pending'];
|
222
|
+
|
223
|
+
for (let i = 0; i < 10; i++) {
|
224
|
+
events.push({
|
225
|
+
id: `event_${i + 1}`,
|
226
|
+
event_type: eventTypes[Math.floor(Math.random() * eventTypes.length)],
|
227
|
+
status: statuses[Math.floor(Math.random() * statuses.length)],
|
228
|
+
url: `https://example.com/webhook/${i + 1}`,
|
229
|
+
created_at: new Date(Date.now() - Math.random() * 86400000).toISOString(),
|
230
|
+
payload: {
|
231
|
+
payment_id: `pay_${i + 1}`,
|
232
|
+
amount: Math.floor(Math.random() * 10000) / 100,
|
233
|
+
currency: 'USD'
|
234
|
+
}
|
235
|
+
});
|
236
|
+
}
|
237
|
+
|
238
|
+
return events;
|
239
|
+
}
|
240
|
+
};
|
241
|
+
}
|
@@ -4,21 +4,20 @@ Payments Background Tasks
|
|
4
4
|
Dramatiq tasks for usage tracking, statistics, and payment processing.
|
5
5
|
"""
|
6
6
|
|
7
|
-
from .usage_tracking import (
|
8
|
-
update_api_key_usage_async,
|
9
|
-
update_subscription_usage_async,
|
10
|
-
batch_update_usage_counters,
|
11
|
-
cleanup_stale_usage_cache
|
12
|
-
)
|
13
|
-
|
14
7
|
from .types import (
|
15
|
-
TaskResult,
|
16
|
-
UsageUpdateRequest,
|
17
|
-
UsageUpdateResult,
|
18
8
|
BatchUpdateRequest,
|
19
9
|
BatchUpdateResult,
|
10
|
+
CacheStats,
|
20
11
|
CleanupResult,
|
21
|
-
|
12
|
+
TaskResult,
|
13
|
+
UsageUpdateRequest,
|
14
|
+
UsageUpdateResult,
|
15
|
+
)
|
16
|
+
from .usage_tracking import (
|
17
|
+
batch_update_usage_counters,
|
18
|
+
cleanup_stale_usage_cache,
|
19
|
+
update_api_key_usage_async,
|
20
|
+
update_subscription_usage_async,
|
22
21
|
)
|
23
22
|
|
24
23
|
__all__ = [
|
@@ -27,7 +26,7 @@ __all__ = [
|
|
27
26
|
'update_subscription_usage_async',
|
28
27
|
'batch_update_usage_counters',
|
29
28
|
'cleanup_stale_usage_cache',
|
30
|
-
|
29
|
+
|
31
30
|
# Pydantic types
|
32
31
|
'TaskResult',
|
33
32
|
'UsageUpdateRequest',
|
@@ -2,15 +2,16 @@
|
|
2
2
|
Pydantic types for background tasks.
|
3
3
|
"""
|
4
4
|
|
5
|
-
from typing import Optional, Dict, Any, List
|
6
|
-
from pydantic import BaseModel, Field, ConfigDict
|
7
5
|
from datetime import datetime
|
6
|
+
from typing import Any, Dict, List, Optional
|
7
|
+
|
8
|
+
from pydantic import BaseModel, ConfigDict, Field
|
8
9
|
|
9
10
|
|
10
11
|
class TaskResult(BaseModel):
|
11
12
|
"""Base result type for all background tasks."""
|
12
13
|
model_config = ConfigDict(validate_assignment=True)
|
13
|
-
|
14
|
+
|
14
15
|
status: str = Field(description="Task execution status")
|
15
16
|
message: Optional[str] = Field(None, description="Human-readable message")
|
16
17
|
error: Optional[str] = Field(None, description="Error message if failed")
|
@@ -21,7 +22,7 @@ class TaskResult(BaseModel):
|
|
21
22
|
class UsageUpdateRequest(BaseModel):
|
22
23
|
"""Request for updating usage counters."""
|
23
24
|
model_config = ConfigDict(validate_assignment=True)
|
24
|
-
|
25
|
+
|
25
26
|
resource_id: str = Field(description="Resource ID (API key or subscription)")
|
26
27
|
increment: int = Field(default=1, description="Amount to increment")
|
27
28
|
ip_address: Optional[str] = Field(None, description="Client IP address")
|
@@ -30,7 +31,7 @@ class UsageUpdateRequest(BaseModel):
|
|
30
31
|
|
31
32
|
class UsageUpdateResult(TaskResult):
|
32
33
|
"""Result of usage update operation."""
|
33
|
-
|
34
|
+
|
34
35
|
resource_id: str = Field(description="Updated resource ID")
|
35
36
|
total_requests: Optional[int] = Field(None, description="Total requests after update")
|
36
37
|
increment: int = Field(description="Amount incremented")
|
@@ -40,7 +41,7 @@ class UsageUpdateResult(TaskResult):
|
|
40
41
|
class BatchUpdateRequest(BaseModel):
|
41
42
|
"""Request for batch usage updates."""
|
42
43
|
model_config = ConfigDict(validate_assignment=True)
|
43
|
-
|
44
|
+
|
44
45
|
api_key_updates: List[UsageUpdateRequest] = Field(default_factory=list, description="API key updates")
|
45
46
|
subscription_updates: List[UsageUpdateRequest] = Field(default_factory=list, description="Subscription updates")
|
46
47
|
force_flush: bool = Field(default=False, description="Force immediate processing")
|
@@ -48,7 +49,7 @@ class BatchUpdateRequest(BaseModel):
|
|
48
49
|
|
49
50
|
class BatchUpdateResult(TaskResult):
|
50
51
|
"""Result of batch update operation."""
|
51
|
-
|
52
|
+
|
52
53
|
api_keys_updated: int = Field(default=0, description="Number of API keys updated")
|
53
54
|
subscriptions_updated: int = Field(default=0, description="Number of subscriptions updated")
|
54
55
|
errors: List[Dict[str, Any]] = Field(default_factory=list, description="Processing errors")
|
@@ -57,7 +58,7 @@ class BatchUpdateResult(TaskResult):
|
|
57
58
|
|
58
59
|
class CleanupResult(TaskResult):
|
59
60
|
"""Result of cleanup operation."""
|
60
|
-
|
61
|
+
|
61
62
|
cleaned_entries: int = Field(default=0, description="Number of entries cleaned")
|
62
63
|
cleanup_type: str = Field(description="Type of cleanup performed")
|
63
64
|
cutoff_date: Optional[datetime] = Field(None, description="Cleanup cutoff date")
|
@@ -66,7 +67,7 @@ class CleanupResult(TaskResult):
|
|
66
67
|
class CacheStats(BaseModel):
|
67
68
|
"""Cache statistics."""
|
68
69
|
model_config = ConfigDict(validate_assignment=True)
|
69
|
-
|
70
|
+
|
70
71
|
total_keys: int = Field(description="Total cache keys")
|
71
72
|
expired_keys: int = Field(description="Expired cache keys")
|
72
73
|
memory_usage_mb: Optional[float] = Field(None, description="Memory usage in MB")
|
@@ -1,29 +1,27 @@
|
|
1
1
|
"""
|
2
2
|
Background tasks for API usage tracking and statistics.
|
3
3
|
"""
|
4
|
-
import dramatiq
|
5
4
|
import logging
|
6
5
|
import time
|
7
|
-
|
6
|
+
|
7
|
+
import dramatiq
|
8
|
+
from django.core.cache import cache
|
8
9
|
from django.db import transaction
|
9
10
|
from django.db.models import F
|
10
11
|
from django.utils import timezone
|
11
|
-
from django.core.cache import cache
|
12
12
|
|
13
13
|
from ..models import APIKey, Subscription
|
14
14
|
from .types import (
|
15
|
-
TaskResult,
|
16
|
-
UsageUpdateRequest,
|
17
|
-
UsageUpdateResult,
|
18
15
|
BatchUpdateResult,
|
19
|
-
CleanupResult
|
16
|
+
CleanupResult,
|
17
|
+
UsageUpdateResult,
|
20
18
|
)
|
21
19
|
|
22
20
|
logger = logging.getLogger(__name__)
|
23
21
|
|
24
22
|
@dramatiq.actor(queue_name="payments")
|
25
23
|
def update_api_key_usage_async(
|
26
|
-
api_key_id: str,
|
24
|
+
api_key_id: str,
|
27
25
|
ip_address: str = None,
|
28
26
|
increment: int = 1
|
29
27
|
) -> UsageUpdateResult:
|
@@ -39,7 +37,7 @@ def update_api_key_usage_async(
|
|
39
37
|
Update result with statistics
|
40
38
|
"""
|
41
39
|
start_time = time.time()
|
42
|
-
|
40
|
+
|
43
41
|
try:
|
44
42
|
with transaction.atomic():
|
45
43
|
# Use F() expressions for atomic updates
|
@@ -48,7 +46,7 @@ def update_api_key_usage_async(
|
|
48
46
|
last_used_at=timezone.now(),
|
49
47
|
updated_at=timezone.now()
|
50
48
|
)
|
51
|
-
|
49
|
+
|
52
50
|
if updated_count == 0:
|
53
51
|
logger.warning(f"API key not found: {api_key_id}")
|
54
52
|
return UsageUpdateResult(
|
@@ -57,13 +55,13 @@ def update_api_key_usage_async(
|
|
57
55
|
resource_id=api_key_id,
|
58
56
|
increment=increment
|
59
57
|
)
|
60
|
-
|
58
|
+
|
61
59
|
# Get updated values for logging
|
62
60
|
api_key = APIKey.objects.get(id=api_key_id)
|
63
|
-
|
61
|
+
|
64
62
|
processing_time = (time.time() - start_time) * 1000
|
65
|
-
|
66
|
-
logger.debug(
|
63
|
+
|
64
|
+
logger.debug("API key usage updated", extra={
|
67
65
|
'api_key_id': api_key_id,
|
68
66
|
'user_id': api_key.user.id,
|
69
67
|
'total_requests': api_key.total_requests,
|
@@ -71,7 +69,7 @@ def update_api_key_usage_async(
|
|
71
69
|
'ip_address': ip_address,
|
72
70
|
'processing_time_ms': round(processing_time, 2)
|
73
71
|
})
|
74
|
-
|
72
|
+
|
75
73
|
return UsageUpdateResult(
|
76
74
|
status='success',
|
77
75
|
resource_id=api_key_id,
|
@@ -80,9 +78,9 @@ def update_api_key_usage_async(
|
|
80
78
|
user_id=api_key.user.id,
|
81
79
|
processing_time_ms=round(processing_time, 2)
|
82
80
|
)
|
83
|
-
|
81
|
+
|
84
82
|
except Exception as e:
|
85
|
-
logger.error(
|
83
|
+
logger.error("Failed to update API key usage", extra={
|
86
84
|
'api_key_id': api_key_id,
|
87
85
|
'error': str(e),
|
88
86
|
'ip_address': ip_address
|
@@ -105,7 +103,7 @@ def update_subscription_usage_async(
|
|
105
103
|
Update result with statistics
|
106
104
|
"""
|
107
105
|
start_time = time.time()
|
108
|
-
|
106
|
+
|
109
107
|
try:
|
110
108
|
with transaction.atomic():
|
111
109
|
# Use F() expressions for atomic updates
|
@@ -114,7 +112,7 @@ def update_subscription_usage_async(
|
|
114
112
|
last_request_at=timezone.now(),
|
115
113
|
updated_at=timezone.now()
|
116
114
|
)
|
117
|
-
|
115
|
+
|
118
116
|
if updated_count == 0:
|
119
117
|
logger.warning(f"Subscription not found: {subscription_id}")
|
120
118
|
return UsageUpdateResult(
|
@@ -123,20 +121,20 @@ def update_subscription_usage_async(
|
|
123
121
|
resource_id=subscription_id,
|
124
122
|
increment=increment
|
125
123
|
)
|
126
|
-
|
124
|
+
|
127
125
|
# Get updated values for logging
|
128
126
|
subscription = Subscription.objects.get(id=subscription_id)
|
129
|
-
|
127
|
+
|
130
128
|
processing_time = (time.time() - start_time) * 1000
|
131
|
-
|
132
|
-
logger.debug(
|
129
|
+
|
130
|
+
logger.debug("Subscription usage updated", extra={
|
133
131
|
'subscription_id': subscription_id,
|
134
132
|
'user_id': subscription.user.id,
|
135
133
|
'total_requests': subscription.total_requests,
|
136
134
|
'increment': increment,
|
137
135
|
'processing_time_ms': round(processing_time, 2)
|
138
136
|
})
|
139
|
-
|
137
|
+
|
140
138
|
return UsageUpdateResult(
|
141
139
|
status='success',
|
142
140
|
resource_id=subscription_id,
|
@@ -145,9 +143,9 @@ def update_subscription_usage_async(
|
|
145
143
|
user_id=subscription.user.id,
|
146
144
|
processing_time_ms=round(processing_time, 2)
|
147
145
|
)
|
148
|
-
|
146
|
+
|
149
147
|
except Exception as e:
|
150
|
-
logger.error(
|
148
|
+
logger.error("Failed to update subscription usage", extra={
|
151
149
|
'subscription_id': subscription_id,
|
152
150
|
'error': str(e)
|
153
151
|
})
|
@@ -168,73 +166,73 @@ def batch_update_usage_counters() -> BatchUpdateResult:
|
|
168
166
|
api_keys_updated = 0
|
169
167
|
subscriptions_updated = 0
|
170
168
|
errors = []
|
171
|
-
|
169
|
+
|
172
170
|
try:
|
173
171
|
# Process API key usage counters
|
174
172
|
api_key_pattern = "api_usage_pending:*"
|
175
173
|
api_key_keys = cache.keys(api_key_pattern)
|
176
|
-
|
174
|
+
|
177
175
|
for cache_key in api_key_keys:
|
178
176
|
try:
|
179
177
|
# Extract API key ID from cache key
|
180
178
|
api_key_id = cache_key.split(':')[-1]
|
181
179
|
pending_count = cache.get(cache_key, 0)
|
182
|
-
|
180
|
+
|
183
181
|
if pending_count > 0:
|
184
182
|
# Update in background
|
185
183
|
update_api_key_usage_async.send(
|
186
184
|
api_key_id=api_key_id,
|
187
185
|
increment=pending_count
|
188
186
|
)
|
189
|
-
|
187
|
+
|
190
188
|
# Clear cache
|
191
189
|
cache.delete(cache_key)
|
192
190
|
api_keys_updated += 1
|
193
|
-
|
191
|
+
|
194
192
|
except Exception as e:
|
195
193
|
errors.append({
|
196
194
|
'type': 'api_key',
|
197
195
|
'cache_key': cache_key,
|
198
196
|
'error': str(e)
|
199
197
|
})
|
200
|
-
|
198
|
+
|
201
199
|
# Process subscription usage counters
|
202
200
|
subscription_pattern = "subscription_usage_pending:*"
|
203
201
|
subscription_keys = cache.keys(subscription_pattern)
|
204
|
-
|
202
|
+
|
205
203
|
for cache_key in subscription_keys:
|
206
204
|
try:
|
207
205
|
# Extract subscription ID from cache key
|
208
206
|
subscription_id = cache_key.split(':')[-1]
|
209
207
|
pending_count = cache.get(cache_key, 0)
|
210
|
-
|
208
|
+
|
211
209
|
if pending_count > 0:
|
212
210
|
# Update in background
|
213
211
|
update_subscription_usage_async.send(
|
214
212
|
subscription_id=subscription_id,
|
215
213
|
increment=pending_count
|
216
214
|
)
|
217
|
-
|
215
|
+
|
218
216
|
# Clear cache
|
219
217
|
cache.delete(cache_key)
|
220
218
|
subscriptions_updated += 1
|
221
|
-
|
219
|
+
|
222
220
|
except Exception as e:
|
223
221
|
errors.append({
|
224
222
|
'type': 'subscription',
|
225
223
|
'cache_key': cache_key,
|
226
224
|
'error': str(e)
|
227
225
|
})
|
228
|
-
|
226
|
+
|
229
227
|
processing_time = (time.time() - start_time) * 1000
|
230
|
-
|
231
|
-
logger.info(
|
228
|
+
|
229
|
+
logger.info("Batch usage update completed", extra={
|
232
230
|
'api_keys_updated': api_keys_updated,
|
233
231
|
'subscriptions_updated': subscriptions_updated,
|
234
232
|
'errors_count': len(errors),
|
235
233
|
'processing_time_ms': round(processing_time, 2)
|
236
234
|
})
|
237
|
-
|
235
|
+
|
238
236
|
return BatchUpdateResult(
|
239
237
|
status='success',
|
240
238
|
api_keys_updated=api_keys_updated,
|
@@ -243,7 +241,7 @@ def batch_update_usage_counters() -> BatchUpdateResult:
|
|
243
241
|
total_items=api_keys_updated + subscriptions_updated,
|
244
242
|
processing_time_ms=round(processing_time, 2)
|
245
243
|
)
|
246
|
-
|
244
|
+
|
247
245
|
except Exception as e:
|
248
246
|
logger.error(f"Batch usage update failed: {e}")
|
249
247
|
errors.append({
|
@@ -272,7 +270,7 @@ def cleanup_stale_usage_cache() -> CleanupResult:
|
|
272
270
|
"""
|
273
271
|
try:
|
274
272
|
cleanup_count = 0
|
275
|
-
|
273
|
+
|
276
274
|
# Cleanup old API key usage cache
|
277
275
|
api_key_keys = cache.keys("api_usage_pending:*")
|
278
276
|
for key in api_key_keys:
|
@@ -281,7 +279,7 @@ def cleanup_stale_usage_cache() -> CleanupResult:
|
|
281
279
|
if ttl is not None and ttl < 3600: # Less than 1 hour remaining
|
282
280
|
cache.delete(key)
|
283
281
|
cleanup_count += 1
|
284
|
-
|
282
|
+
|
285
283
|
# Cleanup old subscription usage cache
|
286
284
|
subscription_keys = cache.keys("subscription_usage_pending:*")
|
287
285
|
for key in subscription_keys:
|
@@ -289,15 +287,15 @@ def cleanup_stale_usage_cache() -> CleanupResult:
|
|
289
287
|
if ttl is not None and ttl < 3600:
|
290
288
|
cache.delete(key)
|
291
289
|
cleanup_count += 1
|
292
|
-
|
290
|
+
|
293
291
|
logger.info(f"Cleaned up {cleanup_count} stale cache entries")
|
294
|
-
|
292
|
+
|
295
293
|
return CleanupResult(
|
296
294
|
status='completed',
|
297
295
|
cleaned_entries=cleanup_count,
|
298
296
|
cleanup_type='stale_usage_cache'
|
299
297
|
)
|
300
|
-
|
298
|
+
|
301
299
|
except Exception as e:
|
302
300
|
logger.error(f"Cache cleanup failed: {e}")
|
303
301
|
return CleanupResult(
|