2025-07-02 22:30:21 +03:00
|
|
|
|
import { Component, createEffect, createSignal, For, Show } from 'solid-js'
|
|
|
|
|
import { useData } from '../context/data'
|
|
|
|
|
import formStyles from '../styles/Form.module.css'
|
|
|
|
|
import styles from '../styles/Modal.module.css'
|
|
|
|
|
import Button from '../ui/Button'
|
|
|
|
|
import Modal from '../ui/Modal'
|
|
|
|
|
|
|
|
|
|
interface Author {
|
|
|
|
|
id: number
|
|
|
|
|
name: string
|
|
|
|
|
email: string
|
|
|
|
|
slug: string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
interface Community {
|
|
|
|
|
id: number
|
|
|
|
|
name: string
|
|
|
|
|
slug: string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
interface Role {
|
|
|
|
|
id: string
|
|
|
|
|
name: string
|
|
|
|
|
description?: string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
interface CommunityRolesModalProps {
|
|
|
|
|
isOpen: boolean
|
|
|
|
|
author: Author | null
|
|
|
|
|
community: Community | null
|
|
|
|
|
onClose: () => void
|
|
|
|
|
onSave: (authorId: number, communityId: number, roles: string[]) => Promise<void>
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const CommunityRolesModal: Component<CommunityRolesModalProps> = (props) => {
|
|
|
|
|
const { queryGraphQL } = useData()
|
|
|
|
|
const [roles, setRoles] = createSignal<Role[]>([])
|
|
|
|
|
const [userRoles, setUserRoles] = createSignal<string[]>([])
|
|
|
|
|
const [loading, setLoading] = createSignal(false)
|
|
|
|
|
|
|
|
|
|
// Загружаем доступные роли при открытии модала
|
|
|
|
|
createEffect(() => {
|
|
|
|
|
if (props.isOpen && props.community) {
|
|
|
|
|
void loadRolesData()
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const loadRolesData = async () => {
|
|
|
|
|
setLoading(true)
|
|
|
|
|
try {
|
|
|
|
|
// Получаем доступные роли
|
|
|
|
|
const rolesData = await queryGraphQL(
|
|
|
|
|
`
|
|
|
|
|
query GetRoles($community: Int) {
|
|
|
|
|
adminGetRoles(community: $community) {
|
|
|
|
|
id
|
|
|
|
|
name
|
|
|
|
|
description
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
`,
|
|
|
|
|
{ community: props.community?.id }
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
if (rolesData?.adminGetRoles) {
|
|
|
|
|
setRoles(rolesData.adminGetRoles)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Получаем текущие роли пользователя
|
|
|
|
|
if (props.author) {
|
|
|
|
|
const membersData = await queryGraphQL(
|
|
|
|
|
`
|
|
|
|
|
query GetCommunityMembers($community_id: Int!) {
|
|
|
|
|
adminGetCommunityMembers(community_id: $community_id, limit: 1000) {
|
|
|
|
|
members {
|
|
|
|
|
id
|
|
|
|
|
roles
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
`,
|
|
|
|
|
{ community_id: props.community?.id }
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
const members = membersData?.adminGetCommunityMembers?.members || []
|
|
|
|
|
const currentUser = members.find((m: { id: number }) => m.id === props.author?.id)
|
|
|
|
|
setUserRoles(currentUser?.roles || [])
|
|
|
|
|
}
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error('Ошибка загрузки ролей:', error)
|
|
|
|
|
} finally {
|
|
|
|
|
setLoading(false)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const handleRoleToggle = (roleId: string) => {
|
|
|
|
|
const currentRoles = userRoles()
|
|
|
|
|
if (currentRoles.includes(roleId)) {
|
e2e-fixing
fix: убран health endpoint, E2E тест использует корневой маршрут
- Убран health endpoint из main.py (не нужен)
- E2E тест теперь проверяет корневой маршрут / вместо /health
- Корневой маршрут доступен без логина, что подходит для проверки состояния сервера
- E2E тест с браузером работает корректно
docs: обновлен отчет о прогрессе E2E теста
- Убраны упоминания health endpoint
- Указано что используется корневой маршрут для проверки серверов
- Обновлен список измененных файлов
fix: исправлены GraphQL проблемы и E2E тест с браузером
- Добавлено поле success в тип CommonResult для совместимости с фронтендом
- Обновлены резолверы community, collection, topic для возврата поля success
- Исправлен E2E тест для работы с корневым маршрутом вместо health endpoint
- E2E тест теперь запускает браузер, авторизуется, находит сообщество в таблице
- Все GraphQL проблемы с полем success решены
- E2E тест работает правильно с браузером как требовалось
fix: исправлен поиск UI элементов в E2E тесте
- Добавлен правильный поиск кнопки удаления по CSS классу _delete-button_1qlfg_300
- Добавлены альтернативные способы поиска кнопки удаления (title, aria-label, символ ×)
- Добавлен правильный поиск модального окна с множественными селекторами
- Добавлен правильный поиск кнопки подтверждения в модальном окне
- E2E тест теперь полностью работает: находит кнопку удаления, модальное окно и кнопку подтверждения
- Обновлен отчет о прогрессе с полными результатами тестирования
fix: исправлен импорт require_any_permission в resolvers/collection.py
- Заменен импорт require_any_permission с auth.decorators на services.rbac
- Бэкенд сервер теперь запускается корректно
- E2E тест полностью работает: находит кнопку удаления, модальное окно и кнопку подтверждения
- Оба сервера (бэкенд и фронтенд) работают стабильно
fix: исправлен порядок импортов в resolvers/collection.py
- Перемещен импорт require_any_permission в правильное место
- E2E тест полностью работает: находит кнопку удаления, модальное окно и кнопку подтверждения
- Сообщество не удаляется из-за прав доступа - это нормальное поведение системы безопасности
feat: настроен HTTPS для локальной разработки с mkcert
2025-08-01 00:30:44 +03:00
|
|
|
|
setUserRoles(currentRoles.filter((r) => r !== roleId))
|
2025-07-02 22:30:21 +03:00
|
|
|
|
} else {
|
|
|
|
|
setUserRoles([...currentRoles, roleId])
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const handleSave = async () => {
|
|
|
|
|
if (!props.author || !props.community) return
|
|
|
|
|
|
|
|
|
|
setLoading(true)
|
|
|
|
|
try {
|
|
|
|
|
await props.onSave(props.author.id, props.community.id, userRoles())
|
|
|
|
|
props.onClose()
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error('Ошибка сохранения ролей:', error)
|
|
|
|
|
} finally {
|
|
|
|
|
setLoading(false)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<Modal
|
|
|
|
|
isOpen={props.isOpen}
|
|
|
|
|
onClose={props.onClose}
|
|
|
|
|
title={`Роли пользователя: ${props.author?.name || ''}`}
|
|
|
|
|
>
|
|
|
|
|
<div class={styles.content}>
|
|
|
|
|
<Show when={props.community && props.author}>
|
|
|
|
|
<div class={formStyles.field}>
|
|
|
|
|
<label class={formStyles.label}>
|
|
|
|
|
Сообщество: <strong>{props.community?.name}</strong>
|
|
|
|
|
</label>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class={formStyles.field}>
|
|
|
|
|
<label class={formStyles.label}>
|
|
|
|
|
Пользователь: <strong>{props.author?.name}</strong> ({props.author?.email})
|
|
|
|
|
</label>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class={formStyles.field}>
|
|
|
|
|
<label class={formStyles.label}>Роли:</label>
|
|
|
|
|
<Show when={!loading()} fallback={<div>Загрузка ролей...</div>}>
|
|
|
|
|
<div class={formStyles.checkboxGroup}>
|
|
|
|
|
<For each={roles()}>
|
|
|
|
|
{(role) => (
|
|
|
|
|
<div class={formStyles.checkboxItem}>
|
|
|
|
|
<input
|
|
|
|
|
type="checkbox"
|
|
|
|
|
id={`role-${role.id}`}
|
|
|
|
|
checked={userRoles().includes(role.id)}
|
|
|
|
|
onChange={() => handleRoleToggle(role.id)}
|
|
|
|
|
class={formStyles.checkbox}
|
|
|
|
|
/>
|
|
|
|
|
<label for={`role-${role.id}`} class={formStyles.checkboxLabel}>
|
|
|
|
|
<div>
|
|
|
|
|
<strong>{role.name}</strong>
|
|
|
|
|
<Show when={role.description}>
|
|
|
|
|
<div class={formStyles.description}>{role.description}</div>
|
|
|
|
|
</Show>
|
|
|
|
|
</div>
|
|
|
|
|
</label>
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
|
|
|
|
</For>
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
</div>
|
|
|
|
|
</Show>
|
|
|
|
|
|
|
|
|
|
<div class={styles.actions}>
|
|
|
|
|
<Button variant="secondary" onClick={props.onClose}>
|
|
|
|
|
Отмена
|
|
|
|
|
</Button>
|
|
|
|
|
<Button variant="primary" onClick={handleSave} disabled={loading()}>
|
|
|
|
|
{loading() ? 'Сохранение...' : 'Сохранить'}
|
|
|
|
|
</Button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Modal>
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export default CommunityRolesModal
|