3224d16197
Co-authored-by: Copilot <copilot@github.com>
81 lines
2.3 KiB
HTML
81 lines
2.3 KiB
HTML
{% extends "base.html" %} {% block title %}Admin — All You Can GET AI{% endblock
|
|
%} {% block content %}
|
|
<div class="card">
|
|
<h1>Admin Dashboard</h1>
|
|
|
|
{% if stats %}
|
|
<div class="stats-grid">
|
|
<div class="stat-box">
|
|
<div class="stat-label">Total users</div>
|
|
<div class="stat-value">{{ stats.get('total_users', 0) }}</div>
|
|
</div>
|
|
<div class="stat-box">
|
|
<div class="stat-label">Active tokens</div>
|
|
<div class="stat-value">{{ stats.get('active_refresh_tokens', 0) }}</div>
|
|
</div>
|
|
<div class="stat-box">
|
|
<div class="stat-label">Admins</div>
|
|
<div class="stat-value">{{ stats.get('admin_users', 0) }}</div>
|
|
</div>
|
|
</div>
|
|
{% endif %}
|
|
|
|
<h2 class="section-title">Users</h2>
|
|
<div class="table-wrap">
|
|
<table>
|
|
<thead>
|
|
<tr>
|
|
<th>Email</th>
|
|
<th>Role</th>
|
|
<th>Actions</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{% for u in users %}
|
|
<tr>
|
|
<td>{{ u.email }}</td>
|
|
<td>
|
|
<span class="role-badge role-{{ u.role }}">{{ u.role }}</span>
|
|
</td>
|
|
<td>
|
|
<div class="table-actions">
|
|
<!-- Role toggle -->
|
|
<form
|
|
method="post"
|
|
action="{{ url_for('admin_set_role', user_id=u.id) }}"
|
|
>
|
|
<input
|
|
type="hidden"
|
|
name="role"
|
|
value="{{ 'user' if u.role == 'admin' else 'admin' }}"
|
|
/>
|
|
<button type="submit" class="btn btn-sm">
|
|
Make {{ 'user' if u.role == 'admin' else 'admin' }}
|
|
</button>
|
|
</form>
|
|
<!-- Delete -->
|
|
{% if u.id != session.get('user_id') %}
|
|
<form
|
|
method="post"
|
|
action="{{ url_for('admin_delete_user', user_id=u.id) }}"
|
|
onsubmit="return confirm('Delete {{ u.email }}?')"
|
|
>
|
|
<button type="submit" class="btn btn-sm btn-danger">
|
|
Delete
|
|
</button>
|
|
</form>
|
|
{% endif %}
|
|
</div>
|
|
</td>
|
|
</tr>
|
|
{% else %}
|
|
<tr>
|
|
<td colspan="3" class="text-muted">No users found.</td>
|
|
</tr>
|
|
{% endfor %}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
{% endblock %}
|