1 changed files with 455 additions and 0 deletions
@ -0,0 +1,455 @@ |
|||||||
|
# Система тестирования сотрудников клиники |
||||||
|
|
||||||
|
Веб-приложение для проведения внутреннего тестирования сотрудников клиники. Руководители подразделений и HR-менеджеры создают тесты и назначают их сотрудникам. Система фиксирует все попытки и результаты. |
||||||
|
|
||||||
|
**Версия ТЗ:** 1.2 |
||||||
|
**Дата:** 2026-03-21 |
||||||
|
**Статус:** Согласовано |
||||||
|
|
||||||
|
--- |
||||||
|
|
||||||
|
## Содержание |
||||||
|
|
||||||
|
- [Технологический стек](#технологический-стек) |
||||||
|
- [Функциональные возможности](#функциональные-возможности) |
||||||
|
- [Роли и права доступа](#роли-и-права-доступа) |
||||||
|
- [База данных](#база-данных) |
||||||
|
- [API Endpoints](#api-endpoints) |
||||||
|
- [Структура проекта](#структура-проекта) |
||||||
|
- [Установка и запуск](#установка-и-запуск) |
||||||
|
- [План разработки](#план-разработки) |
||||||
|
|
||||||
|
--- |
||||||
|
|
||||||
|
## Технологический стек |
||||||
|
|
||||||
|
| Компонент | Выбор | |
||||||
|
|-----------|-------| |
||||||
|
| Backend | Node.js + Express | |
||||||
|
| Frontend | React (SPA) | |
||||||
|
| Database | PostgreSQL | |
||||||
|
| Authentication | JWT (токен в httpOnly cookie) | |
||||||
|
| Password | bcrypt | |
||||||
|
|
||||||
|
--- |
||||||
|
|
||||||
|
## Функциональные возможности |
||||||
|
|
||||||
|
### 4.1 Управление пользователями и подразделениями |
||||||
|
|
||||||
|
- Создание/редактирование/деактивация учётных записей сотрудников |
||||||
|
- Каждый сотрудник принадлежит одному подразделению |
||||||
|
- Создание/редактирование справочника подразделений |
||||||
|
- Назначение роли сотруднику: HR-менеджер / Руководитель подразделения / Сотрудник |
||||||
|
|
||||||
|
### 4.2 Создание и редактирование тестов |
||||||
|
|
||||||
|
**Тест содержит:** |
||||||
|
- Название теста |
||||||
|
- Описание (опционально) |
||||||
|
- Список вопросов (минимум 7) |
||||||
|
- Порог зачёта — минимальный % правильных ответов |
||||||
|
- Таймер прохождения — лимит в минутах (опционально) |
||||||
|
|
||||||
|
**Вопрос содержит:** |
||||||
|
- Текст вопроса |
||||||
|
- Минимум 3 варианта ответа |
||||||
|
- Один или несколько правильных ответов |
||||||
|
|
||||||
|
**Настройки теста:** |
||||||
|
- Разрешить возврат к предыдущему вопросу: да / нет |
||||||
|
|
||||||
|
**Версионирование:** |
||||||
|
- Автор может редактировать тест пока никто его не проходил |
||||||
|
- Если тест уже проходили — создаётся новая версия (`version + 1`), старая сохраняется |
||||||
|
- Все версии теста хранятся; результаты привязаны к конкретной версии |
||||||
|
- Активная версия — та, которую видят сотрудники; автор может вручную переключить активную версию |
||||||
|
- Тест можно деактивировать (скрыть из списка, не удалять) |
||||||
|
|
||||||
|
### 4.3 Назначение теста |
||||||
|
|
||||||
|
- Список получателей (отдел или конкретные сотрудники) |
||||||
|
- Срок сдачи — дата дедлайна |
||||||
|
- Допустимое количество попыток (1 или более) |
||||||
|
|
||||||
|
### 4.4 Прохождение теста |
||||||
|
|
||||||
|
- На главной странице сотрудник видит список назначенных тестов со статусами: |
||||||
|
- `Не начат` — ещё не открывал |
||||||
|
- `В процессе` — начал, не завершил |
||||||
|
- `Завершён` — сдал/не сдал |
||||||
|
- `Просрочен` — дедлайн прошёл, не сдан |
||||||
|
- Если задан таймер — отображается обратный отсчёт, по истечении тест завершается автоматически |
||||||
|
- Порядок вопросов **случайный** при каждом прохождении |
||||||
|
- Возможность вернуться к предыдущему вопросу — определяется настройкой теста |
||||||
|
|
||||||
|
### 4.5 Результаты после завершения теста |
||||||
|
|
||||||
|
- Итоговый балл и процент правильных ответов |
||||||
|
- Факт зачёта: **сдал / не сдал** |
||||||
|
- Разбор ошибок: по каждому вопросу — его ответ и правильный ответ |
||||||
|
|
||||||
|
### 4.6 Трекер попыток |
||||||
|
|
||||||
|
Единый интерфейс просмотра всех попыток прохождения тестов: |
||||||
|
- Фильтрация по подразделению, сотруднику, тесту, статусу, результату |
||||||
|
- Пагинация и сортировка |
||||||
|
|
||||||
|
### 4.7 AI-помощник (DeepSeek) |
||||||
|
|
||||||
|
Интеграция с LLM DeepSeek для помощи при создании тестов: |
||||||
|
|
||||||
|
| Функция | Описание | |
||||||
|
|---------|----------| |
||||||
|
| Генерация теста | AI генерирует готовый набор вопросов с вариантами ответов по теме | |
||||||
|
| Улучшение формулировки | AI переформулирует выбранный вопрос более чётко | |
||||||
|
| Добавление дистракторов | AI генерирует правдоподобные неправильные варианты ответов | |
||||||
|
| Проверка качества | AI анализирует весь тест и выдаёт рекомендации | |
||||||
|
|
||||||
|
--- |
||||||
|
|
||||||
|
## Роли и права доступа |
||||||
|
|
||||||
|
| Роль | Кто | Создаёт тесты | Назначает тесты | Видит результаты | |
||||||
|
|------|-----|:---:|:---:|:---:| |
||||||
|
| **HR-менеджер** | Руководитель службы HR, Директор клиники | ✅ | Всем сотрудникам клиники | Всех сотрудников | |
||||||
|
| **Руководитель подразделения** | Главный врач, рук. службы администраторов | ✅ | Только своему подразделению | Только своего подразделения | |
||||||
|
| **Сотрудник** | Все остальные работники | ❌ | ❌ | Только свои | |
||||||
|
|
||||||
|
--- |
||||||
|
|
||||||
|
## База данных |
||||||
|
|
||||||
|
### Схема таблиц |
||||||
|
|
||||||
|
#### `departments` — Подразделения |
||||||
|
|
||||||
|
| Поле | Тип | Описание | |
||||||
|
|------|-----|----------| |
||||||
|
| id | UUID | PK | |
||||||
|
| name | VARCHAR(255) | Название подразделения | |
||||||
|
| created_at | TIMESTAMP | Дата создания | |
||||||
|
| updated_at | TIMESTAMP | Дата обновления | |
||||||
|
|
||||||
|
#### `users` — Пользователи |
||||||
|
|
||||||
|
| Поле | Тип | Описание | |
||||||
|
|------|-----|----------| |
||||||
|
| id | UUID | PK | |
||||||
|
| login | VARCHAR(100) | Уникальный логин | |
||||||
|
| password_hash | VARCHAR(255) | Хеш пароля | |
||||||
|
| full_name | VARCHAR(255) | ФИО | |
||||||
|
| role | ENUM | 'hr', 'manager', 'employee' | |
||||||
|
| department_id | UUID | FK -> departments | |
||||||
|
| is_active | BOOLEAN | Активна ли учётная запись | |
||||||
|
| created_at | TIMESTAMP | Дата создания | |
||||||
|
| updated_at | TIMESTAMP | Дата обновления | |
||||||
|
|
||||||
|
#### `tests` — Тесты |
||||||
|
|
||||||
|
| Поле | Тип | Описание | |
||||||
|
|------|-----|----------| |
||||||
|
| id | UUID | PK | |
||||||
|
| title | VARCHAR(255) | Название теста | |
||||||
|
| description | TEXT | Описание (опционально) | |
||||||
|
| passing_threshold | INTEGER | Порог зачёта (%) | |
||||||
|
| time_limit | INTEGER | Таймер в минутах (nullable) | |
||||||
|
| allow_back | BOOLEAN | Возврат к предыдущему | |
||||||
|
| is_active | BOOLEAN | Активен | |
||||||
|
| is_versioned | BOOLEAN | Есть версии | |
||||||
|
| created_by | UUID | FK -> users | |
||||||
|
| created_at | TIMESTAMP | Дата создания | |
||||||
|
| updated_at | TIMESTAMP | Дата обновления | |
||||||
|
|
||||||
|
#### `test_versions` — Версии тестов |
||||||
|
|
||||||
|
| Поле | Тип | Описание | |
||||||
|
|------|-----|----------| |
||||||
|
| id | UUID | PK | |
||||||
|
| test_id | UUID | FK -> tests | |
||||||
|
| version | INTEGER | Номер версии | |
||||||
|
| is_active | BOOLEAN | Активная версия | |
||||||
|
| created_at | TIMESTAMP | Дата создания | |
||||||
|
|
||||||
|
#### `questions` — Вопросы |
||||||
|
|
||||||
|
| Поле | Тип | Описание | |
||||||
|
|------|-----|----------| |
||||||
|
| id | UUID | PK | |
||||||
|
| test_version_id | UUID | FK -> test_versions | |
||||||
|
| text | TEXT | Текст вопроса | |
||||||
|
| question_order | INTEGER | Порядок вопроса | |
||||||
|
| has_multiple_answers | BOOLEAN | Несколько правильных | |
||||||
|
|
||||||
|
#### `answer_options` — Варианты ответов |
||||||
|
|
||||||
|
| Поле | Тип | Описание | |
||||||
|
|------|-----|----------| |
||||||
|
| id | UUID | PK | |
||||||
|
| question_id | UUID | FK -> questions | |
||||||
|
| text | TEXT | Текст варианта | |
||||||
|
| is_correct | BOOLEAN | Правильный ответ | |
||||||
|
| option_order | INTEGER | Порядок варианта | |
||||||
|
|
||||||
|
#### `test_assignments` — Назначения тестов |
||||||
|
|
||||||
|
| Поле | Тип | Описание | |
||||||
|
|------|-----|----------| |
||||||
|
| id | UUID | PK | |
||||||
|
| test_version_id | UUID | FK -> test_versions | |
||||||
|
| assigned_by | UUID | FK -> users | |
||||||
|
| deadline | DATE | Срок сдачи | |
||||||
|
| max_attempts | INTEGER | Допустимое число попыток | |
||||||
|
| created_at | TIMESTAMP | Дата назначения | |
||||||
|
|
||||||
|
#### `test_assignment_targets` — Получатели назначений |
||||||
|
|
||||||
|
| Поле | Тип | Описание | |
||||||
|
|------|-----|----------| |
||||||
|
| id | UUID | PK | |
||||||
|
| assignment_id | UUID | FK -> test_assignments | |
||||||
|
| target_type | ENUM | 'department', 'user' | |
||||||
|
| target_id | UUID | FK (department или user) | |
||||||
|
|
||||||
|
#### `test_attempts` — Попытки прохождения |
||||||
|
|
||||||
|
| Поле | Тип | Описание | |
||||||
|
|------|-----|----------| |
||||||
|
| id | UUID | PK | |
||||||
|
| test_version_id | UUID | FK -> test_versions | |
||||||
|
| user_id | UUID | FK -> users | |
||||||
|
| attempt_number | INTEGER | Номер попытки | |
||||||
|
| status | ENUM | 'in_progress', 'completed', 'expired' | |
||||||
|
| started_at | TIMESTAMP | Начало | |
||||||
|
| completed_at | TIMESTAMP | Завершение | |
||||||
|
| correct_count | INTEGER | Правильных ответов | |
||||||
|
| total_questions | INTEGER | Всего вопросов | |
||||||
|
| passed | BOOLEAN | Зачёт | |
||||||
|
|
||||||
|
#### `user_answers` — Ответы пользователя |
||||||
|
|
||||||
|
| Поле | Тип | Описание | |
||||||
|
|------|-----|----------| |
||||||
|
| id | UUID | PK | |
||||||
|
| attempt_id | UUID | FK -> test_attempts | |
||||||
|
| question_id | UUID | FK -> questions | |
||||||
|
| selected_options | UUID[] | Выбранные варианты | |
||||||
|
|
||||||
|
#### `settings` — Настройки |
||||||
|
|
||||||
|
| Поле | Тип | Описание | |
||||||
|
|------|-----|----------| |
||||||
|
| key | VARCHAR(100) | PK | |
||||||
|
| value | TEXT | Значение | |
||||||
|
|
||||||
|
--- |
||||||
|
|
||||||
|
## API Endpoints |
||||||
|
|
||||||
|
### Аутентификация |
||||||
|
|
||||||
|
| Метод | Эндпоинт | Описание | |
||||||
|
|-------|----------|----------| |
||||||
|
| POST | `/api/auth/login` | Вход в систему | |
||||||
|
| POST | `/api/auth/logout` | Выход из системы | |
||||||
|
|
||||||
|
### Подразделения |
||||||
|
|
||||||
|
| Метод | Эндпоинт | Описание | |
||||||
|
|-------|----------|----------| |
||||||
|
| GET | `/api/departments` | Список подразделений | |
||||||
|
| POST | `/api/departments` | Создать подразделение | |
||||||
|
| PUT | `/api/departments/:id` | Редактировать | |
||||||
|
| DELETE | `/api/departments/:id` | Деактивировать | |
||||||
|
|
||||||
|
### Пользователи |
||||||
|
|
||||||
|
| Метод | Эндпоинт | Описание | |
||||||
|
|-------|----------|----------| |
||||||
|
| GET | `/api/users` | Список сотрудников | |
||||||
|
| POST | `/api/users` | Создать сотрудника | |
||||||
|
| GET | `/api/users/:id` | Карточка сотрудника | |
||||||
|
| PUT | `/api/users/:id` | Редактировать | |
||||||
|
| DELETE | `/api/users/:id` | Деактивировать | |
||||||
|
|
||||||
|
### Тесты |
||||||
|
|
||||||
|
| Метод | Эндпоинт | Описание | |
||||||
|
|-------|----------|----------| |
||||||
|
| GET | `/api/tests` | Список тестов | |
||||||
|
| POST | `/api/tests` | Создать тест | |
||||||
|
| GET | `/api/tests/:id` | Тест с вопросами | |
||||||
|
| PUT | `/api/tests/:id` | Редактировать | |
||||||
|
| DELETE | `/api/tests/:id` | Деактивировать | |
||||||
|
|
||||||
|
### Назначения тестов |
||||||
|
|
||||||
|
| Метод | Эндпоинт | Описание | |
||||||
|
|-------|----------|----------| |
||||||
|
| GET | `/api/assignments` | Список назначений | |
||||||
|
| POST | `/api/assignments` | Назначить тест | |
||||||
|
| GET | `/api/assignments/:id` | Детали назначения | |
||||||
|
| PUT | `/api/assignments/:id` | Редактировать | |
||||||
|
| DELETE | `/api/assignments/:id` | Удалить | |
||||||
|
| GET | `/api/my-assignments` | Назначенные тесты (для сотрудника) | |
||||||
|
|
||||||
|
### Прохождение теста |
||||||
|
|
||||||
|
| Метод | Эндпоинт | Описание | |
||||||
|
|-------|----------|----------| |
||||||
|
| POST | `/api/attempts/start` | Начать попытку | |
||||||
|
| GET | `/api/attempts/:id` | Получить вопросы | |
||||||
|
| POST | `/api/attempts/:id/answers` | Отправить ответ | |
||||||
|
| GET | `/api/attempts/:id/answers` | Получить текущий прогресс | |
||||||
|
| POST | `/api/attempts/:id/complete` | Завершить тест | |
||||||
|
|
||||||
|
### Результаты |
||||||
|
|
||||||
|
| Метод | Эндпоинт | Описание | |
||||||
|
|-------|----------|----------| |
||||||
|
| GET | `/api/attempts/:id/result` | Получить результат | |
||||||
|
| GET | `/api/attempts/:id/review` | Получить разбор | |
||||||
|
|
||||||
|
### Трекер попыток |
||||||
|
|
||||||
|
| Метод | Эндпоинт | Описание | |
||||||
|
|-------|----------|----------| |
||||||
|
| GET | `/api/tracker` | Список всех попыток | |
||||||
|
|
||||||
|
### AI-помощник |
||||||
|
|
||||||
|
| Метод | Эндпоинт | Описание | |
||||||
|
|-------|----------|----------| |
||||||
|
| POST | `/api/ai/generate-test` | Сгенерировать тест | |
||||||
|
| POST | `/api/ai/improve-question` | Улучшить вопрос | |
||||||
|
| POST | `/api/ai/add-distractors` | Добавить неправильные варианты | |
||||||
|
| POST | `/api/ai/check-quality` | Проверить качество теста | |
||||||
|
|
||||||
|
### Настройки |
||||||
|
|
||||||
|
| Метод | Эндпоинт | Описание | |
||||||
|
|-------|----------|----------| |
||||||
|
| GET | `/api/settings` | Получить настройки | |
||||||
|
| PUT | `/api/settings` | Обновить настройки | |
||||||
|
| POST | `/api/settings/test-connection` | Проверить подключение к DeepSeek | |
||||||
|
|
||||||
|
--- |
||||||
|
|
||||||
|
## Структура проекта |
||||||
|
|
||||||
|
``` |
||||||
|
TestingWebApp/ |
||||||
|
├── docs/ # Документация |
||||||
|
│ ├── ТЗ.md # Техническое задание |
||||||
|
│ └── шаги/ # План разработки (11 шагов) |
||||||
|
├── backend/ # Node.js + Express |
||||||
|
│ ├── src/ |
||||||
|
│ │ ├── index.js # Точка входа |
||||||
|
│ │ ├── db/ # Работа с БД |
||||||
|
│ │ │ ├── db.js # Подключение к PostgreSQL |
||||||
|
│ │ │ ├── migrate.js # Миграции |
||||||
|
│ │ │ └── migrations/ |
||||||
|
│ │ │ └── 001_initial.sql |
||||||
|
│ │ ├── routes/ # API роуты |
||||||
|
│ │ │ └── auth.js |
||||||
|
│ │ ├── middleware/ # Промежуточное ПО |
||||||
|
│ │ │ └── auth.js |
||||||
|
│ │ └── utils/ # Утилиты |
||||||
|
│ │ └── auth.js |
||||||
|
│ └── package.json |
||||||
|
├── frontend/ # React SPA |
||||||
|
│ ├── src/ |
||||||
|
│ │ ├── main.jsx # Точка входа |
||||||
|
│ │ └── App.jsx # Главный компонент |
||||||
|
│ ├── index.html |
||||||
|
│ ├── vite.config.js |
||||||
|
│ └── package.json |
||||||
|
├── docker-compose.yml # PostgreSQL |
||||||
|
└── README.md # Этот файл |
||||||
|
``` |
||||||
|
|
||||||
|
--- |
||||||
|
|
||||||
|
## Установка и запуск |
||||||
|
|
||||||
|
### Требования |
||||||
|
|
||||||
|
- Node.js 18+ |
||||||
|
- PostgreSQL 14+ |
||||||
|
- Docker (для PostgreSQL) |
||||||
|
|
||||||
|
### Шаги |
||||||
|
|
||||||
|
1. **Клонировать репозиторий** |
||||||
|
|
||||||
|
2. **Настроить окружение** |
||||||
|
|
||||||
|
Создать файл `backend/.env`: |
||||||
|
```env |
||||||
|
DATABASE_URL=postgresql://user:password@localhost:5432/clinic_tests |
||||||
|
JWT_SECRET=your-secret-key |
||||||
|
PORT=3000 |
||||||
|
``` |
||||||
|
|
||||||
|
3. **Запустить PostgreSQL** |
||||||
|
```bash |
||||||
|
docker-compose up -d |
||||||
|
``` |
||||||
|
|
||||||
|
4. **Установить зависимости и запустить backend** |
||||||
|
```bash |
||||||
|
cd backend |
||||||
|
npm install |
||||||
|
npm run dev |
||||||
|
``` |
||||||
|
|
||||||
|
5. **Запустить frontend** |
||||||
|
```bash |
||||||
|
cd frontend |
||||||
|
npm install |
||||||
|
npm run dev |
||||||
|
``` |
||||||
|
|
||||||
|
6. **Открыть приложение** |
||||||
|
- Frontend: http://localhost:5173 |
||||||
|
- Backend API: http://localhost:3000 |
||||||
|
|
||||||
|
--- |
||||||
|
|
||||||
|
## План разработки |
||||||
|
|
||||||
|
| Шаг | Название | Описание | |
||||||
|
|-----|----------|----------| |
||||||
|
| 1 | Проект и инфраструктура | Структура проекта, Docker, настройка окружения | |
||||||
|
| 2 | Проектирование БД | Создание всех таблиц и миграций | |
||||||
|
| 3 | Аутентификация и авторизация | JWT, login/logout, RBAC | |
||||||
|
| 4 | Управление пользователями | CRUD сотрудников и подразделений | |
||||||
|
| 5 | Управление тестами | Создание, редактирование, версионирование | |
||||||
|
| 6 | Назначение тестов | Выбор получателей, дедлайн, лимит попыток | |
||||||
|
| 7 | Интерфейс прохождения | Прохождение с таймером, случайный порядок вопросов | |
||||||
|
| 8 | Результаты и разбор | Итоговый балл, разбор ошибок | |
||||||
|
| 9 | Трекер попыток | Единый интерфейс просмотра попыток | |
||||||
|
| 10 | AI-помощник | Интеграция с DeepSeek | |
||||||
|
| 11 | Страница настроек | Управление API ключами | |
||||||
|
|
||||||
|
--- |
||||||
|
|
||||||
|
## Нефункциональные требования |
||||||
|
|
||||||
|
| Параметр | Значение | |
||||||
|
|----------|----------| |
||||||
|
| Количество пользователей | 50–200 человек | |
||||||
|
| Платформа | Веб-приложение, браузер (desktop-first) | |
||||||
|
| Доступность | Внутренняя сеть клиники | |
||||||
|
| Язык интерфейса | Русский | |
||||||
|
| Время отклика | < 2 секунды | |
||||||
|
|
||||||
|
--- |
||||||
|
|
||||||
|
## Вне scope (не реализуется в данной версии) |
||||||
|
|
||||||
|
- Интеграция с AD/LDAP |
||||||
|
- Мобильное приложение |
||||||
|
- Вопросы с вложениями (изображения, видео) |
||||||
|
- Экспорт отчётов в Excel / PDF |
||||||
|
- Уведомления в MAX (отдельный спринт) |
||||||
Loading…
Reference in new issue