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
@@ -0,0 +1,511 @@
1
+ """
2
+ Payment managers for the Universal Payment System v2.0.
3
+
4
+ Optimized querysets and managers for payment operations.
5
+ """
6
+
7
+ from django.db import models
8
+ from django.utils import timezone
9
+ from django_cfg.modules.django_logger import get_logger
10
+
11
+ logger = get_logger("payment_managers")
12
+
13
+
14
+ class PaymentQuerySet(models.QuerySet):
15
+ """
16
+ Optimized queryset for payment operations.
17
+
18
+ Provides efficient queries with proper indexing and select_related optimization.
19
+ """
20
+
21
+ def optimized(self):
22
+ """
23
+ Prevent N+1 queries with select_related and prefetch_related.
24
+
25
+ Use this for admin interfaces and API responses.
26
+ """
27
+ return self.select_related(
28
+ 'user',
29
+ 'currency',
30
+ 'network'
31
+ ).prefetch_related(
32
+ 'user__payment_balance',
33
+ 'user__payment_transactions'
34
+ )
35
+
36
+ def by_status(self, status):
37
+ """Filter by payment status with index optimization."""
38
+ return self.filter(status=status)
39
+
40
+ def by_provider(self, provider):
41
+ """Filter by payment provider."""
42
+ return self.filter(provider=provider)
43
+
44
+ def by_user(self, user):
45
+ """Filter by user with proper indexing."""
46
+ return self.filter(user=user)
47
+
48
+ def by_amount_range(self, min_amount=None, max_amount=None):
49
+ """
50
+ Filter by USD amount range.
51
+
52
+ Args:
53
+ min_amount: Minimum amount in USD (inclusive)
54
+ max_amount: Maximum amount in USD (inclusive)
55
+ """
56
+ queryset = self
57
+ if min_amount is not None:
58
+ queryset = queryset.filter(amount_usd__gte=min_amount)
59
+ if max_amount is not None:
60
+ queryset = queryset.filter(amount_usd__lte=max_amount)
61
+ return queryset
62
+
63
+ def by_currency(self, currency_code):
64
+ """Filter by currency code."""
65
+ return self.filter(currency__code=currency_code)
66
+
67
+ def by_network(self, network_code):
68
+ """Filter by blockchain network."""
69
+ return self.filter(network__code=network_code)
70
+
71
+ # Status-based filters
72
+ def completed(self):
73
+ """Get completed payments."""
74
+ return self.filter(status='completed')
75
+
76
+ def pending(self):
77
+ """Get pending payments."""
78
+ return self.filter(status='pending')
79
+
80
+ def failed(self):
81
+ """Get failed payments (failed, expired, cancelled)."""
82
+ return self.filter(status__in=['failed', 'expired', 'cancelled'])
83
+
84
+ def confirming(self):
85
+ """Get payments awaiting confirmation."""
86
+ return self.filter(status__in=['confirming', 'confirmed'])
87
+
88
+ def active(self):
89
+ """Get active payments (not failed or completed)."""
90
+ return self.filter(status__in=['pending', 'confirming', 'confirmed'])
91
+
92
+ # Time-based filters
93
+ def recent(self, hours=24):
94
+ """
95
+ Get payments from last N hours.
96
+
97
+ Args:
98
+ hours: Number of hours to look back (default: 24)
99
+ """
100
+ since = timezone.now() - timezone.timedelta(hours=hours)
101
+ return self.filter(created_at__gte=since)
102
+
103
+ def today(self):
104
+ """Get payments created today."""
105
+ today = timezone.now().date()
106
+ return self.filter(created_at__date=today)
107
+
108
+ def this_week(self):
109
+ """Get payments from this week."""
110
+ week_start = timezone.now().date() - timezone.timedelta(days=timezone.now().weekday())
111
+ return self.filter(created_at__date__gte=week_start)
112
+
113
+ def this_month(self):
114
+ """Get payments from this month."""
115
+ month_start = timezone.now().replace(day=1).date()
116
+ return self.filter(created_at__date__gte=month_start)
117
+
118
+ def expired(self):
119
+ """Get expired payments."""
120
+ now = timezone.now()
121
+ return self.filter(
122
+ expires_at__lte=now,
123
+ status__in=['pending', 'confirming']
124
+ )
125
+
126
+ def expiring_soon(self, hours=2):
127
+ """
128
+ Get payments expiring in the next N hours.
129
+
130
+ Args:
131
+ hours: Hours until expiration (default: 2)
132
+ """
133
+ soon = timezone.now() + timezone.timedelta(hours=hours)
134
+ return self.filter(
135
+ expires_at__lte=soon,
136
+ expires_at__gt=timezone.now(),
137
+ status__in=['pending', 'confirming']
138
+ )
139
+
140
+ # Provider-specific filters
141
+ def nowpayments(self):
142
+ """Get NowPayments payments."""
143
+ return self.filter(provider='nowpayments')
144
+
145
+ def crypto_payments(self):
146
+ """Get cryptocurrency payments."""
147
+ return self.filter(currency__currency_type='crypto')
148
+
149
+ def fiat_payments(self):
150
+ """Get fiat currency payments."""
151
+ return self.filter(currency__currency_type='fiat')
152
+
153
+ # Aggregation methods
154
+ def total_amount(self):
155
+ """Get total USD amount for queryset."""
156
+ result = self.aggregate(total=models.Sum('amount_usd'))
157
+ return result['total'] or 0.0
158
+
159
+ def average_amount(self):
160
+ """Get average USD amount for queryset."""
161
+ result = self.aggregate(avg=models.Avg('amount_usd'))
162
+ return result['avg'] or 0.0
163
+
164
+ def count_by_status(self):
165
+ """Get count of payments grouped by status."""
166
+ return self.values('status').annotate(count=models.Count('id')).order_by('status')
167
+
168
+ def count_by_provider(self):
169
+ """Get count of payments grouped by provider."""
170
+ return self.values('provider').annotate(count=models.Count('id')).order_by('provider')
171
+
172
+ def count_by_currency(self):
173
+ """Get count of payments grouped by currency."""
174
+ return self.values('currency__code').annotate(count=models.Count('id')).order_by('currency__code')
175
+
176
+ # Advanced queries
177
+ def with_transactions(self):
178
+ """Include related transaction data."""
179
+ return self.prefetch_related('user__payment_transactions')
180
+
181
+ def with_balance_info(self):
182
+ """Include user balance information."""
183
+ return self.select_related('user__payment_balance')
184
+
185
+ def requiring_confirmation(self):
186
+ """Get payments that need blockchain confirmation."""
187
+ return self.filter(
188
+ status__in=['confirming', 'confirmed'],
189
+ transaction_hash__isnull=False
190
+ )
191
+
192
+ def large_amounts(self, threshold=1000.0):
193
+ """
194
+ Get payments above threshold amount.
195
+
196
+ Args:
197
+ threshold: USD amount threshold (default: $1000)
198
+ """
199
+ return self.filter(amount_usd__gte=threshold)
200
+
201
+ def small_amounts(self, threshold=10.0):
202
+ """
203
+ Get payments below threshold amount.
204
+
205
+ Args:
206
+ threshold: USD amount threshold (default: $10)
207
+ """
208
+ return self.filter(amount_usd__lte=threshold)
209
+
210
+
211
+ class PaymentManager(models.Manager):
212
+ """
213
+ Manager for payment operations with optimized queries.
214
+
215
+ Provides high-level methods for common payment operations.
216
+ """
217
+
218
+ def get_queryset(self):
219
+ """Return optimized queryset by default."""
220
+ return PaymentQuerySet(self.model, using=self._db)
221
+
222
+ def optimized(self):
223
+ """Get optimized queryset for admin/API use."""
224
+ return self.get_queryset().optimized()
225
+
226
+ # Status-based methods
227
+ def by_status(self, status):
228
+ """Get payments by status."""
229
+ return self.get_queryset().by_status(status)
230
+
231
+ def completed(self):
232
+ """Get completed payments."""
233
+ return self.get_queryset().completed()
234
+
235
+ def pending(self):
236
+ """Get pending payments."""
237
+ return self.get_queryset().pending()
238
+
239
+ def failed(self):
240
+ """Get failed payments."""
241
+ return self.get_queryset().failed()
242
+
243
+ def active(self):
244
+ """Get active payments."""
245
+ return self.get_queryset().active()
246
+
247
+ # Provider-based methods
248
+ def by_provider(self, provider):
249
+ """Get payments by provider."""
250
+ return self.get_queryset().by_provider(provider)
251
+
252
+ def nowpayments(self):
253
+ """Get NowPayments payments."""
254
+ return self.get_queryset().nowpayments()
255
+
256
+ # Time-based methods
257
+ def recent(self, hours=24):
258
+ """Get recent payments."""
259
+ return self.get_queryset().recent(hours)
260
+
261
+ def today(self):
262
+ """Get today's payments."""
263
+ return self.get_queryset().today()
264
+
265
+ def this_week(self):
266
+ """Get this week's payments."""
267
+ return self.get_queryset().this_week()
268
+
269
+ def this_month(self):
270
+ """Get this month's payments."""
271
+ return self.get_queryset().this_month()
272
+
273
+ # Maintenance methods
274
+ def expired(self):
275
+ """Get expired payments."""
276
+ return self.get_queryset().expired()
277
+
278
+ def expiring_soon(self, hours=2):
279
+ """Get payments expiring soon."""
280
+ return self.get_queryset().expiring_soon(hours)
281
+
282
+ def requiring_confirmation(self):
283
+ """Get payments needing confirmation."""
284
+ return self.get_queryset().requiring_confirmation()
285
+
286
+ # Statistics methods
287
+ def get_stats(self, days=30):
288
+ """
289
+ Get payment statistics for the last N days.
290
+
291
+ Args:
292
+ days: Number of days to analyze (default: 30)
293
+
294
+ Returns:
295
+ dict: Statistics including totals, averages, and counts
296
+ """
297
+ since = timezone.now() - timezone.timedelta(days=days)
298
+ queryset = self.filter(created_at__gte=since)
299
+
300
+ stats = {
301
+ 'total_payments': queryset.count(),
302
+ 'total_amount_usd': queryset.total_amount(),
303
+ 'average_amount_usd': queryset.average_amount(),
304
+ 'completed_payments': queryset.completed().count(),
305
+ 'pending_payments': queryset.pending().count(),
306
+ 'failed_payments': queryset.failed().count(),
307
+ 'by_status': list(queryset.count_by_status()),
308
+ 'by_provider': list(queryset.count_by_provider()),
309
+ 'by_currency': list(queryset.count_by_currency()),
310
+ }
311
+
312
+ logger.info(f"Generated payment stats for {days} days", extra={
313
+ 'days': days,
314
+ 'total_payments': stats['total_payments'],
315
+ 'total_amount': stats['total_amount_usd']
316
+ })
317
+
318
+ return stats
319
+
320
+ def cleanup_expired(self, dry_run=True):
321
+ """
322
+ Mark expired payments as failed.
323
+
324
+ Args:
325
+ dry_run: If True, only return count without making changes
326
+
327
+ Returns:
328
+ int: Number of payments that would be/were updated
329
+ """
330
+ expired_payments = self.expired()
331
+ count = expired_payments.count()
332
+
333
+ if not dry_run and count > 0:
334
+ expired_payments.update(status='expired')
335
+ logger.info(f"Marked {count} payments as expired")
336
+
337
+ return count
338
+
339
+ def get_user_payment_summary(self, user):
340
+ """
341
+ Get payment summary for a specific user.
342
+
343
+ Args:
344
+ user: User instance
345
+
346
+ Returns:
347
+ dict: User payment summary
348
+ """
349
+ user_payments = self.filter(user=user)
350
+
351
+ summary = {
352
+ 'total_payments': user_payments.count(),
353
+ 'total_amount_usd': user_payments.total_amount(),
354
+ 'completed_payments': user_payments.completed().count(),
355
+ 'pending_payments': user_payments.pending().count(),
356
+ 'failed_payments': user_payments.failed().count(),
357
+ 'last_payment_at': user_payments.first().created_at if user_payments.exists() else None,
358
+ 'average_amount_usd': user_payments.average_amount(),
359
+ }
360
+
361
+ return summary
362
+
363
+ # Business logic methods
364
+ def mark_payment_completed(self, payment_id, actual_amount_usd=None, transaction_hash=None):
365
+ """
366
+ Mark payment as completed (business logic in manager).
367
+
368
+ Args:
369
+ payment_id: Payment ID or instance
370
+ actual_amount_usd: Actual amount received
371
+ transaction_hash: Blockchain transaction hash
372
+
373
+ Returns:
374
+ bool: True if payment was updated successfully
375
+ """
376
+ try:
377
+ if isinstance(payment_id, str):
378
+ payment = self.get(id=payment_id)
379
+ else:
380
+ payment = payment_id
381
+
382
+ # Validate payment can be completed
383
+ if not payment.status in ['pending', 'confirming', 'confirmed']:
384
+ logger.warning(f"Cannot complete payment in status {payment.status}", extra={
385
+ 'payment_id': str(payment.id),
386
+ 'current_status': payment.status
387
+ })
388
+ return False
389
+
390
+ # Update payment
391
+ payment.status = 'completed'
392
+ payment.completed_at = timezone.now()
393
+ if actual_amount_usd:
394
+ payment.actual_amount_usd = actual_amount_usd
395
+ if transaction_hash:
396
+ payment.transaction_hash = transaction_hash
397
+
398
+ payment.save(update_fields=[
399
+ 'status', 'completed_at', 'actual_amount_usd', 'transaction_hash', 'updated_at'
400
+ ])
401
+
402
+ logger.info(f"Payment marked as completed", extra={
403
+ 'payment_id': str(payment.id),
404
+ 'user_id': payment.user.id,
405
+ 'amount_usd': payment.amount_usd,
406
+ 'actual_amount_usd': actual_amount_usd,
407
+ 'transaction_hash': transaction_hash
408
+ })
409
+
410
+ return True
411
+
412
+ except Exception as e:
413
+ logger.error(f"Failed to mark payment as completed: {e}", extra={
414
+ 'payment_id': str(payment_id) if hasattr(payment_id, 'id') else payment_id
415
+ })
416
+ return False
417
+
418
+ def mark_payment_failed(self, payment_id, reason=None, error_code=None):
419
+ """
420
+ Mark payment as failed (business logic in manager).
421
+
422
+ Args:
423
+ payment_id: Payment ID or instance
424
+ reason: Failure reason
425
+ error_code: Error code for categorization
426
+
427
+ Returns:
428
+ bool: True if payment was updated successfully
429
+ """
430
+ try:
431
+ if isinstance(payment_id, str):
432
+ payment = self.get(id=payment_id)
433
+ else:
434
+ payment = payment_id
435
+
436
+ # Update payment
437
+ payment.status = 'failed'
438
+ if reason or error_code:
439
+ if 'error_info' not in payment.provider_data:
440
+ payment.provider_data['error_info'] = {}
441
+ if reason:
442
+ payment.provider_data['error_info']['reason'] = reason
443
+ if error_code:
444
+ payment.provider_data['error_info']['code'] = error_code
445
+ payment.provider_data['error_info']['failed_at'] = timezone.now().isoformat()
446
+
447
+ payment.save(update_fields=['status', 'provider_data', 'updated_at'])
448
+
449
+ logger.warning(f"Payment marked as failed", extra={
450
+ 'payment_id': str(payment.id),
451
+ 'user_id': payment.user.id,
452
+ 'reason': reason,
453
+ 'error_code': error_code
454
+ })
455
+
456
+ return True
457
+
458
+ except Exception as e:
459
+ logger.error(f"Failed to mark payment as failed: {e}", extra={
460
+ 'payment_id': str(payment_id) if hasattr(payment_id, 'id') else payment_id
461
+ })
462
+ return False
463
+
464
+ def cancel_payment(self, payment_id, reason=None):
465
+ """
466
+ Cancel payment (business logic in manager).
467
+
468
+ Args:
469
+ payment_id: Payment ID or instance
470
+ reason: Cancellation reason
471
+
472
+ Returns:
473
+ bool: True if payment was cancelled successfully
474
+ """
475
+ try:
476
+ if isinstance(payment_id, str):
477
+ payment = self.get(id=payment_id)
478
+ else:
479
+ payment = payment_id
480
+
481
+ # Validate payment can be cancelled
482
+ if not payment.status in ['pending', 'confirming']:
483
+ logger.warning(f"Cannot cancel payment in status {payment.status}", extra={
484
+ 'payment_id': str(payment.id),
485
+ 'current_status': payment.status
486
+ })
487
+ return False
488
+
489
+ # Update payment
490
+ payment.status = 'cancelled'
491
+ if reason:
492
+ if 'cancellation_info' not in payment.provider_data:
493
+ payment.provider_data['cancellation_info'] = {}
494
+ payment.provider_data['cancellation_info']['reason'] = reason
495
+ payment.provider_data['cancellation_info']['cancelled_at'] = timezone.now().isoformat()
496
+
497
+ payment.save(update_fields=['status', 'provider_data', 'updated_at'])
498
+
499
+ logger.info(f"Payment cancelled", extra={
500
+ 'payment_id': str(payment.id),
501
+ 'user_id': payment.user.id,
502
+ 'reason': reason
503
+ })
504
+
505
+ return True
506
+
507
+ except Exception as e:
508
+ logger.error(f"Failed to cancel payment: {e}", extra={
509
+ 'payment_id': str(payment_id) if hasattr(payment_id, 'id') else payment_id
510
+ })
511
+ return False