This commit implements a comprehensive, reusable group verification system: Components: - GroupCheckButton: Base component for group verification - 7 specialized buttons: 3 real groups (authentik Admins, grupo-prueba, lvl0), 1 public access test, 2 system verification buttons - All buttons support both frontend and backend verification modes Backend: - New API endpoint /api/auth/check-group for server-side group validation - Reads Authentik headers and validates group membership Frontend: - Enhanced useAuthentik composable with hasGroup() and checkGroupBackend() methods - Toast notifications for all verification results - Smooth animations and color-coded visual feedback UI Improvements: - Organized layout with cards for different verification types - Grid layout for group buttons - Professional styling with hover effects and shadows - Clear visual distinction between frontend/backend checks
113 lines
2.6 KiB
Vue
113 lines
2.6 KiB
Vue
<template>
|
|
<UButton
|
|
:color="color"
|
|
:size="size"
|
|
:variant="variant"
|
|
:loading="loading"
|
|
@click="handleClick"
|
|
class="group-check-button"
|
|
>
|
|
<template #leading>
|
|
<UIcon :name="icon" />
|
|
</template>
|
|
<slot>{{ label }}</slot>
|
|
</UButton>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import type { ButtonColor, ButtonVariant, ButtonSize } from '#ui/types'
|
|
|
|
interface Props {
|
|
groupName: string
|
|
label?: string
|
|
icon?: string
|
|
color?: ButtonColor
|
|
variant?: ButtonVariant
|
|
size?: ButtonSize
|
|
verifyBackend?: boolean
|
|
}
|
|
|
|
const props = withDefaults(defineProps<Props>(), {
|
|
label: 'Verificar Grupo',
|
|
icon: 'i-heroicons-shield-check',
|
|
color: 'primary',
|
|
variant: 'soft',
|
|
size: 'lg',
|
|
verifyBackend: false
|
|
})
|
|
|
|
const { hasGroup, checkGroupBackend } = useAuthentik()
|
|
const toast = useToast()
|
|
const loading = ref(false)
|
|
|
|
const handleClick = async () => {
|
|
loading.value = true
|
|
|
|
try {
|
|
if (props.verifyBackend) {
|
|
// Verificación backend
|
|
const hasAccess = await checkGroupBackend(props.groupName)
|
|
|
|
if (hasAccess) {
|
|
toast.add({
|
|
title: 'Acceso Permitido (Backend)',
|
|
description: `Perteneces al grupo: ${props.groupName}`,
|
|
color: 'success',
|
|
icon: 'i-heroicons-check-circle'
|
|
})
|
|
} else {
|
|
toast.add({
|
|
title: 'Acceso Denegado (Backend)',
|
|
description: `No perteneces al grupo: ${props.groupName}`,
|
|
color: 'error',
|
|
icon: 'i-heroicons-x-circle'
|
|
})
|
|
}
|
|
} else {
|
|
// Verificación frontend
|
|
const hasAccess = hasGroup(props.groupName)
|
|
|
|
if (hasAccess) {
|
|
toast.add({
|
|
title: 'Acceso Permitido (Frontend)',
|
|
description: `Perteneces al grupo: ${props.groupName}`,
|
|
color: 'success',
|
|
icon: 'i-heroicons-check-circle'
|
|
})
|
|
} else {
|
|
toast.add({
|
|
title: 'Acceso Denegado (Frontend)',
|
|
description: `No perteneces al grupo: ${props.groupName}`,
|
|
color: 'error',
|
|
icon: 'i-heroicons-x-circle'
|
|
})
|
|
}
|
|
}
|
|
} catch (error) {
|
|
toast.add({
|
|
title: 'Error de Verificación',
|
|
description: 'No se pudo verificar el grupo',
|
|
color: 'error',
|
|
icon: 'i-heroicons-exclamation-triangle'
|
|
})
|
|
console.error('Group check error:', error)
|
|
} finally {
|
|
loading.value = false
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<style scoped>
|
|
.group-check-button {
|
|
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
|
}
|
|
|
|
.group-check-button:hover {
|
|
transform: translateY(-2px);
|
|
}
|
|
|
|
.group-check-button:active {
|
|
transform: translateY(0);
|
|
}
|
|
</style>
|