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.
- django_cfg/__init__.py +1 -1
- django_cfg/apps/api/health/views.py +4 -2
- django_cfg/apps/knowbase/config/settings.py +16 -15
- django_cfg/apps/payments/README.md +326 -0
- django_cfg/apps/payments/admin/__init__.py +20 -9
- django_cfg/apps/payments/admin/api_keys_admin.py +521 -237
- django_cfg/apps/payments/admin/balance_admin.py +592 -297
- django_cfg/apps/payments/admin/currencies_admin.py +600 -108
- django_cfg/apps/payments/admin/filters.py +306 -199
- django_cfg/apps/payments/admin/payments_admin.py +470 -64
- django_cfg/apps/payments/admin/subscriptions_admin.py +578 -128
- django_cfg/apps/payments/admin_interface/__init__.py +18 -0
- django_cfg/apps/payments/admin_interface/templates/payments/base.html +162 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/dev_tool_card.html +38 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/loading_spinner.html +16 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/notification.html +27 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/provider_card.html +86 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/status_card.html +39 -0
- django_cfg/apps/payments/admin_interface/templates/payments/currency_converter.html +382 -0
- django_cfg/apps/payments/admin_interface/templates/payments/payment_dashboard.html +300 -0
- django_cfg/apps/payments/admin_interface/templates/payments/payment_form.html +303 -0
- django_cfg/apps/payments/admin_interface/templates/payments/payment_list.html +382 -0
- django_cfg/apps/payments/admin_interface/templates/payments/payment_status.html +500 -0
- django_cfg/apps/payments/admin_interface/templates/payments/webhook_dashboard.html +594 -0
- django_cfg/apps/payments/admin_interface/views/__init__.py +23 -0
- django_cfg/apps/payments/admin_interface/views/payment_views.py +259 -0
- django_cfg/apps/payments/admin_interface/views/webhook_dashboard.py +37 -0
- django_cfg/apps/payments/apps.py +34 -9
- django_cfg/apps/payments/config/__init__.py +28 -51
- django_cfg/apps/payments/config/constance/__init__.py +22 -0
- django_cfg/apps/payments/config/constance/config_service.py +123 -0
- django_cfg/apps/payments/config/constance/fields.py +69 -0
- django_cfg/apps/payments/config/constance/settings.py +160 -0
- django_cfg/apps/payments/config/django_cfg_integration.py +202 -0
- django_cfg/apps/payments/config/helpers.py +130 -0
- django_cfg/apps/payments/management/__init__.py +1 -3
- django_cfg/apps/payments/management/commands/__init__.py +1 -3
- django_cfg/apps/payments/management/commands/manage_currencies.py +381 -0
- django_cfg/apps/payments/management/commands/manage_providers.py +408 -0
- django_cfg/apps/payments/middleware/__init__.py +3 -1
- django_cfg/apps/payments/middleware/api_access.py +329 -222
- django_cfg/apps/payments/middleware/rate_limiting.py +343 -163
- django_cfg/apps/payments/middleware/usage_tracking.py +250 -238
- django_cfg/apps/payments/migrations/0001_initial.py +708 -536
- django_cfg/apps/payments/models/__init__.py +16 -20
- django_cfg/apps/payments/models/api_keys.py +121 -43
- django_cfg/apps/payments/models/balance.py +150 -115
- django_cfg/apps/payments/models/base.py +68 -15
- django_cfg/apps/payments/models/currencies.py +207 -67
- django_cfg/apps/payments/models/managers/__init__.py +44 -0
- django_cfg/apps/payments/models/managers/api_key_managers.py +329 -0
- django_cfg/apps/payments/models/managers/balance_managers.py +599 -0
- django_cfg/apps/payments/models/managers/currency_managers.py +385 -0
- django_cfg/apps/payments/models/managers/payment_managers.py +511 -0
- django_cfg/apps/payments/models/managers/subscription_managers.py +641 -0
- django_cfg/apps/payments/models/payments.py +235 -284
- django_cfg/apps/payments/models/subscriptions.py +257 -177
- django_cfg/apps/payments/models/tariffs.py +147 -40
- django_cfg/apps/payments/services/__init__.py +209 -56
- django_cfg/apps/payments/services/cache/__init__.py +6 -6
- django_cfg/apps/payments/services/cache/{simple_cache.py → cache_service.py} +112 -12
- django_cfg/apps/payments/services/core/__init__.py +10 -6
- django_cfg/apps/payments/services/core/balance_service.py +435 -360
- django_cfg/apps/payments/services/core/base.py +166 -0
- django_cfg/apps/payments/services/core/currency_service.py +478 -0
- django_cfg/apps/payments/services/core/payment_service.py +344 -468
- django_cfg/apps/payments/services/core/subscription_service.py +425 -484
- django_cfg/apps/payments/services/core/webhook_service.py +410 -0
- django_cfg/apps/payments/services/integrations/__init__.py +29 -0
- django_cfg/apps/payments/services/integrations/ngrok_service.py +47 -0
- django_cfg/apps/payments/services/integrations/providers_config.py +107 -0
- django_cfg/apps/payments/services/providers/__init__.py +9 -14
- django_cfg/apps/payments/services/providers/base.py +232 -71
- django_cfg/apps/payments/services/providers/nowpayments.py +404 -219
- django_cfg/apps/payments/services/providers/registry.py +429 -80
- django_cfg/apps/payments/services/types/__init__.py +78 -0
- django_cfg/apps/payments/services/types/data.py +177 -0
- django_cfg/apps/payments/services/types/requests.py +150 -0
- django_cfg/apps/payments/services/types/responses.py +156 -0
- django_cfg/apps/payments/services/types/webhooks.py +232 -0
- django_cfg/apps/payments/signals/__init__.py +33 -8
- django_cfg/apps/payments/signals/api_key_signals.py +211 -130
- django_cfg/apps/payments/signals/balance_signals.py +174 -0
- django_cfg/apps/payments/signals/payment_signals.py +129 -98
- django_cfg/apps/payments/signals/subscription_signals.py +195 -143
- django_cfg/apps/payments/static/payments/css/components.css +380 -0
- django_cfg/apps/payments/static/payments/css/dashboard.css +188 -0
- django_cfg/apps/payments/static/payments/js/components.js +545 -0
- django_cfg/apps/payments/static/payments/js/utils.js +412 -0
- django_cfg/apps/payments/templatetags/__init__.py +1 -1
- django_cfg/apps/payments/templatetags/payment_tags.py +466 -0
- django_cfg/apps/payments/urls.py +46 -47
- django_cfg/apps/payments/urls_admin.py +49 -0
- django_cfg/apps/payments/views/api/__init__.py +101 -0
- django_cfg/apps/payments/views/api/api_keys.py +387 -0
- django_cfg/apps/payments/views/api/balances.py +381 -0
- django_cfg/apps/payments/views/api/base.py +298 -0
- django_cfg/apps/payments/views/api/currencies.py +402 -0
- django_cfg/apps/payments/views/api/payments.py +415 -0
- django_cfg/apps/payments/views/api/subscriptions.py +475 -0
- django_cfg/apps/payments/views/api/webhooks.py +476 -0
- django_cfg/apps/payments/views/serializers/__init__.py +99 -0
- django_cfg/apps/payments/views/serializers/api_keys.py +424 -0
- django_cfg/apps/payments/views/serializers/balances.py +300 -0
- django_cfg/apps/payments/views/serializers/currencies.py +335 -0
- django_cfg/apps/payments/views/serializers/payments.py +387 -0
- django_cfg/apps/payments/views/serializers/subscriptions.py +429 -0
- django_cfg/apps/payments/views/serializers/webhooks.py +137 -0
- django_cfg/apps/tasks/urls.py +0 -2
- django_cfg/apps/tasks/urls_admin.py +14 -0
- django_cfg/apps/urls.py +4 -4
- django_cfg/config.py +1 -1
- django_cfg/core/config.py +75 -4
- django_cfg/core/generation.py +25 -4
- django_cfg/core/integration/README.md +363 -0
- django_cfg/core/integration/__init__.py +47 -0
- django_cfg/core/integration/commands_collector.py +239 -0
- django_cfg/core/integration/display/__init__.py +15 -0
- django_cfg/core/integration/display/base.py +157 -0
- django_cfg/core/integration/display/ngrok.py +164 -0
- django_cfg/core/integration/display/startup.py +815 -0
- django_cfg/core/integration/url_integration.py +123 -0
- django_cfg/core/integration/version_checker.py +160 -0
- django_cfg/management/commands/auto_generate.py +4 -0
- django_cfg/management/commands/check_settings.py +6 -0
- django_cfg/management/commands/clear_constance.py +5 -2
- django_cfg/management/commands/create_token.py +6 -0
- django_cfg/management/commands/list_urls.py +6 -0
- django_cfg/management/commands/migrate_all.py +6 -0
- django_cfg/management/commands/migrator.py +3 -0
- django_cfg/management/commands/rundramatiq.py +6 -0
- django_cfg/management/commands/runserver_ngrok.py +51 -29
- django_cfg/management/commands/script.py +6 -0
- django_cfg/management/commands/show_config.py +12 -2
- django_cfg/management/commands/show_urls.py +4 -0
- django_cfg/management/commands/superuser.py +6 -0
- django_cfg/management/commands/task_clear.py +4 -1
- django_cfg/management/commands/task_status.py +3 -1
- django_cfg/management/commands/test_email.py +3 -0
- django_cfg/management/commands/test_telegram.py +6 -0
- django_cfg/management/commands/test_twilio.py +6 -0
- django_cfg/management/commands/tree.py +6 -0
- django_cfg/management/commands/validate_config.py +155 -149
- django_cfg/models/constance.py +31 -11
- django_cfg/models/payments.py +175 -498
- django_cfg/modules/django_currency/__init__.py +16 -11
- django_cfg/modules/django_currency/clients/__init__.py +4 -4
- django_cfg/modules/django_currency/clients/coinpaprika_client.py +289 -0
- django_cfg/modules/django_currency/clients/yahoo_client.py +157 -0
- django_cfg/modules/django_currency/core/__init__.py +1 -7
- django_cfg/modules/django_currency/core/converter.py +18 -23
- django_cfg/modules/django_currency/core/models.py +122 -11
- django_cfg/modules/django_currency/database/__init__.py +4 -4
- django_cfg/modules/django_currency/database/database_loader.py +190 -309
- django_cfg/modules/django_logger.py +160 -146
- django_cfg/modules/django_unfold/dashboard.py +65 -12
- django_cfg/registry/core.py +1 -0
- django_cfg/template_archive/django_sample.zip +0 -0
- django_cfg/templates/admin/components/action_grid.html +9 -9
- django_cfg/templates/admin/components/metric_card.html +5 -5
- django_cfg/templates/admin/components/status_badge.html +2 -2
- django_cfg/templates/admin/layouts/dashboard_with_tabs.html +152 -24
- django_cfg/templates/admin/snippets/components/quick_actions.html +3 -3
- django_cfg/templates/admin/snippets/components/system_health.html +1 -1
- django_cfg/templates/admin/snippets/tabs/overview_tab.html +49 -52
- django_cfg/utils/smart_defaults.py +222 -571
- django_cfg/utils/toolkit.py +51 -11
- {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/METADATA +5 -4
- {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/RECORD +172 -182
- django_cfg/apps/payments/__init__.py +0 -8
- django_cfg/apps/payments/admin/tariffs_admin.py +0 -199
- django_cfg/apps/payments/config/module.py +0 -70
- django_cfg/apps/payments/config/providers.py +0 -105
- django_cfg/apps/payments/config/settings.py +0 -96
- django_cfg/apps/payments/config/utils.py +0 -52
- django_cfg/apps/payments/decorators.py +0 -291
- django_cfg/apps/payments/management/commands/README.md +0 -178
- django_cfg/apps/payments/management/commands/currency_stats.py +0 -323
- django_cfg/apps/payments/management/commands/populate_currencies.py +0 -246
- django_cfg/apps/payments/management/commands/update_currencies.py +0 -336
- django_cfg/apps/payments/managers/__init__.py +0 -22
- django_cfg/apps/payments/managers/api_key_manager.py +0 -35
- django_cfg/apps/payments/managers/balance_manager.py +0 -361
- django_cfg/apps/payments/managers/currency_manager.py +0 -83
- django_cfg/apps/payments/managers/payment_manager.py +0 -44
- django_cfg/apps/payments/managers/subscription_manager.py +0 -37
- django_cfg/apps/payments/managers/tariff_manager.py +0 -29
- django_cfg/apps/payments/models/events.py +0 -73
- django_cfg/apps/payments/serializers/__init__.py +0 -56
- django_cfg/apps/payments/serializers/api_keys.py +0 -51
- django_cfg/apps/payments/serializers/balance.py +0 -59
- django_cfg/apps/payments/serializers/currencies.py +0 -55
- django_cfg/apps/payments/serializers/payments.py +0 -62
- django_cfg/apps/payments/serializers/subscriptions.py +0 -71
- django_cfg/apps/payments/serializers/tariffs.py +0 -56
- django_cfg/apps/payments/services/billing/__init__.py +0 -8
- django_cfg/apps/payments/services/cache/base.py +0 -30
- django_cfg/apps/payments/services/core/fallback_service.py +0 -432
- django_cfg/apps/payments/services/internal_types.py +0 -297
- django_cfg/apps/payments/services/middleware/__init__.py +0 -8
- django_cfg/apps/payments/services/monitoring/__init__.py +0 -22
- django_cfg/apps/payments/services/monitoring/api_schemas.py +0 -222
- django_cfg/apps/payments/services/monitoring/provider_health.py +0 -372
- django_cfg/apps/payments/services/providers/cryptapi.py +0 -273
- django_cfg/apps/payments/services/providers/cryptomus.py +0 -311
- django_cfg/apps/payments/services/security/__init__.py +0 -34
- django_cfg/apps/payments/services/security/error_handler.py +0 -637
- django_cfg/apps/payments/services/security/payment_notifications.py +0 -342
- django_cfg/apps/payments/services/security/webhook_validator.py +0 -475
- django_cfg/apps/payments/services/validators/__init__.py +0 -8
- django_cfg/apps/payments/static/payments/css/payments.css +0 -340
- django_cfg/apps/payments/static/payments/js/notifications.js +0 -202
- django_cfg/apps/payments/static/payments/js/payment-utils.js +0 -318
- django_cfg/apps/payments/static/payments/js/theme.js +0 -86
- django_cfg/apps/payments/tasks/__init__.py +0 -12
- django_cfg/apps/payments/tasks/webhook_processing.py +0 -177
- django_cfg/apps/payments/templates/payments/base.html +0 -182
- django_cfg/apps/payments/templates/payments/components/payment_card.html +0 -201
- django_cfg/apps/payments/templates/payments/components/payment_qr_code.html +0 -109
- django_cfg/apps/payments/templates/payments/components/progress_bar.html +0 -36
- django_cfg/apps/payments/templates/payments/components/provider_stats.html +0 -40
- django_cfg/apps/payments/templates/payments/components/status_badge.html +0 -27
- django_cfg/apps/payments/templates/payments/components/status_overview.html +0 -144
- django_cfg/apps/payments/templates/payments/dashboard.html +0 -346
- django_cfg/apps/payments/templatetags/payments_tags.py +0 -315
- django_cfg/apps/payments/urls_templates.py +0 -52
- django_cfg/apps/payments/utils/__init__.py +0 -45
- django_cfg/apps/payments/utils/billing_utils.py +0 -342
- django_cfg/apps/payments/utils/config_utils.py +0 -245
- django_cfg/apps/payments/utils/middleware_utils.py +0 -228
- django_cfg/apps/payments/utils/validation_utils.py +0 -94
- django_cfg/apps/payments/views/__init__.py +0 -62
- django_cfg/apps/payments/views/api_key_views.py +0 -164
- django_cfg/apps/payments/views/balance_views.py +0 -75
- django_cfg/apps/payments/views/currency_views.py +0 -111
- django_cfg/apps/payments/views/payment_views.py +0 -149
- django_cfg/apps/payments/views/subscription_views.py +0 -135
- django_cfg/apps/payments/views/tariff_views.py +0 -131
- django_cfg/apps/payments/views/templates/__init__.py +0 -25
- django_cfg/apps/payments/views/templates/ajax.py +0 -312
- django_cfg/apps/payments/views/templates/base.py +0 -204
- django_cfg/apps/payments/views/templates/dashboard.py +0 -60
- django_cfg/apps/payments/views/templates/payment_detail.py +0 -102
- django_cfg/apps/payments/views/templates/payment_management.py +0 -164
- django_cfg/apps/payments/views/templates/qr_code.py +0 -174
- django_cfg/apps/payments/views/templates/stats.py +0 -240
- django_cfg/apps/payments/views/templates/utils.py +0 -181
- django_cfg/apps/payments/views/webhook_views.py +0 -266
- django_cfg/apps/payments/viewsets.py +0 -65
- django_cfg/core/integration.py +0 -160
- django_cfg/modules/django_currency/clients/coingecko_client.py +0 -257
- django_cfg/modules/django_currency/clients/yfinance_client.py +0 -246
- django_cfg/template_archive/.gitignore +0 -1
- django_cfg/template_archive/__init__.py +0 -0
- django_cfg/urls.py +0 -33
- {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/WHEEL +0 -0
- {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.dist-info}/entry_points.txt +0 -0
- {django_cfg-1.2.29.dist-info → django_cfg-1.3.1.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
|
+
}
|