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.
Files changed (264) hide show
  1. django_cfg/__init__.py +1 -1
  2. django_cfg/apps/api/health/views.py +4 -2
  3. django_cfg/apps/knowbase/config/settings.py +16 -15
  4. django_cfg/apps/payments/README.md +326 -0
  5. django_cfg/apps/payments/admin/__init__.py +20 -10
  6. django_cfg/apps/payments/admin/api_keys_admin.py +521 -237
  7. django_cfg/apps/payments/admin/balance_admin.py +592 -297
  8. django_cfg/apps/payments/admin/currencies_admin.py +526 -222
  9. django_cfg/apps/payments/admin/filters.py +306 -199
  10. django_cfg/apps/payments/admin/payments_admin.py +465 -70
  11. django_cfg/apps/payments/admin/subscriptions_admin.py +578 -128
  12. django_cfg/apps/payments/admin_interface/__init__.py +18 -0
  13. django_cfg/apps/payments/admin_interface/templates/payments/base.html +162 -0
  14. django_cfg/apps/payments/admin_interface/templates/payments/components/dev_tool_card.html +38 -0
  15. django_cfg/apps/payments/admin_interface/templates/payments/components/loading_spinner.html +16 -0
  16. django_cfg/apps/payments/admin_interface/templates/payments/components/notification.html +27 -0
  17. django_cfg/apps/payments/admin_interface/templates/payments/components/provider_card.html +86 -0
  18. django_cfg/apps/payments/admin_interface/templates/payments/components/status_card.html +39 -0
  19. django_cfg/apps/payments/admin_interface/templates/payments/currency_converter.html +382 -0
  20. django_cfg/apps/payments/admin_interface/templates/payments/payment_dashboard.html +300 -0
  21. django_cfg/apps/payments/admin_interface/templates/payments/payment_form.html +303 -0
  22. django_cfg/apps/payments/admin_interface/templates/payments/payment_list.html +382 -0
  23. django_cfg/apps/payments/admin_interface/templates/payments/payment_status.html +500 -0
  24. django_cfg/apps/payments/admin_interface/templates/payments/webhook_dashboard.html +594 -0
  25. django_cfg/apps/payments/admin_interface/views/__init__.py +23 -0
  26. django_cfg/apps/payments/admin_interface/views/payment_views.py +259 -0
  27. django_cfg/apps/payments/admin_interface/views/webhook_dashboard.py +37 -0
  28. django_cfg/apps/payments/apps.py +34 -9
  29. django_cfg/apps/payments/config/__init__.py +28 -51
  30. django_cfg/apps/payments/config/constance/__init__.py +22 -0
  31. django_cfg/apps/payments/config/constance/config_service.py +123 -0
  32. django_cfg/apps/payments/config/constance/fields.py +69 -0
  33. django_cfg/apps/payments/config/constance/settings.py +160 -0
  34. django_cfg/apps/payments/config/django_cfg_integration.py +202 -0
  35. django_cfg/apps/payments/config/helpers.py +130 -0
  36. django_cfg/apps/payments/management/__init__.py +1 -3
  37. django_cfg/apps/payments/management/commands/__init__.py +1 -3
  38. django_cfg/apps/payments/management/commands/cleanup_expired_data.py +419 -0
  39. django_cfg/apps/payments/management/commands/currency_stats.py +297 -225
  40. django_cfg/apps/payments/management/commands/manage_currencies.py +303 -151
  41. django_cfg/apps/payments/management/commands/manage_providers.py +333 -160
  42. django_cfg/apps/payments/management/commands/process_pending_payments.py +357 -0
  43. django_cfg/apps/payments/management/commands/test_providers.py +434 -0
  44. django_cfg/apps/payments/middleware/__init__.py +3 -1
  45. django_cfg/apps/payments/middleware/api_access.py +329 -222
  46. django_cfg/apps/payments/middleware/rate_limiting.py +342 -152
  47. django_cfg/apps/payments/middleware/usage_tracking.py +249 -240
  48. django_cfg/apps/payments/migrations/0001_initial.py +708 -536
  49. django_cfg/apps/payments/models/__init__.py +13 -18
  50. django_cfg/apps/payments/models/api_keys.py +121 -43
  51. django_cfg/apps/payments/models/balance.py +153 -115
  52. django_cfg/apps/payments/models/base.py +68 -15
  53. django_cfg/apps/payments/models/currencies.py +172 -148
  54. django_cfg/apps/payments/models/managers/__init__.py +44 -0
  55. django_cfg/apps/payments/models/managers/api_key_managers.py +329 -0
  56. django_cfg/apps/payments/models/managers/balance_managers.py +599 -0
  57. django_cfg/apps/payments/models/managers/currency_managers.py +385 -0
  58. django_cfg/apps/payments/models/managers/payment_managers.py +511 -0
  59. django_cfg/apps/payments/models/managers/subscription_managers.py +641 -0
  60. django_cfg/apps/payments/models/payments.py +235 -285
  61. django_cfg/apps/payments/models/subscriptions.py +257 -177
  62. django_cfg/apps/payments/models/tariffs.py +147 -40
  63. django_cfg/apps/payments/services/__init__.py +209 -56
  64. django_cfg/apps/payments/services/cache/__init__.py +6 -6
  65. django_cfg/apps/payments/services/cache_service/__init__.py +143 -0
  66. django_cfg/apps/payments/services/cache_service/api_key_cache.py +37 -0
  67. django_cfg/apps/payments/services/{cache/base.py → cache_service/interfaces.py} +3 -1
  68. django_cfg/apps/payments/services/cache_service/keys.py +49 -0
  69. django_cfg/apps/payments/services/cache_service/rate_limit_cache.py +47 -0
  70. django_cfg/apps/payments/services/cache_service/simple_cache.py +101 -0
  71. django_cfg/apps/payments/services/core/__init__.py +10 -6
  72. django_cfg/apps/payments/services/core/balance_service.py +435 -360
  73. django_cfg/apps/payments/services/core/base.py +166 -0
  74. django_cfg/apps/payments/services/core/currency_service.py +478 -0
  75. django_cfg/apps/payments/services/core/payment_service.py +371 -465
  76. django_cfg/apps/payments/services/core/subscription_service.py +425 -481
  77. django_cfg/apps/payments/services/core/webhook_service.py +410 -0
  78. django_cfg/apps/payments/services/integrations/__init__.py +29 -0
  79. django_cfg/apps/payments/services/integrations/ngrok_service.py +47 -0
  80. django_cfg/apps/payments/services/integrations/providers_config.py +107 -0
  81. django_cfg/apps/payments/services/providers/__init__.py +9 -14
  82. django_cfg/apps/payments/services/providers/base.py +234 -174
  83. django_cfg/apps/payments/services/providers/nowpayments.py +478 -0
  84. django_cfg/apps/payments/services/providers/registry.py +367 -301
  85. django_cfg/apps/payments/services/types/__init__.py +78 -0
  86. django_cfg/apps/payments/services/types/data.py +177 -0
  87. django_cfg/apps/payments/services/types/requests.py +150 -0
  88. django_cfg/apps/payments/services/types/responses.py +156 -0
  89. django_cfg/apps/payments/services/types/webhooks.py +232 -0
  90. django_cfg/apps/payments/signals/__init__.py +33 -8
  91. django_cfg/apps/payments/signals/api_key_signals.py +210 -129
  92. django_cfg/apps/payments/signals/balance_signals.py +174 -0
  93. django_cfg/apps/payments/signals/payment_signals.py +128 -103
  94. django_cfg/apps/payments/signals/subscription_signals.py +194 -142
  95. django_cfg/apps/payments/static/payments/css/components.css +380 -0
  96. django_cfg/apps/payments/static/payments/css/dashboard.css +188 -0
  97. django_cfg/apps/payments/static/payments/js/components.js +545 -0
  98. django_cfg/apps/payments/static/payments/js/utils.js +412 -0
  99. django_cfg/apps/payments/templatetags/__init__.py +1 -1
  100. django_cfg/apps/payments/templatetags/payment_tags.py +466 -0
  101. django_cfg/apps/payments/urls.py +45 -48
  102. django_cfg/apps/payments/urls_admin.py +33 -42
  103. django_cfg/apps/payments/views/api/__init__.py +101 -0
  104. django_cfg/apps/payments/views/api/api_keys.py +387 -0
  105. django_cfg/apps/payments/views/api/balances.py +381 -0
  106. django_cfg/apps/payments/views/api/base.py +298 -0
  107. django_cfg/apps/payments/views/api/currencies.py +402 -0
  108. django_cfg/apps/payments/views/api/payments.py +415 -0
  109. django_cfg/apps/payments/views/api/subscriptions.py +475 -0
  110. django_cfg/apps/payments/views/api/webhooks.py +476 -0
  111. django_cfg/apps/payments/views/serializers/__init__.py +99 -0
  112. django_cfg/apps/payments/views/serializers/api_keys.py +424 -0
  113. django_cfg/apps/payments/views/serializers/balances.py +300 -0
  114. django_cfg/apps/payments/views/serializers/currencies.py +335 -0
  115. django_cfg/apps/payments/views/serializers/payments.py +387 -0
  116. django_cfg/apps/payments/views/serializers/subscriptions.py +429 -0
  117. django_cfg/apps/payments/views/serializers/webhooks.py +137 -0
  118. django_cfg/config.py +1 -1
  119. django_cfg/core/config.py +40 -4
  120. django_cfg/core/generation.py +25 -4
  121. django_cfg/core/integration/README.md +363 -0
  122. django_cfg/core/integration/__init__.py +47 -0
  123. django_cfg/core/integration/commands_collector.py +239 -0
  124. django_cfg/core/integration/display/__init__.py +15 -0
  125. django_cfg/core/integration/display/base.py +157 -0
  126. django_cfg/core/integration/display/ngrok.py +164 -0
  127. django_cfg/core/integration/display/startup.py +815 -0
  128. django_cfg/core/integration/url_integration.py +123 -0
  129. django_cfg/core/integration/version_checker.py +160 -0
  130. django_cfg/management/commands/auto_generate.py +4 -0
  131. django_cfg/management/commands/check_settings.py +6 -0
  132. django_cfg/management/commands/clear_constance.py +5 -2
  133. django_cfg/management/commands/create_token.py +6 -0
  134. django_cfg/management/commands/list_urls.py +6 -0
  135. django_cfg/management/commands/migrate_all.py +6 -0
  136. django_cfg/management/commands/migrator.py +3 -0
  137. django_cfg/management/commands/rundramatiq.py +6 -0
  138. django_cfg/management/commands/runserver_ngrok.py +51 -29
  139. django_cfg/management/commands/script.py +6 -0
  140. django_cfg/management/commands/show_config.py +12 -2
  141. django_cfg/management/commands/show_urls.py +4 -0
  142. django_cfg/management/commands/superuser.py +6 -0
  143. django_cfg/management/commands/task_clear.py +4 -1
  144. django_cfg/management/commands/task_status.py +3 -1
  145. django_cfg/management/commands/test_email.py +3 -0
  146. django_cfg/management/commands/test_telegram.py +6 -0
  147. django_cfg/management/commands/test_twilio.py +6 -0
  148. django_cfg/management/commands/tree.py +6 -0
  149. django_cfg/management/commands/validate_config.py +155 -149
  150. django_cfg/models/constance.py +31 -11
  151. django_cfg/models/payments.py +175 -492
  152. django_cfg/modules/django_logger.py +160 -146
  153. django_cfg/modules/django_unfold/dashboard.py +64 -16
  154. django_cfg/registry/core.py +1 -0
  155. django_cfg/template_archive/django_sample.zip +0 -0
  156. django_cfg/utils/smart_defaults.py +227 -570
  157. django_cfg/utils/toolkit.py +51 -11
  158. {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/METADATA +4 -1
  159. {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/RECORD +162 -185
  160. django_cfg/apps/payments/__init__.py +0 -8
  161. django_cfg/apps/payments/admin/tariffs_admin.py +0 -199
  162. django_cfg/apps/payments/config/module.py +0 -70
  163. django_cfg/apps/payments/config/providers.py +0 -105
  164. django_cfg/apps/payments/config/settings.py +0 -96
  165. django_cfg/apps/payments/config/utils.py +0 -52
  166. django_cfg/apps/payments/decorators.py +0 -291
  167. django_cfg/apps/payments/management/commands/README.md +0 -146
  168. django_cfg/apps/payments/managers/__init__.py +0 -23
  169. django_cfg/apps/payments/managers/api_key_manager.py +0 -35
  170. django_cfg/apps/payments/managers/balance_manager.py +0 -361
  171. django_cfg/apps/payments/managers/currency_manager.py +0 -306
  172. django_cfg/apps/payments/managers/payment_manager.py +0 -192
  173. django_cfg/apps/payments/managers/subscription_manager.py +0 -37
  174. django_cfg/apps/payments/managers/tariff_manager.py +0 -29
  175. django_cfg/apps/payments/migrations/0002_network_providercurrency_and_more.py +0 -241
  176. django_cfg/apps/payments/migrations/0003_add_usd_rate_cache.py +0 -30
  177. django_cfg/apps/payments/models/events.py +0 -73
  178. django_cfg/apps/payments/serializers/__init__.py +0 -57
  179. django_cfg/apps/payments/serializers/api_keys.py +0 -51
  180. django_cfg/apps/payments/serializers/balance.py +0 -59
  181. django_cfg/apps/payments/serializers/currencies.py +0 -63
  182. django_cfg/apps/payments/serializers/payments.py +0 -62
  183. django_cfg/apps/payments/serializers/subscriptions.py +0 -71
  184. django_cfg/apps/payments/serializers/tariffs.py +0 -56
  185. django_cfg/apps/payments/services/billing/__init__.py +0 -8
  186. django_cfg/apps/payments/services/cache/simple_cache.py +0 -135
  187. django_cfg/apps/payments/services/core/fallback_service.py +0 -432
  188. django_cfg/apps/payments/services/internal_types.py +0 -461
  189. django_cfg/apps/payments/services/middleware/__init__.py +0 -8
  190. django_cfg/apps/payments/services/monitoring/__init__.py +0 -22
  191. django_cfg/apps/payments/services/monitoring/api_schemas.py +0 -76
  192. django_cfg/apps/payments/services/monitoring/provider_health.py +0 -372
  193. django_cfg/apps/payments/services/providers/cryptapi/__init__.py +0 -4
  194. django_cfg/apps/payments/services/providers/cryptapi/config.py +0 -8
  195. django_cfg/apps/payments/services/providers/cryptapi/models.py +0 -192
  196. django_cfg/apps/payments/services/providers/cryptapi/provider.py +0 -439
  197. django_cfg/apps/payments/services/providers/cryptomus/__init__.py +0 -4
  198. django_cfg/apps/payments/services/providers/cryptomus/models.py +0 -176
  199. django_cfg/apps/payments/services/providers/cryptomus/provider.py +0 -429
  200. django_cfg/apps/payments/services/providers/cryptomus/provider_v2.py +0 -564
  201. django_cfg/apps/payments/services/providers/models/__init__.py +0 -34
  202. django_cfg/apps/payments/services/providers/models/currencies.py +0 -190
  203. django_cfg/apps/payments/services/providers/nowpayments/__init__.py +0 -4
  204. django_cfg/apps/payments/services/providers/nowpayments/models.py +0 -196
  205. django_cfg/apps/payments/services/providers/nowpayments/provider.py +0 -380
  206. django_cfg/apps/payments/services/providers/stripe/__init__.py +0 -4
  207. django_cfg/apps/payments/services/providers/stripe/models.py +0 -184
  208. django_cfg/apps/payments/services/providers/stripe/provider.py +0 -109
  209. django_cfg/apps/payments/services/security/__init__.py +0 -34
  210. django_cfg/apps/payments/services/security/error_handler.py +0 -635
  211. django_cfg/apps/payments/services/security/payment_notifications.py +0 -342
  212. django_cfg/apps/payments/services/security/webhook_validator.py +0 -474
  213. django_cfg/apps/payments/static/payments/css/payments.css +0 -340
  214. django_cfg/apps/payments/static/payments/js/notifications.js +0 -202
  215. django_cfg/apps/payments/static/payments/js/payment-utils.js +0 -318
  216. django_cfg/apps/payments/static/payments/js/theme.js +0 -86
  217. django_cfg/apps/payments/tasks/__init__.py +0 -12
  218. django_cfg/apps/payments/tasks/webhook_processing.py +0 -177
  219. django_cfg/apps/payments/templates/admin/payments/currency/change_list.html +0 -50
  220. django_cfg/apps/payments/templates/payments/base.html +0 -182
  221. django_cfg/apps/payments/templates/payments/components/payment_card.html +0 -201
  222. django_cfg/apps/payments/templates/payments/components/payment_qr_code.html +0 -109
  223. django_cfg/apps/payments/templates/payments/components/progress_bar.html +0 -43
  224. django_cfg/apps/payments/templates/payments/components/provider_stats.html +0 -40
  225. django_cfg/apps/payments/templates/payments/components/status_badge.html +0 -34
  226. django_cfg/apps/payments/templates/payments/components/status_overview.html +0 -148
  227. django_cfg/apps/payments/templates/payments/dashboard.html +0 -258
  228. django_cfg/apps/payments/templates/payments/dashboard_simple_test.html +0 -35
  229. django_cfg/apps/payments/templates/payments/payment_create.html +0 -579
  230. django_cfg/apps/payments/templates/payments/payment_detail.html +0 -373
  231. django_cfg/apps/payments/templates/payments/payment_list.html +0 -354
  232. django_cfg/apps/payments/templates/payments/stats.html +0 -261
  233. django_cfg/apps/payments/templates/payments/test.html +0 -213
  234. django_cfg/apps/payments/templatetags/payments_tags.py +0 -315
  235. django_cfg/apps/payments/utils/__init__.py +0 -43
  236. django_cfg/apps/payments/utils/billing_utils.py +0 -342
  237. django_cfg/apps/payments/utils/config_utils.py +0 -239
  238. django_cfg/apps/payments/utils/middleware_utils.py +0 -228
  239. django_cfg/apps/payments/utils/validation_utils.py +0 -94
  240. django_cfg/apps/payments/views/__init__.py +0 -63
  241. django_cfg/apps/payments/views/api_key_views.py +0 -164
  242. django_cfg/apps/payments/views/balance_views.py +0 -75
  243. django_cfg/apps/payments/views/currency_views.py +0 -122
  244. django_cfg/apps/payments/views/payment_views.py +0 -149
  245. django_cfg/apps/payments/views/subscription_views.py +0 -135
  246. django_cfg/apps/payments/views/tariff_views.py +0 -131
  247. django_cfg/apps/payments/views/templates/__init__.py +0 -25
  248. django_cfg/apps/payments/views/templates/ajax.py +0 -451
  249. django_cfg/apps/payments/views/templates/base.py +0 -212
  250. django_cfg/apps/payments/views/templates/dashboard.py +0 -60
  251. django_cfg/apps/payments/views/templates/payment_detail.py +0 -102
  252. django_cfg/apps/payments/views/templates/payment_management.py +0 -158
  253. django_cfg/apps/payments/views/templates/qr_code.py +0 -174
  254. django_cfg/apps/payments/views/templates/stats.py +0 -244
  255. django_cfg/apps/payments/views/templates/utils.py +0 -181
  256. django_cfg/apps/payments/views/webhook_views.py +0 -266
  257. django_cfg/apps/payments/viewsets.py +0 -66
  258. django_cfg/core/integration.py +0 -160
  259. django_cfg/template_archive/.gitignore +0 -1
  260. django_cfg/template_archive/__init__.py +0 -0
  261. django_cfg/urls.py +0 -33
  262. {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/WHEEL +0 -0
  263. {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/entry_points.txt +0 -0
  264. {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 for payments.
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
- """Admin interface for universal payments."""
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
- 'payment_display',
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 = ['payment_display']
56
+ list_display_links = ['payment_id_display']
32
57
 
33
58
  search_fields = [
34
- 'internal_payment_id',
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
- 'currency_code',
74
+ 'currency',
52
75
  'created_at'
53
76
  ]
54
77
 
55
78
  readonly_fields = [
56
- 'internal_payment_id',
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': ['user', 'amount_usd', 'currency_code', 'description']
97
+ 'fields': [
98
+ 'id',
99
+ 'user',
100
+ 'amount_usd',
101
+ 'currency',
102
+ 'crypto_amount',
103
+ 'description'
104
+ ]
65
105
  }),
66
- ('Payment Details', {
67
- 'fields': ['internal_payment_id', 'provider_payment_id', 'provider', 'status']
106
+ ('Provider Details', {
107
+ 'fields': [
108
+ 'provider',
109
+ 'provider_payment_id',
110
+ 'payment_url'
111
+ ]
68
112
  }),
69
- ('Crypto Details', {
70
- 'fields': ['pay_address', 'pay_amount', 'network', 'security_nonce', 'transaction_hash', 'sender_address', 'receiver_address', 'crypto_amount', 'confirmations_count'],
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
- ('Provider Data', {
74
- 'fields': ['metadata', 'webhook_data'],
128
+ ('Metadata', {
129
+ 'fields': [
130
+ 'metadata'
131
+ ],
75
132
  'classes': ['collapse']
76
133
  }),
77
134
  ('Timestamps', {
78
- 'fields': ['created_at', 'updated_at', 'expires_at', 'completed_at', 'processed_at'],
135
+ 'fields': [
136
+ 'created_at',
137
+ 'updated_at',
138
+ 'completed_at'
139
+ ],
79
140
  'classes': ['collapse']
80
141
  })
81
142
  ]
82
143
 
83
- @display(description="Payment")
84
- def payment_display(self, obj):
85
- """Display payment ID and description."""
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
- '<strong>#{}</strong><br><small>{}</small>',
88
- obj.internal_payment_id[:8],
89
- obj.description[:40] + '...' if len(obj.description) > 40 else obj.description
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
- return format_html(
96
- '<strong>{}</strong><br><small>{}</small>',
97
- obj.user.get_full_name() or obj.user.email,
98
- obj.user.email
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
- '<span style="font-weight: bold; font-size: 14px;">${}</span><br><small>{}</small>',
106
- f"{float(obj.amount_usd):.2f}",
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 color coding."""
113
- status_colors = {
114
- 'pending': '#ffc107',
115
- 'confirming': '#17a2b8',
116
- 'confirmed': '#28a745',
117
- 'completed': '#28a745',
118
- 'failed': '#dc3545',
119
- 'expired': '#6c757d',
120
- 'cancelled': '#6c757d',
121
- 'refunded': '#fd7e14',
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
- color = status_colors.get(obj.status, '#6c757d')
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 style="background: {}; color: white; padding: 2px 6px; border-radius: 3px; font-size: 11px;">{}</span>',
128
- color,
129
- obj.get_status_display()
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 external ID."""
135
- provider_colors = {
136
- 'nowpayments': '#007bff',
137
- 'stripe': '#6f42c1',
138
- 'internal': '#28a745',
232
+ """Display provider with logo/icon."""
233
+ provider_config = {
234
+ 'nowpayments': ('🟦', 'NowPayments'),
235
+ 'cryptomus': ('🟩', 'Cryptomus'),
236
+ 'cryptapi': ('🟪', 'CryptAPI'),
139
237
  }
140
238
 
141
- color = provider_colors.get(obj.provider, '#6c757d')
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
- '<span style="color: {}; font-weight: bold;">{}</span><br><small>{}</small>',
145
- color,
146
- obj.get_provider_display(),
147
- obj.provider_payment_id[:16] + '...' if obj.provider_payment_id and len(obj.provider_payment_id) > 16 else obj.provider_payment_id or '—'
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 naturaltime(obj.created_at)
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