django-cfg 1.2.29__py3-none-any.whl → 1.3.1__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 (258) 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 -9
  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 +600 -108
  9. django_cfg/apps/payments/admin/filters.py +306 -199
  10. django_cfg/apps/payments/admin/payments_admin.py +470 -64
  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/manage_currencies.py +381 -0
  39. django_cfg/apps/payments/management/commands/manage_providers.py +408 -0
  40. django_cfg/apps/payments/middleware/__init__.py +3 -1
  41. django_cfg/apps/payments/middleware/api_access.py +329 -222
  42. django_cfg/apps/payments/middleware/rate_limiting.py +343 -163
  43. django_cfg/apps/payments/middleware/usage_tracking.py +250 -238
  44. django_cfg/apps/payments/migrations/0001_initial.py +708 -536
  45. django_cfg/apps/payments/models/__init__.py +16 -20
  46. django_cfg/apps/payments/models/api_keys.py +121 -43
  47. django_cfg/apps/payments/models/balance.py +150 -115
  48. django_cfg/apps/payments/models/base.py +68 -15
  49. django_cfg/apps/payments/models/currencies.py +207 -67
  50. django_cfg/apps/payments/models/managers/__init__.py +44 -0
  51. django_cfg/apps/payments/models/managers/api_key_managers.py +329 -0
  52. django_cfg/apps/payments/models/managers/balance_managers.py +599 -0
  53. django_cfg/apps/payments/models/managers/currency_managers.py +385 -0
  54. django_cfg/apps/payments/models/managers/payment_managers.py +511 -0
  55. django_cfg/apps/payments/models/managers/subscription_managers.py +641 -0
  56. django_cfg/apps/payments/models/payments.py +235 -284
  57. django_cfg/apps/payments/models/subscriptions.py +257 -177
  58. django_cfg/apps/payments/models/tariffs.py +147 -40
  59. django_cfg/apps/payments/services/__init__.py +209 -56
  60. django_cfg/apps/payments/services/cache/__init__.py +6 -6
  61. django_cfg/apps/payments/services/cache/{simple_cache.py → cache_service.py} +112 -12
  62. django_cfg/apps/payments/services/core/__init__.py +10 -6
  63. django_cfg/apps/payments/services/core/balance_service.py +435 -360
  64. django_cfg/apps/payments/services/core/base.py +166 -0
  65. django_cfg/apps/payments/services/core/currency_service.py +478 -0
  66. django_cfg/apps/payments/services/core/payment_service.py +344 -468
  67. django_cfg/apps/payments/services/core/subscription_service.py +425 -484
  68. django_cfg/apps/payments/services/core/webhook_service.py +410 -0
  69. django_cfg/apps/payments/services/integrations/__init__.py +29 -0
  70. django_cfg/apps/payments/services/integrations/ngrok_service.py +47 -0
  71. django_cfg/apps/payments/services/integrations/providers_config.py +107 -0
  72. django_cfg/apps/payments/services/providers/__init__.py +9 -14
  73. django_cfg/apps/payments/services/providers/base.py +232 -71
  74. django_cfg/apps/payments/services/providers/nowpayments.py +404 -219
  75. django_cfg/apps/payments/services/providers/registry.py +429 -80
  76. django_cfg/apps/payments/services/types/__init__.py +78 -0
  77. django_cfg/apps/payments/services/types/data.py +177 -0
  78. django_cfg/apps/payments/services/types/requests.py +150 -0
  79. django_cfg/apps/payments/services/types/responses.py +156 -0
  80. django_cfg/apps/payments/services/types/webhooks.py +232 -0
  81. django_cfg/apps/payments/signals/__init__.py +33 -8
  82. django_cfg/apps/payments/signals/api_key_signals.py +211 -130
  83. django_cfg/apps/payments/signals/balance_signals.py +174 -0
  84. django_cfg/apps/payments/signals/payment_signals.py +129 -98
  85. django_cfg/apps/payments/signals/subscription_signals.py +195 -143
  86. django_cfg/apps/payments/static/payments/css/components.css +380 -0
  87. django_cfg/apps/payments/static/payments/css/dashboard.css +188 -0
  88. django_cfg/apps/payments/static/payments/js/components.js +545 -0
  89. django_cfg/apps/payments/static/payments/js/utils.js +412 -0
  90. django_cfg/apps/payments/templatetags/__init__.py +1 -1
  91. django_cfg/apps/payments/templatetags/payment_tags.py +466 -0
  92. django_cfg/apps/payments/urls.py +46 -47
  93. django_cfg/apps/payments/urls_admin.py +49 -0
  94. django_cfg/apps/payments/views/api/__init__.py +101 -0
  95. django_cfg/apps/payments/views/api/api_keys.py +387 -0
  96. django_cfg/apps/payments/views/api/balances.py +381 -0
  97. django_cfg/apps/payments/views/api/base.py +298 -0
  98. django_cfg/apps/payments/views/api/currencies.py +402 -0
  99. django_cfg/apps/payments/views/api/payments.py +415 -0
  100. django_cfg/apps/payments/views/api/subscriptions.py +475 -0
  101. django_cfg/apps/payments/views/api/webhooks.py +476 -0
  102. django_cfg/apps/payments/views/serializers/__init__.py +99 -0
  103. django_cfg/apps/payments/views/serializers/api_keys.py +424 -0
  104. django_cfg/apps/payments/views/serializers/balances.py +300 -0
  105. django_cfg/apps/payments/views/serializers/currencies.py +335 -0
  106. django_cfg/apps/payments/views/serializers/payments.py +387 -0
  107. django_cfg/apps/payments/views/serializers/subscriptions.py +429 -0
  108. django_cfg/apps/payments/views/serializers/webhooks.py +137 -0
  109. django_cfg/apps/tasks/urls.py +0 -2
  110. django_cfg/apps/tasks/urls_admin.py +14 -0
  111. django_cfg/apps/urls.py +4 -4
  112. django_cfg/config.py +1 -1
  113. django_cfg/core/config.py +75 -4
  114. django_cfg/core/generation.py +25 -4
  115. django_cfg/core/integration/README.md +363 -0
  116. django_cfg/core/integration/__init__.py +47 -0
  117. django_cfg/core/integration/commands_collector.py +239 -0
  118. django_cfg/core/integration/display/__init__.py +15 -0
  119. django_cfg/core/integration/display/base.py +157 -0
  120. django_cfg/core/integration/display/ngrok.py +164 -0
  121. django_cfg/core/integration/display/startup.py +815 -0
  122. django_cfg/core/integration/url_integration.py +123 -0
  123. django_cfg/core/integration/version_checker.py +160 -0
  124. django_cfg/management/commands/auto_generate.py +4 -0
  125. django_cfg/management/commands/check_settings.py +6 -0
  126. django_cfg/management/commands/clear_constance.py +5 -2
  127. django_cfg/management/commands/create_token.py +6 -0
  128. django_cfg/management/commands/list_urls.py +6 -0
  129. django_cfg/management/commands/migrate_all.py +6 -0
  130. django_cfg/management/commands/migrator.py +3 -0
  131. django_cfg/management/commands/rundramatiq.py +6 -0
  132. django_cfg/management/commands/runserver_ngrok.py +51 -29
  133. django_cfg/management/commands/script.py +6 -0
  134. django_cfg/management/commands/show_config.py +12 -2
  135. django_cfg/management/commands/show_urls.py +4 -0
  136. django_cfg/management/commands/superuser.py +6 -0
  137. django_cfg/management/commands/task_clear.py +4 -1
  138. django_cfg/management/commands/task_status.py +3 -1
  139. django_cfg/management/commands/test_email.py +3 -0
  140. django_cfg/management/commands/test_telegram.py +6 -0
  141. django_cfg/management/commands/test_twilio.py +6 -0
  142. django_cfg/management/commands/tree.py +6 -0
  143. django_cfg/management/commands/validate_config.py +155 -149
  144. django_cfg/models/constance.py +31 -11
  145. django_cfg/models/payments.py +175 -498
  146. django_cfg/modules/django_currency/__init__.py +16 -11
  147. django_cfg/modules/django_currency/clients/__init__.py +4 -4
  148. django_cfg/modules/django_currency/clients/coinpaprika_client.py +289 -0
  149. django_cfg/modules/django_currency/clients/yahoo_client.py +157 -0
  150. django_cfg/modules/django_currency/core/__init__.py +1 -7
  151. django_cfg/modules/django_currency/core/converter.py +18 -23
  152. django_cfg/modules/django_currency/core/models.py +122 -11
  153. django_cfg/modules/django_currency/database/__init__.py +4 -4
  154. django_cfg/modules/django_currency/database/database_loader.py +190 -309
  155. django_cfg/modules/django_logger.py +160 -146
  156. django_cfg/modules/django_unfold/dashboard.py +65 -12
  157. django_cfg/registry/core.py +1 -0
  158. django_cfg/template_archive/django_sample.zip +0 -0
  159. django_cfg/templates/admin/components/action_grid.html +9 -9
  160. django_cfg/templates/admin/components/metric_card.html +5 -5
  161. django_cfg/templates/admin/components/status_badge.html +2 -2
  162. django_cfg/templates/admin/layouts/dashboard_with_tabs.html +152 -24
  163. django_cfg/templates/admin/snippets/components/quick_actions.html +3 -3
  164. django_cfg/templates/admin/snippets/components/system_health.html +1 -1
  165. django_cfg/templates/admin/snippets/tabs/overview_tab.html +49 -52
  166. django_cfg/utils/smart_defaults.py +222 -571
  167. django_cfg/utils/toolkit.py +51 -11
  168. {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/METADATA +5 -4
  169. {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/RECORD +172 -182
  170. django_cfg/apps/payments/__init__.py +0 -8
  171. django_cfg/apps/payments/admin/tariffs_admin.py +0 -199
  172. django_cfg/apps/payments/config/module.py +0 -70
  173. django_cfg/apps/payments/config/providers.py +0 -105
  174. django_cfg/apps/payments/config/settings.py +0 -96
  175. django_cfg/apps/payments/config/utils.py +0 -52
  176. django_cfg/apps/payments/decorators.py +0 -291
  177. django_cfg/apps/payments/management/commands/README.md +0 -178
  178. django_cfg/apps/payments/management/commands/currency_stats.py +0 -323
  179. django_cfg/apps/payments/management/commands/populate_currencies.py +0 -246
  180. django_cfg/apps/payments/management/commands/update_currencies.py +0 -336
  181. django_cfg/apps/payments/managers/__init__.py +0 -22
  182. django_cfg/apps/payments/managers/api_key_manager.py +0 -35
  183. django_cfg/apps/payments/managers/balance_manager.py +0 -361
  184. django_cfg/apps/payments/managers/currency_manager.py +0 -83
  185. django_cfg/apps/payments/managers/payment_manager.py +0 -44
  186. django_cfg/apps/payments/managers/subscription_manager.py +0 -37
  187. django_cfg/apps/payments/managers/tariff_manager.py +0 -29
  188. django_cfg/apps/payments/models/events.py +0 -73
  189. django_cfg/apps/payments/serializers/__init__.py +0 -56
  190. django_cfg/apps/payments/serializers/api_keys.py +0 -51
  191. django_cfg/apps/payments/serializers/balance.py +0 -59
  192. django_cfg/apps/payments/serializers/currencies.py +0 -55
  193. django_cfg/apps/payments/serializers/payments.py +0 -62
  194. django_cfg/apps/payments/serializers/subscriptions.py +0 -71
  195. django_cfg/apps/payments/serializers/tariffs.py +0 -56
  196. django_cfg/apps/payments/services/billing/__init__.py +0 -8
  197. django_cfg/apps/payments/services/cache/base.py +0 -30
  198. django_cfg/apps/payments/services/core/fallback_service.py +0 -432
  199. django_cfg/apps/payments/services/internal_types.py +0 -297
  200. django_cfg/apps/payments/services/middleware/__init__.py +0 -8
  201. django_cfg/apps/payments/services/monitoring/__init__.py +0 -22
  202. django_cfg/apps/payments/services/monitoring/api_schemas.py +0 -222
  203. django_cfg/apps/payments/services/monitoring/provider_health.py +0 -372
  204. django_cfg/apps/payments/services/providers/cryptapi.py +0 -273
  205. django_cfg/apps/payments/services/providers/cryptomus.py +0 -311
  206. django_cfg/apps/payments/services/security/__init__.py +0 -34
  207. django_cfg/apps/payments/services/security/error_handler.py +0 -637
  208. django_cfg/apps/payments/services/security/payment_notifications.py +0 -342
  209. django_cfg/apps/payments/services/security/webhook_validator.py +0 -475
  210. django_cfg/apps/payments/services/validators/__init__.py +0 -8
  211. django_cfg/apps/payments/static/payments/css/payments.css +0 -340
  212. django_cfg/apps/payments/static/payments/js/notifications.js +0 -202
  213. django_cfg/apps/payments/static/payments/js/payment-utils.js +0 -318
  214. django_cfg/apps/payments/static/payments/js/theme.js +0 -86
  215. django_cfg/apps/payments/tasks/__init__.py +0 -12
  216. django_cfg/apps/payments/tasks/webhook_processing.py +0 -177
  217. django_cfg/apps/payments/templates/payments/base.html +0 -182
  218. django_cfg/apps/payments/templates/payments/components/payment_card.html +0 -201
  219. django_cfg/apps/payments/templates/payments/components/payment_qr_code.html +0 -109
  220. django_cfg/apps/payments/templates/payments/components/progress_bar.html +0 -36
  221. django_cfg/apps/payments/templates/payments/components/provider_stats.html +0 -40
  222. django_cfg/apps/payments/templates/payments/components/status_badge.html +0 -27
  223. django_cfg/apps/payments/templates/payments/components/status_overview.html +0 -144
  224. django_cfg/apps/payments/templates/payments/dashboard.html +0 -346
  225. django_cfg/apps/payments/templatetags/payments_tags.py +0 -315
  226. django_cfg/apps/payments/urls_templates.py +0 -52
  227. django_cfg/apps/payments/utils/__init__.py +0 -45
  228. django_cfg/apps/payments/utils/billing_utils.py +0 -342
  229. django_cfg/apps/payments/utils/config_utils.py +0 -245
  230. django_cfg/apps/payments/utils/middleware_utils.py +0 -228
  231. django_cfg/apps/payments/utils/validation_utils.py +0 -94
  232. django_cfg/apps/payments/views/__init__.py +0 -62
  233. django_cfg/apps/payments/views/api_key_views.py +0 -164
  234. django_cfg/apps/payments/views/balance_views.py +0 -75
  235. django_cfg/apps/payments/views/currency_views.py +0 -111
  236. django_cfg/apps/payments/views/payment_views.py +0 -149
  237. django_cfg/apps/payments/views/subscription_views.py +0 -135
  238. django_cfg/apps/payments/views/tariff_views.py +0 -131
  239. django_cfg/apps/payments/views/templates/__init__.py +0 -25
  240. django_cfg/apps/payments/views/templates/ajax.py +0 -312
  241. django_cfg/apps/payments/views/templates/base.py +0 -204
  242. django_cfg/apps/payments/views/templates/dashboard.py +0 -60
  243. django_cfg/apps/payments/views/templates/payment_detail.py +0 -102
  244. django_cfg/apps/payments/views/templates/payment_management.py +0 -164
  245. django_cfg/apps/payments/views/templates/qr_code.py +0 -174
  246. django_cfg/apps/payments/views/templates/stats.py +0 -240
  247. django_cfg/apps/payments/views/templates/utils.py +0 -181
  248. django_cfg/apps/payments/views/webhook_views.py +0 -266
  249. django_cfg/apps/payments/viewsets.py +0 -65
  250. django_cfg/core/integration.py +0 -160
  251. django_cfg/modules/django_currency/clients/coingecko_client.py +0 -257
  252. django_cfg/modules/django_currency/clients/yfinance_client.py +0 -246
  253. django_cfg/template_archive/.gitignore +0 -1
  254. django_cfg/template_archive/__init__.py +0 -0
  255. django_cfg/urls.py +0 -33
  256. {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/WHEEL +0 -0
  257. {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/entry_points.txt +0 -0
  258. {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.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