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,424 @@
1
+ """
2
+ API Key serializers for the Universal Payment System v2.0.
3
+
4
+ DRF serializers for API key operations with service integration.
5
+ """
6
+
7
+ from rest_framework import serializers
8
+ from typing import Dict, Any
9
+ from django.contrib.auth import get_user_model
10
+
11
+ from ...models import APIKey
12
+ from django_cfg.modules.django_logger import get_logger
13
+
14
+ User = get_user_model()
15
+ logger = get_logger("api_key_serializers")
16
+
17
+
18
+ class APIKeyListSerializer(serializers.ModelSerializer):
19
+ """
20
+ Lightweight API key serializer for lists.
21
+
22
+ Optimized for API key lists with minimal data (no key value).
23
+ """
24
+
25
+ user = serializers.StringRelatedField(read_only=True)
26
+ is_expired = serializers.BooleanField(source='is_expired', read_only=True)
27
+ is_valid = serializers.BooleanField(source='is_valid', read_only=True)
28
+
29
+ class Meta:
30
+ model = APIKey
31
+ fields = [
32
+ 'id',
33
+ 'user',
34
+ 'name',
35
+ 'is_active',
36
+ 'is_expired',
37
+ 'is_valid',
38
+ 'total_requests',
39
+ 'last_used_at',
40
+ 'expires_at',
41
+ 'created_at',
42
+ ]
43
+ read_only_fields = fields
44
+
45
+
46
+ class APIKeySerializer(serializers.ModelSerializer):
47
+ """
48
+ Complete API key serializer with full details.
49
+
50
+ Used for API key detail views (no key value for security).
51
+ """
52
+
53
+ user = serializers.StringRelatedField(read_only=True)
54
+ key_preview = serializers.CharField(source='key_preview', read_only=True)
55
+ is_expired = serializers.BooleanField(source='is_expired', read_only=True)
56
+ is_valid = serializers.BooleanField(source='is_valid', read_only=True)
57
+ days_until_expiry = serializers.IntegerField(source='days_until_expiry', read_only=True)
58
+
59
+ class Meta:
60
+ model = APIKey
61
+ fields = [
62
+ 'id',
63
+ 'user',
64
+ 'name',
65
+ 'key_preview',
66
+ 'is_active',
67
+ 'is_expired',
68
+ 'is_valid',
69
+ 'days_until_expiry',
70
+ 'total_requests',
71
+ 'last_used_at',
72
+ 'expires_at',
73
+ 'created_at',
74
+ 'updated_at',
75
+ ]
76
+ read_only_fields = [
77
+ 'id',
78
+ 'user',
79
+ 'key_preview',
80
+ 'is_expired',
81
+ 'is_valid',
82
+ 'days_until_expiry',
83
+ 'total_requests',
84
+ 'last_used_at',
85
+ 'created_at',
86
+ 'updated_at',
87
+ ]
88
+
89
+
90
+ class APIKeyCreateSerializer(serializers.Serializer):
91
+ """
92
+ API key creation serializer with service integration.
93
+
94
+ Creates new API keys and returns the full key value (only once).
95
+ """
96
+
97
+ name = serializers.CharField(
98
+ max_length=100,
99
+ help_text="Descriptive name for the API key"
100
+ )
101
+ expires_in_days = serializers.IntegerField(
102
+ required=False,
103
+ allow_null=True,
104
+ min_value=1,
105
+ max_value=365,
106
+ help_text="Expiration in days (optional, null for no expiration)"
107
+ )
108
+
109
+ def validate_name(self, value: str) -> str:
110
+ """Validate API key name is unique for user."""
111
+ user = self.context['request'].user
112
+ user_id = self.context.get('user_pk', user.id)
113
+
114
+ if APIKey.objects.filter(user_id=user_id, name=value).exists():
115
+ raise serializers.ValidationError(f"API key with name '{value}' already exists")
116
+
117
+ return value
118
+
119
+ def create(self, validated_data: Dict[str, Any]) -> APIKey:
120
+ """Create API key using APIKey manager."""
121
+ try:
122
+ user = self.context['request'].user
123
+ user_id = self.context.get('user_pk', user.id)
124
+
125
+ # Get user object if creating for different user (admin only)
126
+ if str(user_id) != str(user.id):
127
+ if not user.is_staff:
128
+ raise serializers.ValidationError("Only staff can create API keys for other users")
129
+ user = User.objects.get(id=user_id)
130
+
131
+ # Create API key using manager
132
+ api_key = APIKey.create_for_user(
133
+ user=user,
134
+ name=validated_data['name'],
135
+ expires_in_days=validated_data.get('expires_in_days')
136
+ )
137
+
138
+ if api_key:
139
+ logger.info(f"API key created successfully", extra={
140
+ 'api_key_id': str(api_key.id),
141
+ 'user_id': user.id,
142
+ 'name': validated_data['name']
143
+ })
144
+ return api_key
145
+ else:
146
+ raise serializers.ValidationError("Failed to create API key")
147
+
148
+ except User.DoesNotExist:
149
+ raise serializers.ValidationError(f"User {user_id} not found")
150
+ except Exception as e:
151
+ logger.error(f"API key creation error: {e}")
152
+ raise serializers.ValidationError(f"API key creation failed: {e}")
153
+
154
+ def to_representation(self, instance: APIKey) -> Dict[str, Any]:
155
+ """Return API key data with full key value (only on creation)."""
156
+ data = APIKeySerializer(instance, context=self.context).data
157
+
158
+ # Add full key value only on creation (security: shown only once)
159
+ data['key'] = instance.key
160
+ data['warning'] = "This is the only time the full API key will be shown. Please save it securely."
161
+
162
+ return data
163
+
164
+
165
+ class APIKeyUpdateSerializer(serializers.ModelSerializer):
166
+ """
167
+ API key update serializer for modifying API key properties.
168
+
169
+ Allows updating name and active status only.
170
+ """
171
+
172
+ class Meta:
173
+ model = APIKey
174
+ fields = ['name', 'is_active']
175
+
176
+ def validate_name(self, value: str) -> str:
177
+ """Validate API key name is unique for user."""
178
+ user_id = self.instance.user_id
179
+
180
+ # Check if another API key with same name exists for this user
181
+ existing = APIKey.objects.filter(
182
+ user_id=user_id,
183
+ name=value
184
+ ).exclude(id=self.instance.id)
185
+
186
+ if existing.exists():
187
+ raise serializers.ValidationError(f"API key with name '{value}' already exists")
188
+
189
+ return value
190
+
191
+ def update(self, instance: APIKey, validated_data: Dict[str, Any]) -> APIKey:
192
+ """Update API key with logging."""
193
+ old_name = instance.name
194
+ old_active = instance.is_active
195
+
196
+ instance = super().update(instance, validated_data)
197
+
198
+ # Log changes
199
+ changes = []
200
+ if instance.name != old_name:
201
+ changes.append(f"name: {old_name} → {instance.name}")
202
+ if instance.is_active != old_active:
203
+ changes.append(f"active: {old_active} → {instance.is_active}")
204
+
205
+ if changes:
206
+ logger.info(f"API key updated: {', '.join(changes)}", extra={
207
+ 'api_key_id': str(instance.id),
208
+ 'user_id': instance.user_id
209
+ })
210
+
211
+ return instance
212
+
213
+
214
+ class APIKeyActionSerializer(serializers.Serializer):
215
+ """
216
+ API key action serializer for key operations.
217
+
218
+ Handles deactivation, extension, and usage increment.
219
+ """
220
+
221
+ action = serializers.ChoiceField(
222
+ choices=[
223
+ ('deactivate', 'Deactivate'),
224
+ ('extend', 'Extend Expiration'),
225
+ ('increment_usage', 'Increment Usage'),
226
+ ],
227
+ help_text="Action to perform on API key"
228
+ )
229
+ reason = serializers.CharField(
230
+ required=False,
231
+ allow_blank=True,
232
+ max_length=500,
233
+ help_text="Reason for deactivation"
234
+ )
235
+ days = serializers.IntegerField(
236
+ required=False,
237
+ min_value=1,
238
+ max_value=365,
239
+ help_text="Days to extend expiration"
240
+ )
241
+ ip_address = serializers.IPAddressField(
242
+ required=False,
243
+ allow_null=True,
244
+ help_text="IP address for usage tracking"
245
+ )
246
+
247
+ def validate(self, attrs: Dict[str, Any]) -> Dict[str, Any]:
248
+ """Validate action-specific requirements."""
249
+ action = attrs.get('action')
250
+
251
+ if action == 'extend' and not attrs.get('days'):
252
+ raise serializers.ValidationError("days is required for extend action")
253
+
254
+ return attrs
255
+
256
+ def save(self) -> Dict[str, Any]:
257
+ """Perform API key action."""
258
+ try:
259
+ api_key_id = self.context.get('api_key_id')
260
+ if not api_key_id:
261
+ raise serializers.ValidationError("API key ID is required")
262
+
263
+ api_key = APIKey.objects.get(id=api_key_id)
264
+ action = self.validated_data['action']
265
+
266
+ if action == 'deactivate':
267
+ reason = self.validated_data.get('reason')
268
+ success = api_key.deactivate(reason=reason)
269
+ message = f"API key deactivated successfully"
270
+
271
+ elif action == 'extend':
272
+ days = self.validated_data['days']
273
+ success = api_key.extend_expiry(days=days)
274
+ message = f"API key expiration extended by {days} days"
275
+
276
+ elif action == 'increment_usage':
277
+ ip_address = self.validated_data.get('ip_address')
278
+ success = api_key.increment_usage(ip_address=ip_address)
279
+ message = f"API key usage incremented"
280
+
281
+ else:
282
+ raise serializers.ValidationError(f"Unknown action: {action}")
283
+
284
+ if success:
285
+ # Refresh from database
286
+ api_key.refresh_from_db()
287
+
288
+ return {
289
+ 'success': True,
290
+ 'message': message,
291
+ 'api_key': APIKeySerializer(api_key, context=self.context).data
292
+ }
293
+ else:
294
+ return {
295
+ 'success': False,
296
+ 'error': f"Failed to {action} API key",
297
+ 'error_code': f'{action}_failed'
298
+ }
299
+
300
+ except APIKey.DoesNotExist:
301
+ return {
302
+ 'success': False,
303
+ 'error': 'API key not found',
304
+ 'error_code': 'api_key_not_found'
305
+ }
306
+ except Exception as e:
307
+ logger.error(f"API key action error: {e}")
308
+ return {
309
+ 'success': False,
310
+ 'error': f"API key action failed: {e}",
311
+ 'error_code': 'api_key_action_error'
312
+ }
313
+
314
+
315
+ class APIKeyValidationSerializer(serializers.Serializer):
316
+ """
317
+ API key validation serializer.
318
+
319
+ Validates API key and returns key information.
320
+ """
321
+
322
+ key = serializers.CharField(
323
+ min_length=32,
324
+ max_length=64,
325
+ help_text="API key to validate"
326
+ )
327
+
328
+ def validate_key(self, value: str) -> str:
329
+ """Validate API key format."""
330
+ if not value.startswith('pk_'):
331
+ raise serializers.ValidationError("Invalid API key format")
332
+ return value
333
+
334
+ def save(self) -> Dict[str, Any]:
335
+ """Validate API key and return information."""
336
+ try:
337
+ key_value = self.validated_data['key']
338
+ api_key = APIKey.get_valid_key(key_value)
339
+
340
+ if api_key:
341
+ return {
342
+ 'success': True,
343
+ 'valid': True,
344
+ 'api_key': APIKeySerializer(api_key, context=self.context).data,
345
+ 'message': 'API key is valid'
346
+ }
347
+ else:
348
+ return {
349
+ 'success': True,
350
+ 'valid': False,
351
+ 'api_key': None,
352
+ 'message': 'API key is invalid or expired'
353
+ }
354
+
355
+ except Exception as e:
356
+ logger.error(f"API key validation error: {e}")
357
+ return {
358
+ 'success': False,
359
+ 'error': f"API key validation failed: {e}",
360
+ 'error_code': 'validation_error'
361
+ }
362
+
363
+
364
+ class APIKeyStatsSerializer(serializers.Serializer):
365
+ """
366
+ API key statistics serializer.
367
+
368
+ Used for API key analytics and reporting.
369
+ """
370
+
371
+ days = serializers.IntegerField(
372
+ default=30,
373
+ min_value=1,
374
+ max_value=365,
375
+ help_text="Number of days to analyze"
376
+ )
377
+
378
+ def save(self) -> Dict[str, Any]:
379
+ """Get API key statistics."""
380
+ try:
381
+ from django.db import models
382
+ from django.utils import timezone
383
+ from datetime import timedelta
384
+
385
+ days = self.validated_data['days']
386
+ since_date = timezone.now() - timedelta(days=days)
387
+
388
+ # Get user-specific stats if user context provided
389
+ user_id = self.context.get('user_pk')
390
+ if user_id:
391
+ queryset = APIKey.objects.filter(user_id=user_id)
392
+ else:
393
+ queryset = APIKey.objects.all()
394
+
395
+ stats = queryset.aggregate(
396
+ total_keys=models.Count('id'),
397
+ active_keys=models.Count('id', filter=models.Q(is_active=True)),
398
+ expired_keys=models.Count('id', filter=models.Q(expires_at__lt=timezone.now())),
399
+ total_requests=models.Sum('total_requests'),
400
+ recent_usage=models.Count(
401
+ 'id',
402
+ filter=models.Q(last_used_at__gte=since_date)
403
+ ),
404
+ )
405
+
406
+ return {
407
+ 'success': True,
408
+ 'stats': {
409
+ **stats,
410
+ 'total_requests': stats['total_requests'] or 0,
411
+ 'inactive_keys': stats['total_keys'] - stats['active_keys'],
412
+ 'usage_rate': (stats['recent_usage'] / stats['total_keys'] * 100) if stats['total_keys'] > 0 else 0,
413
+ },
414
+ 'period_days': days,
415
+ 'generated_at': timezone.now().isoformat()
416
+ }
417
+
418
+ except Exception as e:
419
+ logger.error(f"API key stats error: {e}")
420
+ return {
421
+ 'success': False,
422
+ 'error': f"Stats generation failed: {e}",
423
+ 'error_code': 'stats_error'
424
+ }