added README
This commit is contained in:
@@ -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 (отдельный спринт)
|
||||
Reference in New Issue
Block a user