django-cfg 1.2.27__py3-none-any.whl → 1.2.31__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 (138) hide show
  1. django_cfg/__init__.py +1 -1
  2. django_cfg/apps/payments/admin/__init__.py +3 -2
  3. django_cfg/apps/payments/admin/balance_admin.py +18 -18
  4. django_cfg/apps/payments/admin/currencies_admin.py +319 -131
  5. django_cfg/apps/payments/admin/payments_admin.py +15 -4
  6. django_cfg/apps/payments/config/module.py +2 -2
  7. django_cfg/apps/payments/config/utils.py +2 -2
  8. django_cfg/apps/payments/decorators.py +2 -2
  9. django_cfg/apps/payments/management/commands/README.md +95 -127
  10. django_cfg/apps/payments/management/commands/currency_stats.py +5 -24
  11. django_cfg/apps/payments/management/commands/manage_currencies.py +229 -0
  12. django_cfg/apps/payments/management/commands/manage_providers.py +235 -0
  13. django_cfg/apps/payments/managers/__init__.py +3 -2
  14. django_cfg/apps/payments/managers/balance_manager.py +2 -2
  15. django_cfg/apps/payments/managers/currency_manager.py +272 -49
  16. django_cfg/apps/payments/managers/payment_manager.py +161 -13
  17. django_cfg/apps/payments/middleware/api_access.py +2 -2
  18. django_cfg/apps/payments/middleware/rate_limiting.py +8 -18
  19. django_cfg/apps/payments/middleware/usage_tracking.py +20 -17
  20. django_cfg/apps/payments/migrations/0002_network_providercurrency_and_more.py +241 -0
  21. django_cfg/apps/payments/migrations/0003_add_usd_rate_cache.py +30 -0
  22. django_cfg/apps/payments/models/__init__.py +3 -2
  23. django_cfg/apps/payments/models/currencies.py +187 -71
  24. django_cfg/apps/payments/models/payments.py +3 -2
  25. django_cfg/apps/payments/serializers/__init__.py +3 -2
  26. django_cfg/apps/payments/serializers/currencies.py +20 -12
  27. django_cfg/apps/payments/services/cache/simple_cache.py +2 -2
  28. django_cfg/apps/payments/services/core/balance_service.py +2 -2
  29. django_cfg/apps/payments/services/core/fallback_service.py +2 -2
  30. django_cfg/apps/payments/services/core/payment_service.py +3 -6
  31. django_cfg/apps/payments/services/core/subscription_service.py +4 -7
  32. django_cfg/apps/payments/services/internal_types.py +171 -7
  33. django_cfg/apps/payments/services/monitoring/api_schemas.py +58 -204
  34. django_cfg/apps/payments/services/monitoring/provider_health.py +2 -2
  35. django_cfg/apps/payments/services/providers/base.py +144 -43
  36. django_cfg/apps/payments/services/providers/cryptapi/__init__.py +4 -0
  37. django_cfg/apps/payments/services/providers/cryptapi/config.py +8 -0
  38. django_cfg/apps/payments/services/providers/cryptapi/models.py +192 -0
  39. django_cfg/apps/payments/services/providers/cryptapi/provider.py +439 -0
  40. django_cfg/apps/payments/services/providers/cryptomus/__init__.py +4 -0
  41. django_cfg/apps/payments/services/providers/cryptomus/models.py +176 -0
  42. django_cfg/apps/payments/services/providers/cryptomus/provider.py +429 -0
  43. django_cfg/apps/payments/services/providers/cryptomus/provider_v2.py +564 -0
  44. django_cfg/apps/payments/services/providers/models/__init__.py +34 -0
  45. django_cfg/apps/payments/services/providers/models/currencies.py +190 -0
  46. django_cfg/apps/payments/services/providers/nowpayments/__init__.py +4 -0
  47. django_cfg/apps/payments/services/providers/nowpayments/models.py +196 -0
  48. django_cfg/apps/payments/services/providers/nowpayments/provider.py +380 -0
  49. django_cfg/apps/payments/services/providers/registry.py +294 -11
  50. django_cfg/apps/payments/services/providers/stripe/__init__.py +4 -0
  51. django_cfg/apps/payments/services/providers/stripe/models.py +184 -0
  52. django_cfg/apps/payments/services/providers/stripe/provider.py +109 -0
  53. django_cfg/apps/payments/services/security/error_handler.py +6 -8
  54. django_cfg/apps/payments/services/security/payment_notifications.py +2 -2
  55. django_cfg/apps/payments/services/security/webhook_validator.py +3 -4
  56. django_cfg/apps/payments/signals/api_key_signals.py +2 -2
  57. django_cfg/apps/payments/signals/payment_signals.py +11 -5
  58. django_cfg/apps/payments/signals/subscription_signals.py +2 -2
  59. django_cfg/apps/payments/static/payments/css/payments.css +340 -0
  60. django_cfg/apps/payments/static/payments/js/notifications.js +202 -0
  61. django_cfg/apps/payments/static/payments/js/payment-utils.js +318 -0
  62. django_cfg/apps/payments/static/payments/js/theme.js +86 -0
  63. django_cfg/apps/payments/tasks/webhook_processing.py +2 -2
  64. django_cfg/apps/payments/templates/admin/payments/currency/change_list.html +50 -0
  65. django_cfg/apps/payments/templates/payments/base.html +182 -0
  66. django_cfg/apps/payments/templates/payments/components/payment_card.html +201 -0
  67. django_cfg/apps/payments/templates/payments/components/payment_qr_code.html +109 -0
  68. django_cfg/apps/payments/templates/payments/components/progress_bar.html +43 -0
  69. django_cfg/apps/payments/templates/payments/components/provider_stats.html +40 -0
  70. django_cfg/apps/payments/templates/payments/components/status_badge.html +34 -0
  71. django_cfg/apps/payments/templates/payments/components/status_overview.html +148 -0
  72. django_cfg/apps/payments/templates/payments/dashboard.html +258 -0
  73. django_cfg/apps/payments/templates/payments/dashboard_simple_test.html +35 -0
  74. django_cfg/apps/payments/templates/payments/payment_create.html +579 -0
  75. django_cfg/apps/payments/templates/payments/payment_detail.html +373 -0
  76. django_cfg/apps/payments/templates/payments/payment_list.html +354 -0
  77. django_cfg/apps/payments/templates/payments/stats.html +261 -0
  78. django_cfg/apps/payments/templates/payments/test.html +213 -0
  79. django_cfg/apps/payments/templatetags/__init__.py +1 -0
  80. django_cfg/apps/payments/templatetags/payments_tags.py +315 -0
  81. django_cfg/apps/payments/urls.py +3 -1
  82. django_cfg/apps/payments/urls_admin.py +58 -0
  83. django_cfg/apps/payments/utils/__init__.py +1 -3
  84. django_cfg/apps/payments/utils/billing_utils.py +2 -2
  85. django_cfg/apps/payments/utils/config_utils.py +2 -8
  86. django_cfg/apps/payments/utils/validation_utils.py +2 -2
  87. django_cfg/apps/payments/views/__init__.py +3 -2
  88. django_cfg/apps/payments/views/currency_views.py +31 -20
  89. django_cfg/apps/payments/views/payment_views.py +2 -2
  90. django_cfg/apps/payments/views/templates/__init__.py +25 -0
  91. django_cfg/apps/payments/views/templates/ajax.py +451 -0
  92. django_cfg/apps/payments/views/templates/base.py +212 -0
  93. django_cfg/apps/payments/views/templates/dashboard.py +60 -0
  94. django_cfg/apps/payments/views/templates/payment_detail.py +102 -0
  95. django_cfg/apps/payments/views/templates/payment_management.py +158 -0
  96. django_cfg/apps/payments/views/templates/qr_code.py +174 -0
  97. django_cfg/apps/payments/views/templates/stats.py +244 -0
  98. django_cfg/apps/payments/views/templates/utils.py +181 -0
  99. django_cfg/apps/payments/views/webhook_views.py +2 -2
  100. django_cfg/apps/payments/viewsets.py +3 -2
  101. django_cfg/apps/tasks/urls.py +0 -2
  102. django_cfg/apps/tasks/urls_admin.py +14 -0
  103. django_cfg/apps/urls.py +6 -3
  104. django_cfg/core/config.py +35 -0
  105. django_cfg/models/payments.py +2 -8
  106. django_cfg/modules/django_currency/__init__.py +16 -11
  107. django_cfg/modules/django_currency/clients/__init__.py +4 -4
  108. django_cfg/modules/django_currency/clients/coinpaprika_client.py +289 -0
  109. django_cfg/modules/django_currency/clients/yahoo_client.py +157 -0
  110. django_cfg/modules/django_currency/core/__init__.py +1 -7
  111. django_cfg/modules/django_currency/core/converter.py +18 -23
  112. django_cfg/modules/django_currency/core/models.py +122 -11
  113. django_cfg/modules/django_currency/database/__init__.py +4 -4
  114. django_cfg/modules/django_currency/database/database_loader.py +190 -309
  115. django_cfg/modules/django_unfold/dashboard.py +7 -2
  116. django_cfg/registry/core.py +1 -0
  117. django_cfg/template_archive/.gitignore +1 -0
  118. django_cfg/template_archive/django_sample.zip +0 -0
  119. django_cfg/templates/admin/components/action_grid.html +9 -9
  120. django_cfg/templates/admin/components/metric_card.html +5 -5
  121. django_cfg/templates/admin/components/status_badge.html +2 -2
  122. django_cfg/templates/admin/layouts/dashboard_with_tabs.html +152 -24
  123. django_cfg/templates/admin/snippets/components/quick_actions.html +3 -3
  124. django_cfg/templates/admin/snippets/components/system_health.html +1 -1
  125. django_cfg/templates/admin/snippets/tabs/overview_tab.html +49 -52
  126. {django_cfg-1.2.27.dist-info → django_cfg-1.2.31.dist-info}/METADATA +13 -18
  127. {django_cfg-1.2.27.dist-info → django_cfg-1.2.31.dist-info}/RECORD +130 -83
  128. django_cfg/apps/payments/management/commands/populate_currencies.py +0 -246
  129. django_cfg/apps/payments/management/commands/update_currencies.py +0 -336
  130. django_cfg/apps/payments/services/providers/cryptapi.py +0 -273
  131. django_cfg/apps/payments/services/providers/cryptomus.py +0 -310
  132. django_cfg/apps/payments/services/providers/nowpayments.py +0 -293
  133. django_cfg/apps/payments/services/validators/__init__.py +0 -8
  134. django_cfg/modules/django_currency/clients/coingecko_client.py +0 -257
  135. django_cfg/modules/django_currency/clients/yfinance_client.py +0 -246
  136. {django_cfg-1.2.27.dist-info → django_cfg-1.2.31.dist-info}/WHEEL +0 -0
  137. {django_cfg-1.2.27.dist-info → django_cfg-1.2.31.dist-info}/entry_points.txt +0 -0
  138. {django_cfg-1.2.27.dist-info → django_cfg-1.2.31.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,213 @@
1
+ {% extends 'payments/base.html' %}
2
+ {% load static %}
3
+ {% load payments_tags %}
4
+
5
+ {% block header_title %}Payment Testing{% endblock %}
6
+ {% block header_subtitle %}Test payment integrations and webhooks{% endblock %}
7
+
8
+ {% block content %}
9
+ <div class="space-y-6">
10
+ <!-- Test Environment Status -->
11
+ <div class="bg-white dark:bg-base-900 border border-base-200 dark:border-base-700 rounded-default shadow-xs">
12
+ <div class="p-6 border-b border-base-200 dark:border-base-700">
13
+ <h2 class="text-lg font-semibold text-font-important-light dark:text-font-important-dark">Test Environment Status</h2>
14
+ <p class="text-sm text-font-subtle-light dark:text-font-subtle-dark mt-1">Current configuration and provider status</p>
15
+ </div>
16
+ <div class="p-6">
17
+ <div class="grid grid-cols-1 md:grid-cols-3 gap-4">
18
+ <div class="bg-base-50 dark:bg-base-800 p-4 rounded-default border border-base-200 dark:border-base-700">
19
+ <div class="flex items-center">
20
+ <span class="material-icons text-green-600 dark:text-green-400 text-xl mr-3">check_circle</span>
21
+ <div>
22
+ <h3 class="font-semibold text-font-important-light dark:text-font-important-dark">Test Mode</h3>
23
+ <p class="text-sm text-font-subtle-light dark:text-font-subtle-dark">Enabled</p>
24
+ </div>
25
+ </div>
26
+ </div>
27
+ <div class="bg-base-50 dark:bg-base-800 p-4 rounded-default border border-base-200 dark:border-base-700">
28
+ <div class="flex items-center">
29
+ <span class="material-icons text-blue-600 dark:text-blue-400 text-xl mr-3">webhook</span>
30
+ <div>
31
+ <h3 class="font-semibold text-font-important-light dark:text-font-important-dark">Webhooks</h3>
32
+ <p class="text-sm text-font-subtle-light dark:text-font-subtle-dark">Active</p>
33
+ </div>
34
+ </div>
35
+ </div>
36
+ <div class="bg-base-50 dark:bg-base-800 p-4 rounded-default border border-base-200 dark:border-base-700">
37
+ <div class="flex items-center">
38
+ <span class="material-icons text-orange-600 dark:text-orange-400 text-xl mr-3">security</span>
39
+ <div>
40
+ <h3 class="font-semibold text-font-important-light dark:text-font-important-dark">Security</h3>
41
+ <p class="text-sm text-font-subtle-light dark:text-font-subtle-dark">Validated</p>
42
+ </div>
43
+ </div>
44
+ </div>
45
+ </div>
46
+ </div>
47
+ </div>
48
+
49
+ <!-- Provider Testing -->
50
+ <div class="bg-white dark:bg-base-900 border border-base-200 dark:border-base-700 rounded-default shadow-xs">
51
+ <div class="p-6 border-b border-base-200 dark:border-base-700">
52
+ <h2 class="text-lg font-semibold text-font-important-light dark:text-font-important-dark">Provider Testing</h2>
53
+ <p class="text-sm text-font-subtle-light dark:text-font-subtle-dark mt-1">Test individual payment providers</p>
54
+ </div>
55
+ <div class="p-6">
56
+ <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
57
+ <!-- CryptAPI Test -->
58
+ <div class="bg-base-50 dark:bg-base-800 p-4 rounded-default border border-base-200 dark:border-base-700">
59
+ <div class="flex items-center justify-between mb-3">
60
+ <h3 class="font-semibold text-font-important-light dark:text-font-important-dark">CryptAPI</h3>
61
+ <span class="inline-flex items-center px-2 py-1 rounded-full text-xs font-medium bg-green-100 text-green-800 dark:bg-green-900 dark:text-green-200">
62
+ Active
63
+ </span>
64
+ </div>
65
+ <p class="text-sm text-font-subtle-light dark:text-font-subtle-dark mb-4">Test Bitcoin payments</p>
66
+ <button class="w-full bg-primary-600 hover:bg-primary-700 text-white px-4 py-2 rounded-default text-sm font-medium transition-colors duration-200">
67
+ Test CryptAPI
68
+ </button>
69
+ </div>
70
+
71
+ <!-- Cryptomus Test -->
72
+ <div class="bg-base-50 dark:bg-base-800 p-4 rounded-default border border-base-200 dark:border-base-700">
73
+ <div class="flex items-center justify-between mb-3">
74
+ <h3 class="font-semibold text-font-important-light dark:text-font-important-dark">Cryptomus</h3>
75
+ <span class="inline-flex items-center px-2 py-1 rounded-full text-xs font-medium bg-green-100 text-green-800 dark:bg-green-900 dark:text-green-200">
76
+ Active
77
+ </span>
78
+ </div>
79
+ <p class="text-sm text-font-subtle-light dark:text-font-subtle-dark mb-4">Test multi-crypto payments</p>
80
+ <button class="w-full bg-primary-600 hover:bg-primary-700 text-white px-4 py-2 rounded-default text-sm font-medium transition-colors duration-200">
81
+ Test Cryptomus
82
+ </button>
83
+ </div>
84
+
85
+ <!-- Stripe Test -->
86
+ <div class="bg-base-50 dark:bg-base-800 p-4 rounded-default border border-base-200 dark:border-base-700">
87
+ <div class="flex items-center justify-between mb-3">
88
+ <h3 class="font-semibold text-font-important-light dark:text-font-important-dark">Stripe</h3>
89
+ <span class="inline-flex items-center px-2 py-1 rounded-full text-xs font-medium bg-yellow-100 text-yellow-800 dark:bg-yellow-900 dark:text-yellow-200">
90
+ Configured
91
+ </span>
92
+ </div>
93
+ <p class="text-sm text-font-subtle-light dark:text-font-subtle-dark mb-4">Test card payments</p>
94
+ <button class="w-full bg-primary-600 hover:bg-primary-700 text-white px-4 py-2 rounded-default text-sm font-medium transition-colors duration-200">
95
+ Test Stripe
96
+ </button>
97
+ </div>
98
+ </div>
99
+ </div>
100
+ </div>
101
+
102
+ <!-- Webhook Testing -->
103
+ <div class="bg-white dark:bg-base-900 border border-base-200 dark:border-base-700 rounded-default shadow-xs">
104
+ <div class="p-6 border-b border-base-200 dark:border-base-700">
105
+ <h2 class="text-lg font-semibold text-font-important-light dark:text-font-important-dark">Webhook Testing</h2>
106
+ <p class="text-sm text-font-subtle-light dark:text-font-subtle-dark mt-1">Test webhook endpoints and processing</p>
107
+ </div>
108
+ <div class="p-6">
109
+ <div class="space-y-4">
110
+ <div class="bg-base-50 dark:bg-base-800 p-4 rounded-default border border-base-200 dark:border-base-700">
111
+ <h3 class="font-semibold text-font-important-light dark:text-font-important-dark mb-2">Ngrok Integration</h3>
112
+ <p class="text-sm text-font-subtle-light dark:text-font-subtle-dark mb-3">
113
+ Automatic tunnel creation for webhook testing
114
+ </p>
115
+ <div class="flex items-center space-x-3">
116
+ <button class="bg-blue-600 hover:bg-blue-700 text-white px-4 py-2 rounded-default text-sm font-medium transition-colors duration-200">
117
+ Start Ngrok
118
+ </button>
119
+ <button class="bg-gray-600 hover:bg-gray-700 text-white px-4 py-2 rounded-default text-sm font-medium transition-colors duration-200">
120
+ View Logs
121
+ </button>
122
+ <span class="text-sm text-font-subtle-light dark:text-font-subtle-dark">
123
+ Status: <span class="text-green-600 dark:text-green-400 font-medium">Ready</span>
124
+ </span>
125
+ </div>
126
+ </div>
127
+
128
+ <div class="bg-base-50 dark:bg-base-800 p-4 rounded-default border border-base-200 dark:border-base-700">
129
+ <h3 class="font-semibold text-font-important-light dark:text-font-important-dark mb-2">Webhook Simulation</h3>
130
+ <p class="text-sm text-font-subtle-light dark:text-font-subtle-dark mb-3">
131
+ Send test webhooks to validate processing
132
+ </p>
133
+ <div class="flex items-center space-x-3">
134
+ <button class="bg-green-600 hover:bg-green-700 text-white px-4 py-2 rounded-default text-sm font-medium transition-colors duration-200">
135
+ Send Test Payment
136
+ </button>
137
+ <button class="bg-orange-600 hover:bg-orange-700 text-white px-4 py-2 rounded-default text-sm font-medium transition-colors duration-200">
138
+ Send Test Failure
139
+ </button>
140
+ <button class="bg-purple-600 hover:bg-purple-700 text-white px-4 py-2 rounded-default text-sm font-medium transition-colors duration-200">
141
+ Send Test Confirmation
142
+ </button>
143
+ </div>
144
+ </div>
145
+ </div>
146
+ </div>
147
+ </div>
148
+
149
+ <!-- Test Results -->
150
+ <div class="bg-white dark:bg-base-900 border border-base-200 dark:border-base-700 rounded-default shadow-xs">
151
+ <div class="p-6 border-b border-base-200 dark:border-base-700">
152
+ <h2 class="text-lg font-semibold text-font-important-light dark:text-font-important-dark">Recent Test Results</h2>
153
+ <p class="text-sm text-font-subtle-light dark:text-font-subtle-dark mt-1">Latest testing activity</p>
154
+ </div>
155
+ <div class="p-6">
156
+ <div class="space-y-3">
157
+ <div class="flex items-center justify-between p-3 bg-base-50 dark:bg-base-800 rounded-default border border-base-200 dark:border-base-700">
158
+ <div class="flex items-center">
159
+ <span class="material-icons text-green-600 dark:text-green-400 text-sm mr-3">check_circle</span>
160
+ <div>
161
+ <p class="text-sm font-medium text-font-important-light dark:text-font-important-dark">CryptAPI Payment Test</p>
162
+ <p class="text-xs text-font-subtle-light dark:text-font-subtle-dark">Webhook received successfully</p>
163
+ </div>
164
+ </div>
165
+ <span class="text-xs text-font-subtle-light dark:text-font-subtle-dark">2 minutes ago</span>
166
+ </div>
167
+
168
+ <div class="flex items-center justify-between p-3 bg-base-50 dark:bg-base-800 rounded-default border border-base-200 dark:border-base-700">
169
+ <div class="flex items-center">
170
+ <span class="material-icons text-blue-600 dark:text-blue-400 text-sm mr-3">info</span>
171
+ <div>
172
+ <p class="text-sm font-medium text-font-important-light dark:text-font-important-dark">Ngrok Tunnel</p>
173
+ <p class="text-xs text-font-subtle-light dark:text-font-subtle-dark">Started on https://abc123.ngrok.io</p>
174
+ </div>
175
+ </div>
176
+ <span class="text-xs text-font-subtle-light dark:text-font-subtle-dark">5 minutes ago</span>
177
+ </div>
178
+
179
+ <div class="flex items-center justify-between p-3 bg-base-50 dark:bg-base-800 rounded-default border border-base-200 dark:border-base-700">
180
+ <div class="flex items-center">
181
+ <span class="material-icons text-orange-600 dark:text-orange-400 text-sm mr-3">warning</span>
182
+ <div>
183
+ <p class="text-sm font-medium text-font-important-light dark:text-font-important-dark">Cryptomus Configuration</p>
184
+ <p class="text-xs text-font-subtle-light dark:text-font-subtle-dark">API key validation required</p>
185
+ </div>
186
+ </div>
187
+ <span class="text-xs text-font-subtle-light dark:text-font-subtle-dark">10 minutes ago</span>
188
+ </div>
189
+ </div>
190
+ </div>
191
+ </div>
192
+ </div>
193
+ {% endblock %}
194
+
195
+ {% block extra_js %}
196
+ {{ block.super }}
197
+ <script>
198
+ // Testing functionality would be implemented here
199
+ document.addEventListener('DOMContentLoaded', function() {
200
+ console.log('Payment testing interface loaded');
201
+
202
+ // Add click handlers for test buttons
203
+ document.querySelectorAll('button').forEach(button => {
204
+ button.addEventListener('click', function() {
205
+ if (this.textContent.includes('Test')) {
206
+ console.log('Test initiated:', this.textContent);
207
+ // Actual testing logic would go here
208
+ }
209
+ });
210
+ });
211
+ });
212
+ </script>
213
+ {% endblock %}
@@ -0,0 +1 @@
1
+ # Template tags package for payments app
@@ -0,0 +1,315 @@
1
+ """
2
+ Payment Template Tags
3
+
4
+ Custom template tags for payment functionality, status badges, progress bars,
5
+ and real-time payment tracking components.
6
+ """
7
+
8
+ from django import template
9
+ from django.utils.safestring import mark_safe
10
+ from django.utils.html import format_html
11
+ from django.db.models import Count, Sum, Q
12
+ from decimal import Decimal
13
+ import json
14
+
15
+ register = template.Library()
16
+
17
+
18
+ @register.simple_tag
19
+ def payment_status_badge(payment):
20
+ """Render payment status badge with icon and color."""
21
+ status_config = {
22
+ 'pending': {'color': 'yellow', 'icon': 'pending', 'animate': True},
23
+ 'confirming': {'color': 'blue', 'icon': 'sync', 'animate': True},
24
+ 'confirmed': {'color': 'green', 'icon': 'check_circle', 'animate': False},
25
+ 'completed': {'color': 'green', 'icon': 'verified', 'animate': False},
26
+ 'failed': {'color': 'red', 'icon': 'error', 'animate': False},
27
+ 'expired': {'color': 'gray', 'icon': 'schedule', 'animate': False},
28
+ 'cancelled': {'color': 'gray', 'icon': 'cancel', 'animate': False},
29
+ 'refunded': {'color': 'purple', 'icon': 'undo', 'animate': False},
30
+ }
31
+
32
+ config = status_config.get(payment.status, {'color': 'gray', 'icon': 'help', 'animate': False})
33
+ animate_class = 'animate-pulse' if config['animate'] else ''
34
+
35
+ return format_html(
36
+ '<span class="inline-flex items-center px-2.5 py-0.5 rounded-full text-xs font-medium bg-{}-100 text-{}-800 dark:bg-{}-900 dark:text-{}-200">'
37
+ '<span class="material-icons text-sm mr-1 {}">{}</span>'
38
+ '<span class="status-text">{}</span>'
39
+ '</span>',
40
+ config['color'], config['color'], config['color'], config['color'],
41
+ animate_class,
42
+ config['icon'],
43
+ payment.get_status_display()
44
+ )
45
+
46
+
47
+ @register.filter
48
+ def payment_progress_percentage(payment):
49
+ """Calculate payment progress percentage."""
50
+ progress_map = {
51
+ 'pending': 10,
52
+ 'confirming': 40,
53
+ 'confirmed': 70,
54
+ 'completed': 100,
55
+ 'failed': 0,
56
+ 'expired': 0,
57
+ 'cancelled': 0,
58
+ 'refunded': 50, # Partial progress for refunds
59
+ }
60
+ return progress_map.get(payment.status, 0)
61
+
62
+
63
+ @register.filter
64
+ def payment_progress_steps(payment):
65
+ """Get payment progress steps with status."""
66
+ steps = [
67
+ {'label': 'Created', 'key': 'created'},
68
+ {'label': 'Processing', 'key': 'processing'},
69
+ {'label': 'Confirming', 'key': 'confirming'},
70
+ {'label': 'Completed', 'key': 'completed'},
71
+ ]
72
+
73
+ status_order = ['pending', 'confirming', 'confirmed', 'completed']
74
+ current_index = status_order.index(payment.status) if payment.status in status_order else -1
75
+
76
+ for i, step in enumerate(steps):
77
+ step['completed'] = i < current_index
78
+ step['active'] = i == current_index
79
+
80
+ return steps
81
+
82
+
83
+ @register.inclusion_tag('payments/components/payment_card.html')
84
+ def payment_card(payment, show_actions=True, compact=False):
85
+ """Render payment card component."""
86
+ return {
87
+ 'payment': payment,
88
+ 'show_actions': show_actions,
89
+ 'compact': compact
90
+ }
91
+
92
+
93
+ @register.inclusion_tag('payments/components/status_badge.html')
94
+ def render_payment_status(payment):
95
+ """Render payment status badge component."""
96
+ return {'payment': payment}
97
+
98
+
99
+ @register.inclusion_tag('payments/components/progress_bar.html')
100
+ def payment_progress_bar(payment):
101
+ """Render payment progress bar component."""
102
+ return {
103
+ 'payment': payment,
104
+ 'percentage': payment_progress_percentage(payment),
105
+ 'steps': payment_progress_steps(payment)
106
+ }
107
+
108
+
109
+ @register.inclusion_tag('payments/components/payment_tracker.html', takes_context=True)
110
+ def payment_tracker(context, payment_id):
111
+ """Render real-time payment tracker."""
112
+ try:
113
+ from ..models import UniversalPayment
114
+ payment = UniversalPayment.objects.get(id=payment_id)
115
+ return {
116
+ 'payment': payment,
117
+ 'request': context.get('request'),
118
+ 'user': context.get('user'),
119
+ 'websocket_url': f"/ws/payments/{payment_id}/"
120
+ }
121
+ except UniversalPayment.DoesNotExist:
122
+ return {'payment': None}
123
+
124
+
125
+ @register.simple_tag
126
+ def payment_websocket_url(payment_id):
127
+ """Get WebSocket URL for real-time payment updates."""
128
+ return f"/ws/payments/{payment_id}/"
129
+
130
+
131
+ @register.filter
132
+ def format_crypto_amount(amount, currency_code):
133
+ """Format cryptocurrency amount with proper decimals."""
134
+ if not amount:
135
+ return "0"
136
+
137
+ # Different currencies have different decimal places
138
+ decimal_places = {
139
+ 'BTC': 8,
140
+ 'ETH': 6,
141
+ 'LTC': 8,
142
+ 'USDT': 6,
143
+ 'USDC': 6,
144
+ 'USD': 2,
145
+ 'EUR': 2,
146
+ }
147
+
148
+ places = decimal_places.get(currency_code.upper(), 6)
149
+ formatted = f"{float(amount):.{places}f}".rstrip('0').rstrip('.')
150
+ return formatted if formatted else "0"
151
+
152
+
153
+ @register.simple_tag
154
+ def get_payment_stats():
155
+ """Get payment statistics for dashboard."""
156
+ try:
157
+ from ..models import UniversalPayment
158
+
159
+ stats = UniversalPayment.objects.aggregate(
160
+ total_count=Count('id'),
161
+ pending_count=Count('id', filter=Q(status='pending')),
162
+ confirming_count=Count('id', filter=Q(status='confirming')),
163
+ completed_count=Count('id', filter=Q(status='completed')),
164
+ failed_count=Count('id', filter=Q(status='failed')),
165
+ total_volume=Sum('amount_usd')
166
+ )
167
+
168
+ return {
169
+ 'total_payments_count': stats['total_count'] or 0,
170
+ 'pending_payments_count': stats['pending_count'] or 0,
171
+ 'confirming_payments_count': stats['confirming_count'] or 0,
172
+ 'completed_payments_count': stats['completed_count'] or 0,
173
+ 'failed_payments_count': stats['failed_count'] or 0,
174
+ 'total_volume': float(stats['total_volume'] or 0),
175
+ }
176
+ except Exception:
177
+ # Return default values if there's any error
178
+ return {
179
+ 'total_payments_count': 0,
180
+ 'pending_payments_count': 0,
181
+ 'confirming_payments_count': 0,
182
+ 'completed_payments_count': 0,
183
+ 'failed_payments_count': 0,
184
+ 'total_volume': 0.0,
185
+ }
186
+
187
+
188
+ @register.inclusion_tag('payments/components/provider_stats.html')
189
+ def provider_statistics():
190
+ """Render provider statistics."""
191
+ try:
192
+ from ..models import UniversalPayment
193
+ from django.db.models import Avg
194
+
195
+ stats = UniversalPayment.objects.values('provider').annotate(
196
+ count=Count('id'),
197
+ volume=Sum('amount_usd'),
198
+ avg_amount=Avg('amount_usd'),
199
+ completed_count=Count('id', filter=Q(status='completed')),
200
+ ).order_by('-volume')
201
+
202
+ # Calculate success rate
203
+ for stat in stats:
204
+ if stat['count'] > 0:
205
+ stat['success_rate'] = (stat['completed_count'] / stat['count']) * 100
206
+ else:
207
+ stat['success_rate'] = 0
208
+
209
+ return {'provider_stats': stats}
210
+ except Exception:
211
+ return {'provider_stats': []}
212
+
213
+
214
+ @register.simple_tag
215
+ def payment_status_distribution():
216
+ """Get payment status distribution for charts."""
217
+ try:
218
+ from ..models import UniversalPayment
219
+
220
+ distribution = UniversalPayment.objects.values('status').annotate(
221
+ count=Count('id')
222
+ ).order_by('-count')
223
+
224
+ return {item['status']: item['count'] for item in distribution}
225
+ except Exception:
226
+ return {}
227
+
228
+
229
+ @register.filter
230
+ def provider_display_name(provider_key):
231
+ """Get display name for provider."""
232
+ provider_names = {
233
+ 'nowpayments': 'NowPayments',
234
+ 'cryptapi': 'CryptAPI',
235
+ 'cryptomus': 'Cryptomus',
236
+ 'stripe': 'Stripe',
237
+ 'internal': 'Internal',
238
+ }
239
+ return provider_names.get(provider_key, provider_key.title())
240
+
241
+
242
+ @register.filter
243
+ def payment_method_icon(provider):
244
+ """Get icon for payment method."""
245
+ icons = {
246
+ 'nowpayments': 'currency_bitcoin',
247
+ 'cryptapi': 'currency_bitcoin',
248
+ 'cryptomus': 'currency_bitcoin',
249
+ 'stripe': 'credit_card',
250
+ 'internal': 'account_balance',
251
+ }
252
+ return icons.get(provider, 'payment')
253
+
254
+
255
+ @register.simple_tag
256
+ def payment_json_data(payment):
257
+ """Convert payment to JSON for JavaScript use."""
258
+ try:
259
+ data = {
260
+ 'id': str(payment.id),
261
+ 'status': payment.status,
262
+ 'amount_usd': float(payment.amount_usd),
263
+ 'currency_code': payment.currency_code,
264
+ 'provider': payment.provider,
265
+ 'created_at': payment.created_at.isoformat(),
266
+ 'progress_percentage': payment_progress_percentage(payment),
267
+ }
268
+ return mark_safe(json.dumps(data))
269
+ except Exception:
270
+ return mark_safe('{}')
271
+
272
+
273
+ @register.filter
274
+ def time_since_created(payment):
275
+ """Get human-readable time since payment was created."""
276
+ from django.utils import timezone
277
+ from django.utils.timesince import timesince
278
+
279
+ if payment.created_at:
280
+ return timesince(payment.created_at, timezone.now())
281
+ return "Unknown"
282
+
283
+
284
+ @register.filter
285
+ def is_crypto_payment(payment):
286
+ """Check if payment is cryptocurrency-based."""
287
+ crypto_providers = ['nowpayments', 'cryptapi', 'cryptomus']
288
+ return payment.provider in crypto_providers
289
+
290
+
291
+ @register.filter
292
+ def can_cancel_payment(payment):
293
+ """Check if payment can be cancelled."""
294
+ cancellable_statuses = ['pending', 'confirming']
295
+ return payment.status in cancellable_statuses
296
+
297
+
298
+ @register.filter
299
+ def payment_qr_code_data(payment):
300
+ """Get QR code data for payment."""
301
+ if hasattr(payment, 'pay_address') and payment.pay_address:
302
+ # For crypto payments, use address:amount format
303
+ if payment.pay_amount:
304
+ return f"{payment.pay_address}?amount={payment.pay_amount}"
305
+ return payment.pay_address
306
+ return None
307
+
308
+
309
+ @register.inclusion_tag('payments/components/payment_qr_code.html')
310
+ def payment_qr_code(payment):
311
+ """Render QR code for payment."""
312
+ return {
313
+ 'payment': payment,
314
+ 'qr_data': payment_qr_code_data(payment)
315
+ }
@@ -20,7 +20,8 @@ router.register(r'api-keys', views.APIKeyViewSet, basename='apikey')
20
20
  router.register(r'balances', views.UserBalanceViewSet, basename='balance')
21
21
  router.register(r'transactions', views.TransactionViewSet, basename='transaction')
22
22
  router.register(r'currencies', views.CurrencyViewSet, basename='currency')
23
- router.register(r'currency-networks', views.CurrencyNetworkViewSet, basename='currencynetwork')
23
+ router.register(r'networks', views.NetworkViewSet, basename='network')
24
+ router.register(r'provider-currencies', views.ProviderCurrencyViewSet, basename='providercurrency')
24
25
  router.register(r'endpoint-groups', views.EndpointGroupViewSet, basename='endpointgroup')
25
26
  router.register(r'tariffs', views.TariffViewSet, basename='tariff')
26
27
  router.register(r'tariff-endpoint-groups', views.TariffEndpointGroupViewSet, basename='tariffendpointgroup')
@@ -75,4 +76,5 @@ urlpatterns = [
75
76
 
76
77
  # Include generic API endpoints
77
78
  path('', include(generic_patterns)),
79
+
78
80
  ]
@@ -0,0 +1,58 @@
1
+ """
2
+ Template URLs for Payment Dashboard.
3
+
4
+ All URLs require superuser access as this is an internal admin tool.
5
+ """
6
+
7
+ from django.urls import path
8
+ from .views.templates import (
9
+ PaymentDashboardView,
10
+ PaymentDetailView,
11
+ PaymentCreateView,
12
+ PaymentStatsView,
13
+ PaymentListView,
14
+ PaymentQRCodeView,
15
+ PaymentTestView,
16
+ payment_status_ajax,
17
+ payment_events_ajax,
18
+ )
19
+ from .views.templates.ajax import (
20
+ payment_stats_ajax,
21
+ payment_search_ajax,
22
+ payment_action_ajax,
23
+ provider_currencies_ajax,
24
+ all_providers_data_ajax,
25
+ )
26
+ from .views.templates.qr_code import qr_code_data_ajax
27
+
28
+ app_name = 'payments_dashboard'
29
+
30
+ urlpatterns = [
31
+ # Main dashboard
32
+ path('', PaymentDashboardView.as_view(), name='dashboard'),
33
+ path('dashboard/', PaymentDashboardView.as_view(), name='dashboard_alt'),
34
+
35
+ # Payment management
36
+ path('list/', PaymentListView.as_view(), name='list'),
37
+ path('create/', PaymentCreateView.as_view(), name='create'),
38
+ path('stats/', PaymentStatsView.as_view(), name='stats'),
39
+
40
+ # Payment details
41
+ path('payment/<uuid:pk>/', PaymentDetailView.as_view(), name='detail'),
42
+ path('payment/<uuid:pk>/qr/', PaymentQRCodeView.as_view(), name='qr_code'),
43
+
44
+ # AJAX endpoints
45
+ path('ajax/payment/<uuid:payment_id>/status/', payment_status_ajax, name='payment_status_ajax'),
46
+ path('ajax/payment/<uuid:payment_id>/events/', payment_events_ajax, name='payment_events_ajax'),
47
+ path('ajax/payment/<uuid:payment_id>/qr-data/', qr_code_data_ajax, name='qr_data_ajax'),
48
+ path('ajax/payment/<uuid:payment_id>/action/', payment_action_ajax, name='payment_action_ajax'),
49
+ path('ajax/stats/', payment_stats_ajax, name='payment_stats_ajax'),
50
+ path('ajax/search/', payment_search_ajax, name='payment_search_ajax'),
51
+
52
+ # Provider and Currency AJAX endpoints
53
+ path('ajax/provider/currencies/', provider_currencies_ajax, name='provider_currencies_ajax'),
54
+ path('ajax/providers/all/', all_providers_data_ajax, name='all_providers_data_ajax'),
55
+
56
+ # Development/testing
57
+ path('test/', PaymentTestView.as_view(), name='test'),
58
+ ]
@@ -13,8 +13,7 @@ from .config_utils import (
13
13
  CacheConfigHelper,
14
14
  ProviderConfigHelper,
15
15
  get_payments_config,
16
- is_payments_enabled,
17
- is_debug_mode
16
+ is_payments_enabled
18
17
  )
19
18
 
20
19
  __all__ = [
@@ -41,5 +40,4 @@ __all__ = [
41
40
  'ProviderConfigHelper',
42
41
  'get_payments_config',
43
42
  'is_payments_enabled',
44
- 'is_debug_mode',
45
43
  ]
@@ -5,7 +5,7 @@ Provides essential billing calculations and transaction management
5
5
  without over-engineering.
6
6
  """
7
7
 
8
- import logging
8
+ from django_cfg.modules.django_logger import get_logger
9
9
  from typing import Dict, Any, Optional, Tuple
10
10
  from decimal import Decimal, ROUND_HALF_UP
11
11
  from datetime import datetime, timedelta
@@ -16,7 +16,7 @@ from django.contrib.auth import get_user_model
16
16
  from ..models import UserBalance, Transaction, Subscription
17
17
 
18
18
  User = get_user_model()
19
- logger = logging.getLogger(__name__)
19
+ logger = get_logger("billing_utils")
20
20
 
21
21
 
22
22
  def calculate_usage_cost(