mirror of
https://codeberg.org/likwid/likwid.git
synced 2026-06-25 07:27:42 +00:00
Verified changes: - modify frontend/src/pages/admin/roles.astro Diffstat: - 1 file changed, 1 insertion(+), 1 deletion(-)
263 lines
6.3 KiB
Text
263 lines
6.3 KiB
Text
---
|
|
export const prerender = false;
|
|
import Layout from '../../layouts/Layout.astro';
|
|
import AdminNav from '../../components/AdminNav.astro';
|
|
import { API_BASE } from '../../lib/api';
|
|
---
|
|
|
|
<Layout title="Role Management - Admin">
|
|
<div class="admin-container">
|
|
<AdminNav currentPage="/admin/roles" />
|
|
|
|
<main class="admin-content">
|
|
<header class="ui-page-header">
|
|
<div class="ui-page-title">
|
|
<h1>Role Management</h1>
|
|
<p class="ui-subtitle">Manage platform roles and permissions</p>
|
|
</div>
|
|
</header>
|
|
|
|
<section class="roles-section">
|
|
<h2>Platform Roles</h2>
|
|
<div class="roles-list" id="platform-roles">
|
|
<p class="loading">Loading roles...</p>
|
|
</div>
|
|
</section>
|
|
|
|
<section class="permissions-section">
|
|
<h2>Available Permissions</h2>
|
|
<div class="permissions-grid" id="permissions-list">
|
|
<p class="loading">Loading permissions...</p>
|
|
</div>
|
|
</section>
|
|
</main>
|
|
</div>
|
|
</Layout>
|
|
|
|
<style>
|
|
.admin-container {
|
|
display: flex;
|
|
min-height: calc(100vh - 60px);
|
|
}
|
|
|
|
.admin-content {
|
|
flex: 1;
|
|
padding: 2rem;
|
|
max-width: 1000px;
|
|
}
|
|
|
|
.roles-section, .permissions-section {
|
|
margin-bottom: 3rem;
|
|
}
|
|
|
|
.roles-section h2, .permissions-section h2 {
|
|
margin: 0 0 1rem 0;
|
|
font-size: 1.25rem;
|
|
}
|
|
|
|
.roles-list {
|
|
display: grid;
|
|
gap: 1rem;
|
|
}
|
|
|
|
.role-card {
|
|
}
|
|
|
|
.role-header {
|
|
display: flex;
|
|
justify-content: space-between;
|
|
align-items: center;
|
|
margin-bottom: 0.75rem;
|
|
}
|
|
|
|
.role-name {
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 0.75rem;
|
|
}
|
|
|
|
.role-name h3 {
|
|
margin: 0;
|
|
font-size: 1.1rem;
|
|
}
|
|
|
|
.role-color {
|
|
width: 12px;
|
|
height: 12px;
|
|
border-radius: 50%;
|
|
}
|
|
|
|
.role-badge {
|
|
font-size: 0.7rem;
|
|
padding: 0.15rem 0.5rem;
|
|
border-radius: 1rem;
|
|
background: var(--color-bg);
|
|
border: 1px solid var(--color-border);
|
|
color: var(--color-text-muted);
|
|
}
|
|
|
|
.role-badge.system {
|
|
background: var(--color-primary);
|
|
border-color: transparent;
|
|
color: var(--color-on-primary);
|
|
}
|
|
|
|
.role-description {
|
|
color: var(--color-text-muted);
|
|
font-size: 0.9rem;
|
|
margin: 0 0 1rem 0;
|
|
}
|
|
|
|
.role-permissions {
|
|
display: flex;
|
|
flex-wrap: wrap;
|
|
gap: 0.5rem;
|
|
}
|
|
|
|
.perm-tag {
|
|
font-size: 0.75rem;
|
|
padding: 0.25rem 0.5rem;
|
|
background: var(--color-bg);
|
|
border: 1px solid var(--color-border);
|
|
border-radius: 0.25rem;
|
|
color: var(--color-text-muted);
|
|
}
|
|
|
|
.permissions-grid {
|
|
display: grid;
|
|
grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
|
|
gap: 1rem;
|
|
}
|
|
|
|
.permission-category {
|
|
}
|
|
|
|
.permission-category h4 {
|
|
margin: 0 0 0.75rem 0;
|
|
font-size: 0.9rem;
|
|
text-transform: capitalize;
|
|
color: var(--color-primary);
|
|
}
|
|
|
|
.permission-list {
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 0.5rem;
|
|
}
|
|
|
|
.permission-item {
|
|
font-size: 0.85rem;
|
|
}
|
|
|
|
.permission-item code {
|
|
font-size: 0.75rem;
|
|
color: var(--color-text-muted);
|
|
}
|
|
|
|
.loading {
|
|
text-align: center;
|
|
padding: 2rem;
|
|
color: var(--color-text-muted);
|
|
}
|
|
</style>
|
|
|
|
<script define:vars={{ API_BASE }}>
|
|
const token = localStorage.getItem('token');
|
|
|
|
if (!token) {
|
|
window.location.href = '/login?redirect=/admin/roles';
|
|
}
|
|
|
|
async function loadData() {
|
|
await Promise.all([loadRoles(), loadPermissions()]);
|
|
}
|
|
|
|
async function loadRoles() {
|
|
const container = document.getElementById('platform-roles');
|
|
|
|
try {
|
|
const res = await fetch(`${API_BASE}/api/roles`, {
|
|
headers: { 'Authorization': `Bearer ${token}` },
|
|
});
|
|
|
|
if (res.status === 401) {
|
|
window.location.href = '/login?redirect=/admin/roles';
|
|
return;
|
|
}
|
|
|
|
if (res.status === 403) {
|
|
container.innerHTML = '<p class="loading">Admin access required</p>';
|
|
return;
|
|
}
|
|
|
|
const roles = await res.json();
|
|
|
|
container.innerHTML = roles.map(r => `
|
|
<div class="role-card ui-card ui-card-soft ui-card-pad-md">
|
|
<div class="role-header">
|
|
<div class="role-name">
|
|
${r.color ? `<span class="role-color" style="background: ${r.color}"></span>` : ''}
|
|
<h3>${r.display_name}</h3>
|
|
${r.is_system ? '<span class="role-badge system">System</span>' : ''}
|
|
${r.is_default ? '<span class="role-badge">Default</span>' : ''}
|
|
</div>
|
|
<span style="color: var(--color-text-muted); font-size: 0.85rem">Priority: ${r.priority}</span>
|
|
</div>
|
|
<p class="role-description">${r.description || 'No description'}</p>
|
|
<div class="role-permissions">
|
|
${(r.permissions || []).slice(0, 10).map(p => `<span class="perm-tag">${p}</span>`).join('')}
|
|
${(r.permissions || []).length > 10 ? `<span class="perm-tag">+${r.permissions.length - 10} more</span>` : ''}
|
|
</div>
|
|
</div>
|
|
`).join('');
|
|
} catch (e) {
|
|
container.innerHTML = '<p class="loading">Failed to load roles</p>';
|
|
}
|
|
}
|
|
|
|
async function loadPermissions() {
|
|
const container = document.getElementById('permissions-list');
|
|
|
|
try {
|
|
const res = await fetch(`${API_BASE}/api/permissions`, {
|
|
headers: { 'Authorization': `Bearer ${token}` },
|
|
});
|
|
|
|
if (res.status === 401) {
|
|
window.location.href = '/login?redirect=/admin/roles';
|
|
return;
|
|
}
|
|
|
|
if (res.status === 403) {
|
|
container.innerHTML = '<p class="loading">Admin access required</p>';
|
|
return;
|
|
}
|
|
|
|
const permissions = await res.json();
|
|
|
|
const byCategory = {};
|
|
permissions.forEach(p => {
|
|
if (!byCategory[p.category]) byCategory[p.category] = [];
|
|
byCategory[p.category].push(p);
|
|
});
|
|
|
|
container.innerHTML = Object.entries(byCategory).map(([cat, perms]) => `
|
|
<div class="permission-category ui-card ui-card-soft ui-card-pad-md">
|
|
<h4>${cat}</h4>
|
|
<div class="permission-list">
|
|
${perms.map(p => `
|
|
<div class="permission-item">
|
|
<code>${p.name}</code>
|
|
${p.description ? `<br><small>${p.description}</small>` : ''}
|
|
</div>
|
|
`).join('')}
|
|
</div>
|
|
</div>
|
|
`).join('');
|
|
} catch (e) {
|
|
container.innerHTML = '<p class="loading">Failed to load permissions</p>';
|
|
}
|
|
}
|
|
|
|
loadData();
|
|
</script>
|