django-cfg 1.3.3__py3-none-any.whl → 1.3.5__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/payments/admin_interface/old/payments/base.html +175 -0
- django_cfg/apps/payments/admin_interface/old/payments/components/dev_tool_card.html +125 -0
- django_cfg/apps/payments/admin_interface/old/payments/components/ngrok_status_card.html +113 -0
- django_cfg/apps/payments/admin_interface/old/payments/components/status_card.html +35 -0
- django_cfg/apps/payments/admin_interface/old/payments/payment_dashboard.html +309 -0
- django_cfg/apps/payments/admin_interface/old/payments/payment_form.html +303 -0
- django_cfg/apps/payments/admin_interface/old/payments/payment_list.html +382 -0
- django_cfg/apps/payments/admin_interface/old/payments/webhook_dashboard.html +518 -0
- django_cfg/apps/payments/{static → admin_interface/old/static}/payments/css/components.css +248 -9
- django_cfg/apps/payments/admin_interface/old/static/payments/js/ngrok-status.js +163 -0
- django_cfg/apps/payments/admin_interface/serializers/__init__.py +39 -0
- django_cfg/apps/payments/admin_interface/serializers/payment_serializers.py +149 -0
- django_cfg/apps/payments/admin_interface/serializers/webhook_serializers.py +114 -0
- django_cfg/apps/payments/admin_interface/templates/payments/base.html +55 -90
- django_cfg/apps/payments/admin_interface/templates/payments/components/dialog.html +81 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/ngrok_help_dialog.html +112 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/ngrok_status.html +175 -0
- django_cfg/apps/payments/admin_interface/templates/payments/components/status_card.html +21 -17
- django_cfg/apps/payments/admin_interface/templates/payments/payment_dashboard.html +123 -250
- django_cfg/apps/payments/admin_interface/templates/payments/payment_form.html +170 -269
- django_cfg/apps/payments/admin_interface/templates/payments/payment_list.html +152 -355
- django_cfg/apps/payments/admin_interface/templates/payments/webhook_dashboard.html +202 -551
- django_cfg/apps/payments/admin_interface/views/__init__.py +25 -14
- django_cfg/apps/payments/admin_interface/views/api/__init__.py +20 -0
- django_cfg/apps/payments/admin_interface/views/api/payments.py +191 -0
- django_cfg/apps/payments/admin_interface/views/api/stats.py +206 -0
- django_cfg/apps/payments/admin_interface/views/api/users.py +60 -0
- django_cfg/apps/payments/admin_interface/views/api/webhook_admin.py +257 -0
- django_cfg/apps/payments/admin_interface/views/api/webhook_public.py +70 -0
- django_cfg/apps/payments/admin_interface/views/base.py +114 -0
- django_cfg/apps/payments/admin_interface/views/dashboard.py +60 -0
- django_cfg/apps/payments/admin_interface/views/forms.py +94 -0
- django_cfg/apps/payments/config/helpers.py +2 -2
- django_cfg/apps/payments/management/commands/cleanup_expired_data.py +16 -6
- django_cfg/apps/payments/management/commands/currency_stats.py +72 -5
- django_cfg/apps/payments/management/commands/manage_currencies.py +9 -20
- django_cfg/apps/payments/management/commands/manage_providers.py +5 -5
- django_cfg/apps/payments/middleware/api_access.py +35 -34
- django_cfg/apps/payments/migrations/0001_initial.py +1 -1
- django_cfg/apps/payments/models/managers/api_key_managers.py +4 -0
- django_cfg/apps/payments/models/managers/payment_managers.py +5 -0
- django_cfg/apps/payments/models/subscriptions.py +0 -24
- django_cfg/apps/payments/services/cache/__init__.py +1 -1
- django_cfg/apps/payments/services/core/balance_service.py +13 -2
- django_cfg/apps/payments/services/integrations/ngrok_service.py +3 -3
- django_cfg/apps/payments/services/providers/registry.py +20 -0
- django_cfg/apps/payments/signals/balance_signals.py +7 -4
- django_cfg/apps/payments/static/payments/js/api-client.js +385 -0
- django_cfg/apps/payments/static/payments/js/ngrok-status.js +58 -0
- django_cfg/apps/payments/static/payments/js/payment-dashboard.js +50 -0
- django_cfg/apps/payments/static/payments/js/payment-form.js +175 -0
- django_cfg/apps/payments/static/payments/js/payment-list.js +95 -0
- django_cfg/apps/payments/static/payments/js/webhook-dashboard.js +154 -0
- django_cfg/apps/payments/urls.py +4 -0
- django_cfg/apps/payments/urls_admin.py +37 -18
- django_cfg/apps/payments/views/api/api_keys.py +14 -0
- django_cfg/apps/payments/views/api/base.py +1 -0
- django_cfg/apps/payments/views/api/currencies.py +2 -2
- django_cfg/apps/payments/views/api/payments.py +11 -5
- django_cfg/apps/payments/views/api/subscriptions.py +36 -31
- django_cfg/apps/payments/views/overview/__init__.py +40 -0
- django_cfg/apps/payments/views/overview/serializers.py +205 -0
- django_cfg/apps/payments/views/overview/services.py +439 -0
- django_cfg/apps/payments/views/overview/urls.py +27 -0
- django_cfg/apps/payments/views/overview/views.py +231 -0
- django_cfg/apps/payments/views/serializers/api_keys.py +20 -6
- django_cfg/apps/payments/views/serializers/balances.py +5 -8
- django_cfg/apps/payments/views/serializers/currencies.py +2 -6
- django_cfg/apps/payments/views/serializers/payments.py +37 -32
- django_cfg/apps/payments/views/serializers/subscriptions.py +4 -26
- django_cfg/apps/urls.py +2 -1
- django_cfg/core/config.py +25 -15
- django_cfg/core/generation.py +12 -12
- django_cfg/core/integration/display/startup.py +1 -1
- django_cfg/core/validation.py +4 -4
- django_cfg/management/commands/show_config.py +2 -2
- django_cfg/management/commands/tree.py +1 -3
- django_cfg/middleware/__init__.py +2 -0
- django_cfg/middleware/static_nocache.py +55 -0
- django_cfg/models/payments.py +13 -15
- django_cfg/models/security.py +15 -0
- django_cfg/modules/django_ngrok.py +6 -0
- django_cfg/modules/django_unfold/dashboard.py +1 -3
- django_cfg/utils/smart_defaults.py +41 -1
- {django_cfg-1.3.3.dist-info → django_cfg-1.3.5.dist-info}/METADATA +1 -1
- {django_cfg-1.3.3.dist-info → django_cfg-1.3.5.dist-info}/RECORD +98 -65
- django_cfg/apps/payments/admin_interface/templates/payments/components/dev_tool_card.html +0 -38
- django_cfg/apps/payments/admin_interface/views/payment_views.py +0 -259
- django_cfg/apps/payments/admin_interface/views/webhook_dashboard.py +0 -37
- /django_cfg/apps/payments/admin_interface/{templates → old}/payments/components/loading_spinner.html +0 -0
- /django_cfg/apps/payments/admin_interface/{templates → old}/payments/components/notification.html +0 -0
- /django_cfg/apps/payments/admin_interface/{templates → old}/payments/components/provider_card.html +0 -0
- /django_cfg/apps/payments/admin_interface/{templates → old}/payments/currency_converter.html +0 -0
- /django_cfg/apps/payments/admin_interface/{templates → old}/payments/payment_status.html +0 -0
- /django_cfg/apps/payments/{static → admin_interface/old/static}/payments/css/dashboard.css +0 -0
- /django_cfg/apps/payments/{static → admin_interface/old/static}/payments/js/components.js +0 -0
- /django_cfg/apps/payments/{static → admin_interface/old/static}/payments/js/utils.js +0 -0
- {django_cfg-1.3.3.dist-info → django_cfg-1.3.5.dist-info}/WHEEL +0 -0
- {django_cfg-1.3.3.dist-info → django_cfg-1.3.5.dist-info}/entry_points.txt +0 -0
- {django_cfg-1.3.3.dist-info → django_cfg-1.3.5.dist-info}/licenses/LICENSE +0 -0
@@ -1,594 +1,245 @@
|
|
1
1
|
{% extends 'payments/base.html' %}
|
2
2
|
{% load static %}
|
3
3
|
|
4
|
-
{% block title %}Webhook Dashboard -
|
4
|
+
{% block title %}Webhook Dashboard - Payment Admin{% endblock %}
|
5
5
|
|
6
|
-
{% block
|
7
|
-
|
8
|
-
.notification {
|
9
|
-
position: fixed;
|
10
|
-
top: 20px;
|
11
|
-
right: 20px;
|
12
|
-
z-index: 1000;
|
13
|
-
padding: 12px 16px;
|
14
|
-
border-radius: 8px;
|
15
|
-
color: white;
|
16
|
-
font-weight: 500;
|
17
|
-
min-width: 300px;
|
18
|
-
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
|
19
|
-
}
|
20
|
-
|
21
|
-
.notification.success { background-color: #10b981; }
|
22
|
-
.notification.error { background-color: #ef4444; }
|
23
|
-
.notification.warning { background-color: #f59e0b; }
|
24
|
-
.notification.info { background-color: #3b82f6; }
|
25
|
-
|
26
|
-
.slide-in {
|
27
|
-
animation: slideIn 0.3s ease-out;
|
28
|
-
}
|
29
|
-
|
30
|
-
@keyframes slideIn {
|
31
|
-
from {
|
32
|
-
transform: translateX(100%);
|
33
|
-
opacity: 0;
|
34
|
-
}
|
35
|
-
to {
|
36
|
-
transform: translateX(0);
|
37
|
-
opacity: 1;
|
38
|
-
}
|
39
|
-
}
|
40
|
-
|
41
|
-
.status-indicator {
|
42
|
-
width: 8px;
|
43
|
-
height: 8px;
|
44
|
-
border-radius: 50%;
|
45
|
-
display: inline-block;
|
46
|
-
margin-right: 8px;
|
47
|
-
}
|
48
|
-
|
49
|
-
.status-active { background-color: #10b981; }
|
50
|
-
.status-inactive { background-color: #ef4444; }
|
51
|
-
.status-warning { background-color: #f59e0b; }
|
52
|
-
</style>
|
53
|
-
{% endblock %}
|
6
|
+
{% block page_title %}Webhook Dashboard{% endblock %}
|
7
|
+
{% block page_subtitle %}Monitor and test webhook endpoints{% endblock %}
|
54
8
|
|
55
9
|
{% block content %}
|
56
|
-
<div x-data="webhookDashboard()" x-init="init()"
|
57
|
-
|
58
|
-
|
59
|
-
<div>
|
60
|
-
|
61
|
-
<
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
-
|
75
|
-
🧪 Test Webhook
|
76
|
-
</button>
|
77
|
-
</div>
|
78
|
-
</div>
|
79
|
-
|
80
|
-
<!-- Status Overview -->
|
81
|
-
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-6">
|
82
|
-
<!-- Ngrok Status -->
|
83
|
-
<div class="bg-white dark:bg-gray-800 rounded-lg shadow p-6">
|
84
|
-
<div class="flex items-center">
|
85
|
-
<div class="flex-shrink-0">
|
86
|
-
<div class="w-8 h-8 bg-blue-100 dark:bg-blue-900 rounded-full flex items-center justify-center">
|
87
|
-
<span class="text-blue-600 dark:text-blue-400">🌐</span>
|
10
|
+
<div x-data="webhookDashboard()" x-init="init()">
|
11
|
+
<div class="grid grid-cols-1 lg:grid-cols-3 gap-8">
|
12
|
+
<!-- Main Content -->
|
13
|
+
<div class="lg:col-span-2 space-y-8">
|
14
|
+
<!-- Webhook Events -->
|
15
|
+
<div class="bg-white dark:bg-gray-800 shadow rounded-lg">
|
16
|
+
<div class="px-6 py-4 border-b border-gray-200 dark:border-gray-700">
|
17
|
+
<div class="flex items-center justify-between">
|
18
|
+
<h3 class="text-lg font-medium text-gray-900 dark:text-white">Recent Webhook Events</h3>
|
19
|
+
<div class="flex items-center space-x-2">
|
20
|
+
<button @click="refreshEvents()"
|
21
|
+
class="text-sm text-blue-600 hover:text-blue-800 dark:text-blue-400 dark:hover:text-blue-300">
|
22
|
+
<span class="material-icons-outlined" :class="{ 'animate-spin': loading }">refresh</span>
|
23
|
+
</button>
|
24
|
+
<button @click="clearEvents()"
|
25
|
+
class="text-sm text-red-600 hover:text-red-800 dark:text-red-400 dark:hover:text-red-300">
|
26
|
+
Clear All
|
27
|
+
</button>
|
28
|
+
</div>
|
88
29
|
</div>
|
89
30
|
</div>
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
31
|
+
|
32
|
+
<div class="p-6">
|
33
|
+
<!-- Filters -->
|
34
|
+
<div class="mb-6 flex flex-wrap gap-4">
|
35
|
+
<select x-model="filters.event_type"
|
36
|
+
@change="applyFilters()"
|
37
|
+
class="px-3 py-2 border border-gray-300 dark:border-gray-600 rounded-md text-sm bg-white dark:bg-gray-700 text-gray-900 dark:text-white">
|
38
|
+
<option value="">All Events</option>
|
39
|
+
<option value="payment.created">Payment Created</option>
|
40
|
+
<option value="payment.completed">Payment Completed</option>
|
41
|
+
<option value="payment.failed">Payment Failed</option>
|
42
|
+
</select>
|
43
|
+
|
44
|
+
<select x-model="filters.status"
|
45
|
+
@change="applyFilters()"
|
46
|
+
class="px-3 py-2 border border-gray-300 dark:border-gray-600 rounded-md text-sm bg-white dark:bg-gray-700 text-gray-900 dark:text-white">
|
47
|
+
<option value="">All Status</option>
|
48
|
+
<option value="success">Success</option>
|
49
|
+
<option value="failed">Failed</option>
|
50
|
+
<option value="pending">Pending</option>
|
51
|
+
</select>
|
106
52
|
</div>
|
107
|
-
</div>
|
108
|
-
<div class="ml-4">
|
109
|
-
<h3 class="text-sm font-medium text-gray-900 dark:text-white">Active Providers</h3>
|
110
|
-
<p class="text-2xl font-semibold text-gray-900 dark:text-white" x-text="providers.length"></p>
|
111
|
-
<p class="text-xs text-gray-500 dark:text-gray-400">Payment providers configured</p>
|
112
|
-
</div>
|
113
|
-
</div>
|
114
|
-
</div>
|
115
53
|
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
|
120
|
-
<div class="w-8 h-8 bg-green-100 dark:bg-green-900 rounded-full flex items-center justify-center">
|
121
|
-
<span class="text-green-600 dark:text-green-400">💚</span>
|
54
|
+
<!-- Events List -->
|
55
|
+
<div x-show="filteredEvents.length === 0 && !loading" class="text-center py-8">
|
56
|
+
<span class="material-icons-outlined text-gray-400 text-4xl mb-2">webhook</span>
|
57
|
+
<p class="text-gray-500 dark:text-gray-400">No webhook events yet</p>
|
122
58
|
</div>
|
123
|
-
|
124
|
-
|
125
|
-
|
126
|
-
|
127
|
-
<span x-text="healthStatus.healthy ? 'Healthy' : 'Issues'"></span>
|
128
|
-
</p>
|
129
|
-
<p class="text-xs text-gray-500 dark:text-gray-400" x-text="healthStatus.description"></p>
|
130
|
-
</div>
|
131
|
-
</div>
|
132
|
-
</div>
|
133
|
-
|
134
|
-
<!-- Recent Activity -->
|
135
|
-
<div class="bg-white dark:bg-gray-800 rounded-lg shadow p-6">
|
136
|
-
<div class="flex items-center">
|
137
|
-
<div class="flex-shrink-0">
|
138
|
-
<div class="w-8 h-8 bg-yellow-100 dark:bg-yellow-900 rounded-full flex items-center justify-center">
|
139
|
-
<span class="text-yellow-600 dark:text-yellow-400">📊</span>
|
59
|
+
|
60
|
+
<div x-show="loading" class="text-center py-8">
|
61
|
+
<div class="inline-block animate-spin rounded-full h-6 w-6 border-b-2 border-blue-600"></div>
|
62
|
+
<p class="mt-2 text-sm text-gray-500 dark:text-gray-400">Loading events...</p>
|
140
63
|
</div>
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
159
|
-
|
160
|
-
|
161
|
-
|
162
|
-
|
163
|
-
|
164
|
-
|
165
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
173
|
-
|
174
|
-
|
175
|
-
|
176
|
-
|
177
|
-
|
178
|
-
|
179
|
-
Active
|
180
|
-
</span>
|
181
|
-
</div>
|
182
|
-
|
183
|
-
<div class="space-y-2 text-sm">
|
184
|
-
<div>
|
185
|
-
<label class="text-gray-600 dark:text-gray-400">Webhook URL:</label>
|
186
|
-
<div class="flex items-center mt-1">
|
187
|
-
<code class="flex-1 px-2 py-1 bg-gray-100 dark:bg-gray-700 text-gray-800 dark:text-gray-200 rounded text-xs font-mono break-all" x-text="provider.webhook_url"></code>
|
188
|
-
<button @click="copyToClipboard(provider.webhook_url)"
|
189
|
-
class="ml-2 p-1 text-gray-400 hover:text-gray-600 dark:hover:text-gray-200 flex-shrink-0"
|
190
|
-
title="Copy to clipboard">
|
191
|
-
📋
|
192
|
-
</button>
|
193
|
-
</div>
|
194
|
-
</div>
|
195
|
-
|
196
|
-
<div class="grid grid-cols-2 gap-4">
|
197
|
-
<div>
|
198
|
-
<label class="text-gray-600 dark:text-gray-400">Signature Header:</label>
|
199
|
-
<code class="block mt-1 text-xs font-mono text-gray-800 dark:text-gray-200" x-text="provider.signature_header"></code>
|
64
|
+
|
65
|
+
<div class="space-y-4">
|
66
|
+
<template x-for="event in filteredEvents.slice(0, 10)" :key="event.id">
|
67
|
+
<div class="border border-gray-200 dark:border-gray-700 rounded-lg p-4 hover:bg-gray-50 dark:hover:bg-gray-700">
|
68
|
+
<div class="flex items-start justify-between">
|
69
|
+
<div class="flex items-start">
|
70
|
+
<div class="w-8 h-8 rounded-full flex items-center justify-center mr-3"
|
71
|
+
:class="{
|
72
|
+
'bg-green-100 dark:bg-green-900/20': event.status === 'success',
|
73
|
+
'bg-red-100 dark:bg-red-900/20': event.status === 'failed',
|
74
|
+
'bg-yellow-100 dark:bg-yellow-900/20': event.status === 'pending'
|
75
|
+
}">
|
76
|
+
<span class="material-icons-outlined text-sm"
|
77
|
+
:class="{
|
78
|
+
'text-green-600 dark:text-green-400': event.status === 'success',
|
79
|
+
'text-red-600 dark:text-red-400': event.status === 'failed',
|
80
|
+
'text-yellow-600 dark:text-yellow-400': event.status === 'pending'
|
81
|
+
}">
|
82
|
+
<span x-show="event.status === 'success'">check_circle</span>
|
83
|
+
<span x-show="event.status === 'failed'">error</span>
|
84
|
+
<span x-show="event.status === 'pending'">schedule</span>
|
85
|
+
</span>
|
86
|
+
</div>
|
87
|
+
<div class="flex-1">
|
88
|
+
<div class="flex items-center space-x-2">
|
89
|
+
<span class="text-sm font-medium text-gray-900 dark:text-white" x-text="event.event_type"></span>
|
90
|
+
<span class="inline-flex px-2 py-1 text-xs font-semibold rounded-full"
|
91
|
+
:class="{
|
92
|
+
'bg-green-100 text-green-800 dark:bg-green-900/20 dark:text-green-400': event.status === 'success',
|
93
|
+
'bg-red-100 text-red-800 dark:bg-red-900/20 dark:text-red-400': event.status === 'failed',
|
94
|
+
'bg-yellow-100 text-yellow-800 dark:bg-yellow-900/20 dark:text-yellow-400': event.status === 'pending'
|
95
|
+
}"
|
96
|
+
x-text="event.status">
|
97
|
+
</span>
|
98
|
+
</div>
|
99
|
+
<p class="text-sm text-gray-500 dark:text-gray-400 mt-1" x-text="event.url"></p>
|
100
|
+
<p class="text-xs text-gray-400 dark:text-gray-500 mt-1" x-text="formatDate(event.created_at)"></p>
|
101
|
+
</div>
|
200
102
|
</div>
|
201
|
-
<div>
|
202
|
-
<
|
203
|
-
|
103
|
+
<div class="flex items-center space-x-2">
|
104
|
+
<button @click="viewEvent(event)"
|
105
|
+
class="text-gray-400 hover:text-gray-600 dark:hover:text-gray-200">
|
106
|
+
<span class="material-icons-outlined text-sm">visibility</span>
|
107
|
+
</button>
|
108
|
+
<button @click="retryEvent(event.id)"
|
109
|
+
x-show="event.status === 'failed'"
|
110
|
+
class="text-blue-600 hover:text-blue-800 dark:text-blue-400 dark:hover:text-blue-300">
|
111
|
+
<span class="material-icons-outlined text-sm">refresh</span>
|
112
|
+
</button>
|
204
113
|
</div>
|
205
114
|
</div>
|
206
115
|
</div>
|
207
|
-
</
|
208
|
-
</
|
116
|
+
</template>
|
117
|
+
</div>
|
209
118
|
</div>
|
210
119
|
</div>
|
211
|
-
</div>
|
212
120
|
|
213
|
-
|
214
|
-
|
215
|
-
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
|
224
|
-
|
225
|
-
|
226
|
-
|
227
|
-
|
121
|
+
<!-- Webhook Testing -->
|
122
|
+
<div class="bg-white dark:bg-gray-800 shadow rounded-lg">
|
123
|
+
<div class="px-6 py-4 border-b border-gray-200 dark:border-gray-700">
|
124
|
+
<h3 class="text-lg font-medium text-gray-900 dark:text-white">Test Webhook</h3>
|
125
|
+
</div>
|
126
|
+
|
127
|
+
<div class="p-6">
|
128
|
+
<form @submit.prevent="testWebhook()" class="space-y-4">
|
129
|
+
<div>
|
130
|
+
<label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
|
131
|
+
Webhook URL
|
132
|
+
</label>
|
133
|
+
<input type="url"
|
134
|
+
x-model="testForm.url"
|
135
|
+
class="block w-full px-3 py-2 border border-gray-300 dark:border-gray-600 rounded-md shadow-sm focus:ring-blue-500 focus:border-blue-500 dark:bg-gray-700 dark:text-white"
|
136
|
+
placeholder="https://your-site.com/webhook"
|
137
|
+
required>
|
228
138
|
</div>
|
229
|
-
|
230
|
-
|
231
|
-
<
|
139
|
+
|
140
|
+
<div>
|
141
|
+
<label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">
|
142
|
+
Event Type
|
143
|
+
</label>
|
144
|
+
<select x-model="testForm.event_type"
|
145
|
+
class="block w-full px-3 py-2 border border-gray-300 dark:border-gray-600 rounded-md shadow-sm focus:ring-blue-500 focus:border-blue-500 dark:bg-gray-700 dark:text-white"
|
146
|
+
required>
|
147
|
+
<option value="">Select event type...</option>
|
148
|
+
<option value="payment.created">Payment Created</option>
|
149
|
+
<option value="payment.completed">Payment Completed</option>
|
150
|
+
<option value="payment.failed">Payment Failed</option>
|
151
|
+
</select>
|
232
152
|
</div>
|
233
|
-
|
234
|
-
|
235
|
-
|
236
|
-
|
237
|
-
|
238
|
-
|
153
|
+
|
154
|
+
<button type="submit"
|
155
|
+
:disabled="testLoading"
|
156
|
+
class="w-full px-4 py-2 bg-blue-600 text-white text-sm font-medium rounded-md hover:bg-blue-700 disabled:opacity-50">
|
157
|
+
<span x-show="!testLoading">Send Test Webhook</span>
|
158
|
+
<span x-show="testLoading">Sending...</span>
|
159
|
+
</button>
|
160
|
+
</form>
|
239
161
|
</div>
|
162
|
+
</div>
|
163
|
+
</div>
|
240
164
|
|
241
|
-
|
242
|
-
|
243
|
-
|
244
|
-
|
245
|
-
|
246
|
-
|
247
|
-
|
248
|
-
|
249
|
-
|
250
|
-
|
251
|
-
|
252
|
-
|
253
|
-
|
165
|
+
<!-- Sidebar -->
|
166
|
+
<div class="space-y-8">
|
167
|
+
<!-- Ngrok Status -->
|
168
|
+
{% include 'payments/components/ngrok_status.html' %}
|
169
|
+
|
170
|
+
<!-- Webhook Stats -->
|
171
|
+
<div class="bg-white dark:bg-gray-800 shadow rounded-lg">
|
172
|
+
<div class="px-6 py-4 border-b border-gray-200 dark:border-gray-700">
|
173
|
+
<h3 class="text-lg font-medium text-gray-900 dark:text-white">Webhook Stats</h3>
|
174
|
+
</div>
|
175
|
+
|
176
|
+
<div class="p-6 space-y-4">
|
177
|
+
<div class="flex items-center justify-between">
|
178
|
+
<span class="text-sm text-gray-600 dark:text-gray-400">Total Events</span>
|
179
|
+
<span class="text-sm font-medium text-gray-900 dark:text-white" x-text="stats.total"></span>
|
254
180
|
</div>
|
255
181
|
|
256
|
-
<
|
257
|
-
|
258
|
-
|
259
|
-
</
|
260
|
-
</div>
|
261
|
-
|
262
|
-
<!-- Health Monitor -->
|
263
|
-
<div class="p-4 border border-gray-200 dark:border-gray-700 rounded-lg">
|
264
|
-
<h4 class="font-medium text-gray-900 dark:text-white mb-2">Health Monitor</h4>
|
182
|
+
<div class="flex items-center justify-between">
|
183
|
+
<span class="text-sm text-gray-600 dark:text-gray-400">Successful</span>
|
184
|
+
<span class="text-sm font-medium text-green-600 dark:text-green-400" x-text="stats.successful"></span>
|
185
|
+
</div>
|
265
186
|
|
266
|
-
<div class="
|
267
|
-
<
|
268
|
-
|
269
|
-
<span :class="healthStatus.healthy ? 'text-green-600' : 'text-red-600'" x-text="healthStatus.healthy ? 'Healthy' : 'Issues'"></span>
|
270
|
-
</div>
|
271
|
-
<div class="flex justify-between">
|
272
|
-
<span class="text-gray-600 dark:text-gray-400">Last Check:</span>
|
273
|
-
<span class="text-xs" x-text="healthStatus.last_check || 'Never'"></span>
|
274
|
-
</div>
|
187
|
+
<div class="flex items-center justify-between">
|
188
|
+
<span class="text-sm text-gray-600 dark:text-gray-400">Failed</span>
|
189
|
+
<span class="text-sm font-medium text-red-600 dark:text-red-400" x-text="stats.failed"></span>
|
275
190
|
</div>
|
276
191
|
|
277
|
-
<
|
278
|
-
|
279
|
-
|
280
|
-
</
|
192
|
+
<div class="flex items-center justify-between">
|
193
|
+
<span class="text-sm text-gray-600 dark:text-gray-400">Success Rate</span>
|
194
|
+
<span class="text-sm font-medium text-gray-900 dark:text-white" x-text="stats.successRate + '%'"></span>
|
195
|
+
</div>
|
281
196
|
</div>
|
282
197
|
</div>
|
283
|
-
</div>
|
284
|
-
</div>
|
285
198
|
|
286
|
-
|
287
|
-
|
288
|
-
|
289
|
-
|
290
|
-
<p class="text-sm text-gray-600 dark:text-gray-400">Recent webhook activity and performance metrics</p>
|
291
|
-
</div>
|
292
|
-
<div class="p-6">
|
293
|
-
<div class="grid grid-cols-1 md:grid-cols-3 gap-6">
|
294
|
-
<div class="text-center">
|
295
|
-
<div class="text-2xl font-bold text-blue-600" x-text="stats.total_webhooks || 0"></div>
|
296
|
-
<div class="text-sm text-gray-600 dark:text-gray-400">Total Webhooks</div>
|
297
|
-
</div>
|
298
|
-
<div class="text-center">
|
299
|
-
<div class="text-2xl font-bold text-green-600" x-text="stats.successful_webhooks || 0"></div>
|
300
|
-
<div class="text-sm text-gray-600 dark:text-gray-400">Successful</div>
|
301
|
-
</div>
|
302
|
-
<div class="text-center">
|
303
|
-
<div class="text-2xl font-bold text-red-600" x-text="stats.failed_webhooks || 0"></div>
|
304
|
-
<div class="text-sm text-gray-600 dark:text-gray-400">Failed</div>
|
199
|
+
<!-- Quick Actions -->
|
200
|
+
<div class="bg-white dark:bg-gray-800 shadow rounded-lg">
|
201
|
+
<div class="px-6 py-4 border-b border-gray-200 dark:border-gray-700">
|
202
|
+
<h3 class="text-lg font-medium text-gray-900 dark:text-white">Quick Actions</h3>
|
305
203
|
</div>
|
306
|
-
|
307
|
-
|
308
|
-
|
309
|
-
|
310
|
-
|
311
|
-
|
312
|
-
|
313
|
-
|
314
|
-
|
315
|
-
|
316
|
-
|
317
|
-
<div class="text-xs text-gray-500 dark:text-gray-400" x-text="activity.timestamp"></div>
|
318
|
-
</div>
|
319
|
-
</template>
|
320
|
-
</div>
|
321
|
-
<div x-show="!stats.recent_activity || stats.recent_activity.length === 0" class="text-center py-4">
|
322
|
-
<p class="text-gray-500 dark:text-gray-400">No recent webhook activity</p>
|
204
|
+
|
205
|
+
<div class="p-6 space-y-3">
|
206
|
+
<button @click="retryFailedEvents()"
|
207
|
+
class="w-full px-4 py-2 bg-yellow-600 text-white text-sm font-medium rounded-md hover:bg-yellow-700">
|
208
|
+
Retry Failed Events
|
209
|
+
</button>
|
210
|
+
|
211
|
+
<a href="{% url 'cfg_payments_admin:dashboard' %}"
|
212
|
+
class="block w-full px-4 py-2 bg-blue-600 text-white text-sm font-medium rounded-md hover:bg-blue-700 text-center">
|
213
|
+
Back to Dashboard
|
214
|
+
</a>
|
323
215
|
</div>
|
324
216
|
</div>
|
325
217
|
</div>
|
326
218
|
</div>
|
327
219
|
|
328
|
-
<!--
|
329
|
-
<div x-show="
|
330
|
-
x-
|
331
|
-
|
332
|
-
|
333
|
-
|
334
|
-
|
335
|
-
|
336
|
-
|
337
|
-
|
338
|
-
|
339
|
-
|
340
|
-
<div class="
|
341
|
-
<
|
342
|
-
|
343
|
-
|
344
|
-
|
345
|
-
<select x-model="testProvider" class="w-full px-3 py-2 border border-gray-300 dark:border-gray-600 rounded-md bg-white dark:bg-gray-700 text-gray-900 dark:text-white">
|
346
|
-
<option value="">Select provider...</option>
|
347
|
-
<template x-for="provider in providers" :key="provider.name">
|
348
|
-
<option :value="provider.name" x-text="provider.display_name"></option>
|
349
|
-
</template>
|
350
|
-
</select>
|
351
|
-
</div>
|
352
|
-
|
353
|
-
<div class="mb-4">
|
354
|
-
<label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">Test Data:</label>
|
355
|
-
<textarea x-model="testData"
|
356
|
-
class="w-full px-3 py-2 border border-gray-300 dark:border-gray-600 rounded-md bg-white dark:bg-gray-700 text-gray-900 dark:text-white"
|
357
|
-
rows="4"
|
358
|
-
placeholder='{"payment_id": "test_123", "status": "completed"}'></textarea>
|
359
|
-
</div>
|
360
|
-
|
361
|
-
<div class="flex justify-end space-x-3">
|
362
|
-
<button @click="showTestModal = false"
|
363
|
-
class="px-4 py-2 bg-gray-300 dark:bg-gray-600 text-gray-700 dark:text-gray-300 rounded-md hover:bg-gray-400 dark:hover:bg-gray-500">
|
364
|
-
Cancel
|
365
|
-
</button>
|
366
|
-
<button @click="sendTestWebhook()"
|
367
|
-
:disabled="!testProvider || testLoading"
|
368
|
-
class="px-4 py-2 bg-blue-600 text-white rounded-md hover:bg-blue-700 disabled:opacity-50 disabled:cursor-not-allowed">
|
369
|
-
<span x-show="!testLoading">Send Test</span>
|
370
|
-
<span x-show="testLoading">Testing...</span>
|
371
|
-
</button>
|
372
|
-
</div>
|
220
|
+
<!-- Event Detail Modal -->
|
221
|
+
<div x-show="showEventModal"
|
222
|
+
x-cloak
|
223
|
+
class="fixed inset-0 bg-gray-600 bg-opacity-50 flex items-center justify-center z-50"
|
224
|
+
@click.away="showEventModal = false">
|
225
|
+
<div class="bg-white dark:bg-gray-800 rounded-lg shadow-xl max-w-2xl w-full mx-4 max-h-96 overflow-y-auto">
|
226
|
+
<div class="px-6 py-4 border-b border-gray-200 dark:border-gray-700">
|
227
|
+
<h3 class="text-lg font-medium text-gray-900 dark:text-white">Event Details</h3>
|
228
|
+
</div>
|
229
|
+
<div class="p-6">
|
230
|
+
<pre class="text-sm bg-gray-100 dark:bg-gray-700 p-4 rounded-md overflow-x-auto" x-text="JSON.stringify(selectedEvent, null, 2)"></pre>
|
231
|
+
</div>
|
232
|
+
<div class="px-6 py-4 border-t border-gray-200 dark:border-gray-700 text-right">
|
233
|
+
<button @click="showEventModal = false"
|
234
|
+
class="px-4 py-2 bg-gray-600 text-white text-sm rounded-md hover:bg-gray-700">
|
235
|
+
Close
|
236
|
+
</button>
|
373
237
|
</div>
|
374
238
|
</div>
|
375
239
|
</div>
|
376
|
-
|
377
|
-
<!-- Notifications Container -->
|
378
|
-
<div id="notifications-container"></div>
|
379
240
|
</div>
|
241
|
+
{% endblock %}
|
380
242
|
|
381
|
-
|
382
|
-
|
383
|
-
return {
|
384
|
-
loading: true,
|
385
|
-
providers: [],
|
386
|
-
ngrokStatus: {
|
387
|
-
active: false,
|
388
|
-
url: null,
|
389
|
-
description: 'Checking...'
|
390
|
-
},
|
391
|
-
healthStatus: {
|
392
|
-
healthy: false,
|
393
|
-
description: 'Checking...',
|
394
|
-
last_check: null
|
395
|
-
},
|
396
|
-
stats: {
|
397
|
-
total_webhooks: 0,
|
398
|
-
successful_webhooks: 0,
|
399
|
-
failed_webhooks: 0,
|
400
|
-
recent_count: 0,
|
401
|
-
recent_activity: []
|
402
|
-
},
|
403
|
-
showTestModal: false,
|
404
|
-
testProvider: '',
|
405
|
-
testData: '{"payment_id": "test_123", "status": "completed", "amount": "10.00"}',
|
406
|
-
testLoading: false,
|
407
|
-
lastTest: null,
|
408
|
-
testStatus: null,
|
409
|
-
|
410
|
-
async init() {
|
411
|
-
await this.loadData();
|
412
|
-
},
|
413
|
-
|
414
|
-
async loadData() {
|
415
|
-
this.loading = true;
|
416
|
-
try {
|
417
|
-
await Promise.all([
|
418
|
-
this.loadProviders(),
|
419
|
-
this.checkHealth(),
|
420
|
-
this.loadStats()
|
421
|
-
]);
|
422
|
-
} catch (error) {
|
423
|
-
console.error('Failed to load dashboard data:', error);
|
424
|
-
this.showNotification('error', 'Failed to load dashboard data');
|
425
|
-
} finally {
|
426
|
-
this.loading = false;
|
427
|
-
}
|
428
|
-
},
|
429
|
-
|
430
|
-
async loadProviders() {
|
431
|
-
try {
|
432
|
-
const response = await fetch('/api/payments/webhooks/providers/');
|
433
|
-
const data = await response.json();
|
434
|
-
|
435
|
-
if (data.success) {
|
436
|
-
this.providers = data.providers || [];
|
437
|
-
} else {
|
438
|
-
throw new Error(data.error || 'Failed to load providers');
|
439
|
-
}
|
440
|
-
} catch (error) {
|
441
|
-
console.error('Failed to load providers:', error);
|
442
|
-
this.providers = [];
|
443
|
-
}
|
444
|
-
},
|
445
|
-
|
446
|
-
async checkHealth() {
|
447
|
-
try {
|
448
|
-
const response = await fetch('/api/payments/webhooks/health/');
|
449
|
-
const data = await response.json();
|
450
|
-
|
451
|
-
this.healthStatus = {
|
452
|
-
healthy: data.success && data.status === 'healthy',
|
453
|
-
description: data.message || 'Unknown status',
|
454
|
-
last_check: new Date().toLocaleString()
|
455
|
-
};
|
456
|
-
|
457
|
-
// Update ngrok status
|
458
|
-
this.ngrokStatus = {
|
459
|
-
active: data.ngrok_available || false,
|
460
|
-
url: data.ngrok_url || null,
|
461
|
-
description: data.ngrok_available ? 'Tunnel active' : 'Tunnel inactive'
|
462
|
-
};
|
463
|
-
} catch (error) {
|
464
|
-
console.error('Failed to check health:', error);
|
465
|
-
this.healthStatus = {
|
466
|
-
healthy: false,
|
467
|
-
description: 'Health check failed',
|
468
|
-
last_check: new Date().toLocaleString()
|
469
|
-
};
|
470
|
-
}
|
471
|
-
},
|
472
|
-
|
473
|
-
async loadStats() {
|
474
|
-
try {
|
475
|
-
const response = await fetch('/api/payments/webhooks/stats/');
|
476
|
-
const data = await response.json();
|
477
|
-
|
478
|
-
if (data.success) {
|
479
|
-
this.stats = {
|
480
|
-
total_webhooks: data.stats.total_webhooks || 0,
|
481
|
-
successful_webhooks: data.stats.successful_webhooks || 0,
|
482
|
-
failed_webhooks: data.stats.failed_webhooks || 0,
|
483
|
-
recent_count: data.stats.recent_count || 0,
|
484
|
-
recent_activity: data.stats.recent_activity || []
|
485
|
-
};
|
486
|
-
}
|
487
|
-
} catch (error) {
|
488
|
-
console.error('Failed to load stats:', error);
|
489
|
-
this.stats = {
|
490
|
-
total_webhooks: 0,
|
491
|
-
successful_webhooks: 0,
|
492
|
-
failed_webhooks: 0,
|
493
|
-
recent_count: 0,
|
494
|
-
recent_activity: []
|
495
|
-
};
|
496
|
-
}
|
497
|
-
},
|
498
|
-
|
499
|
-
async refreshData() {
|
500
|
-
await this.loadData();
|
501
|
-
this.showNotification('success', 'Dashboard data refreshed');
|
502
|
-
},
|
503
|
-
|
504
|
-
async checkNgrokStatus() {
|
505
|
-
await this.checkHealth();
|
506
|
-
this.showNotification('info', 'Ngrok status updated');
|
507
|
-
},
|
508
|
-
|
509
|
-
async sendTestWebhook() {
|
510
|
-
if (!this.testProvider) {
|
511
|
-
this.showNotification('warning', 'Please select a provider');
|
512
|
-
return;
|
513
|
-
}
|
514
|
-
|
515
|
-
this.testLoading = true;
|
516
|
-
try {
|
517
|
-
const provider = this.providers.find(p => p.name === this.testProvider);
|
518
|
-
if (!provider) {
|
519
|
-
throw new Error('Provider not found');
|
520
|
-
}
|
521
|
-
|
522
|
-
const response = await fetch(provider.webhook_url, {
|
523
|
-
method: 'POST',
|
524
|
-
headers: {
|
525
|
-
'Content-Type': 'application/json',
|
526
|
-
[provider.signature_header]: 'test-signature'
|
527
|
-
},
|
528
|
-
body: this.testData
|
529
|
-
});
|
530
|
-
|
531
|
-
this.lastTest = new Date().toLocaleString();
|
532
|
-
this.testStatus = response.ok ? 'Success' : 'Failed';
|
533
|
-
|
534
|
-
this.showNotification(
|
535
|
-
response.ok ? 'success' : 'error',
|
536
|
-
`Test webhook ${response.ok ? 'sent successfully' : 'failed'}`
|
537
|
-
);
|
538
|
-
|
539
|
-
this.showTestModal = false;
|
540
|
-
} catch (error) {
|
541
|
-
console.error('Test webhook failed:', error);
|
542
|
-
this.testStatus = 'Error';
|
543
|
-
this.showNotification('error', 'Failed to send test webhook');
|
544
|
-
} finally {
|
545
|
-
this.testLoading = false;
|
546
|
-
}
|
547
|
-
},
|
548
|
-
|
549
|
-
copyToClipboard(text) {
|
550
|
-
navigator.clipboard.writeText(text).then(() => {
|
551
|
-
this.showNotification('success', 'Copied to clipboard');
|
552
|
-
}).catch(() => {
|
553
|
-
this.showNotification('error', 'Failed to copy to clipboard');
|
554
|
-
});
|
555
|
-
},
|
556
|
-
|
557
|
-
showNotification(type, message) {
|
558
|
-
const container = document.getElementById('notifications-container');
|
559
|
-
const notification = document.createElement('div');
|
560
|
-
|
561
|
-
const icons = {
|
562
|
-
success: '✅',
|
563
|
-
error: '❌',
|
564
|
-
warning: '⚠️',
|
565
|
-
info: 'ℹ️'
|
566
|
-
};
|
567
|
-
|
568
|
-
notification.innerHTML = `
|
569
|
-
<div class="notification ${type} slide-in">
|
570
|
-
<div class="flex items-center">
|
571
|
-
<div class="flex-shrink-0">${icons[type] || icons.info}</div>
|
572
|
-
<div class="ml-2">
|
573
|
-
<p class="text-sm font-medium">${message}</p>
|
574
|
-
</div>
|
575
|
-
<div class="ml-auto">
|
576
|
-
<button onclick="this.parentElement.parentElement.parentElement.remove()" class="text-white hover:text-gray-200">✕</button>
|
577
|
-
</div>
|
578
|
-
</div>
|
579
|
-
</div>
|
580
|
-
`;
|
581
|
-
|
582
|
-
container.appendChild(notification);
|
583
|
-
|
584
|
-
// Auto-remove after 5 seconds
|
585
|
-
setTimeout(() => {
|
586
|
-
if (notification.parentElement) {
|
587
|
-
notification.remove();
|
588
|
-
}
|
589
|
-
}, 5000);
|
590
|
-
}
|
591
|
-
}
|
592
|
-
}
|
593
|
-
</script>
|
243
|
+
{% block extra_js %}
|
244
|
+
<script src="{% static 'payments/js/webhook-dashboard.js' %}"></script>
|
594
245
|
{% endblock %}
|