django-cfg 1.2.31__py3-none-any.whl → 1.3.3__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- django_cfg/__init__.py +1 -1
- django_cfg/apps/api/health/views.py +4 -2
- django_cfg/apps/knowbase/config/settings.py +16 -15
- django_cfg/apps/payments/README.md +326 -0
- django_cfg/apps/payments/admin/__init__.py +20 -10
- django_cfg/apps/payments/admin/api_keys_admin.py +521 -237
- django_cfg/apps/payments/admin/balance_admin.py +592 -297
- django_cfg/apps/payments/admin/currencies_admin.py +526 -222
- django_cfg/apps/payments/admin/filters.py +306 -199
- django_cfg/apps/payments/admin/payments_admin.py +465 -70
- django_cfg/apps/payments/admin/subscriptions_admin.py +578 -128
- django_cfg/apps/payments/admin_interface/__init__.py +18 -0
- django_cfg/apps/payments/admin_interface/templates/payments/base.html +162 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/dev_tool_card.html +38 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/loading_spinner.html +16 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/notification.html +27 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/provider_card.html +86 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/status_card.html +39 -0
- django_cfg/apps/payments/admin_interface/templates/payments/currency_converter.html +382 -0
- django_cfg/apps/payments/admin_interface/templates/payments/payment_dashboard.html +300 -0
- django_cfg/apps/payments/admin_interface/templates/payments/payment_form.html +303 -0
- django_cfg/apps/payments/admin_interface/templates/payments/payment_list.html +382 -0
- django_cfg/apps/payments/admin_interface/templates/payments/payment_status.html +500 -0
- django_cfg/apps/payments/admin_interface/templates/payments/webhook_dashboard.html +594 -0
- django_cfg/apps/payments/admin_interface/views/__init__.py +23 -0
- django_cfg/apps/payments/admin_interface/views/payment_views.py +259 -0
- django_cfg/apps/payments/admin_interface/views/webhook_dashboard.py +37 -0
- django_cfg/apps/payments/apps.py +34 -9
- django_cfg/apps/payments/config/__init__.py +28 -51
- django_cfg/apps/payments/config/constance/__init__.py +22 -0
- django_cfg/apps/payments/config/constance/config_service.py +123 -0
- django_cfg/apps/payments/config/constance/fields.py +69 -0
- django_cfg/apps/payments/config/constance/settings.py +160 -0
- django_cfg/apps/payments/config/django_cfg_integration.py +202 -0
- django_cfg/apps/payments/config/helpers.py +130 -0
- django_cfg/apps/payments/management/__init__.py +1 -3
- django_cfg/apps/payments/management/commands/__init__.py +1 -3
- django_cfg/apps/payments/management/commands/cleanup_expired_data.py +419 -0
- django_cfg/apps/payments/management/commands/currency_stats.py +297 -225
- django_cfg/apps/payments/management/commands/manage_currencies.py +303 -151
- django_cfg/apps/payments/management/commands/manage_providers.py +333 -160
- django_cfg/apps/payments/management/commands/process_pending_payments.py +357 -0
- django_cfg/apps/payments/management/commands/test_providers.py +434 -0
- django_cfg/apps/payments/middleware/__init__.py +3 -1
- django_cfg/apps/payments/middleware/api_access.py +329 -222
- django_cfg/apps/payments/middleware/rate_limiting.py +342 -152
- django_cfg/apps/payments/middleware/usage_tracking.py +249 -240
- django_cfg/apps/payments/migrations/0001_initial.py +708 -536
- django_cfg/apps/payments/models/__init__.py +13 -18
- django_cfg/apps/payments/models/api_keys.py +121 -43
- django_cfg/apps/payments/models/balance.py +153 -115
- django_cfg/apps/payments/models/base.py +68 -15
- django_cfg/apps/payments/models/currencies.py +172 -148
- django_cfg/apps/payments/models/managers/__init__.py +44 -0
- django_cfg/apps/payments/models/managers/api_key_managers.py +329 -0
- django_cfg/apps/payments/models/managers/balance_managers.py +599 -0
- django_cfg/apps/payments/models/managers/currency_managers.py +385 -0
- django_cfg/apps/payments/models/managers/payment_managers.py +511 -0
- django_cfg/apps/payments/models/managers/subscription_managers.py +641 -0
- django_cfg/apps/payments/models/payments.py +235 -285
- django_cfg/apps/payments/models/subscriptions.py +257 -177
- django_cfg/apps/payments/models/tariffs.py +147 -40
- django_cfg/apps/payments/services/__init__.py +209 -56
- django_cfg/apps/payments/services/cache/__init__.py +6 -6
- django_cfg/apps/payments/services/cache_service/__init__.py +143 -0
- django_cfg/apps/payments/services/cache_service/api_key_cache.py +37 -0
- django_cfg/apps/payments/services/{cache/base.py → cache_service/interfaces.py} +3 -1
- django_cfg/apps/payments/services/cache_service/keys.py +49 -0
- django_cfg/apps/payments/services/cache_service/rate_limit_cache.py +47 -0
- django_cfg/apps/payments/services/cache_service/simple_cache.py +101 -0
- django_cfg/apps/payments/services/core/__init__.py +10 -6
- django_cfg/apps/payments/services/core/balance_service.py +435 -360
- django_cfg/apps/payments/services/core/base.py +166 -0
- django_cfg/apps/payments/services/core/currency_service.py +478 -0
- django_cfg/apps/payments/services/core/payment_service.py +371 -465
- django_cfg/apps/payments/services/core/subscription_service.py +425 -481
- django_cfg/apps/payments/services/core/webhook_service.py +410 -0
- django_cfg/apps/payments/services/integrations/__init__.py +29 -0
- django_cfg/apps/payments/services/integrations/ngrok_service.py +47 -0
- django_cfg/apps/payments/services/integrations/providers_config.py +107 -0
- django_cfg/apps/payments/services/providers/__init__.py +9 -14
- django_cfg/apps/payments/services/providers/base.py +234 -174
- django_cfg/apps/payments/services/providers/nowpayments.py +478 -0
- django_cfg/apps/payments/services/providers/registry.py +367 -301
- django_cfg/apps/payments/services/types/__init__.py +78 -0
- django_cfg/apps/payments/services/types/data.py +177 -0
- django_cfg/apps/payments/services/types/requests.py +150 -0
- django_cfg/apps/payments/services/types/responses.py +156 -0
- django_cfg/apps/payments/services/types/webhooks.py +232 -0
- django_cfg/apps/payments/signals/__init__.py +33 -8
- django_cfg/apps/payments/signals/api_key_signals.py +210 -129
- django_cfg/apps/payments/signals/balance_signals.py +174 -0
- django_cfg/apps/payments/signals/payment_signals.py +128 -103
- django_cfg/apps/payments/signals/subscription_signals.py +194 -142
- django_cfg/apps/payments/static/payments/css/components.css +380 -0
- django_cfg/apps/payments/static/payments/css/dashboard.css +188 -0
- django_cfg/apps/payments/static/payments/js/components.js +545 -0
- django_cfg/apps/payments/static/payments/js/utils.js +412 -0
- django_cfg/apps/payments/templatetags/__init__.py +1 -1
- django_cfg/apps/payments/templatetags/payment_tags.py +466 -0
- django_cfg/apps/payments/urls.py +45 -48
- django_cfg/apps/payments/urls_admin.py +33 -42
- django_cfg/apps/payments/views/api/__init__.py +101 -0
- django_cfg/apps/payments/views/api/api_keys.py +387 -0
- django_cfg/apps/payments/views/api/balances.py +381 -0
- django_cfg/apps/payments/views/api/base.py +298 -0
- django_cfg/apps/payments/views/api/currencies.py +402 -0
- django_cfg/apps/payments/views/api/payments.py +415 -0
- django_cfg/apps/payments/views/api/subscriptions.py +475 -0
- django_cfg/apps/payments/views/api/webhooks.py +476 -0
- django_cfg/apps/payments/views/serializers/__init__.py +99 -0
- django_cfg/apps/payments/views/serializers/api_keys.py +424 -0
- django_cfg/apps/payments/views/serializers/balances.py +300 -0
- django_cfg/apps/payments/views/serializers/currencies.py +335 -0
- django_cfg/apps/payments/views/serializers/payments.py +387 -0
- django_cfg/apps/payments/views/serializers/subscriptions.py +429 -0
- django_cfg/apps/payments/views/serializers/webhooks.py +137 -0
- django_cfg/config.py +1 -1
- django_cfg/core/config.py +40 -4
- django_cfg/core/generation.py +25 -4
- django_cfg/core/integration/README.md +363 -0
- django_cfg/core/integration/__init__.py +47 -0
- django_cfg/core/integration/commands_collector.py +239 -0
- django_cfg/core/integration/display/__init__.py +15 -0
- django_cfg/core/integration/display/base.py +157 -0
- django_cfg/core/integration/display/ngrok.py +164 -0
- django_cfg/core/integration/display/startup.py +815 -0
- django_cfg/core/integration/url_integration.py +123 -0
- django_cfg/core/integration/version_checker.py +160 -0
- django_cfg/management/commands/auto_generate.py +4 -0
- django_cfg/management/commands/check_settings.py +6 -0
- django_cfg/management/commands/clear_constance.py +5 -2
- django_cfg/management/commands/create_token.py +6 -0
- django_cfg/management/commands/list_urls.py +6 -0
- django_cfg/management/commands/migrate_all.py +6 -0
- django_cfg/management/commands/migrator.py +3 -0
- django_cfg/management/commands/rundramatiq.py +6 -0
- django_cfg/management/commands/runserver_ngrok.py +51 -29
- django_cfg/management/commands/script.py +6 -0
- django_cfg/management/commands/show_config.py +12 -2
- django_cfg/management/commands/show_urls.py +4 -0
- django_cfg/management/commands/superuser.py +6 -0
- django_cfg/management/commands/task_clear.py +4 -1
- django_cfg/management/commands/task_status.py +3 -1
- django_cfg/management/commands/test_email.py +3 -0
- django_cfg/management/commands/test_telegram.py +6 -0
- django_cfg/management/commands/test_twilio.py +6 -0
- django_cfg/management/commands/tree.py +6 -0
- django_cfg/management/commands/validate_config.py +155 -149
- django_cfg/models/constance.py +31 -11
- django_cfg/models/payments.py +175 -492
- django_cfg/modules/django_logger.py +160 -146
- django_cfg/modules/django_unfold/dashboard.py +64 -16
- django_cfg/registry/core.py +1 -0
- django_cfg/template_archive/django_sample.zip +0 -0
- django_cfg/utils/smart_defaults.py +227 -570
- django_cfg/utils/toolkit.py +51 -11
- {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/METADATA +4 -1
- {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/RECORD +162 -185
- django_cfg/apps/payments/__init__.py +0 -8
- django_cfg/apps/payments/admin/tariffs_admin.py +0 -199
- django_cfg/apps/payments/config/module.py +0 -70
- django_cfg/apps/payments/config/providers.py +0 -105
- django_cfg/apps/payments/config/settings.py +0 -96
- django_cfg/apps/payments/config/utils.py +0 -52
- django_cfg/apps/payments/decorators.py +0 -291
- django_cfg/apps/payments/management/commands/README.md +0 -146
- django_cfg/apps/payments/managers/__init__.py +0 -23
- django_cfg/apps/payments/managers/api_key_manager.py +0 -35
- django_cfg/apps/payments/managers/balance_manager.py +0 -361
- django_cfg/apps/payments/managers/currency_manager.py +0 -306
- django_cfg/apps/payments/managers/payment_manager.py +0 -192
- django_cfg/apps/payments/managers/subscription_manager.py +0 -37
- django_cfg/apps/payments/managers/tariff_manager.py +0 -29
- django_cfg/apps/payments/migrations/0002_network_providercurrency_and_more.py +0 -241
- django_cfg/apps/payments/migrations/0003_add_usd_rate_cache.py +0 -30
- django_cfg/apps/payments/models/events.py +0 -73
- django_cfg/apps/payments/serializers/__init__.py +0 -57
- django_cfg/apps/payments/serializers/api_keys.py +0 -51
- django_cfg/apps/payments/serializers/balance.py +0 -59
- django_cfg/apps/payments/serializers/currencies.py +0 -63
- django_cfg/apps/payments/serializers/payments.py +0 -62
- django_cfg/apps/payments/serializers/subscriptions.py +0 -71
- django_cfg/apps/payments/serializers/tariffs.py +0 -56
- django_cfg/apps/payments/services/billing/__init__.py +0 -8
- django_cfg/apps/payments/services/cache/simple_cache.py +0 -135
- django_cfg/apps/payments/services/core/fallback_service.py +0 -432
- django_cfg/apps/payments/services/internal_types.py +0 -461
- django_cfg/apps/payments/services/middleware/__init__.py +0 -8
- django_cfg/apps/payments/services/monitoring/__init__.py +0 -22
- django_cfg/apps/payments/services/monitoring/api_schemas.py +0 -76
- django_cfg/apps/payments/services/monitoring/provider_health.py +0 -372
- django_cfg/apps/payments/services/providers/cryptapi/__init__.py +0 -4
- django_cfg/apps/payments/services/providers/cryptapi/config.py +0 -8
- django_cfg/apps/payments/services/providers/cryptapi/models.py +0 -192
- django_cfg/apps/payments/services/providers/cryptapi/provider.py +0 -439
- django_cfg/apps/payments/services/providers/cryptomus/__init__.py +0 -4
- django_cfg/apps/payments/services/providers/cryptomus/models.py +0 -176
- django_cfg/apps/payments/services/providers/cryptomus/provider.py +0 -429
- django_cfg/apps/payments/services/providers/cryptomus/provider_v2.py +0 -564
- django_cfg/apps/payments/services/providers/models/__init__.py +0 -34
- django_cfg/apps/payments/services/providers/models/currencies.py +0 -190
- django_cfg/apps/payments/services/providers/nowpayments/__init__.py +0 -4
- django_cfg/apps/payments/services/providers/nowpayments/models.py +0 -196
- django_cfg/apps/payments/services/providers/nowpayments/provider.py +0 -380
- django_cfg/apps/payments/services/providers/stripe/__init__.py +0 -4
- django_cfg/apps/payments/services/providers/stripe/models.py +0 -184
- django_cfg/apps/payments/services/providers/stripe/provider.py +0 -109
- django_cfg/apps/payments/services/security/__init__.py +0 -34
- django_cfg/apps/payments/services/security/error_handler.py +0 -635
- django_cfg/apps/payments/services/security/payment_notifications.py +0 -342
- django_cfg/apps/payments/services/security/webhook_validator.py +0 -474
- django_cfg/apps/payments/static/payments/css/payments.css +0 -340
- django_cfg/apps/payments/static/payments/js/notifications.js +0 -202
- django_cfg/apps/payments/static/payments/js/payment-utils.js +0 -318
- django_cfg/apps/payments/static/payments/js/theme.js +0 -86
- django_cfg/apps/payments/tasks/__init__.py +0 -12
- django_cfg/apps/payments/tasks/webhook_processing.py +0 -177
- django_cfg/apps/payments/templates/admin/payments/currency/change_list.html +0 -50
- django_cfg/apps/payments/templates/payments/base.html +0 -182
- django_cfg/apps/payments/templates/payments/components/payment_card.html +0 -201
- django_cfg/apps/payments/templates/payments/components/payment_qr_code.html +0 -109
- django_cfg/apps/payments/templates/payments/components/progress_bar.html +0 -43
- django_cfg/apps/payments/templates/payments/components/provider_stats.html +0 -40
- django_cfg/apps/payments/templates/payments/components/status_badge.html +0 -34
- django_cfg/apps/payments/templates/payments/components/status_overview.html +0 -148
- django_cfg/apps/payments/templates/payments/dashboard.html +0 -258
- django_cfg/apps/payments/templates/payments/dashboard_simple_test.html +0 -35
- django_cfg/apps/payments/templates/payments/payment_create.html +0 -579
- django_cfg/apps/payments/templates/payments/payment_detail.html +0 -373
- django_cfg/apps/payments/templates/payments/payment_list.html +0 -354
- django_cfg/apps/payments/templates/payments/stats.html +0 -261
- django_cfg/apps/payments/templates/payments/test.html +0 -213
- django_cfg/apps/payments/templatetags/payments_tags.py +0 -315
- django_cfg/apps/payments/utils/__init__.py +0 -43
- django_cfg/apps/payments/utils/billing_utils.py +0 -342
- django_cfg/apps/payments/utils/config_utils.py +0 -239
- django_cfg/apps/payments/utils/middleware_utils.py +0 -228
- django_cfg/apps/payments/utils/validation_utils.py +0 -94
- django_cfg/apps/payments/views/__init__.py +0 -63
- django_cfg/apps/payments/views/api_key_views.py +0 -164
- django_cfg/apps/payments/views/balance_views.py +0 -75
- django_cfg/apps/payments/views/currency_views.py +0 -122
- django_cfg/apps/payments/views/payment_views.py +0 -149
- django_cfg/apps/payments/views/subscription_views.py +0 -135
- django_cfg/apps/payments/views/tariff_views.py +0 -131
- django_cfg/apps/payments/views/templates/__init__.py +0 -25
- django_cfg/apps/payments/views/templates/ajax.py +0 -451
- django_cfg/apps/payments/views/templates/base.py +0 -212
- django_cfg/apps/payments/views/templates/dashboard.py +0 -60
- django_cfg/apps/payments/views/templates/payment_detail.py +0 -102
- django_cfg/apps/payments/views/templates/payment_management.py +0 -158
- django_cfg/apps/payments/views/templates/qr_code.py +0 -174
- django_cfg/apps/payments/views/templates/stats.py +0 -244
- django_cfg/apps/payments/views/templates/utils.py +0 -181
- django_cfg/apps/payments/views/webhook_views.py +0 -266
- django_cfg/apps/payments/viewsets.py +0 -66
- django_cfg/core/integration.py +0 -160
- django_cfg/template_archive/.gitignore +0 -1
- django_cfg/template_archive/__init__.py +0 -0
- django_cfg/urls.py +0 -33
- {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/WHEEL +0 -0
- {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/entry_points.txt +0 -0
- {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/licenses/LICENSE +0 -0
@@ -1,12 +1,23 @@
|
|
1
1
|
"""
|
2
|
-
Admin interface
|
2
|
+
Payment Admin interface with Unfold integration.
|
3
|
+
|
4
|
+
Advanced payment management with filtering, actions, and monitoring.
|
3
5
|
"""
|
4
6
|
|
5
7
|
from django.contrib import admin
|
6
8
|
from django.utils.html import format_html
|
7
|
-
from django.contrib.humanize.templatetags.humanize import naturaltime
|
9
|
+
from django.contrib.humanize.templatetags.humanize import naturaltime, intcomma
|
10
|
+
from django.contrib import messages
|
11
|
+
from django.shortcuts import redirect
|
12
|
+
from django.utils.safestring import mark_safe
|
13
|
+
from django.db.models import Count, Sum, Q
|
14
|
+
from django.utils import timezone
|
15
|
+
from datetime import timedelta
|
16
|
+
from typing import Optional
|
17
|
+
|
8
18
|
from unfold.admin import ModelAdmin
|
9
|
-
from unfold.decorators import display
|
19
|
+
from unfold.decorators import display, action
|
20
|
+
from unfold.enums import ActionVariant
|
10
21
|
|
11
22
|
from ..models import UniversalPayment
|
12
23
|
from .filters import PaymentStatusFilter, PaymentAmountFilter, UserEmailFilter, RecentActivityFilter
|
@@ -17,137 +28,521 @@ logger = get_logger("payments_admin")
|
|
17
28
|
|
18
29
|
@admin.register(UniversalPayment)
|
19
30
|
class UniversalPaymentAdmin(ModelAdmin):
|
20
|
-
"""
|
31
|
+
"""
|
32
|
+
Advanced Payment admin with Unfold styling and comprehensive management features.
|
33
|
+
|
34
|
+
Features:
|
35
|
+
- Real-time status tracking with visual indicators
|
36
|
+
- Advanced filtering and search capabilities
|
37
|
+
- Bulk operations for payment management
|
38
|
+
- Provider-specific information display
|
39
|
+
- Financial statistics and monitoring
|
40
|
+
"""
|
41
|
+
|
42
|
+
# Custom template for payment statistics
|
43
|
+
change_list_template = 'admin/payments/payment/change_list.html'
|
21
44
|
|
22
45
|
list_display = [
|
23
|
-
'
|
46
|
+
'payment_id_display',
|
24
47
|
'user_display',
|
25
48
|
'amount_display',
|
26
49
|
'status_display',
|
27
50
|
'provider_display',
|
51
|
+
'currency_display',
|
52
|
+
'progress_display',
|
28
53
|
'created_at_display'
|
29
54
|
]
|
30
55
|
|
31
|
-
list_display_links = ['
|
56
|
+
list_display_links = ['payment_id_display']
|
32
57
|
|
33
58
|
search_fields = [
|
34
|
-
'
|
59
|
+
'id',
|
35
60
|
'provider_payment_id',
|
36
61
|
'user__email',
|
37
62
|
'user__first_name',
|
38
|
-
'user__last_name'
|
63
|
+
'user__last_name',
|
64
|
+
'user__username',
|
65
|
+
'description'
|
39
66
|
]
|
40
67
|
|
41
|
-
def get_queryset(self, request):
|
42
|
-
"""Optimize queryset to prevent N+1 queries."""
|
43
|
-
return super().get_queryset(request).optimized()
|
44
|
-
|
45
68
|
list_filter = [
|
46
69
|
PaymentStatusFilter,
|
47
70
|
PaymentAmountFilter,
|
48
71
|
UserEmailFilter,
|
49
72
|
RecentActivityFilter,
|
50
73
|
'provider',
|
51
|
-
'
|
74
|
+
'currency',
|
52
75
|
'created_at'
|
53
76
|
]
|
54
77
|
|
55
78
|
readonly_fields = [
|
56
|
-
'
|
79
|
+
'id',
|
57
80
|
'provider_payment_id',
|
81
|
+
'payment_url',
|
58
82
|
'created_at',
|
59
|
-
'updated_at'
|
83
|
+
'updated_at',
|
84
|
+
'completed_at'
|
85
|
+
]
|
86
|
+
|
87
|
+
# Unfold actions
|
88
|
+
actions_list = [
|
89
|
+
'check_payment_status',
|
90
|
+
'cancel_selected_payments',
|
91
|
+
'mark_as_completed',
|
92
|
+
'export_payment_data'
|
60
93
|
]
|
61
94
|
|
62
95
|
fieldsets = [
|
63
96
|
('Payment Information', {
|
64
|
-
'fields': [
|
97
|
+
'fields': [
|
98
|
+
'id',
|
99
|
+
'user',
|
100
|
+
'amount_usd',
|
101
|
+
'currency',
|
102
|
+
'crypto_amount',
|
103
|
+
'description'
|
104
|
+
]
|
65
105
|
}),
|
66
|
-
('
|
67
|
-
'fields': [
|
106
|
+
('Provider Details', {
|
107
|
+
'fields': [
|
108
|
+
'provider',
|
109
|
+
'provider_payment_id',
|
110
|
+
'payment_url'
|
111
|
+
]
|
68
112
|
}),
|
69
|
-
('
|
70
|
-
'fields': [
|
113
|
+
('Status & Tracking', {
|
114
|
+
'fields': [
|
115
|
+
'status',
|
116
|
+
'error_code',
|
117
|
+
'error_message',
|
118
|
+
'expires_at'
|
119
|
+
]
|
120
|
+
}),
|
121
|
+
('URLs & Callbacks', {
|
122
|
+
'fields': [
|
123
|
+
'callback_url',
|
124
|
+
'cancel_url'
|
125
|
+
],
|
71
126
|
'classes': ['collapse']
|
72
127
|
}),
|
73
|
-
('
|
74
|
-
'fields': [
|
128
|
+
('Metadata', {
|
129
|
+
'fields': [
|
130
|
+
'metadata'
|
131
|
+
],
|
75
132
|
'classes': ['collapse']
|
76
133
|
}),
|
77
134
|
('Timestamps', {
|
78
|
-
'fields': [
|
135
|
+
'fields': [
|
136
|
+
'created_at',
|
137
|
+
'updated_at',
|
138
|
+
'completed_at'
|
139
|
+
],
|
79
140
|
'classes': ['collapse']
|
80
141
|
})
|
81
142
|
]
|
82
143
|
|
83
|
-
|
84
|
-
|
85
|
-
|
144
|
+
def get_queryset(self, request):
|
145
|
+
"""Optimize queryset with user data."""
|
146
|
+
return super().get_queryset(request).select_related('user')
|
147
|
+
|
148
|
+
@display(description="Payment ID", ordering='id')
|
149
|
+
def payment_id_display(self, obj):
|
150
|
+
"""Display payment ID with copy functionality."""
|
151
|
+
short_id = str(obj.id)[:8]
|
86
152
|
return format_html(
|
87
|
-
'<
|
88
|
-
|
89
|
-
obj.
|
153
|
+
'<span class="font-mono text-sm bg-gray-100 dark:bg-gray-800 px-2 py-1 rounded" '
|
154
|
+
'title="Click to copy full ID: {}">{}</span>',
|
155
|
+
obj.id,
|
156
|
+
short_id
|
90
157
|
)
|
91
158
|
|
92
|
-
@display(description="User")
|
159
|
+
@display(description="User", ordering='user__email')
|
93
160
|
def user_display(self, obj):
|
94
|
-
"""Display user information."""
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
|
99
|
-
|
161
|
+
"""Display user information with avatar."""
|
162
|
+
if obj.user:
|
163
|
+
display_name = obj.user.get_full_name() or obj.user.username
|
164
|
+
return format_html(
|
165
|
+
'<div class="flex items-center space-x-2">'
|
166
|
+
'<div class="w-8 h-8 bg-blue-500 rounded-full flex items-center justify-center text-white text-xs font-bold">'
|
167
|
+
'{}'
|
168
|
+
'</div>'
|
169
|
+
'<div>'
|
170
|
+
'<div class="font-medium text-gray-900 dark:text-gray-100">{}</div>'
|
171
|
+
'<div class="text-xs text-gray-500">{}</div>'
|
172
|
+
'</div>'
|
173
|
+
'</div>',
|
174
|
+
display_name[0].upper() if display_name else 'U',
|
175
|
+
display_name,
|
176
|
+
obj.user.email
|
177
|
+
)
|
178
|
+
return format_html('<span class="text-gray-500">No user</span>')
|
100
179
|
|
101
|
-
@display(description="Amount")
|
180
|
+
@display(description="Amount", ordering='amount_usd')
|
102
181
|
def amount_display(self, obj):
|
103
|
-
"""Display amount with currency."""
|
182
|
+
"""Display amount with currency conversion."""
|
183
|
+
usd_amount = f"${obj.amount_usd:,.2f}"
|
184
|
+
|
185
|
+
if obj.amount_crypto and obj.currency:
|
186
|
+
crypto_display = f"{obj.amount_crypto:.8f}".rstrip('0').rstrip('.')
|
187
|
+
return format_html(
|
188
|
+
'<div class="text-right">'
|
189
|
+
'<div class="font-bold text-green-600 dark:text-green-400">{}</div>'
|
190
|
+
'<div class="text-xs text-gray-500">{} {}</div>'
|
191
|
+
'</div>',
|
192
|
+
usd_amount,
|
193
|
+
crypto_display,
|
194
|
+
obj.currency.code
|
195
|
+
)
|
196
|
+
|
104
197
|
return format_html(
|
105
|
-
'<
|
106
|
-
|
107
|
-
obj.currency_code
|
198
|
+
'<div class="text-right font-bold text-green-600 dark:text-green-400">{}</div>',
|
199
|
+
usd_amount
|
108
200
|
)
|
109
201
|
|
110
|
-
@display(description="Status")
|
202
|
+
@display(description="Status", ordering='status')
|
111
203
|
def status_display(self, obj):
|
112
|
-
"""Display status with
|
113
|
-
|
114
|
-
'
|
115
|
-
'
|
116
|
-
'
|
117
|
-
'
|
118
|
-
'
|
119
|
-
'
|
120
|
-
'
|
121
|
-
'
|
204
|
+
"""Display status with colored badge and icon."""
|
205
|
+
status_config = {
|
206
|
+
UniversalPayment.PaymentStatus.PENDING: ('🟡', 'bg-yellow-100 text-yellow-800 dark:bg-yellow-900 dark:text-yellow-200', 'Pending'),
|
207
|
+
UniversalPayment.PaymentStatus.WAITING_FOR_PAYMENT: ('⏰', 'bg-blue-100 text-blue-800 dark:bg-blue-900 dark:text-blue-200', 'Waiting'),
|
208
|
+
UniversalPayment.PaymentStatus.CONFIRMING: ('🔄', 'bg-purple-100 text-purple-800 dark:bg-purple-900 dark:text-purple-200', 'Confirming'),
|
209
|
+
UniversalPayment.PaymentStatus.COMPLETED: ('✅', 'bg-green-100 text-green-800 dark:bg-green-900 dark:text-green-200', 'Completed'),
|
210
|
+
UniversalPayment.PaymentStatus.FAILED: ('❌', 'bg-red-100 text-red-800 dark:bg-red-900 dark:text-red-200', 'Failed'),
|
211
|
+
UniversalPayment.PaymentStatus.CANCELLED: ('🚫', 'bg-gray-100 text-gray-800 dark:bg-gray-900 dark:text-gray-200', 'Cancelled'),
|
212
|
+
UniversalPayment.PaymentStatus.EXPIRED: ('⌛', 'bg-orange-100 text-orange-800 dark:bg-orange-900 dark:text-orange-200', 'Expired'),
|
213
|
+
UniversalPayment.PaymentStatus.REFUNDED: ('↩️', 'bg-indigo-100 text-indigo-800 dark:bg-indigo-900 dark:text-indigo-200', 'Refunded'),
|
122
214
|
}
|
123
215
|
|
124
|
-
|
216
|
+
icon, color_class, label = status_config.get(
|
217
|
+
obj.status,
|
218
|
+
('❓', 'bg-gray-100 text-gray-800', 'Unknown')
|
219
|
+
)
|
125
220
|
|
126
221
|
return format_html(
|
127
|
-
'<span
|
128
|
-
|
129
|
-
|
222
|
+
'<span class="inline-flex items-center rounded-full px-2.5 py-0.5 text-xs font-medium {}">'
|
223
|
+
'{} {}'
|
224
|
+
'</span>',
|
225
|
+
color_class,
|
226
|
+
icon,
|
227
|
+
label
|
130
228
|
)
|
131
229
|
|
132
|
-
@display(description="Provider")
|
230
|
+
@display(description="Provider", ordering='provider')
|
133
231
|
def provider_display(self, obj):
|
134
|
-
"""Display provider with
|
135
|
-
|
136
|
-
'nowpayments': '
|
137
|
-
'
|
138
|
-
'
|
232
|
+
"""Display provider with logo/icon."""
|
233
|
+
provider_config = {
|
234
|
+
'nowpayments': ('🟦', 'NowPayments'),
|
235
|
+
'cryptomus': ('🟩', 'Cryptomus'),
|
236
|
+
'cryptapi': ('🟪', 'CryptAPI'),
|
139
237
|
}
|
140
238
|
|
141
|
-
|
239
|
+
icon, name = provider_config.get(obj.provider, ('🔷', obj.provider.title()))
|
240
|
+
|
241
|
+
return format_html(
|
242
|
+
'<span class="inline-flex items-center space-x-1">'
|
243
|
+
'<span>{}</span>'
|
244
|
+
'<span class="text-sm font-medium">{}</span>'
|
245
|
+
'</span>',
|
246
|
+
icon,
|
247
|
+
name
|
248
|
+
)
|
249
|
+
|
250
|
+
@display(description="Currency", ordering='currency__code')
|
251
|
+
def currency_display(self, obj):
|
252
|
+
"""Display currency with type indicator."""
|
253
|
+
if obj.currency:
|
254
|
+
# Use currency type from model
|
255
|
+
is_crypto = obj.currency.currency_type == 'crypto'
|
256
|
+
|
257
|
+
icon = '₿' if is_crypto else '💰'
|
258
|
+
|
259
|
+
return format_html(
|
260
|
+
'<span class="inline-flex items-center space-x-1">'
|
261
|
+
'<span>{}</span>'
|
262
|
+
'<span class="font-mono font-bold">{}</span>'
|
263
|
+
'</span>',
|
264
|
+
icon,
|
265
|
+
obj.currency.code
|
266
|
+
)
|
267
|
+
|
268
|
+
return format_html('<span class="text-gray-500">—</span>')
|
269
|
+
|
270
|
+
@display(description="Progress")
|
271
|
+
def progress_display(self, obj):
|
272
|
+
"""Display payment progress with time information."""
|
273
|
+
now = timezone.now()
|
274
|
+
|
275
|
+
# Calculate time since creation
|
276
|
+
time_elapsed = now - obj.created_at
|
277
|
+
|
278
|
+
# Check if expired
|
279
|
+
if obj.expires_at and now > obj.expires_at:
|
280
|
+
return format_html(
|
281
|
+
'<div class="text-red-500 text-xs">'
|
282
|
+
'⌛ Expired<br>'
|
283
|
+
'<span class="text-gray-400">{}</span>'
|
284
|
+
'</div>',
|
285
|
+
naturaltime(obj.expires_at)
|
286
|
+
)
|
142
287
|
|
288
|
+
# Show time remaining if has expiry
|
289
|
+
if obj.expires_at:
|
290
|
+
time_remaining = obj.expires_at - now
|
291
|
+
if time_remaining.total_seconds() > 0:
|
292
|
+
return format_html(
|
293
|
+
'<div class="text-orange-500 text-xs">'
|
294
|
+
'⏰ {} left<br>'
|
295
|
+
'<span class="text-gray-400">Created {}</span>'
|
296
|
+
'</div>',
|
297
|
+
naturaltime(now + time_remaining),
|
298
|
+
naturaltime(obj.created_at)
|
299
|
+
)
|
300
|
+
|
301
|
+
# Default: show creation time
|
143
302
|
return format_html(
|
144
|
-
'<
|
145
|
-
|
146
|
-
|
147
|
-
|
303
|
+
'<div class="text-gray-500 text-xs">'
|
304
|
+
'Created<br>'
|
305
|
+
'<span>{}</span>'
|
306
|
+
'</div>',
|
307
|
+
naturaltime(obj.created_at)
|
148
308
|
)
|
149
309
|
|
150
|
-
@display(description="Created")
|
310
|
+
@display(description="Created", ordering='created_at')
|
151
311
|
def created_at_display(self, obj):
|
152
|
-
"""Display creation date."""
|
153
|
-
return
|
312
|
+
"""Display creation date with relative time."""
|
313
|
+
return format_html(
|
314
|
+
'<div class="text-xs">'
|
315
|
+
'<div class="font-medium">{}</div>'
|
316
|
+
'<div class="text-gray-500">{}</div>'
|
317
|
+
'</div>',
|
318
|
+
obj.created_at.strftime('%Y-%m-%d %H:%M'),
|
319
|
+
naturaltime(obj.created_at)
|
320
|
+
)
|
321
|
+
|
322
|
+
def changelist_view(self, request, extra_context=None):
|
323
|
+
"""Add payment statistics to changelist context."""
|
324
|
+
extra_context = extra_context or {}
|
325
|
+
|
326
|
+
try:
|
327
|
+
# Basic statistics
|
328
|
+
total_payments = UniversalPayment.objects.count()
|
329
|
+
|
330
|
+
# Status distribution
|
331
|
+
status_stats = {}
|
332
|
+
for status in UniversalPayment.PaymentStatus:
|
333
|
+
count = UniversalPayment.objects.filter(status=status).count()
|
334
|
+
status_stats[status] = count
|
335
|
+
|
336
|
+
# Financial statistics
|
337
|
+
total_amount = UniversalPayment.objects.aggregate(
|
338
|
+
total=Sum('amount_usd')
|
339
|
+
)['total'] or 0
|
340
|
+
|
341
|
+
completed_amount = UniversalPayment.objects.filter(
|
342
|
+
status=UniversalPayment.PaymentStatus.COMPLETED
|
343
|
+
).aggregate(total=Sum('amount_usd'))['total'] or 0
|
344
|
+
|
345
|
+
# Recent activity (24 hours)
|
346
|
+
recent_threshold = timezone.now() - timedelta(hours=24)
|
347
|
+
recent_payments = UniversalPayment.objects.filter(
|
348
|
+
created_at__gte=recent_threshold
|
349
|
+
).count()
|
350
|
+
|
351
|
+
recent_amount = UniversalPayment.objects.filter(
|
352
|
+
created_at__gte=recent_threshold
|
353
|
+
).aggregate(total=Sum('amount_usd'))['total'] or 0
|
354
|
+
|
355
|
+
# Provider statistics
|
356
|
+
provider_stats = UniversalPayment.objects.values('provider').annotate(
|
357
|
+
count=Count('id'),
|
358
|
+
amount=Sum('amount_usd')
|
359
|
+
).order_by('-count')
|
360
|
+
|
361
|
+
# Success rate
|
362
|
+
completed_count = status_stats.get(UniversalPayment.PaymentStatus.COMPLETED, 0)
|
363
|
+
success_rate = (completed_count / total_payments * 100) if total_payments > 0 else 0
|
364
|
+
|
365
|
+
extra_context.update({
|
366
|
+
'payment_stats': {
|
367
|
+
'total_payments': total_payments,
|
368
|
+
'status_stats': status_stats,
|
369
|
+
'total_amount': total_amount,
|
370
|
+
'completed_amount': completed_amount,
|
371
|
+
'recent_payments': recent_payments,
|
372
|
+
'recent_amount': recent_amount,
|
373
|
+
'provider_stats': provider_stats,
|
374
|
+
'success_rate': success_rate,
|
375
|
+
}
|
376
|
+
})
|
377
|
+
|
378
|
+
except Exception as e:
|
379
|
+
logger.warning(f"Failed to generate payment statistics: {e}")
|
380
|
+
extra_context['payment_stats'] = None
|
381
|
+
|
382
|
+
return super().changelist_view(request, extra_context)
|
383
|
+
|
384
|
+
# ===== ADMIN ACTIONS =====
|
385
|
+
|
386
|
+
@action(
|
387
|
+
description="🔍 Check Payment Status",
|
388
|
+
icon="refresh",
|
389
|
+
variant=ActionVariant.INFO
|
390
|
+
)
|
391
|
+
def check_payment_status(self, request, queryset):
|
392
|
+
"""Check payment status with providers."""
|
393
|
+
|
394
|
+
updated_count = 0
|
395
|
+
error_count = 0
|
396
|
+
|
397
|
+
for payment in queryset:
|
398
|
+
try:
|
399
|
+
# Use payment service to check status
|
400
|
+
from ..services.core.payment_service import PaymentService
|
401
|
+
|
402
|
+
service = PaymentService()
|
403
|
+
result = service.check_payment_status(payment.id)
|
404
|
+
|
405
|
+
if result.success:
|
406
|
+
updated_count += 1
|
407
|
+
else:
|
408
|
+
error_count += 1
|
409
|
+
|
410
|
+
except Exception as e:
|
411
|
+
error_count += 1
|
412
|
+
logger.error(f"Failed to check payment status for {payment.id}: {e}")
|
413
|
+
|
414
|
+
if updated_count > 0:
|
415
|
+
messages.success(
|
416
|
+
request,
|
417
|
+
f"✅ Checked status for {updated_count} payments"
|
418
|
+
)
|
419
|
+
|
420
|
+
if error_count > 0:
|
421
|
+
messages.warning(
|
422
|
+
request,
|
423
|
+
f"⚠️ Failed to check {error_count} payments"
|
424
|
+
)
|
425
|
+
|
426
|
+
@action(
|
427
|
+
description="🚫 Cancel Selected Payments",
|
428
|
+
icon="cancel",
|
429
|
+
variant=ActionVariant.WARNING
|
430
|
+
)
|
431
|
+
def cancel_selected_payments(self, request, queryset):
|
432
|
+
"""Cancel selected payments."""
|
433
|
+
|
434
|
+
# Only allow cancellation of pending/waiting payments
|
435
|
+
cancelable_payments = queryset.filter(
|
436
|
+
status__in=[
|
437
|
+
UniversalPayment.PaymentStatus.PENDING,
|
438
|
+
UniversalPayment.PaymentStatus.WAITING_FOR_PAYMENT
|
439
|
+
]
|
440
|
+
)
|
441
|
+
|
442
|
+
cancelled_count = 0
|
443
|
+
|
444
|
+
for payment in cancelable_payments:
|
445
|
+
try:
|
446
|
+
payment.mark_cancelled(reason="Cancelled by admin")
|
447
|
+
cancelled_count += 1
|
448
|
+
|
449
|
+
except Exception as e:
|
450
|
+
logger.error(f"Failed to cancel payment {payment.id}: {e}")
|
451
|
+
|
452
|
+
if cancelled_count > 0:
|
453
|
+
messages.success(
|
454
|
+
request,
|
455
|
+
f"🚫 Cancelled {cancelled_count} payments"
|
456
|
+
)
|
457
|
+
|
458
|
+
skipped_count = queryset.count() - cancelled_count
|
459
|
+
if skipped_count > 0:
|
460
|
+
messages.info(
|
461
|
+
request,
|
462
|
+
f"ℹ️ Skipped {skipped_count} payments (not cancelable)"
|
463
|
+
)
|
464
|
+
|
465
|
+
@action(
|
466
|
+
description="✅ Mark as Completed",
|
467
|
+
icon="check_circle",
|
468
|
+
variant=ActionVariant.SUCCESS
|
469
|
+
)
|
470
|
+
def mark_as_completed(self, request, queryset):
|
471
|
+
"""Mark selected payments as completed (admin override)."""
|
472
|
+
|
473
|
+
# Only allow completion of pending/waiting/confirming payments
|
474
|
+
completable_payments = queryset.filter(
|
475
|
+
status__in=[
|
476
|
+
UniversalPayment.PaymentStatus.PENDING,
|
477
|
+
UniversalPayment.PaymentStatus.WAITING_FOR_PAYMENT,
|
478
|
+
UniversalPayment.PaymentStatus.CONFIRMING
|
479
|
+
]
|
480
|
+
)
|
481
|
+
|
482
|
+
completed_count = 0
|
483
|
+
|
484
|
+
for payment in completable_payments:
|
485
|
+
try:
|
486
|
+
payment.mark_completed()
|
487
|
+
completed_count += 1
|
488
|
+
|
489
|
+
except Exception as e:
|
490
|
+
logger.error(f"Failed to complete payment {payment.id}: {e}")
|
491
|
+
|
492
|
+
if completed_count > 0:
|
493
|
+
messages.success(
|
494
|
+
request,
|
495
|
+
f"✅ Marked {completed_count} payments as completed"
|
496
|
+
)
|
497
|
+
messages.warning(
|
498
|
+
request,
|
499
|
+
"⚠️ Admin override used - ensure payments were actually received!"
|
500
|
+
)
|
501
|
+
|
502
|
+
skipped_count = queryset.count() - completed_count
|
503
|
+
if skipped_count > 0:
|
504
|
+
messages.info(
|
505
|
+
request,
|
506
|
+
f"ℹ️ Skipped {skipped_count} payments (not completable)"
|
507
|
+
)
|
508
|
+
|
509
|
+
@action(
|
510
|
+
description="📊 Export Payment Data",
|
511
|
+
icon="download",
|
512
|
+
variant=ActionVariant.INFO
|
513
|
+
)
|
514
|
+
def export_payment_data(self, request, queryset):
|
515
|
+
"""Export selected payments to CSV."""
|
516
|
+
|
517
|
+
import csv
|
518
|
+
from django.http import HttpResponse
|
519
|
+
|
520
|
+
response = HttpResponse(content_type='text/csv')
|
521
|
+
response['Content-Disposition'] = f'attachment; filename="payments_{timezone.now().strftime("%Y%m%d_%H%M%S")}.csv"'
|
522
|
+
|
523
|
+
writer = csv.writer(response)
|
524
|
+
writer.writerow([
|
525
|
+
'ID', 'User Email', 'Amount USD', 'Currency', 'Crypto Amount',
|
526
|
+
'Status', 'Provider', 'Created', 'Completed', 'Description'
|
527
|
+
])
|
528
|
+
|
529
|
+
for payment in queryset:
|
530
|
+
writer.writerow([
|
531
|
+
str(payment.id),
|
532
|
+
payment.user.email if payment.user else '',
|
533
|
+
payment.amount_usd,
|
534
|
+
payment.currency.code if payment.currency else '',
|
535
|
+
payment.amount_crypto or '',
|
536
|
+
payment.status,
|
537
|
+
payment.provider,
|
538
|
+
payment.created_at.isoformat(),
|
539
|
+
payment.completed_at.isoformat() if payment.completed_at else '',
|
540
|
+
payment.description or ''
|
541
|
+
])
|
542
|
+
|
543
|
+
messages.success(
|
544
|
+
request,
|
545
|
+
f"📊 Exported {queryset.count()} payments to CSV"
|
546
|
+
)
|
547
|
+
|
548
|
+
return response
|