Compare commits
38 Commits
personal-s
...
4915e6f33b
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4915e6f33b | ||
|
|
798d61c7ea | ||
|
|
0817961d97 | ||
|
|
49ca2e17b6 | ||
|
|
c07e49ca98 | ||
|
|
b89d1c7f72 | ||
| 6774ebb766 | |||
| f7fb524bb0 | |||
| d78e675a71 | |||
|
|
8cf086d3e9 | ||
| f39c3d1bbb | |||
|
|
dc1c343174 | ||
| 74fcd07e25 | |||
|
|
8ced8ae669 | ||
|
|
f519650bbb | ||
|
|
7fac9f744d | ||
|
|
18d099460d | ||
|
|
59b6704be9 | ||
|
|
220b99594f | ||
|
|
c10198515c | ||
|
|
a8144acb8b | ||
|
|
04feb5a3c3 | ||
|
|
d69eab1c12 | ||
|
|
f3ea05cd17 | ||
|
|
9f124c52a5 | ||
|
|
10c06e726a | ||
|
|
9d2de1faaf | ||
|
|
59caa9d6cc | ||
|
|
bad1215341 | ||
|
|
ccdc371c3a | ||
|
|
4c2293b620 | ||
|
|
6ea420e529 | ||
|
|
75b1ad166e | ||
|
|
abad3776db | ||
|
|
13b3a5c481 | ||
|
|
3579ef9f1c | ||
|
|
14cc006f06 | ||
| 9e55472de7 |
0
.gitea/workflows/docker-build.yaml
Normal file → Executable file
0
.gitea/workflows/docker-build.yaml
Normal file → Executable file
1
.gitignore
vendored
1
.gitignore
vendored
@@ -11,5 +11,4 @@ frontend/dist/
|
|||||||
.idea/
|
.idea/
|
||||||
.vscode/
|
.vscode/
|
||||||
*.DS_Store
|
*.DS_Store
|
||||||
AGENTS.md
|
|
||||||
GEMINI.md
|
GEMINI.md
|
||||||
201
AGENTS.md
Executable file
201
AGENTS.md
Executable file
@@ -0,0 +1,201 @@
|
|||||||
|
# AGENTS.md - Руководство для агентных помощников
|
||||||
|
|
||||||
|
## Обзор проекта
|
||||||
|
|
||||||
|
Проект представляет собой систему управления университетским расписанием.
|
||||||
|
- **Backend**: Java 17, Spring Boot 3.2.5 (Мультитенантная архитектура: отдельная БД для каждого клиента)
|
||||||
|
- **Frontend**: Vanilla JavaScript + HTML/CSS (без фреймворков)
|
||||||
|
- **Database**: PostgreSQL (множество БД, управляются через Flyway)
|
||||||
|
- **Локальный URL**: localhost:80
|
||||||
|
- **Продакшн URL**: https://magistr.zuev.company
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Структура директорий
|
||||||
|
|
||||||
|
```
|
||||||
|
magistr/
|
||||||
|
├── backend/ # Java Spring Boot приложение
|
||||||
|
│ └── src/main/java/com/magistr/app/
|
||||||
|
│ ├── controller/ # REST контроллеры
|
||||||
|
│ ├── model/ # JPA сущности
|
||||||
|
│ ├── dto/ # Data Transfer Objects
|
||||||
|
│ ├── repository/ # Spring Data JPA репозитории
|
||||||
|
│ ├── config/ # Конфигурация приложения
|
||||||
|
│ ├── utils/ # Утилиты
|
||||||
|
│ └── src/main/resources/db/migration/ # Flyway SQL миграции (версионирование схемы БД)
|
||||||
|
├── frontend/ # Статические файлы
|
||||||
|
│ ├── admin/ # Интерфейс администратора
|
||||||
|
│ ├── teacher/ # Интерфейс преподавателя
|
||||||
|
│ └── student/ # Интерфейс студента
|
||||||
|
├── compose.yaml # Docker Compose конфигурация
|
||||||
|
└── .env # Переменные окружения
|
||||||
|
```
|
||||||
|
|
||||||
|
**Внешние зависимости (родительская директория)**
|
||||||
|
|
||||||
|
На уровень выше расположен `../caddy-proxy/`. Это реверс-прокси, обрабатывающий трафик для `magistr.zuev.company`. Если возникают проблемы с доменом или внешним доступом, проверяйте `Caddyfile` там.
|
||||||
|
|
||||||
|
так же на уровень выше расположен конфиг kubernetes `../k8s/`, все файлы сборки на проде расположены там.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Команды сборки и запуска
|
||||||
|
|
||||||
|
### Docker Compose (основной способ)
|
||||||
|
|
||||||
|
Сборка и запуск всех сервисов (backend, frontend, PostgreSQL) выполняется через Docker Compose.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Сборка и запуск всех сервисов
|
||||||
|
docker compose up -d --build
|
||||||
|
|
||||||
|
# Остановка всех сервисов
|
||||||
|
docker compose down
|
||||||
|
|
||||||
|
# Просмотр логов всех сервисов
|
||||||
|
docker compose logs -f
|
||||||
|
|
||||||
|
# Просмотр логов конкретного сервиса
|
||||||
|
docker compose logs -f backend
|
||||||
|
|
||||||
|
# Пересоздать контейнер базы данных (полный сброс данных и повтор миграций Flyway)
|
||||||
|
docker compose down -v
|
||||||
|
docker compose up -d db
|
||||||
|
```
|
||||||
|
|
||||||
|
### Frontend
|
||||||
|
|
||||||
|
Статические файлы обслуживаются через Apache (httpd:alpine). Изменения в файлах frontend требуют пересборки контейнера.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Соглашения о коде (Code Style)
|
||||||
|
|
||||||
|
### Java (Backend)
|
||||||
|
|
||||||
|
**Именование:**
|
||||||
|
- Классы: PascalCase (например, `LessonsController`, `LessonResponse`)
|
||||||
|
- Методы и переменные: camelCase
|
||||||
|
- Константы: UPPER_SNAKE_CASE
|
||||||
|
- Пакеты: lowercase (например, `com.magistr.app.controller`)
|
||||||
|
|
||||||
|
**Импорты:**
|
||||||
|
- Группировка: static imports, затем external packages, затем internal
|
||||||
|
- Используйте wildcard imports для пакетов того же модуля: `import com.magistr.app.model.*;`
|
||||||
|
- Порядок: java.*, javax.*, external.*, internal.*
|
||||||
|
|
||||||
|
**Форматирование:**
|
||||||
|
- Отступы: 4 пробела (стандарт Java)
|
||||||
|
- Фигурные скобки: K&R style (открывающая на той же строке)
|
||||||
|
- Длина строки: до 120 символов
|
||||||
|
- Всегда используйте фигурные скобки для if/for/while
|
||||||
|
|
||||||
|
**Типы и аннотации:**
|
||||||
|
- Используйте явные типы вместо `var` для возвращаемых значений публичных методов
|
||||||
|
- Аннотации JPA: `@Entity`, `@Table`, `@Id`, `@GeneratedValue`, `@Column`
|
||||||
|
- Используйте `@JsonInclude(JsonInclude.Include.NON_NULL)` для DTO
|
||||||
|
- Для логгирования используйте SLF4J: `LoggerFactory.getLogger(ClassName.class)`
|
||||||
|
|
||||||
|
**Обработка ошибок:**
|
||||||
|
- Возвращайте `ResponseEntity<?>` с соответствующим HTTP статусом
|
||||||
|
- Логируйте ошибки с полным стектрейсом: `logger.error("msg: {}", e.getMessage(), e)`
|
||||||
|
- Для валидации используйте отдельные классы-валидаторы (см. `DayAndWeekValidator`)
|
||||||
|
|
||||||
|
**Архитектура контроллеров:**
|
||||||
|
- Используйте constructor injection для зависимостей
|
||||||
|
- Все endpoints имеют префикс `/api/`
|
||||||
|
- Возвращайте понятные сообщения об ошибках на русском языке
|
||||||
|
|
||||||
|
### Frontend (JavaScript)
|
||||||
|
|
||||||
|
**Именование:**
|
||||||
|
- Файлы: kebab-case (например, `main.js`, `schedule-view.js`)
|
||||||
|
- Функции и переменные: camelCase
|
||||||
|
- Константы: UPPER_SNAKE_CASE
|
||||||
|
|
||||||
|
**Модули:**
|
||||||
|
- Используйте ES6 modules с `import`/`export`
|
||||||
|
- Всегда указывайте расширение при импорте: `import { x } from './api.js';`
|
||||||
|
|
||||||
|
**Форматирование:**
|
||||||
|
- Отступы: 4 пробела
|
||||||
|
- Используйте template literals вместо конкатенации строк
|
||||||
|
- Предпочитайте `const` переменные, используйте `let` только при необходимости переприсваивания
|
||||||
|
|
||||||
|
**Лучшие практики:**
|
||||||
|
- Используйте `async/await` для асинхронных операций
|
||||||
|
- Всегда обрабатывайте ошибки в блоках `catch`
|
||||||
|
- Используйте деструктуризацию объектов
|
||||||
|
- Кешируйте DOM-элементы в переменные
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Работа с базой данных и мультитенантностью
|
||||||
|
|
||||||
|
**Мультитенантность:**
|
||||||
|
- Приложение поддерживает множество клиентов (университетов). Каждый клиент имеет свою изолированную базу данных PostgreSQL.
|
||||||
|
- Маршрутизация к нужной БД происходит динамически на основе поддомена (`TenantInterceptor` -> `TenantContext` -> `TenantRoutingDataSource`).
|
||||||
|
- Список клиентов хранится в Kubernetes `ConfigMap` (`tenants-config`), который монтируется в под бэкенда как `/config/tenants.json`.
|
||||||
|
- Локально список берётся из файла `backend/tenants.json`.
|
||||||
|
- При добавлении нового клиента в интерфейсе `DatabaseController` через K8s API обновляет `ConfigMap`. Все реплики бэкенда заметят изменения и в фоне инициализируют новый пул соединений (`TenantConfigWatcher`).
|
||||||
|
|
||||||
|
**Миграции схемы (Flyway):**
|
||||||
|
- Автогенерация Hibernate ОТКЛЮЧЕНА (`ddl-auto=none`). Структура баз данных управляется строго через **Flyway**.
|
||||||
|
- Все изменения схемы БД вносятся путем создания новых файлов в `backend/src/main/resources/db/migration/` (название строго `V2__add_new_table.sql` и т.д.).
|
||||||
|
- **ЗАПРЕЩЕНО** изменять существующие файлы миграций (например, `V1__init.sql`), которые уже закоммичены. Это сломает контрольные суммы Flyway.
|
||||||
|
- Flyway запускается программно при первом обращении к базе тенанта. Чтобы запустить Flyway для уже существующих тенантов (накатить V2), необходимо перезапустить бэкенд: `kubectl rollout restart deployment backend -n magistr`.
|
||||||
|
- Для локального сброса базы до изначального состояния: `docker compose down -v && docker compose up -d`.
|
||||||
|
|
||||||
|
**Сущности и связи:**
|
||||||
|
- Foreign keys с `ON DELETE CASCADE` для поддержания целостности
|
||||||
|
- Используйте расширение `pgcrypto` для хеширования паролей (bcrypt)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Функциональные требования к системе (Бизнес-логика)
|
||||||
|
|
||||||
|
### 1. Ролевая модель
|
||||||
|
- **Администратор (Деканат)**: Полный доступ, настройка топологии университета, управление аудиторным фондом, подтверждение переносов, регистрация инцидентов.
|
||||||
|
- **Преподаватель**: Просмотр своего расписания, подача заявок на перенос, отметка о своём отсутствии.
|
||||||
|
- **Студент**: Только просмотр расписания (Read-only).
|
||||||
|
|
||||||
|
### 2. Управление ресурсами и топология
|
||||||
|
- **Управление аудиториями**:
|
||||||
|
- Указание вместимости.
|
||||||
|
- Привязка доступного оборудования (через сущность Equipments: Проектор, ПК, Лаборатория).
|
||||||
|
- Установка статуса "Не доступно" (блокирует назначение пар в этот период).
|
||||||
|
- **Управление группами**:
|
||||||
|
- Управление списком студентов (и возможность деления на подгруппы).
|
||||||
|
- **Управление дисциплинами**:
|
||||||
|
- Создание предметов и привязка их к преподавателям (какие дисциплины имеет право вести конкретный преподаватель).
|
||||||
|
|
||||||
|
### 3. Логика расписания
|
||||||
|
- **Сетка**: 7 фиксированных слотов по 1.5 часа (08:00 - 09:30, и т.д.) + поддержка кастомного времени.
|
||||||
|
- **Проверка конфликтов**:
|
||||||
|
- *Критический конфликт*: Преподаватель не может находиться в двух разных аудиториях одновременно.
|
||||||
|
- *Уточнение по преподавателям*: Преподаватель может иметь несколько пар одновременно (для разных групп), только если они проходят в одной и той же аудитории (потоковая лекция).
|
||||||
|
- **Потоковые занятия**:
|
||||||
|
- Возможность назначить одну лекцию сразу нескольким группам (технически — несколько записей в БД или одна запись со списком групп).
|
||||||
|
- Проверка вместимости: вместимость аудитории должна покрывать суммарную численность всех групп, находящихся в этой аудитории в данный слот.
|
||||||
|
|
||||||
|
### 4. Управление инцидентами (Инклюзия отсутствия)
|
||||||
|
- **Отсутствие (Sickness/Business Trip)**: Регистрация отсутствия преподавателя (с указанием причины и периода дат).
|
||||||
|
- **Обнаружение коллизий**: Автоматическая подсветка конфликтующих пар в расписании (Red Zone).
|
||||||
|
- **Система разрешения конфликтов (Resolution Wizard)**:
|
||||||
|
- Предложение подходящей замены преподавателя на этот слот.
|
||||||
|
- Предложение переноса занятия на другое время или в другую аудиторию.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Языковые требования
|
||||||
|
|
||||||
|
- **Все ответы и комментарии на русском языке**
|
||||||
|
- Сообщения об ошибках и логи на русском
|
||||||
|
- Пользовательский интерфейс на русском
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Существующие правила проекта
|
||||||
|
|
||||||
|
См. `.agent/rules/main.md` и `.agent/rules/database_schema.md` для полного контекста о функциональных требованиях и схеме БД.
|
||||||
@@ -4,9 +4,16 @@ COPY pom.xml .
|
|||||||
RUN mvn dependency:go-offline -B
|
RUN mvn dependency:go-offline -B
|
||||||
COPY src ./src
|
COPY src ./src
|
||||||
RUN mvn package -DskipTests -B
|
RUN mvn package -DskipTests -B
|
||||||
|
RUN curl -L -o opentelemetry-javaagent.jar https://github.com/open-telemetry/opentelemetry-java-instrumentation/releases/latest/download/opentelemetry-javaagent.jar
|
||||||
|
|
||||||
FROM eclipse-temurin:17-jre-alpine
|
FROM eclipse-temurin:17-jre-alpine
|
||||||
|
|
||||||
|
# Best practice: run as a non-root user
|
||||||
|
RUN addgroup -S spring && adduser -S spring -G spring
|
||||||
|
USER spring:spring
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY --from=build /app/target/app.jar app.jar
|
COPY --from=build /app/target/app.jar app.jar
|
||||||
|
COPY --from=build /app/opentelemetry-javaagent.jar opentelemetry-javaagent.jar
|
||||||
EXPOSE 8080
|
EXPOSE 8080
|
||||||
ENTRYPOINT ["java", "-jar", "app.jar"]
|
ENTRYPOINT ["java", "-javaagent:opentelemetry-javaagent.jar", "-jar", "app.jar"]
|
||||||
|
|||||||
77
backend/README.md
Normal file
77
backend/README.md
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
# Руководство Backend-разработчика Magistr
|
||||||
|
|
||||||
|
Добро пожаловать в проект Magistr! Этот бэкенд построен на **Spring Boot** и имеет сложную **мультитенантную архитектуру**, где одно приложение обслуживает множество независимых университетов, каждый со своей базой данных. В проекте также есть интеграция с Kubernetes для "горячего" управления этими тенантами.
|
||||||
|
|
||||||
|
Здесь описано, как тут всё устроено, чтобы вы ничего не сломали.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 1. Архитектура мультитенантности
|
||||||
|
|
||||||
|
Мы используем подход **Separate Database per Tenant** (Отдельная БД для каждого клиента).
|
||||||
|
|
||||||
|
- **Как приложение понимает, к какой базе обращаться?**
|
||||||
|
Все запросы с фронтенда приходят с заголовком `Host` (например, `swsu.zuev.company`).
|
||||||
|
В классе `TenantInterceptor` (находится в `config/tenant/TenantInterceptor.java`) мы перехватываем этот запрос ДО того, как он дойдёт до контроллеров, вытаскиваем поддомен (`swsu`) и сохраняем его в `ThreadLocal` переменную через класс `TenantContext`.
|
||||||
|
|
||||||
|
- **Как переключаются базы данных?**
|
||||||
|
Класс `TenantRoutingDataSource` наследуется от спринговского `AbstractRoutingDataSource`. Перед каждым запросом в базу (любой `findById` или `save` из репозитория) Spring спрашивает этот класс: *"Какой сейчас ключ тенанта?"*. Класс берёт имя из `TenantContext` и переключает коннект на нужную БД на лету.
|
||||||
|
|
||||||
|
> **Важно:** Вся логика переключения абсолютно прозрачна для бизнес-кода. В контроллерах и сервисах вы пишете обычный код (`userRepository.findAll()`), и он сам выполнится в нужной базе.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. Динамическое управление тенантами (Kubernetes / ConfigMap)
|
||||||
|
|
||||||
|
Бэкенд спроектирован для работы в **Kubernetes с несколькими репликами (replicas: 2+)**.
|
||||||
|
|
||||||
|
Список тенантов не зашит в код:
|
||||||
|
- В K8s он лежит в специальном `ConfigMap`, который монтируется внутрь пода как файл `tenants.json`.
|
||||||
|
- В классе `DatabaseController` находится API для добавления нового тенанта из админки.
|
||||||
|
- Чтобы изменения применились ко **всем подам** без перезагрузки, `DatabaseController` вызывает `ConfigMapUpdater`. Этот класс обращается напрямую к **Kubernetes API** (используя ServiceAccount токен пода) и патчит `ConfigMap`.
|
||||||
|
- В фоне работает планировщик `TenantConfigWatcher` (каждые 30 секунд). Он следит за изменениями `tenants.json` и, если видит нового тенанта, на лету поднимает для него новый `HikariCP` пул соединений и добавляет в маршрутизатор баз данных.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. Базы данных и Миграции (Flyway)
|
||||||
|
|
||||||
|
Мы **НЕ используем** автоматическую генерацию таблиц через Hibernate (`spring.jpa.hibernate.ddl-auto=none`).
|
||||||
|
Структурой баз данных правит **Flyway**.
|
||||||
|
|
||||||
|
Поскольку баз данных много (они создаются динамически), стандартный Spring Boot Flyway отключён. Вместо этого `TenantConfigWatcher` вызывает Flyway **программно** в момент первого подключения нового тенанта.
|
||||||
|
|
||||||
|
### 🛑 ПРАВИЛА ИЗМЕНЕНИЯ СТРУКТУРЫ БД:
|
||||||
|
|
||||||
|
Если вам нужно добавить новую таблицу, колонку или изменить тип поля:
|
||||||
|
|
||||||
|
1. **Запрещено трогать старые файлы миграций!**
|
||||||
|
Запомните: файл `V1__init.sql` (и любые другие V-файлы, которые уже попали в коммит) — **СВЯЩЕНЕН**. Если вы его измените, бэкенд не запустится на сервере с ошибкой `Migration checksum mismatch`.
|
||||||
|
|
||||||
|
2. **Как правильно добавить таблицу?**
|
||||||
|
- Зайдите в папку `src/main/resources/db/migration/`.
|
||||||
|
- Создайте новый файл. Название **строго** по формату: `V<Номер>__<Описание>.sql`. Например: `V2__add_student_rating_table.sql`.
|
||||||
|
- Напишите в нём ваш SQL (`CREATE TABLE ...`, `ALTER TABLE ...`).
|
||||||
|
- Сохраните и запустите проект. Flyway **сам** пройдёт по всем базам данных тенантов и накатит этот скрипт.
|
||||||
|
|
||||||
|
3. **Что если локально я накосячил в V2?**
|
||||||
|
Пока файл `V2_...` не залит в Git и крутится только у вас на локалке, вы можете его переписывать. Но для этого вам нужно зайти в вашу локальную БД (через DBeaver/pgAdmin), вручную откатить свои кривые изменения (удалить таблицу) и **удалить запись из истории Flyway**:
|
||||||
|
`DELETE FROM flyway_schema_history WHERE version = '2';`
|
||||||
|
Либо, что проще: удалите контейнер с локальной БД (`docker compose down -v`) и поднимите заново пустую.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 4. Как запускать проект локально
|
||||||
|
|
||||||
|
В корневой папке репозитория (где лежит `docker-compose.yaml`) поднимите инфраструктуру:
|
||||||
|
```bash
|
||||||
|
docker compose up -d
|
||||||
|
```
|
||||||
|
Соберется и запустится:
|
||||||
|
- Фронтенд
|
||||||
|
- Бэкенд
|
||||||
|
- Ваша локальная тестовая PostgreSQL-база данных (на порту 5432, имя базы `app_db`, юзер `myuser`, логин/пароль см. в compose файле).
|
||||||
|
|
||||||
|
Файл `backend/tenants.json` нужен для локальной разработки. Если вы запускаете бэкенд в Docker Compose, вы можете указать URL `jdbc:postgresql://db:5432/app_db` (где `db` — имя контейнера в compose сети).
|
||||||
|
Либо, если вы тестируете взаимодействие бэкенда с вашим текущим IP-адресом (например, `192.168.1.87`), вы можете использовать этот IP. Оба варианта рабочие! Проект сразу подхватит настройки и накатит таблицы через Flyway.
|
||||||
|
|
||||||
|
Контроллеры и бизнес-логику пишите как в обычном Spring Boot проекте. Главное, помните: у каждого тенанта — своё изолированное хранилище!
|
||||||
@@ -32,6 +32,12 @@
|
|||||||
<groupId>org.springframework.boot</groupId>
|
<groupId>org.springframework.boot</groupId>
|
||||||
<artifactId>spring-boot-starter-data-jpa</artifactId>
|
<artifactId>spring-boot-starter-data-jpa</artifactId>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
|
||||||
|
<!-- Flyway Database Migrations -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.flywaydb</groupId>
|
||||||
|
<artifactId>flyway-core</artifactId>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.postgresql</groupId>
|
<groupId>org.postgresql</groupId>
|
||||||
<artifactId>postgresql</artifactId>
|
<artifactId>postgresql</artifactId>
|
||||||
@@ -43,6 +49,20 @@
|
|||||||
<groupId>org.springframework.security</groupId>
|
<groupId>org.springframework.security</groupId>
|
||||||
<artifactId>spring-security-crypto</artifactId>
|
<artifactId>spring-security-crypto</artifactId>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
|
||||||
|
<!-- H2 in-memory DB (fallback когда нет настроенных тенантов) -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.h2database</groupId>
|
||||||
|
<artifactId>h2</artifactId>
|
||||||
|
<scope>runtime</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<!-- OpenTelemetry API for custom span attributes -->
|
||||||
|
<dependency>
|
||||||
|
<groupId>io.opentelemetry</groupId>
|
||||||
|
<artifactId>opentelemetry-api</artifactId>
|
||||||
|
<version>1.49.0</version>
|
||||||
|
</dependency>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
|
|
||||||
<build>
|
<build>
|
||||||
|
|||||||
@@ -2,8 +2,12 @@ package com.magistr.app;
|
|||||||
|
|
||||||
import org.springframework.boot.SpringApplication;
|
import org.springframework.boot.SpringApplication;
|
||||||
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
||||||
|
import org.springframework.boot.autoconfigure.flyway.FlywayAutoConfiguration;
|
||||||
|
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
|
||||||
|
import org.springframework.scheduling.annotation.EnableScheduling;
|
||||||
|
|
||||||
@SpringBootApplication
|
@SpringBootApplication(exclude = {DataSourceAutoConfiguration.class, FlywayAutoConfiguration.class})
|
||||||
|
@EnableScheduling
|
||||||
public class Application {
|
public class Application {
|
||||||
|
|
||||||
public static void main(String[] args) {
|
public static void main(String[] args) {
|
||||||
|
|||||||
@@ -1,50 +1,39 @@
|
|||||||
package com.magistr.app.config;
|
package com.magistr.app.config;
|
||||||
|
|
||||||
import com.magistr.app.model.Role;
|
import com.magistr.app.config.tenant.TenantConfig;
|
||||||
import com.magistr.app.model.User;
|
import com.magistr.app.config.tenant.TenantConfigWatcher;
|
||||||
import com.magistr.app.repository.UserRepository;
|
import com.magistr.app.config.tenant.TenantRoutingDataSource;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.boot.CommandLineRunner;
|
import org.springframework.boot.CommandLineRunner;
|
||||||
import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder;
|
|
||||||
import org.springframework.stereotype.Component;
|
import org.springframework.stereotype.Component;
|
||||||
|
|
||||||
import java.util.Optional;
|
/**
|
||||||
|
* При запуске приложения инициализирует БД для каждого тенанта.
|
||||||
|
* Делегирует инициализацию в TenantConfigWatcher.initDatabaseForTenant().
|
||||||
|
*/
|
||||||
@Component
|
@Component
|
||||||
public class DataInitializer implements CommandLineRunner {
|
public class DataInitializer implements CommandLineRunner {
|
||||||
|
|
||||||
private static final Logger log = LoggerFactory.getLogger(DataInitializer.class);
|
private static final Logger log = LoggerFactory.getLogger(DataInitializer.class);
|
||||||
|
|
||||||
private final UserRepository userRepository;
|
private final TenantRoutingDataSource routingDataSource;
|
||||||
private final BCryptPasswordEncoder passwordEncoder;
|
private final TenantConfigWatcher configWatcher;
|
||||||
|
|
||||||
public DataInitializer(UserRepository userRepository, BCryptPasswordEncoder passwordEncoder) {
|
public DataInitializer(TenantRoutingDataSource routingDataSource,
|
||||||
this.userRepository = userRepository;
|
TenantConfigWatcher configWatcher) {
|
||||||
this.passwordEncoder = passwordEncoder;
|
this.routingDataSource = routingDataSource;
|
||||||
|
this.configWatcher = configWatcher;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void run(String... args) {
|
public void run(String... args) {
|
||||||
Optional<User> existing = userRepository.findByUsername("admin");
|
log.info("Initializing databases for {} tenant(s)...", routingDataSource.getTenantConfigs().size());
|
||||||
|
|
||||||
if (existing.isEmpty()) {
|
for (TenantConfig tenant : routingDataSource.getTenantConfigs().values()) {
|
||||||
User admin = new User();
|
configWatcher.initDatabaseForTenant(tenant);
|
||||||
admin.setUsername("admin");
|
|
||||||
admin.setPassword(passwordEncoder.encode("admin"));
|
|
||||||
admin.setRole(Role.ADMIN);
|
|
||||||
userRepository.save(admin);
|
|
||||||
log.info("Created default admin user");
|
|
||||||
} else {
|
|
||||||
User admin = existing.get();
|
|
||||||
if (!passwordEncoder.matches("admin", admin.getPassword())) {
|
|
||||||
admin.setPassword(passwordEncoder.encode("admin"));
|
|
||||||
admin.setRole(Role.ADMIN);
|
|
||||||
userRepository.save(admin);
|
|
||||||
log.info("Reset admin password (hash was invalid)");
|
|
||||||
} else {
|
|
||||||
log.info("Admin user already exists with correct password");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
log.info("Database initialization complete");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
127
backend/src/main/java/com/magistr/app/config/tenant/ConfigMapUpdater.java
Executable file
127
backend/src/main/java/com/magistr/app/config/tenant/ConfigMapUpdater.java
Executable file
@@ -0,0 +1,127 @@
|
|||||||
|
package com.magistr.app.config.tenant;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.core.JsonProcessingException;
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.URI;
|
||||||
|
import java.net.http.HttpClient;
|
||||||
|
import java.net.http.HttpRequest;
|
||||||
|
import java.net.http.HttpResponse;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
import java.security.SecureRandom;
|
||||||
|
import java.security.cert.X509Certificate;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import javax.net.ssl.SSLContext;
|
||||||
|
import javax.net.ssl.TrustManager;
|
||||||
|
import javax.net.ssl.X509TrustManager;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Обновляет K8s ConfigMap tenants-config через Kubernetes REST API.
|
||||||
|
*
|
||||||
|
* Работает ТОЛЬКО внутри K8s пода (использует ServiceAccount token).
|
||||||
|
* При запуске вне K8s (локальная разработка) — просто логирует предупреждение.
|
||||||
|
*/
|
||||||
|
@Service
|
||||||
|
public class ConfigMapUpdater {
|
||||||
|
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(ConfigMapUpdater.class);
|
||||||
|
|
||||||
|
private static final String TOKEN_PATH = "/var/run/secrets/kubernetes.io/serviceaccount/token";
|
||||||
|
private static final String NAMESPACE_PATH = "/var/run/secrets/kubernetes.io/serviceaccount/namespace";
|
||||||
|
private static final String K8S_API_BASE = "https://kubernetes.default.svc";
|
||||||
|
private static final String CONFIGMAP_NAME = "tenants-config";
|
||||||
|
|
||||||
|
private final ObjectMapper objectMapper = new ObjectMapper();
|
||||||
|
private final boolean runningInK8s;
|
||||||
|
|
||||||
|
public ConfigMapUpdater() {
|
||||||
|
this.runningInK8s = Files.exists(Path.of(TOKEN_PATH));
|
||||||
|
if (!runningInK8s) {
|
||||||
|
log.info("Not running in K8s — ConfigMap updates will be skipped");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Обновляет ConfigMap tenants-config с новым списком тенантов.
|
||||||
|
* @return true если обновление успешно (или мы не в K8s)
|
||||||
|
*/
|
||||||
|
public boolean updateTenantsConfig(List<TenantConfig> tenants) {
|
||||||
|
if (!runningInK8s) {
|
||||||
|
log.warn("Not in K8s, skipping ConfigMap update");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
String token = Files.readString(Path.of(TOKEN_PATH)).trim();
|
||||||
|
String namespace = Files.readString(Path.of(NAMESPACE_PATH)).trim();
|
||||||
|
|
||||||
|
// Формируем JSON для тенантов
|
||||||
|
String tenantsJson = objectMapper.writerWithDefaultPrettyPrinter()
|
||||||
|
.writeValueAsString(tenants);
|
||||||
|
|
||||||
|
// Strategic merge patch для ConfigMap
|
||||||
|
String patchBody = objectMapper.writeValueAsString(Map.of(
|
||||||
|
"data", Map.of("tenants.json", tenantsJson)
|
||||||
|
));
|
||||||
|
|
||||||
|
String url = String.format("%s/api/v1/namespaces/%s/configmaps/%s",
|
||||||
|
K8S_API_BASE, namespace, CONFIGMAP_NAME);
|
||||||
|
|
||||||
|
// Создаём HttpClient с отключённой проверкой сертификатов
|
||||||
|
// (внутри кластера используется self-signed CA)
|
||||||
|
HttpClient client = createInsecureClient();
|
||||||
|
|
||||||
|
HttpRequest request = HttpRequest.newBuilder()
|
||||||
|
.uri(URI.create(url))
|
||||||
|
.header("Authorization", "Bearer " + token)
|
||||||
|
.header("Content-Type", "application/strategic-merge-patch+json")
|
||||||
|
.method("PATCH", HttpRequest.BodyPublishers.ofString(patchBody))
|
||||||
|
.build();
|
||||||
|
|
||||||
|
HttpResponse<String> response = client.send(request, HttpResponse.BodyHandlers.ofString());
|
||||||
|
|
||||||
|
if (response.statusCode() == 200) {
|
||||||
|
log.info("ConfigMap '{}' updated successfully ({} tenants)", CONFIGMAP_NAME, tenants.size());
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
log.error("Failed to update ConfigMap: HTTP {} — {}", response.statusCode(), response.body());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("Error updating ConfigMap: {}", e.getMessage());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Создаёт HttpClient, который доверяет self-signed сертификатам K8s API.
|
||||||
|
*/
|
||||||
|
private HttpClient createInsecureClient() {
|
||||||
|
try {
|
||||||
|
TrustManager[] trustAll = new TrustManager[]{
|
||||||
|
new X509TrustManager() {
|
||||||
|
public X509Certificate[] getAcceptedIssuers() { return new X509Certificate[0]; }
|
||||||
|
public void checkClientTrusted(X509Certificate[] certs, String authType) {}
|
||||||
|
public void checkServerTrusted(X509Certificate[] certs, String authType) {}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
SSLContext sslContext = SSLContext.getInstance("TLS");
|
||||||
|
sslContext.init(null, trustAll, new SecureRandom());
|
||||||
|
|
||||||
|
return HttpClient.newBuilder()
|
||||||
|
.sslContext(sslContext)
|
||||||
|
.build();
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.warn("Failed to create insecure client, using default: {}", e.getMessage());
|
||||||
|
return HttpClient.newHttpClient();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
41
backend/src/main/java/com/magistr/app/config/tenant/TenantConfig.java
Executable file
41
backend/src/main/java/com/magistr/app/config/tenant/TenantConfig.java
Executable file
@@ -0,0 +1,41 @@
|
|||||||
|
package com.magistr.app.config.tenant;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Модель конфигурации тенанта (университета).
|
||||||
|
*/
|
||||||
|
@JsonIgnoreProperties(ignoreUnknown = true)
|
||||||
|
public class TenantConfig {
|
||||||
|
|
||||||
|
private String name; // "ЮЗГУ", "МГУ"
|
||||||
|
private String domain; // "swsu", "mgu" (поддомен)
|
||||||
|
private String url; // "jdbc:postgresql://192.168.1.50:5432/magistr_db"
|
||||||
|
private String username;
|
||||||
|
private String password;
|
||||||
|
|
||||||
|
public TenantConfig() {}
|
||||||
|
|
||||||
|
public TenantConfig(String name, String domain, String url, String username, String password) {
|
||||||
|
this.name = name;
|
||||||
|
this.domain = domain;
|
||||||
|
this.url = url;
|
||||||
|
this.username = username;
|
||||||
|
this.password = password;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() { return name; }
|
||||||
|
public void setName(String name) { this.name = name; }
|
||||||
|
|
||||||
|
public String getDomain() { return domain; }
|
||||||
|
public void setDomain(String domain) { this.domain = domain; }
|
||||||
|
|
||||||
|
public String getUrl() { return url; }
|
||||||
|
public void setUrl(String url) { this.url = url; }
|
||||||
|
|
||||||
|
public String getUsername() { return username; }
|
||||||
|
public void setUsername(String username) { this.username = username; }
|
||||||
|
|
||||||
|
public String getPassword() { return password; }
|
||||||
|
public void setPassword(String password) { this.password = password; }
|
||||||
|
}
|
||||||
156
backend/src/main/java/com/magistr/app/config/tenant/TenantConfigWatcher.java
Executable file
156
backend/src/main/java/com/magistr/app/config/tenant/TenantConfigWatcher.java
Executable file
@@ -0,0 +1,156 @@
|
|||||||
|
package com.magistr.app.config.tenant;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.core.type.TypeReference;
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.beans.factory.annotation.Value;
|
||||||
|
import org.springframework.core.io.ClassPathResource;
|
||||||
|
import org.springframework.scheduling.annotation.Scheduled;
|
||||||
|
import org.springframework.stereotype.Component;
|
||||||
|
|
||||||
|
import javax.sql.DataSource;
|
||||||
|
import java.io.BufferedReader;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.InputStreamReader;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.sql.Connection;
|
||||||
|
import java.sql.ResultSet;
|
||||||
|
import java.sql.Statement;
|
||||||
|
import java.util.*;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Периодически перечитывает tenants.json (mounted ConfigMap).
|
||||||
|
* Если ConfigMap был обновлён через K8s API, этот компонент
|
||||||
|
* подхватит изменения и синхронизирует in-memory datasource'ы.
|
||||||
|
*
|
||||||
|
* Также отвечает за инициализацию БД (init.sql) для новых тенантов.
|
||||||
|
*/
|
||||||
|
@Component
|
||||||
|
public class TenantConfigWatcher {
|
||||||
|
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(TenantConfigWatcher.class);
|
||||||
|
|
||||||
|
private final TenantRoutingDataSource routingDataSource;
|
||||||
|
private final DataSource dataSource;
|
||||||
|
private final ObjectMapper objectMapper = new ObjectMapper();
|
||||||
|
|
||||||
|
@Value("${app.tenants.config-path:tenants.json}")
|
||||||
|
private String tenantsConfigPath;
|
||||||
|
|
||||||
|
// Хеш последнего прочитанного конфига — чтобы не перезагружать зря
|
||||||
|
private String lastConfigHash = "";
|
||||||
|
|
||||||
|
public TenantConfigWatcher(TenantRoutingDataSource routingDataSource, DataSource dataSource) {
|
||||||
|
this.routingDataSource = routingDataSource;
|
||||||
|
this.dataSource = dataSource;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Каждые 30 секунд проверяет, изменился ли tenants.json.
|
||||||
|
*/
|
||||||
|
@Scheduled(fixedDelay = 30_000, initialDelay = 30_000)
|
||||||
|
public void watchForChanges() {
|
||||||
|
try {
|
||||||
|
File file = new File(tenantsConfigPath);
|
||||||
|
if (!file.exists()) return;
|
||||||
|
|
||||||
|
String content = new String(java.nio.file.Files.readAllBytes(file.toPath()), StandardCharsets.UTF_8);
|
||||||
|
String hash = Integer.toHexString(content.hashCode());
|
||||||
|
|
||||||
|
if (hash.equals(lastConfigHash)) {
|
||||||
|
return; // Ничего не изменилось
|
||||||
|
}
|
||||||
|
|
||||||
|
log.info("Detected tenants.json change (hash: {} -> {}), reloading...", lastConfigHash, hash);
|
||||||
|
lastConfigHash = hash;
|
||||||
|
|
||||||
|
List<TenantConfig> newTenants = objectMapper.readValue(content, new TypeReference<>() {});
|
||||||
|
syncTenants(newTenants);
|
||||||
|
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("Error watching tenants config: {}", e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Обновляет хеш конфига (вызывается после ручного обновления ConfigMap с этого же пода).
|
||||||
|
*/
|
||||||
|
public void refreshHash() {
|
||||||
|
try {
|
||||||
|
File file = new File(tenantsConfigPath);
|
||||||
|
if (file.exists()) {
|
||||||
|
String content = new String(java.nio.file.Files.readAllBytes(file.toPath()), StandardCharsets.UTF_8);
|
||||||
|
lastConfigHash = Integer.toHexString(content.hashCode());
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.warn("Failed to refresh config hash: {}", e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Синхронизирует in-memory тенантов с конфигом из файла.
|
||||||
|
*/
|
||||||
|
private void syncTenants(List<TenantConfig> newTenants) {
|
||||||
|
Map<String, TenantConfig> current = routingDataSource.getTenantConfigs();
|
||||||
|
Set<String> newDomains = newTenants.stream()
|
||||||
|
.map(t -> t.getDomain().toLowerCase())
|
||||||
|
.collect(Collectors.toSet());
|
||||||
|
|
||||||
|
// Добавить новые тенанты
|
||||||
|
for (TenantConfig tenant : newTenants) {
|
||||||
|
String domain = tenant.getDomain().toLowerCase();
|
||||||
|
if (!current.containsKey(domain)) {
|
||||||
|
log.info("Adding new tenant '{}' from ConfigMap update", domain);
|
||||||
|
routingDataSource.addTenant(tenant);
|
||||||
|
// Инициализируем БД для нового тенанта
|
||||||
|
initDatabaseForTenant(tenant);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Удалить тенанты, которых больше нет в конфиге
|
||||||
|
for (String existingDomain : new ArrayList<>(current.keySet())) {
|
||||||
|
if (!newDomains.contains(existingDomain)) {
|
||||||
|
log.info("Removing tenant '{}' (no longer in ConfigMap)", existingDomain);
|
||||||
|
routingDataSource.removeTenant(existingDomain);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Выполняет миграции Flyway для конкретного тенанта пи подключении.
|
||||||
|
* Если БД уже существует, но история Flyway пуста —
|
||||||
|
* делает baseline (считает V1_init.sql уже выполненным).
|
||||||
|
*/
|
||||||
|
public void initDatabaseForTenant(TenantConfig tenant) {
|
||||||
|
String domain = tenant.getDomain();
|
||||||
|
try {
|
||||||
|
TenantContext.setCurrentTenant(domain);
|
||||||
|
|
||||||
|
log.info("[{}] Starting Flyway migrations...", domain);
|
||||||
|
|
||||||
|
// Получаем DataSource конкретно для этого тенанта
|
||||||
|
javax.sql.DataSource tenantDs = routingDataSource.getResolvedDataSources().get(domain);
|
||||||
|
if (tenantDs == null) {
|
||||||
|
// Если ещё не resolve'нулся (первый запуск), берём обёртку
|
||||||
|
tenantDs = dataSource;
|
||||||
|
}
|
||||||
|
|
||||||
|
org.flywaydb.core.Flyway flyway = org.flywaydb.core.Flyway.configure()
|
||||||
|
.dataSource(tenantDs)
|
||||||
|
.baselineOnMigrate(true)
|
||||||
|
.baselineVersion("1")
|
||||||
|
.load();
|
||||||
|
|
||||||
|
flyway.migrate();
|
||||||
|
log.info("[{}] Flyway migrations completed successfully", domain);
|
||||||
|
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("[{}] Flyway migration failed: {}", domain, e.getMessage());
|
||||||
|
} finally {
|
||||||
|
TenantContext.clear();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
22
backend/src/main/java/com/magistr/app/config/tenant/TenantContext.java
Executable file
22
backend/src/main/java/com/magistr/app/config/tenant/TenantContext.java
Executable file
@@ -0,0 +1,22 @@
|
|||||||
|
package com.magistr.app.config.tenant;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ThreadLocal хранилище текущего тенанта (домена).
|
||||||
|
* Устанавливается в TenantInterceptor на каждый HTTP-запрос.
|
||||||
|
*/
|
||||||
|
public class TenantContext {
|
||||||
|
|
||||||
|
private static final ThreadLocal<String> CURRENT_TENANT = new ThreadLocal<>();
|
||||||
|
|
||||||
|
public static String getCurrentTenant() {
|
||||||
|
return CURRENT_TENANT.get();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void setCurrentTenant(String tenant) {
|
||||||
|
CURRENT_TENANT.set(tenant);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void clear() {
|
||||||
|
CURRENT_TENANT.remove();
|
||||||
|
}
|
||||||
|
}
|
||||||
156
backend/src/main/java/com/magistr/app/config/tenant/TenantDataSourceConfig.java
Executable file
156
backend/src/main/java/com/magistr/app/config/tenant/TenantDataSourceConfig.java
Executable file
@@ -0,0 +1,156 @@
|
|||||||
|
package com.magistr.app.config.tenant;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.core.type.TypeReference;
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.beans.factory.annotation.Value;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
import org.springframework.context.annotation.Primary;
|
||||||
|
import org.springframework.orm.jpa.JpaTransactionManager;
|
||||||
|
import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean;
|
||||||
|
import org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter;
|
||||||
|
import org.springframework.transaction.PlatformTransactionManager;
|
||||||
|
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
|
||||||
|
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
|
||||||
|
|
||||||
|
import jakarta.persistence.EntityManagerFactory;
|
||||||
|
import javax.sql.DataSource;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Конфигурация мультитенантного DataSource.
|
||||||
|
* Загружает тенанты из JSON-файла (mounted ConfigMap).
|
||||||
|
*
|
||||||
|
* Если нет ни одного настроенного тенанта — создаёт H2 in-memory БД
|
||||||
|
* как заглушку, чтобы Spring JPA мог инициализироваться.
|
||||||
|
*/
|
||||||
|
@Configuration
|
||||||
|
public class TenantDataSourceConfig implements WebMvcConfigurer {
|
||||||
|
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(TenantDataSourceConfig.class);
|
||||||
|
|
||||||
|
@Value("${app.tenants.config-path:tenants.json}")
|
||||||
|
private String tenantsConfigPath;
|
||||||
|
|
||||||
|
@Value("${spring.datasource.url:}")
|
||||||
|
private String defaultDbUrl;
|
||||||
|
|
||||||
|
@Value("${spring.datasource.username:}")
|
||||||
|
private String defaultDbUsername;
|
||||||
|
|
||||||
|
@Value("${spring.datasource.password:}")
|
||||||
|
private String defaultDbPassword;
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@Primary
|
||||||
|
public DataSource dataSource() {
|
||||||
|
TenantRoutingDataSource routingDataSource = new TenantRoutingDataSource();
|
||||||
|
|
||||||
|
// Загружаем тенантов из JSON (read-only ConfigMap mount)
|
||||||
|
List<TenantConfig> tenants = loadTenantsFromFile();
|
||||||
|
|
||||||
|
// Если нет тенантов и есть дефолтный datasource — создаём "default" тенант
|
||||||
|
if (tenants.isEmpty() && defaultDbUrl != null && !defaultDbUrl.isBlank()) {
|
||||||
|
TenantConfig defaultTenant = new TenantConfig(
|
||||||
|
"Default", "default", defaultDbUrl, defaultDbUsername, defaultDbPassword
|
||||||
|
);
|
||||||
|
tenants.add(defaultTenant);
|
||||||
|
log.info("No tenants config found, using default datasource: {}", defaultDbUrl);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Регистрируем тенантов
|
||||||
|
for (TenantConfig tenant : tenants) {
|
||||||
|
try {
|
||||||
|
routingDataSource.addTenant(tenant);
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("Failed to add tenant '{}': {}", tenant.getDomain(), e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Если всё ещё нет ни одного тенанта — H2 in-memory заглушка
|
||||||
|
if (routingDataSource.getTenantConfigs().isEmpty()) {
|
||||||
|
log.warn("=== НЕТ НАСТРОЕННЫХ ТЕНАНТОВ ===");
|
||||||
|
log.warn("Создаём H2 in-memory заглушку для запуска приложения.");
|
||||||
|
log.warn("Добавьте тенант через POST /api/database/tenants");
|
||||||
|
|
||||||
|
TenantConfig h2Fallback = new TenantConfig(
|
||||||
|
"H2 Placeholder", "default",
|
||||||
|
"jdbc:h2:mem:placeholder;DB_CLOSE_DELAY=-1",
|
||||||
|
"sa", ""
|
||||||
|
);
|
||||||
|
routingDataSource.addTenant(h2Fallback);
|
||||||
|
}
|
||||||
|
|
||||||
|
return routingDataSource;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public TenantRoutingDataSource tenantRoutingDataSource(DataSource dataSource) {
|
||||||
|
return (TenantRoutingDataSource) dataSource;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@Primary
|
||||||
|
public LocalContainerEntityManagerFactoryBean entityManagerFactory(DataSource dataSource) {
|
||||||
|
LocalContainerEntityManagerFactoryBean em = new LocalContainerEntityManagerFactoryBean();
|
||||||
|
em.setDataSource(dataSource);
|
||||||
|
em.setPackagesToScan("com.magistr.app.model");
|
||||||
|
|
||||||
|
HibernateJpaVendorAdapter vendorAdapter = new HibernateJpaVendorAdapter();
|
||||||
|
vendorAdapter.setGenerateDdl(false);
|
||||||
|
vendorAdapter.setDatabasePlatform("org.hibernate.dialect.PostgreSQLDialect");
|
||||||
|
em.setJpaVendorAdapter(vendorAdapter);
|
||||||
|
|
||||||
|
Map<String, Object> props = new HashMap<>();
|
||||||
|
props.put("hibernate.hbm2ddl.auto", "none");
|
||||||
|
props.put("hibernate.show_sql", "false");
|
||||||
|
em.setJpaPropertyMap(props);
|
||||||
|
|
||||||
|
return em;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@Primary
|
||||||
|
public PlatformTransactionManager transactionManager(EntityManagerFactory emf) {
|
||||||
|
return new JpaTransactionManager(emf);
|
||||||
|
}
|
||||||
|
|
||||||
|
@org.springframework.context.annotation.Lazy
|
||||||
|
@org.springframework.beans.factory.annotation.Autowired
|
||||||
|
private TenantRoutingDataSource tenantRoutingDataSource;
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public TenantInterceptor tenantInterceptor(TenantRoutingDataSource routingDataSource) {
|
||||||
|
TenantInterceptor interceptor = new TenantInterceptor();
|
||||||
|
interceptor.setRoutingDataSource(routingDataSource);
|
||||||
|
return interceptor;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void addInterceptors(InterceptorRegistry registry) {
|
||||||
|
// Вызываем метод-бин с переданным параметром (будет перехвачен CGLIB)
|
||||||
|
registry.addInterceptor(tenantInterceptor(tenantRoutingDataSource)).addPathPatterns("/**");
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<TenantConfig> loadTenantsFromFile() {
|
||||||
|
File file = new File(tenantsConfigPath);
|
||||||
|
if (!file.exists()) {
|
||||||
|
log.info("Tenants config file not found: {}", tenantsConfigPath);
|
||||||
|
return new ArrayList<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
ObjectMapper mapper = new ObjectMapper();
|
||||||
|
List<TenantConfig> list = mapper.readValue(file, new TypeReference<>() {});
|
||||||
|
log.info("Loaded {} tenant(s) from {}", list.size(), tenantsConfigPath);
|
||||||
|
return list;
|
||||||
|
} catch (IOException e) {
|
||||||
|
log.error("Failed to read tenants config: {}", e.getMessage());
|
||||||
|
return new ArrayList<>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
106
backend/src/main/java/com/magistr/app/config/tenant/TenantInterceptor.java
Executable file
106
backend/src/main/java/com/magistr/app/config/tenant/TenantInterceptor.java
Executable file
@@ -0,0 +1,106 @@
|
|||||||
|
package com.magistr.app.config.tenant;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
|
import jakarta.servlet.http.HttpServletRequest;
|
||||||
|
import jakarta.servlet.http.HttpServletResponse;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.web.servlet.HandlerInterceptor;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Map;
|
||||||
|
import org.slf4j.MDC;
|
||||||
|
import io.opentelemetry.api.trace.Span;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interceptor: извлекает поддомен из Host header и кладёт в TenantContext.
|
||||||
|
*
|
||||||
|
* Если тенант не настроен в TenantRoutingDataSource —
|
||||||
|
* сразу возвращает HTTP 404 (не допускает fallback на чужой тенант).
|
||||||
|
*
|
||||||
|
* Примеры:
|
||||||
|
* "swsu.zuev.company" → tenant = "swsu"
|
||||||
|
* "mgu.zuev.company" → tenant = "mgu"
|
||||||
|
* "localhost" → tenant = "default"
|
||||||
|
* "localhost:8080" → tenant = "default"
|
||||||
|
*
|
||||||
|
* API управления тенантами (/api/database/**) пропускается без проверки,
|
||||||
|
* чтобы администратор мог добавлять тенантов с любого домена.
|
||||||
|
*/
|
||||||
|
public class TenantInterceptor implements HandlerInterceptor {
|
||||||
|
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(TenantInterceptor.class);
|
||||||
|
|
||||||
|
private TenantRoutingDataSource routingDataSource;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Устанавливается после создания бина (из TenantDataSourceConfig).
|
||||||
|
*/
|
||||||
|
public void setRoutingDataSource(TenantRoutingDataSource routingDataSource) {
|
||||||
|
this.routingDataSource = routingDataSource;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws IOException {
|
||||||
|
String host = request.getHeader("Host");
|
||||||
|
String tenant = resolveTenant(host);
|
||||||
|
String path = request.getRequestURI();
|
||||||
|
|
||||||
|
// API управления тенантами — всегда пропускаем
|
||||||
|
// (нужно чтобы админ мог добавить тенант даже если его домен не настроен)
|
||||||
|
if (path.startsWith("/api/database")) {
|
||||||
|
TenantContext.setCurrentTenant(tenant);
|
||||||
|
MDC.put("tenant.id", tenant);
|
||||||
|
Span.current().setAttribute("tenant.id", tenant);
|
||||||
|
log.debug("Database API request, tenant '{}' (no strict check)", tenant);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Проверяем, существует ли тенант
|
||||||
|
if (routingDataSource != null && !routingDataSource.hasTenant(tenant)) {
|
||||||
|
log.warn("Unknown tenant '{}' from Host '{}' — returning 404", tenant, host);
|
||||||
|
response.setStatus(404);
|
||||||
|
response.setContentType("application/json;charset=UTF-8");
|
||||||
|
new ObjectMapper().writeValue(response.getOutputStream(), Map.of(
|
||||||
|
"error", "Тенант не найден",
|
||||||
|
"tenant", tenant,
|
||||||
|
"message", "Домен " + host + " не настроен. Обратитесь к администратору."
|
||||||
|
));
|
||||||
|
return false; // Останавливаем обработку запроса
|
||||||
|
}
|
||||||
|
|
||||||
|
TenantContext.setCurrentTenant(tenant);
|
||||||
|
MDC.put("tenant.id", tenant);
|
||||||
|
Span.current().setAttribute("tenant.id", tenant);
|
||||||
|
log.debug("Resolved tenant '{}' from Host '{}'", tenant, host);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) {
|
||||||
|
TenantContext.clear();
|
||||||
|
MDC.remove("tenant.id");
|
||||||
|
}
|
||||||
|
|
||||||
|
private String resolveTenant(String host) {
|
||||||
|
if (host == null || host.isBlank()) {
|
||||||
|
return "default";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Убираем порт (localhost:8080 → localhost)
|
||||||
|
String hostname = host.contains(":") ? host.substring(0, host.indexOf(':')) : host;
|
||||||
|
|
||||||
|
// localhost или IP → default
|
||||||
|
if ("localhost".equalsIgnoreCase(hostname) || hostname.matches("\\d+\\.\\d+\\.\\d+\\.\\d+")) {
|
||||||
|
return "default";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Извлекаем первый поддомен: swsu.zuev.company → swsu
|
||||||
|
int firstDot = hostname.indexOf('.');
|
||||||
|
if (firstDot > 0) {
|
||||||
|
return hostname.substring(0, firstDot).toLowerCase();
|
||||||
|
}
|
||||||
|
|
||||||
|
return "default";
|
||||||
|
}
|
||||||
|
}
|
||||||
150
backend/src/main/java/com/magistr/app/config/tenant/TenantRoutingDataSource.java
Executable file
150
backend/src/main/java/com/magistr/app/config/tenant/TenantRoutingDataSource.java
Executable file
@@ -0,0 +1,150 @@
|
|||||||
|
package com.magistr.app.config.tenant;
|
||||||
|
|
||||||
|
import com.zaxxer.hikari.HikariDataSource;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;
|
||||||
|
|
||||||
|
import javax.sql.DataSource;
|
||||||
|
import java.sql.Connection;
|
||||||
|
import java.sql.SQLException;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* DataSource, который переключается между БД разных тенантов.
|
||||||
|
* На каждый запрос determineCurrentLookupKey() возвращает текущий тенант из TenantContext.
|
||||||
|
*/
|
||||||
|
public class TenantRoutingDataSource extends AbstractRoutingDataSource {
|
||||||
|
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(TenantRoutingDataSource.class);
|
||||||
|
|
||||||
|
private final Map<String, TenantConfig> tenantConfigs = new ConcurrentHashMap<>();
|
||||||
|
private final Map<Object, Object> dataSources = new ConcurrentHashMap<>();
|
||||||
|
private boolean initialized = false;
|
||||||
|
|
||||||
|
public TenantRoutingDataSource() {
|
||||||
|
// Устанавливаем пустой map чтобы afterPropertiesSet не падал
|
||||||
|
setTargetDataSources(new HashMap<>());
|
||||||
|
setLenientFallback(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Object determineCurrentLookupKey() {
|
||||||
|
String tenant = TenantContext.getCurrentTenant();
|
||||||
|
|
||||||
|
if (tenant == null) {
|
||||||
|
// Нет HTTP контекста (JPA init, background tasks) — берём первый доступный
|
||||||
|
if (!dataSources.isEmpty()) {
|
||||||
|
return dataSources.keySet().iterator().next().toString();
|
||||||
|
}
|
||||||
|
return "default";
|
||||||
|
}
|
||||||
|
|
||||||
|
// HTTP запрос — возвращаем точный ключ тенанта
|
||||||
|
// Если тенанта нет — TenantInterceptor уже вернул 404
|
||||||
|
return tenant;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Добавляет тенант и создаёт для него HikariCP пул.
|
||||||
|
*/
|
||||||
|
public void addTenant(TenantConfig config) {
|
||||||
|
String domain = config.getDomain().toLowerCase();
|
||||||
|
HikariDataSource ds = createDataSource(config);
|
||||||
|
|
||||||
|
dataSources.put(domain, ds);
|
||||||
|
tenantConfigs.put(domain, config);
|
||||||
|
|
||||||
|
// Обновляем target data sources
|
||||||
|
setTargetDataSources(dataSources);
|
||||||
|
afterPropertiesSet();
|
||||||
|
initialized = true;
|
||||||
|
|
||||||
|
log.info("Added tenant '{}' -> {}", domain, config.getUrl());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Удаляет тенант и закрывает его пул соединений.
|
||||||
|
*/
|
||||||
|
public void removeTenant(String domain) {
|
||||||
|
domain = domain.toLowerCase();
|
||||||
|
Object removed = dataSources.remove(domain);
|
||||||
|
tenantConfigs.remove(domain);
|
||||||
|
|
||||||
|
if (removed instanceof HikariDataSource ds) {
|
||||||
|
ds.close();
|
||||||
|
log.info("Removed and closed tenant '{}'", domain);
|
||||||
|
}
|
||||||
|
|
||||||
|
setTargetDataSources(dataSources);
|
||||||
|
afterPropertiesSet();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Проверяет подключение к БД для указанного тенанта.
|
||||||
|
*/
|
||||||
|
public boolean testConnection(String domain) {
|
||||||
|
DataSource ds = (DataSource) dataSources.get(domain.toLowerCase());
|
||||||
|
if (ds == null) return false;
|
||||||
|
|
||||||
|
try (Connection conn = ds.getConnection()) {
|
||||||
|
return conn.isValid(5);
|
||||||
|
} catch (SQLException e) {
|
||||||
|
log.warn("Connection test failed for tenant '{}': {}", domain, e.getMessage());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Тестирует подключение по произвольным параметрам (без регистрации тенанта).
|
||||||
|
*/
|
||||||
|
public String testExternalConnection(String url, String username, String password) {
|
||||||
|
HikariDataSource ds = new HikariDataSource();
|
||||||
|
ds.setJdbcUrl(url);
|
||||||
|
ds.setUsername(username);
|
||||||
|
ds.setPassword(password);
|
||||||
|
ds.setMaximumPoolSize(1);
|
||||||
|
ds.setConnectionTimeout(5000);
|
||||||
|
|
||||||
|
try (Connection conn = ds.getConnection()) {
|
||||||
|
if (conn.isValid(5)) {
|
||||||
|
return "OK";
|
||||||
|
}
|
||||||
|
return "Подключение не валидно";
|
||||||
|
} catch (Exception e) {
|
||||||
|
return e.getMessage();
|
||||||
|
} finally {
|
||||||
|
ds.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public Map<String, TenantConfig> getTenantConfigs() {
|
||||||
|
return tenantConfigs;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean hasTenant(String domain) {
|
||||||
|
return tenantConfigs.containsKey(domain.toLowerCase());
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isInitialized() {
|
||||||
|
return initialized && !dataSources.isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
private HikariDataSource createDataSource(TenantConfig config) {
|
||||||
|
HikariDataSource ds = new HikariDataSource();
|
||||||
|
ds.setJdbcUrl(config.getUrl());
|
||||||
|
ds.setUsername(config.getUsername());
|
||||||
|
ds.setPassword(config.getPassword());
|
||||||
|
ds.setPoolName("tenant-" + config.getDomain());
|
||||||
|
ds.setMaximumPoolSize(10);
|
||||||
|
ds.setMinimumIdle(2);
|
||||||
|
ds.setConnectionTimeout(10000);
|
||||||
|
ds.setIdleTimeout(300000);
|
||||||
|
ds.setMaxLifetime(600000);
|
||||||
|
// Не падать при инициализации если БД недоступна
|
||||||
|
ds.setInitializationFailTimeout(-1);
|
||||||
|
return ds;
|
||||||
|
}
|
||||||
|
}
|
||||||
181
backend/src/main/java/com/magistr/app/controller/DatabaseController.java
Executable file
181
backend/src/main/java/com/magistr/app/controller/DatabaseController.java
Executable file
@@ -0,0 +1,181 @@
|
|||||||
|
package com.magistr.app.controller;
|
||||||
|
|
||||||
|
import com.magistr.app.config.tenant.ConfigMapUpdater;
|
||||||
|
import com.magistr.app.config.tenant.TenantConfig;
|
||||||
|
import com.magistr.app.config.tenant.TenantConfigWatcher;
|
||||||
|
import com.magistr.app.config.tenant.TenantContext;
|
||||||
|
import com.magistr.app.config.tenant.TenantRoutingDataSource;
|
||||||
|
import org.springframework.http.ResponseEntity;
|
||||||
|
import org.springframework.web.bind.annotation.*;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* API управления подключениями к базам данных (тенантами).
|
||||||
|
* Доступно только для ADMIN.
|
||||||
|
*
|
||||||
|
* При добавлении/удалении тенанта:
|
||||||
|
* 1. Обновляется in-memory DataSource (мгновенно на этом поде)
|
||||||
|
* 2. Обновляется K8s ConfigMap (через ConfigMapUpdater)
|
||||||
|
* 3. Другие поды подхватят изменения через TenantConfigWatcher (~30 сек)
|
||||||
|
*/
|
||||||
|
@RestController
|
||||||
|
@RequestMapping("/api/database")
|
||||||
|
public class DatabaseController {
|
||||||
|
|
||||||
|
private final TenantRoutingDataSource routingDataSource;
|
||||||
|
private final ConfigMapUpdater configMapUpdater;
|
||||||
|
private final TenantConfigWatcher configWatcher;
|
||||||
|
|
||||||
|
public DatabaseController(TenantRoutingDataSource routingDataSource,
|
||||||
|
ConfigMapUpdater configMapUpdater,
|
||||||
|
TenantConfigWatcher configWatcher) {
|
||||||
|
this.routingDataSource = routingDataSource;
|
||||||
|
this.configMapUpdater = configMapUpdater;
|
||||||
|
this.configWatcher = configWatcher;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Статус текущего подключения (по домену запроса).
|
||||||
|
*/
|
||||||
|
@GetMapping("/status")
|
||||||
|
public ResponseEntity<Map<String, Object>> getStatus() {
|
||||||
|
String currentTenant = TenantContext.getCurrentTenant();
|
||||||
|
boolean connected = routingDataSource.testConnection(currentTenant);
|
||||||
|
|
||||||
|
TenantConfig config = routingDataSource.getTenantConfigs().get(currentTenant);
|
||||||
|
|
||||||
|
Map<String, Object> result = new HashMap<>();
|
||||||
|
result.put("tenant", currentTenant);
|
||||||
|
result.put("connected", connected);
|
||||||
|
result.put("configured", config != null);
|
||||||
|
if (config != null) {
|
||||||
|
result.put("name", config.getName());
|
||||||
|
result.put("url", config.getUrl());
|
||||||
|
}
|
||||||
|
return ResponseEntity.ok(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Список всех тенантов.
|
||||||
|
*/
|
||||||
|
@GetMapping("/tenants")
|
||||||
|
public ResponseEntity<List<Map<String, Object>>> getTenants() {
|
||||||
|
List<Map<String, Object>> result = new ArrayList<>();
|
||||||
|
|
||||||
|
for (TenantConfig config : routingDataSource.getTenantConfigs().values()) {
|
||||||
|
Map<String, Object> tenant = new HashMap<>();
|
||||||
|
tenant.put("name", config.getName());
|
||||||
|
tenant.put("domain", config.getDomain());
|
||||||
|
tenant.put("url", config.getUrl());
|
||||||
|
tenant.put("username", config.getUsername());
|
||||||
|
tenant.put("connected", routingDataSource.testConnection(config.getDomain()));
|
||||||
|
result.add(tenant);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ResponseEntity.ok(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Добавить новый тенант.
|
||||||
|
*/
|
||||||
|
@PostMapping("/tenants")
|
||||||
|
public ResponseEntity<Map<String, Object>> addTenant(@RequestBody TenantConfig config) {
|
||||||
|
Map<String, Object> result = new HashMap<>();
|
||||||
|
|
||||||
|
if (config.getDomain() == null || config.getDomain().isBlank()) {
|
||||||
|
result.put("success", false);
|
||||||
|
result.put("message", "Домен не может быть пустым");
|
||||||
|
return ResponseEntity.badRequest().body(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (config.getUrl() == null || config.getUrl().isBlank()) {
|
||||||
|
result.put("success", false);
|
||||||
|
result.put("message", "URL базы данных не может быть пустым");
|
||||||
|
return ResponseEntity.badRequest().body(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (routingDataSource.hasTenant(config.getDomain())) {
|
||||||
|
routingDataSource.removeTenant(config.getDomain());
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// 1. Добавить в in-memory (мгновенно на этом поде)
|
||||||
|
routingDataSource.addTenant(config);
|
||||||
|
|
||||||
|
// 2. Инициализировать БД (init.sql) если нужно
|
||||||
|
configWatcher.initDatabaseForTenant(config);
|
||||||
|
|
||||||
|
// 3. Обновить K8s ConfigMap (другие поды подхватят через ~30 сек)
|
||||||
|
persistToConfigMap();
|
||||||
|
|
||||||
|
result.put("success", true);
|
||||||
|
result.put("message", "Тенант '" + config.getDomain() + "' добавлен");
|
||||||
|
return ResponseEntity.ok(result);
|
||||||
|
} catch (Exception e) {
|
||||||
|
result.put("success", false);
|
||||||
|
result.put("message", "Ошибка: " + e.getMessage());
|
||||||
|
return ResponseEntity.internalServerError().body(result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Удалить тенант.
|
||||||
|
*/
|
||||||
|
@DeleteMapping("/tenants/{domain}")
|
||||||
|
public ResponseEntity<Map<String, Object>> removeTenant(@PathVariable String domain) {
|
||||||
|
Map<String, Object> result = new HashMap<>();
|
||||||
|
|
||||||
|
if (!routingDataSource.hasTenant(domain)) {
|
||||||
|
result.put("success", false);
|
||||||
|
result.put("message", "Тенант '" + domain + "' не найден");
|
||||||
|
return ResponseEntity.status(404).body(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
routingDataSource.removeTenant(domain);
|
||||||
|
persistToConfigMap();
|
||||||
|
|
||||||
|
result.put("success", true);
|
||||||
|
result.put("message", "Тенант '" + domain + "' удалён");
|
||||||
|
return ResponseEntity.ok(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Тест подключения к произвольной БД.
|
||||||
|
*/
|
||||||
|
@PostMapping("/test")
|
||||||
|
public ResponseEntity<Map<String, Object>> testConnection(@RequestBody Map<String, String> params) {
|
||||||
|
String url = params.get("url");
|
||||||
|
String username = params.get("username");
|
||||||
|
String password = params.get("password");
|
||||||
|
|
||||||
|
Map<String, Object> result = new HashMap<>();
|
||||||
|
|
||||||
|
if (url == null || url.isBlank()) {
|
||||||
|
result.put("success", false);
|
||||||
|
result.put("message", "URL не указан");
|
||||||
|
return ResponseEntity.badRequest().body(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
String testResult = routingDataSource.testExternalConnection(url, username, password);
|
||||||
|
boolean success = "OK".equals(testResult);
|
||||||
|
|
||||||
|
result.put("success", success);
|
||||||
|
result.put("message", success ? "Подключение успешно!" : testResult);
|
||||||
|
return ResponseEntity.ok(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Сохраняет текущий список тенантов в K8s ConfigMap.
|
||||||
|
*/
|
||||||
|
private void persistToConfigMap() {
|
||||||
|
List<TenantConfig> tenants = new ArrayList<>(routingDataSource.getTenantConfigs().values());
|
||||||
|
boolean ok = configMapUpdater.updateTenantsConfig(tenants);
|
||||||
|
if (ok) {
|
||||||
|
configWatcher.refreshHash(); // Чтобы watcher не перезагрузил те же данные
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,106 @@
|
|||||||
|
package com.magistr.app.controller;
|
||||||
|
|
||||||
|
import com.magistr.app.dto.CreateDepartmentRequest;
|
||||||
|
import com.magistr.app.dto.DepartmentResponse;
|
||||||
|
import com.magistr.app.model.Department;
|
||||||
|
import com.magistr.app.repository.DepartmentRepository;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.http.HttpStatus;
|
||||||
|
import org.springframework.http.ResponseEntity;
|
||||||
|
import org.springframework.web.bind.annotation.*;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
@RestController
|
||||||
|
@RequestMapping("/api/departments")
|
||||||
|
public class DepartmentController {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(DepartmentController.class);
|
||||||
|
|
||||||
|
private final DepartmentRepository departmentRepository;
|
||||||
|
|
||||||
|
public DepartmentController(DepartmentRepository departmentRepository) {
|
||||||
|
this.departmentRepository = departmentRepository;
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping
|
||||||
|
public List<Department> getAllDepartments() {
|
||||||
|
logger.info("Получен запрос на получение списка кафедр");
|
||||||
|
try {
|
||||||
|
List<Department> departments = departmentRepository.findAll();
|
||||||
|
List<Department> response = departments.stream()
|
||||||
|
.map( d -> new Department(
|
||||||
|
d.getId(),
|
||||||
|
d.getDepartmentName(),
|
||||||
|
d.getDepartmentCode()
|
||||||
|
))
|
||||||
|
.toList();
|
||||||
|
logger.info("Получено {} кафедр", response.size());
|
||||||
|
return response;
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Ошибка при получении списка кафедр: {}", e.getMessage(), e);
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping
|
||||||
|
public ResponseEntity<?> createDepartment(@RequestBody CreateDepartmentRequest request) {
|
||||||
|
logger.info("Получен запрос на создание кафедры: name = {}, code = {}", request.getDepartmentName(), request.getDepartmentCode());
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (request.getDepartmentName() == null || request.getDepartmentName().isBlank()){
|
||||||
|
String errorMessage = "Название кафедры обязательно";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (departmentRepository.findByDepartmentName(request.getDepartmentName().trim()).isPresent()) {
|
||||||
|
String errorMessage = "Кафедра с таким названием уже существует";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (request.getDepartmentCode() == null || request.getDepartmentCode() == 0) {
|
||||||
|
String errorMessage = "Код кафедры обязателен";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (departmentRepository.findByDepartmentCode(request.getDepartmentCode()).isPresent()) {
|
||||||
|
String errorMessage = "Кафедра с таким кодом уже существует";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
|
||||||
|
Department department = new Department();
|
||||||
|
department.setDepartmentName(request.getDepartmentName());
|
||||||
|
department.setDepartmentCode(request.getDepartmentCode());
|
||||||
|
departmentRepository.save(department);
|
||||||
|
|
||||||
|
logger.info("Кафедра успешно создана с ID: {}", department.getId());
|
||||||
|
|
||||||
|
return ResponseEntity.ok(
|
||||||
|
new DepartmentResponse(
|
||||||
|
department.getId(),
|
||||||
|
department.getDepartmentName(),
|
||||||
|
department.getDepartmentCode()
|
||||||
|
)
|
||||||
|
);
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Ошибка при создании кафедры: {}", e.getMessage(), e);
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR)
|
||||||
|
.body(Map.of("message", "Произошла ошибка при создании кафедры " + e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@DeleteMapping("/id")
|
||||||
|
public ResponseEntity<?> deleteDepartment(@PathVariable Long id) {
|
||||||
|
logger.info("Получен запрос на удаление кафедры с ID: {}", id);
|
||||||
|
if (!departmentRepository.existsById(id)) {
|
||||||
|
logger.info("Кафедра с ID - {} не найдена", id);
|
||||||
|
return ResponseEntity.notFound().build();
|
||||||
|
}
|
||||||
|
departmentRepository.deleteById(id);
|
||||||
|
logger.info("Кафедра с ID - {} успешно удалена", id);
|
||||||
|
return ResponseEntity.ok(Map.of("message", "Кафедра удалена"));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -6,6 +6,9 @@ import com.magistr.app.model.EducationForm;
|
|||||||
import com.magistr.app.model.StudentGroup;
|
import com.magistr.app.model.StudentGroup;
|
||||||
import com.magistr.app.repository.EducationFormRepository;
|
import com.magistr.app.repository.EducationFormRepository;
|
||||||
import com.magistr.app.repository.GroupRepository;
|
import com.magistr.app.repository.GroupRepository;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.http.HttpStatus;
|
||||||
import org.springframework.http.ResponseEntity;
|
import org.springframework.http.ResponseEntity;
|
||||||
import org.springframework.web.bind.annotation.*;
|
import org.springframework.web.bind.annotation.*;
|
||||||
|
|
||||||
@@ -17,6 +20,8 @@ import java.util.Optional;
|
|||||||
@RequestMapping("/api/groups")
|
@RequestMapping("/api/groups")
|
||||||
public class GroupController {
|
public class GroupController {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(GroupController.class);
|
||||||
|
|
||||||
private final GroupRepository groupRepository;
|
private final GroupRepository groupRepository;
|
||||||
private final EducationFormRepository educationFormRepository;
|
private final EducationFormRepository educationFormRepository;
|
||||||
|
|
||||||
@@ -28,56 +33,127 @@ public class GroupController {
|
|||||||
|
|
||||||
@GetMapping
|
@GetMapping
|
||||||
public List<GroupResponse> getAllGroups() {
|
public List<GroupResponse> getAllGroups() {
|
||||||
return groupRepository.findAll().stream()
|
logger.info("Получен запрос на получение всех групп");
|
||||||
.map(g -> new GroupResponse(
|
|
||||||
|
try {
|
||||||
|
List<StudentGroup> groups = groupRepository.findAll();
|
||||||
|
|
||||||
|
List<GroupResponse> response = groups.stream()
|
||||||
|
.map(g -> new GroupResponse(
|
||||||
g.getId(),
|
g.getId(),
|
||||||
g.getName(),
|
g.getName(),
|
||||||
g.getGroupSize(),
|
g.getGroupSize(),
|
||||||
g.getEducationForm().getId(),
|
g.getEducationForm().getId(),
|
||||||
g.getEducationForm().getName()))
|
g.getEducationForm().getName(),
|
||||||
.toList();
|
g.getDepartmentId(),
|
||||||
|
g.getCourse()
|
||||||
|
))
|
||||||
|
.toList();
|
||||||
|
logger.info("Получено {} групп", response.size());
|
||||||
|
return response;
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Ошибка при получении списка групп: {}", e.getMessage(), e);
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping("/{departmentId}")
|
||||||
|
public ResponseEntity<?> getGroupsByDepartmentId(@PathVariable Long departmentId) {
|
||||||
|
logger.info("Получен запрос на получение списка групп для кафедры с ID - {}", departmentId);
|
||||||
|
try {
|
||||||
|
List<StudentGroup> groups = groupRepository.findByDepartmentId(departmentId);
|
||||||
|
|
||||||
|
if(groups.isEmpty()) {
|
||||||
|
logger.info("Группы для кафедры с ID - {} не найдены", departmentId);
|
||||||
|
return ResponseEntity.status(HttpStatus.NOT_FOUND)
|
||||||
|
.body("Группы для указанной кафедры не найдены");
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.info("Найдено {} групп для кафедры с ID - {}", groups.size(), departmentId);
|
||||||
|
|
||||||
|
return ResponseEntity.ok(groups);
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Получена ошибка при получении списка групп для кафедры с ID - {}: {}", departmentId, e.getMessage());
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR)
|
||||||
|
.body("Произошла ошибка при получении списка групп");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@PostMapping
|
@PostMapping
|
||||||
public ResponseEntity<?> createGroup(@RequestBody CreateGroupRequest request) {
|
public ResponseEntity<?> createGroup(@RequestBody CreateGroupRequest request) {
|
||||||
if (request.getName() == null || request.getName().isBlank()) {
|
logger.info("Получен запрос на создание новой группы: name = {}, groupSize = {}, educationFormId = {}, departmentId = {}, course = {}",
|
||||||
return ResponseEntity.badRequest().body(Map.of("message", "Название группы обязательно"));
|
request.getName(), request.getGroupSize(), request.getEducationFormId(), request.getDepartmentId(), request.getCourse());
|
||||||
}
|
try {
|
||||||
if (groupRepository.findByName(request.getName().trim()).isPresent()) {
|
if (request.getName() == null || request.getName().isBlank()) {
|
||||||
return ResponseEntity.badRequest().body(Map.of("message", "Группа с таким названием уже существует"));
|
String errorMessage = "Название группы обязательно";
|
||||||
}
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
if (request.getGroupSize() == null) {
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
return ResponseEntity.badRequest().body(Map.of("message", "Численность группы обязательна"));
|
}
|
||||||
}
|
if (groupRepository.findByName(request.getName().trim()).isPresent()) {
|
||||||
if (request.getEducationFormId() == null) {
|
String errorMessage = "Группа с таким названием уже существует";
|
||||||
return ResponseEntity.badRequest().body(Map.of("message", "Форма обучения обязательна"));
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
}
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (request.getGroupSize() == null) {
|
||||||
|
String errorMessage = "Численность группы обязательна";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (request.getEducationFormId() == null) {
|
||||||
|
String errorMessage = "Форма обучения обязательна";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (request.getDepartmentId() == null || request.getDepartmentId() == 0) {
|
||||||
|
String errorMessage = "ID кафедры обязателен";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (request.getCourse() == null || request.getCourse() == 0) {
|
||||||
|
String errorMessage = "Курс обязателен";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
|
||||||
Optional<EducationForm> efOpt = educationFormRepository.findById(request.getEducationFormId());
|
Optional<EducationForm> efOpt = educationFormRepository.findById(request.getEducationFormId());
|
||||||
if (efOpt.isEmpty()) {
|
if (efOpt.isEmpty()) {
|
||||||
return ResponseEntity.badRequest().body(Map.of("message", "Форма обучения не найдена"));
|
return ResponseEntity.badRequest().body(Map.of("message", "Форма обучения не найдена"));
|
||||||
|
}
|
||||||
|
|
||||||
|
StudentGroup group = new StudentGroup();
|
||||||
|
group.setName(request.getName().trim());
|
||||||
|
group.setGroupSize(request.getGroupSize());
|
||||||
|
group.setEducationForm(efOpt.get());
|
||||||
|
group.setDepartmentId(request.getDepartmentId());
|
||||||
|
group.setCourse(request.getCourse());
|
||||||
|
groupRepository.save(group);
|
||||||
|
|
||||||
|
logger.info("Группа успешно создана с ID - {}", group.getId());
|
||||||
|
|
||||||
|
return ResponseEntity.ok(new GroupResponse(
|
||||||
|
group.getId(),
|
||||||
|
group.getName(),
|
||||||
|
group.getGroupSize(),
|
||||||
|
group.getEducationForm().getId(),
|
||||||
|
group.getEducationForm().getName(),
|
||||||
|
group.getDepartmentId(),
|
||||||
|
group.getCourse()));
|
||||||
|
} catch (Exception e ) {
|
||||||
|
logger.error("Ошибка при создании группы: {}", e.getMessage(), e);
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR)
|
||||||
|
.body(Map.of("message", "Произошла ошибка при создании группы: " + e.getMessage()));
|
||||||
}
|
}
|
||||||
|
|
||||||
StudentGroup group = new StudentGroup();
|
|
||||||
group.setName(request.getName().trim());
|
|
||||||
group.setGroupSize(request.getGroupSize());
|
|
||||||
group.setEducationForm(efOpt.get());
|
|
||||||
groupRepository.save(group);
|
|
||||||
|
|
||||||
return ResponseEntity.ok(new GroupResponse(
|
|
||||||
group.getId(),
|
|
||||||
group.getName(),
|
|
||||||
group.getGroupSize(),
|
|
||||||
group.getEducationForm().getId(),
|
|
||||||
group.getEducationForm().getName()));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@DeleteMapping("/{id}")
|
@DeleteMapping("/{id}")
|
||||||
public ResponseEntity<?> deleteGroup(@PathVariable Long id) {
|
public ResponseEntity<?> deleteGroup(@PathVariable Long id) {
|
||||||
|
logger.info("Получен запрос на удаление группы с ID - {}", id);
|
||||||
if (!groupRepository.existsById(id)) {
|
if (!groupRepository.existsById(id)) {
|
||||||
|
logger.info("Группа с ID - {} не найдена", id);
|
||||||
return ResponseEntity.notFound().build();
|
return ResponseEntity.notFound().build();
|
||||||
}
|
}
|
||||||
groupRepository.deleteById(id);
|
groupRepository.deleteById(id);
|
||||||
|
logger.info("Группа с ID - {} успешно удалена", id);
|
||||||
return ResponseEntity.ok(Map.of("message", "Группа удалена"));
|
return ResponseEntity.ok(Map.of("message", "Группа удалена"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,54 @@
|
|||||||
|
package com.magistr.app.controller;
|
||||||
|
|
||||||
|
import com.magistr.app.model.Department;
|
||||||
|
import com.magistr.app.model.ScheduleData;
|
||||||
|
import com.magistr.app.repository.DepartmentRepository;
|
||||||
|
import com.magistr.app.repository.ScheduleDataRepository;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.web.bind.annotation.GetMapping;
|
||||||
|
import org.springframework.web.bind.annotation.RequestMapping;
|
||||||
|
import org.springframework.web.bind.annotation.RestController;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
@RestController
|
||||||
|
@RequestMapping("/api/scheduledata")
|
||||||
|
public class ScheduleDataController {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(ScheduleDataController.class);
|
||||||
|
|
||||||
|
private final ScheduleDataRepository scheduleDataRepository;
|
||||||
|
|
||||||
|
public ScheduleDataController(ScheduleDataRepository scheduleDataRepository) {
|
||||||
|
this.scheduleDataRepository = scheduleDataRepository;
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping
|
||||||
|
public List<ScheduleData> getAllScheduleDataList() {
|
||||||
|
logger.info("Получен запрос на получение списка данных расписаний");
|
||||||
|
try {
|
||||||
|
List<ScheduleData> scheduleData = scheduleDataRepository.findAll();
|
||||||
|
List<ScheduleData> response = scheduleData.stream()
|
||||||
|
.map(s -> new ScheduleData(
|
||||||
|
s.getId(),
|
||||||
|
s.getDepartmentId(),
|
||||||
|
s.getSemester(),
|
||||||
|
s.getGroupId(),
|
||||||
|
s.getSubjectsId(),
|
||||||
|
s.getLessonTypeId(),
|
||||||
|
s.getNumberOfHours(),
|
||||||
|
s.getDivision(),
|
||||||
|
s.getTeacherId(),
|
||||||
|
s.getSemesterType(),
|
||||||
|
s.getPeriod()
|
||||||
|
))
|
||||||
|
.toList();
|
||||||
|
logger.info("Получено {} записей", response.size());
|
||||||
|
return response;
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Ошибка при получении списка данных расписаний: {}", e.getMessage(), e);
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,106 @@
|
|||||||
|
package com.magistr.app.controller;
|
||||||
|
|
||||||
|
import com.magistr.app.dto.CreateSpecialityRequest;
|
||||||
|
import com.magistr.app.dto.SpecialityResponse;
|
||||||
|
import com.magistr.app.model.Speciality;
|
||||||
|
import com.magistr.app.repository.SpecialtiesRepository;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.http.HttpStatus;
|
||||||
|
import org.springframework.http.ResponseEntity;
|
||||||
|
import org.springframework.web.bind.annotation.*;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
@RestController
|
||||||
|
@RequestMapping("/api/specialties")
|
||||||
|
public class SpecialityController {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(SpecialityController.class);
|
||||||
|
|
||||||
|
private final SpecialtiesRepository specialtiesRepository;
|
||||||
|
|
||||||
|
public SpecialityController(SpecialtiesRepository specialtiesRepository) {
|
||||||
|
this.specialtiesRepository = specialtiesRepository;
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping
|
||||||
|
public List<Speciality> getAllSpecialties() {
|
||||||
|
logger.info("Получен запрос на получение списка специальностей");
|
||||||
|
try {
|
||||||
|
List<Speciality> specialities = specialtiesRepository.findAll();
|
||||||
|
List<Speciality> response = specialities.stream()
|
||||||
|
.map( s -> new Speciality(
|
||||||
|
s.getId(),
|
||||||
|
s.getSpecialityName(),
|
||||||
|
s.getSpecialityCode()
|
||||||
|
))
|
||||||
|
.toList();
|
||||||
|
logger.info("Получено {} специальностей", response.size());
|
||||||
|
return response;
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Ошибка при получении списка специальностей: {}", e.getMessage(), e);
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping
|
||||||
|
public ResponseEntity<?> createSpeciality(@RequestBody CreateSpecialityRequest request) {
|
||||||
|
logger.info("Получен запрос на создание специальности: name = {}, code = {}", request.getSpecialityName(), request.getSpecialityCode());
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (request.getSpecialityName() == null || request.getSpecialityName().isBlank()) {
|
||||||
|
String errorMessage = "Название специальности обязательно";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (specialtiesRepository.findBySpecialityName(request.getSpecialityName().trim()).isPresent()) {
|
||||||
|
String errorMessage = "Специальность с таким названием уже существует";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (request.getSpecialityCode() == null || request.getSpecialityCode().isBlank()) {
|
||||||
|
String errorMessage = "Код специальности обязателен";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (specialtiesRepository.findBySpecialityCode(request.getSpecialityCode().trim()).isPresent()) {
|
||||||
|
String errorMessage = "Специальность с таким кодом уже существует";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
|
||||||
|
Speciality speciality = new Speciality();
|
||||||
|
speciality.setSpecialityName(request.getSpecialityName());
|
||||||
|
speciality.setSpecialityCode(request.getSpecialityCode());
|
||||||
|
specialtiesRepository.save(speciality);
|
||||||
|
|
||||||
|
logger.info("Специальность успешно создана с ID: {}", speciality.getId());
|
||||||
|
|
||||||
|
return ResponseEntity.ok(
|
||||||
|
new SpecialityResponse(
|
||||||
|
speciality.getId(),
|
||||||
|
speciality.getSpecialityName(),
|
||||||
|
speciality.getSpecialityCode()
|
||||||
|
)
|
||||||
|
);
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Ошибка при создании специальности: {}", e.getMessage(), e);
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR)
|
||||||
|
.body(Map.of("message", "Произошла ошибка при создании специальности " + e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@DeleteMapping("/id")
|
||||||
|
public ResponseEntity<?> deleteSpeciality(@PathVariable Long id) {
|
||||||
|
logger.info("Получен запрос на удаление специальности с ID: {}", id);
|
||||||
|
if (!specialtiesRepository.existsById(id)) {
|
||||||
|
logger.info("Специальность с ID - {} не найдена", id);
|
||||||
|
return ResponseEntity.notFound().build();
|
||||||
|
}
|
||||||
|
specialtiesRepository.deleteById(id);
|
||||||
|
logger.info("Специальность с ID - {} успешно удалена", id);
|
||||||
|
return ResponseEntity.ok(Map.of("message", "Специальнсть удалена"));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,7 +1,13 @@
|
|||||||
package com.magistr.app.controller;
|
package com.magistr.app.controller;
|
||||||
|
|
||||||
|
import com.magistr.app.dto.CreateSubjectRequest;
|
||||||
|
import com.magistr.app.dto.SubjectResponse;
|
||||||
import com.magistr.app.model.Subject;
|
import com.magistr.app.model.Subject;
|
||||||
import com.magistr.app.repository.SubjectRepository;
|
import com.magistr.app.repository.SubjectRepository;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.http.HttpStatus;
|
||||||
|
import org.springframework.http.HttpStatusCode;
|
||||||
import org.springframework.http.ResponseEntity;
|
import org.springframework.http.ResponseEntity;
|
||||||
import org.springframework.web.bind.annotation.*;
|
import org.springframework.web.bind.annotation.*;
|
||||||
|
|
||||||
@@ -12,6 +18,8 @@ import java.util.Map;
|
|||||||
@RequestMapping("/api/subjects")
|
@RequestMapping("/api/subjects")
|
||||||
public class SubjectController {
|
public class SubjectController {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(SubjectController.class);
|
||||||
|
|
||||||
private final SubjectRepository subjectRepository;
|
private final SubjectRepository subjectRepository;
|
||||||
|
|
||||||
public SubjectController(SubjectRepository subjectRepository) {
|
public SubjectController(SubjectRepository subjectRepository) {
|
||||||
@@ -20,32 +28,105 @@ public class SubjectController {
|
|||||||
|
|
||||||
@GetMapping
|
@GetMapping
|
||||||
public List<Subject> getAllSubjects() {
|
public List<Subject> getAllSubjects() {
|
||||||
return subjectRepository.findAll();
|
logger.info("Получен запрос на получение всех дисциплин");
|
||||||
|
try {
|
||||||
|
List<Subject> subjects = subjectRepository.findAll();
|
||||||
|
List<Subject> response = subjects.stream()
|
||||||
|
.map(s -> new Subject(
|
||||||
|
s.getId(),
|
||||||
|
s.getName(),
|
||||||
|
s.getCode(),
|
||||||
|
s.getDepartmentId()
|
||||||
|
))
|
||||||
|
.toList();
|
||||||
|
logger.info("Получено {} дисциплин", response.size());
|
||||||
|
return response;
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Ошибка при получении списка дисциплин: {}", e.getMessage(), e);
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping("/{departmentId}")
|
||||||
|
public ResponseEntity<?> getSubjectsByDepartmentId(@PathVariable Long departmentId) {
|
||||||
|
logger.info("Получен запрос на получение дисциплин для кафедры с ID - {}", departmentId);
|
||||||
|
try{
|
||||||
|
List<Subject> subjects = subjectRepository.findByDepartmentId(departmentId);
|
||||||
|
|
||||||
|
if(subjects.isEmpty()){
|
||||||
|
logger.info("Дисциплины для кафедры с ID - {} не найдены", departmentId);
|
||||||
|
return ResponseEntity.status(HttpStatus.NOT_FOUND)
|
||||||
|
.body("Дисциплины для указанной кафедры не найдены");
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.info("Найдено {} дисциплин для кафедры с ID - {}", subjects.size(), departmentId);
|
||||||
|
return ResponseEntity.ok(subjects);
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Произошла ошибка при получении списка дисциплин для кафедры с ID - {}", departmentId);
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR)
|
||||||
|
.body("Произошла ошибка при получении списка дисциплин");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@PostMapping
|
@PostMapping
|
||||||
public ResponseEntity<?> createSubject(@RequestBody Map<String, String> request) {
|
public ResponseEntity<?> createSubject(@RequestBody CreateSubjectRequest request) {
|
||||||
String name = request.get("name");
|
logger.info("Получен запрос на создание дисциплины: name = {}, code = {}, departmentId = {}",
|
||||||
if (name == null || name.isBlank()) {
|
request.getName(), request.getCode(), request.getDepartmentId());
|
||||||
return ResponseEntity.badRequest().body(Map.of("message", "Название обязательно"));
|
|
||||||
}
|
|
||||||
if (subjectRepository.findByName(name.trim()).isPresent()) {
|
|
||||||
return ResponseEntity.badRequest().body(Map.of("message", "Дисциплина с таким названием уже существует"));
|
|
||||||
}
|
|
||||||
|
|
||||||
Subject subject = new Subject();
|
try {
|
||||||
subject.setName(name.trim());
|
if (request.getName() == null || request.getName().isBlank()) {
|
||||||
subjectRepository.save(subject);
|
String errorMessage = "Название обязательно";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (subjectRepository.findByName(request.getName().trim()).isPresent()) {
|
||||||
|
String errorMessage = "Дисциплина с таким названием уже существует";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (request.getCode() == null || request.getCode().isBlank()) {
|
||||||
|
String errorMessage = "Код дисциплины обязателен";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (request.getDepartmentId() == null || request.getDepartmentId() == 0) {
|
||||||
|
String errorMessage = "ID кафедры не может быть равен 0 или пустым";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
|
||||||
return ResponseEntity.ok(subject);
|
Subject subject = new Subject();
|
||||||
|
subject.setName(request.getName());
|
||||||
|
subject.setCode(request.getCode());
|
||||||
|
subject.setDepartmentId(request.getDepartmentId());
|
||||||
|
subjectRepository.save(subject);
|
||||||
|
|
||||||
|
logger.info("Дисциплина успешно создана с ID: {}", subject.getId());
|
||||||
|
|
||||||
|
return ResponseEntity.ok(
|
||||||
|
new SubjectResponse(
|
||||||
|
subject.getId(),
|
||||||
|
subject.getName(),
|
||||||
|
subject.getCode(),
|
||||||
|
subject.getDepartmentId()
|
||||||
|
)
|
||||||
|
);
|
||||||
|
} catch (Exception e){
|
||||||
|
logger.error("Ошибка при создании дисциплины: {}", e.getMessage(), e);
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR)
|
||||||
|
.body(Map.of("message", "Произошла ошибка при создании дисциплины " + e.getMessage()));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@DeleteMapping("/{id}")
|
@DeleteMapping("/{id}")
|
||||||
public ResponseEntity<?> deleteSubject(@PathVariable Long id) {
|
public ResponseEntity<?> deleteSubject(@PathVariable Long id) {
|
||||||
|
logger.info("Получен запрос на удаление дисциплины с ID: {}", id);
|
||||||
if (!subjectRepository.existsById(id)) {
|
if (!subjectRepository.existsById(id)) {
|
||||||
|
logger.info("Дисциплина с ID - {} не найдена", id);
|
||||||
return ResponseEntity.notFound().build();
|
return ResponseEntity.notFound().build();
|
||||||
}
|
}
|
||||||
subjectRepository.deleteById(id);
|
subjectRepository.deleteById(id);
|
||||||
|
logger.info("Дисциплина с ID - {} успешно удалена", id);
|
||||||
return ResponseEntity.ok(Map.of("message", "Дисциплина удалена"));
|
return ResponseEntity.ok(Map.of("message", "Дисциплина удалена"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,10 @@ import com.magistr.app.dto.UserResponse;
|
|||||||
import com.magistr.app.model.Role;
|
import com.magistr.app.model.Role;
|
||||||
import com.magistr.app.model.User;
|
import com.magistr.app.model.User;
|
||||||
import com.magistr.app.repository.UserRepository;
|
import com.magistr.app.repository.UserRepository;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.http.HttpStatus;
|
||||||
|
import org.springframework.http.HttpStatusCode;
|
||||||
import org.springframework.http.ResponseEntity;
|
import org.springframework.http.ResponseEntity;
|
||||||
import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder;
|
import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder;
|
||||||
import org.springframework.web.bind.annotation.*;
|
import org.springframework.web.bind.annotation.*;
|
||||||
@@ -16,6 +20,7 @@ import java.util.Map;
|
|||||||
@RequestMapping("/api/users")
|
@RequestMapping("/api/users")
|
||||||
public class UserController {
|
public class UserController {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(UserController.class);
|
||||||
private final UserRepository userRepository;
|
private final UserRepository userRepository;
|
||||||
private final BCryptPasswordEncoder passwordEncoder;
|
private final BCryptPasswordEncoder passwordEncoder;
|
||||||
|
|
||||||
@@ -26,52 +31,159 @@ public class UserController {
|
|||||||
|
|
||||||
@GetMapping
|
@GetMapping
|
||||||
public List<UserResponse> getAllUsers() {
|
public List<UserResponse> getAllUsers() {
|
||||||
return userRepository.findAll().stream()
|
logger.info("Получен запрос на получение всех пользователей");
|
||||||
.map(u -> new UserResponse(u.getId(), u.getUsername(), u.getRole().name()))
|
try {
|
||||||
.toList();
|
List<User> users = userRepository.findAll();
|
||||||
|
|
||||||
|
List<UserResponse> response = users.stream()
|
||||||
|
.map(u -> new UserResponse(
|
||||||
|
u.getId(),
|
||||||
|
u.getUsername(),
|
||||||
|
u.getRole().name(),
|
||||||
|
u.getFullName(),
|
||||||
|
u.getJobTitle(),
|
||||||
|
u.getDepartmentId()
|
||||||
|
))
|
||||||
|
.toList();
|
||||||
|
logger.info("Получено {} пользователей", response.size());
|
||||||
|
return response;
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Ошибка при получении списка пользователей: {}", e.getMessage(),e);
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@GetMapping("/teachers")
|
@GetMapping("/teachers")
|
||||||
public List<UserResponse> getTeachers() {
|
public List<UserResponse> getTeachers() {
|
||||||
return userRepository.findByRole(Role.TEACHER).stream()
|
logger.info("Запрос на получение пользователей с ролью 'Преподаватель'");
|
||||||
.map(u -> new UserResponse(u.getId(), u.getUsername(), u.getRole().name()))
|
|
||||||
.toList();
|
try {
|
||||||
|
List<User> users = userRepository.findByRole(Role.TEACHER);
|
||||||
|
|
||||||
|
List<UserResponse> response = users.stream()
|
||||||
|
.map(u -> new UserResponse(
|
||||||
|
u.getId(),
|
||||||
|
u.getUsername(),
|
||||||
|
u.getRole().name(),
|
||||||
|
u.getFullName(),
|
||||||
|
u.getJobTitle(),
|
||||||
|
u.getDepartmentId()
|
||||||
|
))
|
||||||
|
.toList();
|
||||||
|
logger.info("Получено {} преподавателей", response.size());
|
||||||
|
return response;
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Ошибка при получении списка преподавателей: {}", e.getMessage(),e);
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping("/teachers/{departmentId}")
|
||||||
|
public ResponseEntity<?> getTeachersByDepartmentId(@PathVariable Long departmentId){
|
||||||
|
logger.info("Получен запрос на получение преподавателей для кафедры с ID - {}", departmentId);
|
||||||
|
try {
|
||||||
|
List<User> users = userRepository.findByRoleAndDepartmentId(Role.TEACHER, departmentId);
|
||||||
|
|
||||||
|
if (users.isEmpty()) {
|
||||||
|
logger.info("Преподаватели для кафедры с ID - {} не найдены", departmentId);
|
||||||
|
return ResponseEntity.status(HttpStatus.NOT_FOUND)
|
||||||
|
.body("Преподаватели для указанной кафедры не найдены");
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.info("Найдено {} преподавателей для кафедры с ID - {}", users.size(), departmentId);
|
||||||
|
|
||||||
|
List<UserResponse> userResponses = users.stream()
|
||||||
|
.map( user -> {
|
||||||
|
|
||||||
|
return new UserResponse(
|
||||||
|
user.getId(),
|
||||||
|
user.getRole().name(),
|
||||||
|
user.getFullName(),
|
||||||
|
user.getJobTitle(),
|
||||||
|
user.getDepartmentId()
|
||||||
|
);
|
||||||
|
}).toList();
|
||||||
|
|
||||||
|
return ResponseEntity.ok(userResponses);
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Произошла ошибка при получении списка преподавателей для кафедры с ID - {}: {}",departmentId, e.getMessage());
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR)
|
||||||
|
.body("Произошла ошибка при получении списка преподавателей");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@PostMapping
|
@PostMapping
|
||||||
public ResponseEntity<?> createUser(@RequestBody CreateUserRequest request) {
|
public ResponseEntity<?> createUser(@RequestBody CreateUserRequest request) {
|
||||||
if (request.getUsername() == null || request.getUsername().isBlank()) {
|
logger.info("Получен запрос на создание нового пользователя: username = {}, fullName = {}, jobTitle = {}, departmentId = {}", request.getUsername(), request.getFullName(), request.getJobTitle(), request.getDepartmentId());
|
||||||
return ResponseEntity.badRequest().body(Map.of("message", "Имя пользователя обязательно"));
|
|
||||||
}
|
|
||||||
if (request.getPassword() == null || request.getPassword().length() < 4) {
|
|
||||||
return ResponseEntity.badRequest().body(Map.of("message", "Пароль минимум 4 символа"));
|
|
||||||
}
|
|
||||||
if (userRepository.findByUsername(request.getUsername()).isPresent()) {
|
|
||||||
return ResponseEntity.badRequest().body(Map.of("message", "Пользователь уже существует"));
|
|
||||||
}
|
|
||||||
|
|
||||||
Role role;
|
|
||||||
try {
|
try {
|
||||||
role = Role.valueOf(request.getRole());
|
if (request.getUsername() == null || request.getUsername().isBlank()) {
|
||||||
} catch (Exception e) {
|
String errorMessage = "Имя пользователя обязательно";
|
||||||
return ResponseEntity.badRequest().body(Map.of("message", "Недопустимая роль"));
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (request.getPassword() == null || request.getPassword().length() < 4) {
|
||||||
|
String errorMessage = "Пароль минимум 4 символа";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (userRepository.findByUsername(request.getUsername()).isPresent()) {
|
||||||
|
String errorMessage = "Пользователь уже существует";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (request.getFullName() == null || request.getFullName().isBlank()) {
|
||||||
|
String errorMessage = "Имя пользователя обязательно";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
if (request.getJobTitle() == null || request.getJobTitle().isBlank()) {
|
||||||
|
logger.info("Должность не была указана, установлено значение по умолчанию: 'Не указано'");
|
||||||
|
request.setJobTitle("Не указано");
|
||||||
|
}
|
||||||
|
if (request.getDepartmentId() == null || request.getDepartmentId() == 0) {
|
||||||
|
String errorMessage = "ID кафедры не может быть равен 0 или пустым";
|
||||||
|
logger.error("Ошибка валидации: {}", errorMessage);
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", errorMessage));
|
||||||
|
}
|
||||||
|
|
||||||
|
Role role;
|
||||||
|
try {
|
||||||
|
role = Role.valueOf(request.getRole());
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.error("Ошибка при преобразовании роли: {}", e.getMessage());
|
||||||
|
return ResponseEntity.badRequest().body(Map.of("message", "Недопустимая роль"));
|
||||||
|
}
|
||||||
|
|
||||||
|
User user = new User();
|
||||||
|
user.setUsername(request.getUsername());
|
||||||
|
user.setPassword(passwordEncoder.encode(request.getPassword()));
|
||||||
|
user.setRole(role);
|
||||||
|
user.setFullName(request.getFullName());
|
||||||
|
user.setJobTitle(request.getJobTitle());
|
||||||
|
user.setDepartmentId(request.getDepartmentId());
|
||||||
|
userRepository.save(user);
|
||||||
|
|
||||||
|
logger.info("Пользователь успешно создан с ID: {}", user.getId());
|
||||||
|
|
||||||
|
return ResponseEntity.ok(new UserResponse(user.getId(), user.getUsername(), user.getRole().name(), user.getFullName(), user.getJobTitle(), user.getDepartmentId()));
|
||||||
|
} catch (Exception e ) {
|
||||||
|
logger.error("Ошибка при создании пользователя: {}", e.getMessage(), e);
|
||||||
|
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR)
|
||||||
|
.body(Map.of("message", "Произошла ошибка при создании пользователя: " + e.getMessage()));
|
||||||
}
|
}
|
||||||
|
|
||||||
User user = new User();
|
|
||||||
user.setUsername(request.getUsername());
|
|
||||||
user.setPassword(passwordEncoder.encode(request.getPassword()));
|
|
||||||
user.setRole(role);
|
|
||||||
userRepository.save(user);
|
|
||||||
|
|
||||||
return ResponseEntity.ok(new UserResponse(user.getId(), user.getUsername(), user.getRole().name()));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@DeleteMapping("/{id}")
|
@DeleteMapping("/{id}")
|
||||||
public ResponseEntity<?> deleteUser(@PathVariable Long id) {
|
public ResponseEntity<?> deleteUser(@PathVariable Long id) {
|
||||||
|
logger.info("Получен запрос на удаление пользователя с ID: {}", id);
|
||||||
if (!userRepository.existsById(id)) {
|
if (!userRepository.existsById(id)) {
|
||||||
|
logger.info("Пользователь с ID - {} не найден", id);
|
||||||
return ResponseEntity.notFound().build();
|
return ResponseEntity.notFound().build();
|
||||||
}
|
}
|
||||||
userRepository.deleteById(id);
|
userRepository.deleteById(id);
|
||||||
|
logger.info("Пользователь с ID - {} успешно удалён", id);
|
||||||
return ResponseEntity.ok(Map.of("message", "Пользователь удалён"));
|
return ResponseEntity.ok(Map.of("message", "Пользователь удалён"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,25 @@
|
|||||||
|
package com.magistr.app.dto;
|
||||||
|
|
||||||
|
public class CreateDepartmentRequest {
|
||||||
|
|
||||||
|
private String departmentName;
|
||||||
|
private Long departmentCode;
|
||||||
|
|
||||||
|
public CreateDepartmentRequest() {}
|
||||||
|
|
||||||
|
public String getDepartmentName() {
|
||||||
|
return departmentName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentName(String departmentName) {
|
||||||
|
this.departmentName = departmentName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentCode() {
|
||||||
|
return departmentCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentCode(Long departmentCode) {
|
||||||
|
this.departmentCode = departmentCode;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,6 +5,8 @@ public class CreateGroupRequest {
|
|||||||
private String name;
|
private String name;
|
||||||
private Long groupSize;
|
private Long groupSize;
|
||||||
private Long educationFormId;
|
private Long educationFormId;
|
||||||
|
private Long departmentId;
|
||||||
|
private Integer course;
|
||||||
|
|
||||||
public String getName() {
|
public String getName() {
|
||||||
return name;
|
return name;
|
||||||
@@ -29,4 +31,20 @@ public class CreateGroupRequest {
|
|||||||
public void setEducationFormId(Long educationFormId) {
|
public void setEducationFormId(Long educationFormId) {
|
||||||
this.educationFormId = educationFormId;
|
this.educationFormId = educationFormId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentId() {
|
||||||
|
return departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentId(Long departmentId) {
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Integer getCourse() {
|
||||||
|
return course;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCourse(Integer course) {
|
||||||
|
this.course = course;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,103 @@
|
|||||||
|
package com.magistr.app.dto;
|
||||||
|
|
||||||
|
public class CreateScheduleDataRequest {
|
||||||
|
private Long id;
|
||||||
|
private Long departmentId;
|
||||||
|
private Long semester;
|
||||||
|
private Long groupId;
|
||||||
|
private Long subjectsId;
|
||||||
|
private Long lessonTypeId;
|
||||||
|
private Long numberOfHours;
|
||||||
|
private Boolean isDivision;
|
||||||
|
private Long teacherId;
|
||||||
|
private String semesterType;
|
||||||
|
private String period;
|
||||||
|
|
||||||
|
public Long getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(Long id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentId() {
|
||||||
|
return departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentId(Long departmentId) {
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getSemester() {
|
||||||
|
return semester;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSemester(Long semester) {
|
||||||
|
this.semester = semester;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getGroupId() {
|
||||||
|
return groupId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setGroupId(Long groupId) {
|
||||||
|
this.groupId = groupId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getSubjectsId() {
|
||||||
|
return subjectsId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSubjectsId(Long subjectsId) {
|
||||||
|
this.subjectsId = subjectsId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getLessonTypeId() {
|
||||||
|
return lessonTypeId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setLessonTypeId(Long lessonTypeId) {
|
||||||
|
this.lessonTypeId = lessonTypeId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getNumberOfHours() {
|
||||||
|
return numberOfHours;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setNumberOfHours(Long numberOfHours) {
|
||||||
|
this.numberOfHours = numberOfHours;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Boolean getDivision() {
|
||||||
|
return isDivision;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDivision(Boolean division) {
|
||||||
|
isDivision = division;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getTeacherId() {
|
||||||
|
return teacherId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setTeacherId(Long teacherId) {
|
||||||
|
this.teacherId = teacherId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSemesterType() {
|
||||||
|
return semesterType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSemesterType(String semesterType) {
|
||||||
|
this.semesterType = semesterType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getPeriod() {
|
||||||
|
return period;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setPeriod(String period) {
|
||||||
|
this.period = period;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,25 @@
|
|||||||
|
package com.magistr.app.dto;
|
||||||
|
|
||||||
|
public class CreateSpecialityRequest {
|
||||||
|
|
||||||
|
private String specialityName;
|
||||||
|
private String specialityCode;
|
||||||
|
|
||||||
|
public CreateSpecialityRequest() {}
|
||||||
|
|
||||||
|
public String getSpecialityName() {
|
||||||
|
return specialityName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSpecialityName(String specialityName) {
|
||||||
|
this.specialityName = specialityName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSpecialityCode() {
|
||||||
|
return specialityCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSpecialityCode(String specialityCode) {
|
||||||
|
this.specialityCode = specialityCode;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,50 @@
|
|||||||
|
package com.magistr.app.dto;
|
||||||
|
|
||||||
|
public class CreateSubjectRequest {
|
||||||
|
|
||||||
|
private Long id;
|
||||||
|
private String name;
|
||||||
|
private String code;
|
||||||
|
private Long departmentId;
|
||||||
|
|
||||||
|
public CreateSubjectRequest() {};
|
||||||
|
|
||||||
|
public CreateSubjectRequest(Long id, String name, String code, Long departmentId) {
|
||||||
|
this.id = id;
|
||||||
|
this.name = name;
|
||||||
|
this.code = code;
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(Long id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setName(String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getCode() {
|
||||||
|
return code;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCode(String code) {
|
||||||
|
this.code = code;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentId() {
|
||||||
|
return departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentId(Long departmentId) {
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,6 +5,9 @@ public class CreateUserRequest {
|
|||||||
private String username;
|
private String username;
|
||||||
private String password;
|
private String password;
|
||||||
private String role;
|
private String role;
|
||||||
|
private String fullName;
|
||||||
|
private String jobTitle;
|
||||||
|
private Long departmentId;
|
||||||
|
|
||||||
public CreateUserRequest() {
|
public CreateUserRequest() {
|
||||||
}
|
}
|
||||||
@@ -32,4 +35,28 @@ public class CreateUserRequest {
|
|||||||
public void setRole(String role) {
|
public void setRole(String role) {
|
||||||
this.role = role;
|
this.role = role;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getFullName() {
|
||||||
|
return fullName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setFullName(String fullName) {
|
||||||
|
this.fullName = fullName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getJobTitle() {
|
||||||
|
return jobTitle;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setJobTitle(String jobTitle) {
|
||||||
|
this.jobTitle = jobTitle;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentId() {
|
||||||
|
return departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentId(Long departmentId) {
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,38 @@
|
|||||||
|
package com.magistr.app.dto;
|
||||||
|
|
||||||
|
public class DepartmentResponse {
|
||||||
|
|
||||||
|
private Long id;
|
||||||
|
private String departmentName;
|
||||||
|
private Long departmentCode;
|
||||||
|
|
||||||
|
public DepartmentResponse(Long id, String departmentName, Long departmentCode) {
|
||||||
|
this.id = id;
|
||||||
|
this.departmentName = departmentName;
|
||||||
|
this.departmentCode = departmentCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(Long id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getDepartmentName() {
|
||||||
|
return departmentName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentName(String departmentName) {
|
||||||
|
this.departmentName = departmentName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentCode() {
|
||||||
|
return departmentCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentCode(Long departmentCode) {
|
||||||
|
this.departmentCode = departmentCode;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -7,13 +7,17 @@ public class GroupResponse {
|
|||||||
private Long groupSize;
|
private Long groupSize;
|
||||||
private Long educationFormId;
|
private Long educationFormId;
|
||||||
private String educationFormName;
|
private String educationFormName;
|
||||||
|
private Long departmentId;
|
||||||
|
private Integer course;
|
||||||
|
|
||||||
public GroupResponse(Long id, String name, Long groupSize, Long educationFormId, String educationFormName) {
|
public GroupResponse(Long id, String name, Long groupSize, Long educationFormId, String educationFormName, Long departmentId, Integer course) {
|
||||||
this.id = id;
|
this.id = id;
|
||||||
this.name = name;
|
this.name = name;
|
||||||
this.groupSize = groupSize;
|
this.groupSize = groupSize;
|
||||||
this.educationFormId = educationFormId;
|
this.educationFormId = educationFormId;
|
||||||
this.educationFormName = educationFormName;
|
this.educationFormName = educationFormName;
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
this.course = course;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Long getId() {
|
public Long getId() {
|
||||||
@@ -35,4 +39,12 @@ public class GroupResponse {
|
|||||||
public String getEducationFormName() {
|
public String getEducationFormName() {
|
||||||
return educationFormName;
|
return educationFormName;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentId() {
|
||||||
|
return departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Integer getCourse() {
|
||||||
|
return course;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
120
backend/src/main/java/com/magistr/app/dto/ScheduleResponse.java
Normal file
120
backend/src/main/java/com/magistr/app/dto/ScheduleResponse.java
Normal file
@@ -0,0 +1,120 @@
|
|||||||
|
package com.magistr.app.dto;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonInclude;
|
||||||
|
|
||||||
|
@JsonInclude(JsonInclude.Include.NON_NULL)
|
||||||
|
public class ScheduleResponse {
|
||||||
|
private Long id;
|
||||||
|
private Long departmentId;
|
||||||
|
private Long semester;
|
||||||
|
private Long groupId;
|
||||||
|
private Long subjectsId;
|
||||||
|
private Long lessonTypeId;
|
||||||
|
private Long numberOfHours;
|
||||||
|
private Boolean isDivision;
|
||||||
|
private Long teacherId;
|
||||||
|
private String semesterType;
|
||||||
|
private String period;
|
||||||
|
|
||||||
|
public ScheduleResponse(Long id, Long departmentId, Long semester, Long groupId, Long subjectsId, Long lessonTypeId, Long numberOfHours, Boolean isDivision, Long teacherId, String semesterType, String period) {
|
||||||
|
this.id = id;
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
this.semester = semester;
|
||||||
|
this.groupId = groupId;
|
||||||
|
this.subjectsId = subjectsId;
|
||||||
|
this.lessonTypeId = lessonTypeId;
|
||||||
|
this.numberOfHours = numberOfHours;
|
||||||
|
this.isDivision = isDivision;
|
||||||
|
this.teacherId = teacherId;
|
||||||
|
this.semesterType = semesterType;
|
||||||
|
this.period = period;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(Long id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentId() {
|
||||||
|
return departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentId(Long departmentId) {
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getSemester() {
|
||||||
|
return semester;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSemester(Long semester) {
|
||||||
|
this.semester = semester;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getGroupId() {
|
||||||
|
return groupId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setGroupId(Long groupId) {
|
||||||
|
this.groupId = groupId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getSubjectsId() {
|
||||||
|
return subjectsId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSubjectsId(Long subjectsId) {
|
||||||
|
this.subjectsId = subjectsId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getLessonTypeId() {
|
||||||
|
return lessonTypeId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setLessonTypeId(Long lessonTypeId) {
|
||||||
|
this.lessonTypeId = lessonTypeId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getNumberOfHours() {
|
||||||
|
return numberOfHours;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setNumberOfHours(Long numberOfHours) {
|
||||||
|
this.numberOfHours = numberOfHours;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Boolean getDivision() {
|
||||||
|
return isDivision;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDivision(Boolean division) {
|
||||||
|
isDivision = division;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getTeacherId() {
|
||||||
|
return teacherId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setTeacherId(Long teacherId) {
|
||||||
|
this.teacherId = teacherId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSemesterType() {
|
||||||
|
return semesterType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSemesterType(String semesterType) {
|
||||||
|
this.semesterType = semesterType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getPeriod() {
|
||||||
|
return period;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setPeriod(String period) {
|
||||||
|
this.period = period;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,38 @@
|
|||||||
|
package com.magistr.app.dto;
|
||||||
|
|
||||||
|
public class SpecialityResponse {
|
||||||
|
|
||||||
|
private Long id;
|
||||||
|
private String specialityName;
|
||||||
|
private String specialityCode;
|
||||||
|
|
||||||
|
public SpecialityResponse(Long id, String specialityName, String specialityCode) {
|
||||||
|
this.id = id;
|
||||||
|
this.specialityName = specialityName;
|
||||||
|
this.specialityCode = specialityCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(Long id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSpecialityName() {
|
||||||
|
return specialityName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSpecialityName(String specialityName) {
|
||||||
|
this.specialityName = specialityName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSpecialityCode() {
|
||||||
|
return specialityCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSpecialityCode(String specialityCode) {
|
||||||
|
this.specialityCode = specialityCode;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
package com.magistr.app.dto;
|
||||||
|
|
||||||
|
public class SubjectResponse {
|
||||||
|
|
||||||
|
private Long id;
|
||||||
|
private String name;
|
||||||
|
private String code;
|
||||||
|
private Long departmentId;
|
||||||
|
|
||||||
|
public SubjectResponse() {};
|
||||||
|
|
||||||
|
public SubjectResponse(Long id, String name, String code, Long departmentId) {
|
||||||
|
this.id = id;
|
||||||
|
this.name = name;
|
||||||
|
this.code = code;
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getCode() {
|
||||||
|
return code;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentId() {
|
||||||
|
return departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -1,18 +1,35 @@
|
|||||||
package com.magistr.app.dto;
|
package com.magistr.app.dto;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonInclude;
|
||||||
|
|
||||||
|
@JsonInclude(JsonInclude.Include.NON_NULL)
|
||||||
public class UserResponse {
|
public class UserResponse {
|
||||||
|
|
||||||
private Long id;
|
private Long id;
|
||||||
private String username;
|
private String username;
|
||||||
private String role;
|
private String role;
|
||||||
|
private String fullName;
|
||||||
|
private String jobTitle;
|
||||||
|
private Long departmentId;
|
||||||
|
|
||||||
public UserResponse() {
|
public UserResponse() {
|
||||||
}
|
}
|
||||||
|
|
||||||
public UserResponse(Long id, String username, String role) {
|
public UserResponse(Long id, String username, String role, String fullName, String jobTitle, Long departmentId) {
|
||||||
this.id = id;
|
this.id = id;
|
||||||
this.username = username;
|
this.username = username;
|
||||||
this.role = role;
|
this.role = role;
|
||||||
|
this.fullName = fullName;
|
||||||
|
this.jobTitle = jobTitle;
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public UserResponse(Long id, String role, String fullName, String jobTitle, Long departmentId) {
|
||||||
|
this.id = id;
|
||||||
|
this.role = role;
|
||||||
|
this.fullName = fullName;
|
||||||
|
this.jobTitle = jobTitle;
|
||||||
|
this.departmentId = departmentId;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Long getId() {
|
public Long getId() {
|
||||||
@@ -38,4 +55,28 @@ public class UserResponse {
|
|||||||
public void setRole(String role) {
|
public void setRole(String role) {
|
||||||
this.role = role;
|
this.role = role;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getFullName() {
|
||||||
|
return fullName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setFullName(String fullName) {
|
||||||
|
this.fullName = fullName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getJobTitle() {
|
||||||
|
return jobTitle;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setJobTitle(String jobTitle) {
|
||||||
|
this.jobTitle = jobTitle;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentId() {
|
||||||
|
return departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentId(Long departmentId) {
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
50
backend/src/main/java/com/magistr/app/model/Department.java
Normal file
50
backend/src/main/java/com/magistr/app/model/Department.java
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
package com.magistr.app.model;
|
||||||
|
|
||||||
|
import jakarta.persistence.*;
|
||||||
|
|
||||||
|
@Entity
|
||||||
|
@Table(name="departments")
|
||||||
|
public class Department {
|
||||||
|
|
||||||
|
@Id
|
||||||
|
@GeneratedValue(strategy = GenerationType.IDENTITY)
|
||||||
|
private Long id;
|
||||||
|
|
||||||
|
@Column(name = "name", nullable = false)
|
||||||
|
private String departmentName;
|
||||||
|
|
||||||
|
@Column(name = "code", nullable = false)
|
||||||
|
private Long departmentCode;
|
||||||
|
|
||||||
|
public Department() {}
|
||||||
|
|
||||||
|
public Department(Long id, String departmentName, Long departmentCode) {
|
||||||
|
this.id = id;
|
||||||
|
this.departmentName = departmentName;
|
||||||
|
this.departmentCode = departmentCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(Long id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getDepartmentName() {
|
||||||
|
return departmentName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentName(String departmentName) {
|
||||||
|
this.departmentName = departmentName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentCode() {
|
||||||
|
return departmentCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentCode(Long departmentCode) {
|
||||||
|
this.departmentCode = departmentCode;
|
||||||
|
}
|
||||||
|
}
|
||||||
146
backend/src/main/java/com/magistr/app/model/ScheduleData.java
Normal file
146
backend/src/main/java/com/magistr/app/model/ScheduleData.java
Normal file
@@ -0,0 +1,146 @@
|
|||||||
|
package com.magistr.app.model;
|
||||||
|
|
||||||
|
import jakarta.persistence.*;
|
||||||
|
|
||||||
|
@Entity
|
||||||
|
@Table(name="schedule_data")
|
||||||
|
public class ScheduleData {
|
||||||
|
|
||||||
|
@Id
|
||||||
|
@GeneratedValue(strategy = GenerationType.IDENTITY)
|
||||||
|
private Long id;
|
||||||
|
|
||||||
|
@Column(name="department_id", nullable = false)
|
||||||
|
private Long departmentId;
|
||||||
|
|
||||||
|
@Column(name="semester", nullable = false)
|
||||||
|
private Long semester;
|
||||||
|
|
||||||
|
@Column(name="group_id", nullable = false)
|
||||||
|
private Long groupId;
|
||||||
|
|
||||||
|
@Column(name="subjects_id", nullable = false)
|
||||||
|
private Long subjectsId;
|
||||||
|
|
||||||
|
@Column(name="lesson_type_id", nullable = false)
|
||||||
|
private Long lessonTypeId;
|
||||||
|
|
||||||
|
@Column(name="number_of_hours", nullable = false)
|
||||||
|
private Long numberOfHours;
|
||||||
|
|
||||||
|
@Column(name="is_division", nullable = false)
|
||||||
|
private Boolean isDivision;
|
||||||
|
|
||||||
|
@Column(name="teacher_id", nullable = false)
|
||||||
|
private Long teacherId;
|
||||||
|
|
||||||
|
@Column(name="semester_type", nullable = false)
|
||||||
|
private String semesterType;
|
||||||
|
|
||||||
|
@Column(name="period", nullable = false)
|
||||||
|
private String period;
|
||||||
|
|
||||||
|
public ScheduleData() {}
|
||||||
|
|
||||||
|
public ScheduleData(Long id, Long departmentId, Long semester, Long groupId, Long subjectsId, Long lessonTypeId, Long numberOfHours, Boolean isDivision, Long teacherId, String semesterType, String period) {
|
||||||
|
this.id = id;
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
this.semester = semester;
|
||||||
|
this.groupId = groupId;
|
||||||
|
this.subjectsId = subjectsId;
|
||||||
|
this.lessonTypeId = lessonTypeId;
|
||||||
|
this.numberOfHours = numberOfHours;
|
||||||
|
this.isDivision = isDivision;
|
||||||
|
this.teacherId = teacherId;
|
||||||
|
this.semesterType = semesterType;
|
||||||
|
this.period = period;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(Long id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentId() {
|
||||||
|
return departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentId(Long departmentId) {
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getSemester() {
|
||||||
|
return semester;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSemester(Long semester) {
|
||||||
|
this.semester = semester;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getGroupId() {
|
||||||
|
return groupId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setGroupId(Long groupId) {
|
||||||
|
this.groupId = groupId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getSubjectsId() {
|
||||||
|
return subjectsId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSubjectsId(Long subjectsId) {
|
||||||
|
this.subjectsId = subjectsId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getLessonTypeId() {
|
||||||
|
return lessonTypeId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setLessonTypeId(Long lessonTypeId) {
|
||||||
|
this.lessonTypeId = lessonTypeId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getNumberOfHours() {
|
||||||
|
return numberOfHours;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setNumberOfHours(Long numberOfHours) {
|
||||||
|
this.numberOfHours = numberOfHours;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Boolean getDivision() {
|
||||||
|
return isDivision;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDivision(Boolean division) {
|
||||||
|
isDivision = division;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getTeacherId() {
|
||||||
|
return teacherId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setTeacherId(Long teacherId) {
|
||||||
|
this.teacherId = teacherId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSemesterType() {
|
||||||
|
return semesterType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSemesterType(String semesterType) {
|
||||||
|
this.semesterType = semesterType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getPeriod() {
|
||||||
|
return period;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setPeriod(String period) {
|
||||||
|
this.period = period;
|
||||||
|
}
|
||||||
|
}
|
||||||
50
backend/src/main/java/com/magistr/app/model/Speciality.java
Normal file
50
backend/src/main/java/com/magistr/app/model/Speciality.java
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
package com.magistr.app.model;
|
||||||
|
|
||||||
|
import jakarta.persistence.*;
|
||||||
|
|
||||||
|
@Entity
|
||||||
|
@Table(name="specialties")
|
||||||
|
public class Speciality {
|
||||||
|
|
||||||
|
@Id
|
||||||
|
@GeneratedValue(strategy = GenerationType.IDENTITY)
|
||||||
|
private Long id;
|
||||||
|
|
||||||
|
@Column(name="name", nullable = false)
|
||||||
|
private String specialityName;
|
||||||
|
|
||||||
|
@Column(name="specialty_code",nullable = false)
|
||||||
|
private String specialityCode;
|
||||||
|
|
||||||
|
public Speciality() {}
|
||||||
|
|
||||||
|
public Speciality(Long id, String specialityName, String specialityCode) {
|
||||||
|
this.id = id;
|
||||||
|
this.specialityName = specialityName;
|
||||||
|
this.specialityCode = specialityCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(Long id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSpecialityName() {
|
||||||
|
return specialityName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSpecialityName(String specialityName) {
|
||||||
|
this.specialityName = specialityName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSpecialityCode() {
|
||||||
|
return specialityCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSpecialityCode(String specialityCode) {
|
||||||
|
this.specialityCode = specialityCode;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -20,6 +20,12 @@ public class StudentGroup {
|
|||||||
@JoinColumn(name = "education_form_id", nullable = false)
|
@JoinColumn(name = "education_form_id", nullable = false)
|
||||||
private EducationForm educationForm;
|
private EducationForm educationForm;
|
||||||
|
|
||||||
|
@Column(name = "department_id", nullable = false)
|
||||||
|
private Long departmentId;
|
||||||
|
|
||||||
|
@Column(name = "course", nullable = false)
|
||||||
|
private Integer course;
|
||||||
|
|
||||||
public StudentGroup() {
|
public StudentGroup() {
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -54,4 +60,20 @@ public class StudentGroup {
|
|||||||
public void setEducationForm(EducationForm educationForm) {
|
public void setEducationForm(EducationForm educationForm) {
|
||||||
this.educationForm = educationForm;
|
this.educationForm = educationForm;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentId() {
|
||||||
|
return departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentId(Long departmentId) {
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Integer getCourse() {
|
||||||
|
return course;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCourse(Integer course) {
|
||||||
|
this.course = course;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,12 +13,20 @@ public class Subject {
|
|||||||
@Column(unique = true, nullable = false, length = 200)
|
@Column(unique = true, nullable = false, length = 200)
|
||||||
private String name;
|
private String name;
|
||||||
|
|
||||||
|
@Column(name = "code")
|
||||||
|
private String code;
|
||||||
|
|
||||||
|
@Column(name = "department_id", nullable = false)
|
||||||
|
private Long departmentId;
|
||||||
|
|
||||||
public Subject() {
|
public Subject() {
|
||||||
}
|
}
|
||||||
|
|
||||||
public Subject(Long id, String name) {
|
public Subject(Long id, String name, String code, Long departmentId) {
|
||||||
this.id = id;
|
this.id = id;
|
||||||
this.name = name;
|
this.name = name;
|
||||||
|
this.code = code;
|
||||||
|
this.departmentId = departmentId;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Long getId() {
|
public Long getId() {
|
||||||
@@ -36,4 +44,20 @@ public class Subject {
|
|||||||
public void setName(String name) {
|
public void setName(String name) {
|
||||||
this.name = name;
|
this.name = name;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getCode() {
|
||||||
|
return code;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCode(String code) {
|
||||||
|
this.code = code;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentId() {
|
||||||
|
return departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentId(Long departmentId) {
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,6 +20,15 @@ public class User {
|
|||||||
@Column(nullable = false, length = 20)
|
@Column(nullable = false, length = 20)
|
||||||
private Role role = Role.STUDENT;
|
private Role role = Role.STUDENT;
|
||||||
|
|
||||||
|
@Column(name = "full_name", nullable = false)
|
||||||
|
private String fullName;
|
||||||
|
|
||||||
|
@Column(name="job_title", nullable = false)
|
||||||
|
private String jobTitle;
|
||||||
|
|
||||||
|
@Column(name="department_id", nullable = false)
|
||||||
|
private Long departmentId;
|
||||||
|
|
||||||
public User() {
|
public User() {
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -54,4 +63,28 @@ public class User {
|
|||||||
public void setRole(Role role) {
|
public void setRole(Role role) {
|
||||||
this.role = role;
|
this.role = role;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getFullName() {
|
||||||
|
return fullName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setFullName(String fullName) {
|
||||||
|
this.fullName = fullName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getJobTitle() {
|
||||||
|
return jobTitle;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setJobTitle(String jobTitle) {
|
||||||
|
this.jobTitle = jobTitle;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Long getDepartmentId() {
|
||||||
|
return departmentId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDepartmentId(Long departmentId) {
|
||||||
|
this.departmentId = departmentId;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,13 @@
|
|||||||
|
package com.magistr.app.repository;
|
||||||
|
|
||||||
|
import com.magistr.app.model.Department;
|
||||||
|
import org.springframework.data.jpa.repository.JpaRepository;
|
||||||
|
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
public interface DepartmentRepository extends JpaRepository<Department, Long> {
|
||||||
|
|
||||||
|
Optional<Department> findByDepartmentName(String departmentName);
|
||||||
|
|
||||||
|
Optional<Department> findByDepartmentCode(Long departmentCode);
|
||||||
|
}
|
||||||
@@ -11,4 +11,6 @@ public interface GroupRepository extends JpaRepository<StudentGroup, Long> {
|
|||||||
Optional<StudentGroup> findByName(String name);
|
Optional<StudentGroup> findByName(String name);
|
||||||
|
|
||||||
List<StudentGroup> findByEducationFormId(Long educationFormId);
|
List<StudentGroup> findByEducationFormId(Long educationFormId);
|
||||||
|
|
||||||
|
List<StudentGroup> findByDepartmentId(Long departmentId);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,7 @@
|
|||||||
|
package com.magistr.app.repository;
|
||||||
|
|
||||||
|
import com.magistr.app.model.ScheduleData;
|
||||||
|
import org.springframework.data.jpa.repository.JpaRepository;
|
||||||
|
|
||||||
|
public interface ScheduleDataRepository extends JpaRepository<ScheduleData, Long> {
|
||||||
|
}
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
package com.magistr.app.repository;
|
||||||
|
|
||||||
|
import com.magistr.app.model.Speciality;
|
||||||
|
import org.springframework.data.jpa.repository.JpaRepository;
|
||||||
|
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
public interface SpecialtiesRepository extends JpaRepository<Speciality, Long> {
|
||||||
|
|
||||||
|
Optional<Speciality> findBySpecialityName(String specialityName);
|
||||||
|
|
||||||
|
Optional<Speciality> findBySpecialityCode(String specialityCode);
|
||||||
|
}
|
||||||
@@ -3,8 +3,11 @@ package com.magistr.app.repository;
|
|||||||
import com.magistr.app.model.Subject;
|
import com.magistr.app.model.Subject;
|
||||||
import org.springframework.data.jpa.repository.JpaRepository;
|
import org.springframework.data.jpa.repository.JpaRepository;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
|
|
||||||
public interface SubjectRepository extends JpaRepository<Subject, Long> {
|
public interface SubjectRepository extends JpaRepository<Subject, Long> {
|
||||||
Optional<Subject> findByName(String name);
|
Optional<Subject> findByName(String name);
|
||||||
|
|
||||||
|
List<Subject> findByDepartmentId(Long departmentId);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,4 +12,6 @@ public interface UserRepository extends JpaRepository<User, Long> {
|
|||||||
Optional<User> findByUsername(String username);
|
Optional<User> findByUsername(String username);
|
||||||
|
|
||||||
List<User> findByRole(Role role);
|
List<User> findByRole(Role role);
|
||||||
|
|
||||||
|
List<User> findByRoleAndDepartmentId(Role role, Long departmentId);
|
||||||
}
|
}
|
||||||
|
|||||||
0
backend/src/main/java/com/magistr/app/utils/TypeAndFormatLessonValidator.java
Normal file → Executable file
0
backend/src/main/java/com/magistr/app/utils/TypeAndFormatLessonValidator.java
Normal file → Executable file
@@ -1,15 +1,18 @@
|
|||||||
server.port=8080
|
server.port=8080
|
||||||
|
|
||||||
# PostgreSQL
|
# PostgreSQL (дефолтный — для локальной разработки через Docker Compose)
|
||||||
spring.datasource.url=jdbc:postgresql://db:5432/app_db
|
spring.datasource.url=jdbc:postgresql://db:5432/app_db
|
||||||
spring.datasource.username=${POSTGRES_USER}
|
spring.datasource.username=${POSTGRES_USER:myuser}
|
||||||
spring.datasource.password=${POSTGRES_PASSWORD}
|
spring.datasource.password=${POSTGRES_PASSWORD:supersecretpassword}
|
||||||
spring.datasource.driver-class-name=org.postgresql.Driver
|
spring.datasource.driver-class-name=org.postgresql.Driver
|
||||||
|
|
||||||
# JPA
|
# JPA
|
||||||
spring.jpa.hibernate.ddl-auto=validate
|
spring.jpa.hibernate.ddl-auto=none
|
||||||
spring.jpa.show-sql=false
|
spring.jpa.show-sql=false
|
||||||
spring.jpa.open-in-view=false
|
spring.jpa.open-in-view=false
|
||||||
|
|
||||||
#Eta nastroyka otvechayet za vklyucheniye vidimosti logov urovnya DEBUG v logakh BE, poka vyklyuchil chtoby ne zasoryat'. Zapisi INFO otobrazhat'sya budut
|
# Мультитенантность
|
||||||
|
app.tenants.config-path=${TENANTS_CONFIG_PATH:tenants.json}
|
||||||
|
|
||||||
#logging.level.root=DEBUG
|
#logging.level.root=DEBUG
|
||||||
|
|
||||||
|
|||||||
386
backend/src/main/resources/db/migration/V1__init.sql
Executable file
386
backend/src/main/resources/db/migration/V1__init.sql
Executable file
@@ -0,0 +1,386 @@
|
|||||||
|
-- ==========================================
|
||||||
|
-- Инициализация расширений
|
||||||
|
-- ==========================================
|
||||||
|
CREATE EXTENSION IF NOT EXISTS pgcrypto;
|
||||||
|
|
||||||
|
-- ===============================
|
||||||
|
-- Справочники высшего уровня
|
||||||
|
-- ===============================
|
||||||
|
CREATE TABLE IF NOT EXISTS departments (
|
||||||
|
id BIGSERIAL UNIQUE PRIMARY KEY NOT NULL,
|
||||||
|
name VARCHAR(255) NOT NULL,
|
||||||
|
code BIGINT UNIQUE NOT NULL
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO departments (name, code) VALUES
|
||||||
|
('Кафедра ИБ', 1),
|
||||||
|
('Кафедра ВТ', 2),
|
||||||
|
('Кафедра КТ', 3);
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS specialties (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
name VARCHAR(255) NOT NULL,
|
||||||
|
specialty_code VARCHAR(255) NOT NULL
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO specialties (name, specialty_code) VALUES
|
||||||
|
('Информационная безопасность', '10.03.01'),
|
||||||
|
('Информатика и вычислительная техника', '09.03.01'),
|
||||||
|
('Программная инженерия', '09.03.04');
|
||||||
|
|
||||||
|
-- ==========================================
|
||||||
|
-- Пользователи и роли
|
||||||
|
-- ==========================================
|
||||||
|
CREATE TABLE IF NOT EXISTS users (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
username VARCHAR(50) UNIQUE NOT NULL,
|
||||||
|
password VARCHAR(255) NOT NULL,
|
||||||
|
role VARCHAR(20) NOT NULL DEFAULT 'STUDENT',
|
||||||
|
full_name VARCHAR(255) NOT NULL,
|
||||||
|
job_title VARCHAR(255) NOT NULL,
|
||||||
|
department_id BIGINT NOT NULL REFERENCES departments(id),
|
||||||
|
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||||
|
);
|
||||||
|
|
||||||
|
-- Админ по умолчанию: admin / admin (bcrypt через pgcrypto)
|
||||||
|
INSERT INTO users (username, password, role, full_name, job_title, department_id)
|
||||||
|
VALUES ('admin', crypt('admin', gen_salt('bf', 10)), 'ADMIN', 'Иванов Админ Иванович', 'Доцент', 1),
|
||||||
|
('Тестовый преподаватель', crypt('1234567890', gen_salt('bf', 10)), 'TEACHER', 'Петров Препод Петрович', 'Профессор', 2)
|
||||||
|
ON CONFLICT (username) DO NOTHING;
|
||||||
|
|
||||||
|
-- ==========================================
|
||||||
|
-- Образовательные формы
|
||||||
|
-- ==========================================
|
||||||
|
CREATE TABLE IF NOT EXISTS education_forms (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
name VARCHAR(100) UNIQUE NOT NULL,
|
||||||
|
description TEXT,
|
||||||
|
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO education_forms (name) VALUES
|
||||||
|
('Бакалавриат'),
|
||||||
|
('Магистратура'),
|
||||||
|
('Специалитет')
|
||||||
|
ON CONFLICT (name) DO NOTHING;
|
||||||
|
|
||||||
|
-- ==========================================
|
||||||
|
-- Учебные группы
|
||||||
|
-- ==========================================
|
||||||
|
CREATE TABLE IF NOT EXISTS student_groups (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
name VARCHAR(100) UNIQUE NOT NULL,
|
||||||
|
group_size BIGINT NOT NULL,
|
||||||
|
education_form_id BIGINT NOT NULL REFERENCES education_forms(id),
|
||||||
|
department_id BIGINT NOT NULL REFERENCES departments(id),
|
||||||
|
course INT CHECK (course BETWEEN 1 AND 6),
|
||||||
|
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||||
|
);
|
||||||
|
|
||||||
|
-- Тестовая базовая группа для работы
|
||||||
|
INSERT INTO student_groups (name, group_size, education_form_id, department_id, course)
|
||||||
|
VALUES ('ИВТ-21-1', 25, 1, 1, 3),
|
||||||
|
('ИБ-41м', 15, 2, 1, 2)
|
||||||
|
ON CONFLICT (name) DO NOTHING;
|
||||||
|
|
||||||
|
-- ==========================================
|
||||||
|
-- Подгруппы (например: "ИВТ-21-1 Подгруппа 1")
|
||||||
|
-- ==========================================
|
||||||
|
CREATE TABLE IF NOT EXISTS subgroups (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
group_id BIGINT NOT NULL REFERENCES student_groups(id) ON DELETE CASCADE,
|
||||||
|
name VARCHAR(100) NOT NULL,
|
||||||
|
student_capacity INT,
|
||||||
|
UNIQUE(group_id, name)
|
||||||
|
);
|
||||||
|
|
||||||
|
-- ==========================================
|
||||||
|
-- Справочники
|
||||||
|
-- ==========================================
|
||||||
|
|
||||||
|
-- Дисциплины
|
||||||
|
CREATE TABLE IF NOT EXISTS subjects (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
name VARCHAR(200) UNIQUE NOT NULL,
|
||||||
|
code VARCHAR(20),
|
||||||
|
department_id BIGINT NOT NULL REFERENCES departments(id),
|
||||||
|
description TEXT,
|
||||||
|
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO subjects (name, department_id) VALUES
|
||||||
|
('Высшая математика', 1),
|
||||||
|
('Философия', 1),
|
||||||
|
('Информатика', 1),
|
||||||
|
('Базы данных', 1),
|
||||||
|
('Английский язык', 1)
|
||||||
|
ON CONFLICT (name) DO NOTHING;
|
||||||
|
|
||||||
|
-- Типы занятий
|
||||||
|
CREATE TABLE IF NOT EXISTS lesson_types (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
name VARCHAR(50) UNIQUE NOT NULL,
|
||||||
|
color_code VARCHAR(7) DEFAULT '#3788d8',
|
||||||
|
duration_minutes INT DEFAULT 90
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO lesson_types (name, color_code) VALUES
|
||||||
|
('Лекция', '#FF6B6B'),
|
||||||
|
('Практика', '#4ECDC4'),
|
||||||
|
('Лабораторная работа', '#45B7D1')
|
||||||
|
ON CONFLICT (name) DO NOTHING;
|
||||||
|
|
||||||
|
-- Оборудование
|
||||||
|
CREATE TABLE IF NOT EXISTS equipments (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
name VARCHAR(50) UNIQUE NOT NULL,
|
||||||
|
description TEXT,
|
||||||
|
inventory_number VARCHAR(50)
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO equipments (name) VALUES
|
||||||
|
('Проектор'),
|
||||||
|
('ПК'),
|
||||||
|
('Лаборатория'),
|
||||||
|
('Интерактивная доска'),
|
||||||
|
('Документ-камера'),
|
||||||
|
('Аудиосистема')
|
||||||
|
ON CONFLICT (name) DO NOTHING;
|
||||||
|
|
||||||
|
-- Аудитории
|
||||||
|
CREATE TABLE IF NOT EXISTS classrooms (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
name VARCHAR(50) UNIQUE NOT NULL,
|
||||||
|
capacity INT NOT NULL CHECK (capacity > 0),
|
||||||
|
building VARCHAR(50),
|
||||||
|
floor INT,
|
||||||
|
is_available BOOLEAN DEFAULT TRUE,
|
||||||
|
description TEXT,
|
||||||
|
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO classrooms (name, capacity, building, floor) VALUES
|
||||||
|
('101 Ленинская', 120, 'Главный корпус', 1),
|
||||||
|
('202 IT Lab', 20, 'Корпус IT', 2),
|
||||||
|
('303 Обычная', 30, 'Главный корпус', 3)
|
||||||
|
ON CONFLICT (name) DO NOTHING;
|
||||||
|
|
||||||
|
-- Привязка оборудования к аудиториям (Many-to-Many)
|
||||||
|
CREATE TABLE IF NOT EXISTS classroom_equipments (
|
||||||
|
classroom_id BIGINT NOT NULL REFERENCES classrooms(id) ON DELETE CASCADE,
|
||||||
|
equipment_id BIGINT NOT NULL REFERENCES equipments(id) ON DELETE CASCADE,
|
||||||
|
quantity INT DEFAULT 1 CHECK (quantity > 0),
|
||||||
|
notes TEXT,
|
||||||
|
PRIMARY KEY (classroom_id, equipment_id)
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO classroom_equipments (classroom_id, equipment_id, quantity)
|
||||||
|
SELECT c.id, e.id,
|
||||||
|
CASE
|
||||||
|
WHEN e.name = 'ПК' AND c.name = '202 IT Lab' THEN 15
|
||||||
|
WHEN e.name = 'ПК' THEN 1
|
||||||
|
ELSE 1
|
||||||
|
END
|
||||||
|
FROM classrooms c, equipments e
|
||||||
|
WHERE
|
||||||
|
(c.name = '101 Ленинская' AND e.name IN ('Проектор', 'Интерактивная доска', 'Аудиосистема'))
|
||||||
|
OR (c.name = '202 IT Lab' AND e.name IN ('ПК', 'Проектор', 'Лаборатория', 'Интерактивная доска'))
|
||||||
|
OR (c.name = '303 Обычная' AND e.name IN ('Проектор'))
|
||||||
|
ON CONFLICT (classroom_id, equipment_id) DO NOTHING;
|
||||||
|
|
||||||
|
-- ==========================================
|
||||||
|
-- Связи для преподавателей
|
||||||
|
-- ==========================================
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS teacher_subjects (
|
||||||
|
user_id BIGINT NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||||
|
subject_id BIGINT NOT NULL REFERENCES subjects(id) ON DELETE CASCADE,
|
||||||
|
qualification_level VARCHAR(50),
|
||||||
|
experience_years INT,
|
||||||
|
PRIMARY KEY(user_id, subject_id)
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE TABLE IF NOT EXISTS teacher_lesson_types (
|
||||||
|
user_id BIGINT NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||||
|
subject_id BIGINT NOT NULL REFERENCES subjects(id) ON DELETE CASCADE,
|
||||||
|
lesson_type_id BIGINT NOT NULL REFERENCES lesson_types(id) ON DELETE CASCADE,
|
||||||
|
PRIMARY KEY (user_id, subject_id, lesson_type_id)
|
||||||
|
);
|
||||||
|
|
||||||
|
-- ==========================================
|
||||||
|
-- Основная таблица Расписания (Lessons)
|
||||||
|
-- ==========================================
|
||||||
|
CREATE TABLE IF NOT EXISTS lessons (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
teacher_id BIGINT NOT NULL REFERENCES users(id),
|
||||||
|
group_id BIGINT NOT NULL REFERENCES student_groups(id),
|
||||||
|
subject_id BIGINT NOT NULL REFERENCES subjects(id),
|
||||||
|
lesson_format VARCHAR(255) NOT NULL,
|
||||||
|
type_lesson VARCHAR(255) NOT NULL,
|
||||||
|
classroom_id BIGINT NOT NULL REFERENCES classrooms(id),
|
||||||
|
day VARCHAR(255) NOT NULL,
|
||||||
|
week VARCHAR(255) NOT NULL,
|
||||||
|
time VARCHAR(255) NOT NULL
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO lessons (teacher_id, group_id, subject_id, lesson_format, type_lesson, classroom_id, day, week, time) VALUES
|
||||||
|
(2, 1, 1, 'Очно', 'Лекция', 1, 'Понедельник', 'Верхняя', '11:40 - 13:10'),
|
||||||
|
(1, 1, 2, 'Онлайн', 'Практическая работа', 2, 'Вторник', 'Нижняя', '15:00 - 16:30'),
|
||||||
|
(2, 1, 3, 'Очно', 'Лабораторная работа', 3, 'Среда', 'Верхняя', '8:00 - 9:30'),
|
||||||
|
(1, 1, 4, 'Онлайн', 'Лекция', 1, 'Четверг', 'Нижняя', '11:40 - 13:10'),
|
||||||
|
(2, 1, 5, 'Очно', 'Практическая работа', 2, 'Пятница', 'Верхняя', '15:00 - 16:30'),
|
||||||
|
(1, 1, 3, 'Онлайн', 'Лабораторная работа', 3, 'Суббота', 'Нижняя', '8:00 - 9:30');
|
||||||
|
|
||||||
|
-- ===============================
|
||||||
|
-- Создание таблицы данных расписания (schedule_data)
|
||||||
|
-- ===============================
|
||||||
|
CREATE TABLE IF NOT EXISTS schedule_data (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
department_id BIGINT NOT NULL REFERENCES departments(id),
|
||||||
|
semester INT NOT NULL,
|
||||||
|
group_id BIGINT NOT NULL REFERENCES student_groups(id),
|
||||||
|
subjects_id BIGINT NOT NULL REFERENCES subjects(id),
|
||||||
|
lesson_type_id BIGINT NOT NULL REFERENCES lesson_types(id),
|
||||||
|
number_of_hours INT NOT NULL,
|
||||||
|
is_division BOOLEAN NOT NULL DEFAULT FALSE,
|
||||||
|
teacher_id BIGINT NOT NULL REFERENCES users(id),
|
||||||
|
semester_type VARCHAR(255) NOT NULL,
|
||||||
|
period VARCHAR(255) NOT NULL
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO schedule_data (department_id, semester, group_id, subjects_id, lesson_type_id, number_of_hours, is_division, teacher_id, semester_type, period)
|
||||||
|
VALUES (1, 1, 1, 1, 3, 2, true, 1, 'Весенний', '2024/2025'),
|
||||||
|
(2, 4, 2, 3, 2, 1, false, 2, 'Осенний', '2025/2026'),
|
||||||
|
(3, 5, 1, 2, 1, 3, true, 1, 'Весенний', '2023/2024');
|
||||||
|
|
||||||
|
-- ==========================================
|
||||||
|
-- Функция обновления timestamp
|
||||||
|
-- ==========================================
|
||||||
|
CREATE OR REPLACE FUNCTION update_updated_at_column()
|
||||||
|
RETURNS TRIGGER AS $$
|
||||||
|
BEGIN
|
||||||
|
NEW.updated_at = CURRENT_TIMESTAMP;
|
||||||
|
RETURN NEW;
|
||||||
|
END;
|
||||||
|
$$ LANGUAGE plpgsql;
|
||||||
|
|
||||||
|
CREATE TRIGGER update_users_updated_at
|
||||||
|
BEFORE UPDATE ON users
|
||||||
|
FOR EACH ROW
|
||||||
|
EXECUTE FUNCTION update_updated_at_column();
|
||||||
|
|
||||||
|
-- ==========================================
|
||||||
|
-- Комментарии к таблицам и полям (для документации)
|
||||||
|
-- ==========================================
|
||||||
|
COMMENT ON TABLE users IS 'Пользователи системы (студенты, преподаватели, администраторы)';
|
||||||
|
COMMENT ON TABLE lessons IS 'Основное расписание занятий';
|
||||||
|
COMMENT ON TABLE departments IS 'Кафедры';
|
||||||
|
COMMENT ON TABLE specialties IS 'Специальности';
|
||||||
|
COMMENT ON TABLE schedule_data IS 'Данные к составлению расписания';
|
||||||
|
COMMENT ON COLUMN schedule_data.department_id IS 'Идентификатор кафедры';
|
||||||
|
COMMENT ON COLUMN schedule_data.semester IS 'Номер семестра';
|
||||||
|
COMMENT ON COLUMN schedule_data.group_id IS 'Идентификатор группы';
|
||||||
|
COMMENT ON COLUMN schedule_data.subjects_id IS 'Идентификатор предмета';
|
||||||
|
COMMENT ON COLUMN schedule_data.lesson_type_id IS 'Идентификатор типа занятия';
|
||||||
|
COMMENT ON COLUMN schedule_data.number_of_hours IS 'Количество часов';
|
||||||
|
COMMENT ON COLUMN schedule_data.is_division IS 'Является ли занятие разделенным';
|
||||||
|
COMMENT ON COLUMN schedule_data.teacher_id IS 'Идентификатор преподавателя';
|
||||||
|
COMMENT ON COLUMN schedule_data.semester_type IS 'Тип семестра (Весенний, Осенний)';
|
||||||
|
COMMENT ON COLUMN schedule_data.period IS 'Период занятий (год/год)';
|
||||||
|
|
||||||
|
COMMENT ON TABLE education_forms IS 'Формы обучения';
|
||||||
|
COMMENT ON TABLE subgroups IS 'Подгруппы';
|
||||||
|
COMMENT ON TABLE lesson_types IS 'Типы занятий';
|
||||||
|
COMMENT ON TABLE equipments IS 'Оборудование';
|
||||||
|
COMMENT ON TABLE classrooms IS 'Аудитории';
|
||||||
|
COMMENT ON TABLE classroom_equipments IS 'Привязка оборудования к аудиториям';
|
||||||
|
COMMENT ON TABLE teacher_subjects IS 'Привязка преподавателей к дисциплинам';
|
||||||
|
COMMENT ON TABLE teacher_lesson_types IS 'Типы занятий преподавателя';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN users.id IS 'ID пользователя';
|
||||||
|
COMMENT ON COLUMN users.username IS 'Логин пользователя';
|
||||||
|
COMMENT ON COLUMN users.password IS 'Хэш пароля пользователя';
|
||||||
|
COMMENT ON COLUMN users.role IS 'Роль пользователя';
|
||||||
|
COMMENT ON COLUMN users.created_at IS 'Дата и время создания';
|
||||||
|
COMMENT ON COLUMN users.updated_at IS 'Дата и время последнего обновления';
|
||||||
|
COMMENT ON COLUMN users.full_name IS 'ФИО пользователя';
|
||||||
|
COMMENT ON COLUMN users.job_title IS 'Должность пользователя';
|
||||||
|
COMMENT ON COLUMN users.department_id IS 'ID кафедры';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN education_forms.id IS 'ID формы обучения';
|
||||||
|
COMMENT ON COLUMN education_forms.name IS 'Название формы обучения';
|
||||||
|
COMMENT ON COLUMN education_forms.description IS 'Описание';
|
||||||
|
COMMENT ON COLUMN education_forms.created_at IS 'Дата и время создания';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN student_groups.id IS 'ID учебной группы';
|
||||||
|
COMMENT ON COLUMN student_groups.name IS 'Название группы';
|
||||||
|
COMMENT ON COLUMN student_groups.group_size IS 'Количество студентов';
|
||||||
|
COMMENT ON COLUMN student_groups.education_form_id IS 'ID формы обучения, к которой относится группа';
|
||||||
|
COMMENT ON COLUMN student_groups.department_id IS 'ID кафедры';
|
||||||
|
COMMENT ON COLUMN student_groups.course IS 'Курс';
|
||||||
|
COMMENT ON COLUMN student_groups.created_at IS 'Дата и время создания';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN subgroups.id IS 'ID подгруппы';
|
||||||
|
COMMENT ON COLUMN subgroups.group_id IS 'ID учебной группы, к которой относится подгруппа';
|
||||||
|
COMMENT ON COLUMN subgroups.name IS 'Название подгруппы';
|
||||||
|
COMMENT ON COLUMN subgroups.student_capacity IS 'Количество студентов в подгруппе';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN subjects.id IS 'ID предмета';
|
||||||
|
COMMENT ON COLUMN subjects.name IS 'Название предмета';
|
||||||
|
COMMENT ON COLUMN subjects.code IS 'Код предмета';
|
||||||
|
COMMENT ON COLUMN subjects.department_id IS 'ID кафедры';
|
||||||
|
COMMENT ON COLUMN subjects.description IS 'Описание предмета';
|
||||||
|
COMMENT ON COLUMN subjects.created_at IS 'Дата и время создания';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN lesson_types.id IS 'ID урока';
|
||||||
|
COMMENT ON COLUMN lesson_types.name IS 'Название типа урока';
|
||||||
|
COMMENT ON COLUMN lesson_types.color_code IS 'Цветовой код для типа урока';
|
||||||
|
COMMENT ON COLUMN lesson_types.duration_minutes IS 'Длительность урока в минутах';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN equipments.id IS 'ID оборудования';
|
||||||
|
COMMENT ON COLUMN equipments.name IS 'Название оборудования';
|
||||||
|
COMMENT ON COLUMN equipments.description IS 'Описание оборудования';
|
||||||
|
COMMENT ON COLUMN equipments.inventory_number IS 'Инвентарный номер оборудования';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN classrooms.id IS 'ID аудитории';
|
||||||
|
COMMENT ON COLUMN classrooms.name IS 'Название аудитории';
|
||||||
|
COMMENT ON COLUMN classrooms.capacity IS 'Вместимость аудитории';
|
||||||
|
COMMENT ON COLUMN classrooms.building IS 'Корпус';
|
||||||
|
COMMENT ON COLUMN classrooms.floor IS 'Этаж';
|
||||||
|
COMMENT ON COLUMN classrooms.is_available IS 'Доступность аудитории';
|
||||||
|
COMMENT ON COLUMN classrooms.description IS 'Описание аудитории';
|
||||||
|
COMMENT ON COLUMN classrooms.created_at IS 'Дата и время создания';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN classroom_equipments.classroom_id IS 'ID аудитории';
|
||||||
|
COMMENT ON COLUMN classroom_equipments.equipment_id IS 'ID оборудования';
|
||||||
|
COMMENT ON COLUMN classroom_equipments.quantity IS 'Дата и время создания'; -- Так было в V2
|
||||||
|
COMMENT ON COLUMN classroom_equipments.notes IS 'Примечания к записи';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN teacher_subjects.user_id IS 'ID преподавателя';
|
||||||
|
COMMENT ON COLUMN teacher_subjects.subject_id IS 'ID предмета';
|
||||||
|
COMMENT ON COLUMN teacher_subjects.qualification_level IS 'Уровень квалификации преподавателя';
|
||||||
|
COMMENT ON COLUMN teacher_subjects.experience_years IS 'Опыт преподавания';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN lessons.id IS 'ID урока';
|
||||||
|
COMMENT ON COLUMN lessons.teacher_id IS 'Идентификатор преподавателя, который проводит урок';
|
||||||
|
COMMENT ON COLUMN lessons.group_id IS 'ID группы, в которой проходит урок';
|
||||||
|
COMMENT ON COLUMN lessons.subject_id IS 'ID предмета, который преподается';
|
||||||
|
COMMENT ON COLUMN lessons.lesson_format IS 'Формат урока';
|
||||||
|
COMMENT ON COLUMN lessons.type_lesson IS 'Тип урока';
|
||||||
|
COMMENT ON COLUMN lessons.classroom_id IS 'ID аудитории, в которой проходит урок';
|
||||||
|
COMMENT ON COLUMN lessons.day IS 'День недели, в который проходит урок';
|
||||||
|
COMMENT ON COLUMN lessons.week IS 'Номер недели, в которой проходит урок';
|
||||||
|
COMMENT ON COLUMN lessons.time IS 'Время урока';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN departments.id IS 'ID кафедры';
|
||||||
|
COMMENT ON COLUMN departments.name IS 'Название кафедры';
|
||||||
|
COMMENT ON COLUMN departments.code IS 'Код кафедры';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN specialties.id IS 'ID специальности';
|
||||||
|
COMMENT ON COLUMN specialties.name IS 'Название специальности';
|
||||||
|
COMMENT ON COLUMN specialties.specialty_code IS 'Код специальности';
|
||||||
|
|
||||||
|
COMMENT ON COLUMN teacher_lesson_types.user_id IS 'ID преподавателя';
|
||||||
|
COMMENT ON COLUMN teacher_lesson_types.subject_id IS 'ID предмета';
|
||||||
|
COMMENT ON COLUMN teacher_lesson_types.lesson_type_id IS 'ID типа занятия';
|
||||||
9
backend/tenants.json
Executable file
9
backend/tenants.json
Executable file
@@ -0,0 +1,9 @@
|
|||||||
|
[
|
||||||
|
{
|
||||||
|
"name": "Default (dev)",
|
||||||
|
"domain": "default",
|
||||||
|
"url": "jdbc:postgresql://db:5432/app_db",
|
||||||
|
"username": "myuser",
|
||||||
|
"password": "supersecretpassword"
|
||||||
|
}
|
||||||
|
]
|
||||||
20
compose.yaml
20
compose.yaml
@@ -10,9 +10,7 @@ services:
|
|||||||
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
|
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
|
||||||
networks:
|
networks:
|
||||||
- proxy
|
- proxy
|
||||||
depends_on:
|
|
||||||
db:
|
|
||||||
condition: service_healthy
|
|
||||||
frontend:
|
frontend:
|
||||||
container_name: frontend
|
container_name: frontend
|
||||||
restart: always
|
restart: always
|
||||||
@@ -23,6 +21,7 @@ services:
|
|||||||
- proxy
|
- proxy
|
||||||
depends_on:
|
depends_on:
|
||||||
- backend
|
- backend
|
||||||
|
|
||||||
db:
|
db:
|
||||||
image: postgres:alpine3.23
|
image: postgres:alpine3.23
|
||||||
container_name: db
|
container_name: db
|
||||||
@@ -30,21 +29,12 @@ services:
|
|||||||
ports:
|
ports:
|
||||||
- "5432:5432"
|
- "5432:5432"
|
||||||
environment:
|
environment:
|
||||||
POSTGRES_USER: ${POSTGRES_USER}
|
POSTGRES_USER: myuser
|
||||||
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
|
POSTGRES_PASSWORD: supersecretpassword
|
||||||
POSTGRES_DB: app_db
|
POSTGRES_DB: app_db
|
||||||
volumes:
|
|
||||||
- ./db/data:/var/lib/postgresql
|
|
||||||
- ./db/init:/docker-entrypoint-initdb.d:ro
|
|
||||||
networks:
|
networks:
|
||||||
- proxy
|
- proxy
|
||||||
healthcheck:
|
|
||||||
test:
|
|
||||||
- CMD-SHELL
|
|
||||||
- pg_isready -U ${POSTGRES_USER} -d app_db
|
|
||||||
interval: 10s
|
|
||||||
timeout: 5s
|
|
||||||
retries: 5
|
|
||||||
networks:
|
networks:
|
||||||
proxy:
|
proxy:
|
||||||
external: true
|
external: true
|
||||||
|
|||||||
229
db/init/init.sql
229
db/init/init.sql
@@ -1,229 +0,0 @@
|
|||||||
-- ==========================================
|
|
||||||
-- Инициализация расширений
|
|
||||||
-- ==========================================
|
|
||||||
CREATE EXTENSION IF NOT EXISTS pgcrypto;
|
|
||||||
|
|
||||||
-- ==========================================
|
|
||||||
-- Пользователи и роли
|
|
||||||
-- ==========================================
|
|
||||||
CREATE TABLE IF NOT EXISTS users (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
username VARCHAR(50) UNIQUE NOT NULL,
|
|
||||||
password VARCHAR(255) NOT NULL,
|
|
||||||
role VARCHAR(20) NOT NULL DEFAULT 'STUDENT',
|
|
||||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
|
||||||
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
|
||||||
);
|
|
||||||
|
|
||||||
-- Админ по умолчанию: admin / admin (bcrypt через pgcrypto)
|
|
||||||
INSERT INTO users (username, password, role)
|
|
||||||
VALUES ('admin', crypt('admin', gen_salt('bf', 10)), 'ADMIN'),
|
|
||||||
('Тестовый преподаватель', '1234567890', 'TEACHER')
|
|
||||||
ON CONFLICT (username) DO NOTHING;
|
|
||||||
|
|
||||||
-- ==========================================
|
|
||||||
-- Образовательные формы
|
|
||||||
-- ==========================================
|
|
||||||
CREATE TABLE IF NOT EXISTS education_forms (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
name VARCHAR(100) UNIQUE NOT NULL,
|
|
||||||
description TEXT,
|
|
||||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
|
||||||
);
|
|
||||||
|
|
||||||
INSERT INTO education_forms (name) VALUES
|
|
||||||
('Бакалавриат'),
|
|
||||||
('Магистратура'),
|
|
||||||
('Специалитет')
|
|
||||||
ON CONFLICT (name) DO NOTHING;
|
|
||||||
|
|
||||||
-- ==========================================
|
|
||||||
-- Учебные группы
|
|
||||||
-- ==========================================
|
|
||||||
CREATE TABLE IF NOT EXISTS student_groups (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
name VARCHAR(100) UNIQUE NOT NULL,
|
|
||||||
group_size BIGINT NOT NULL,
|
|
||||||
education_form_id BIGINT NOT NULL REFERENCES education_forms(id),
|
|
||||||
course INT CHECK (course BETWEEN 1 AND 6),
|
|
||||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
|
||||||
);
|
|
||||||
|
|
||||||
-- Тестовая базовая группа для работы
|
|
||||||
INSERT INTO student_groups (name, group_size, education_form_id, course)
|
|
||||||
VALUES ('ИВТ-21-1', 25, 1, 3),
|
|
||||||
('ИБ-41м', 15, 2, 2)
|
|
||||||
ON CONFLICT (name) DO NOTHING;
|
|
||||||
|
|
||||||
-- ==========================================
|
|
||||||
-- Подгруппы (например: "ИВТ-21-1 Подгруппа 1")
|
|
||||||
-- ==========================================
|
|
||||||
CREATE TABLE IF NOT EXISTS subgroups (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
group_id BIGINT NOT NULL REFERENCES student_groups(id) ON DELETE CASCADE,
|
|
||||||
name VARCHAR(100) NOT NULL,
|
|
||||||
student_capacity INT,
|
|
||||||
UNIQUE(group_id, name)
|
|
||||||
);
|
|
||||||
|
|
||||||
-- ==========================================
|
|
||||||
-- Справочники
|
|
||||||
-- ==========================================
|
|
||||||
|
|
||||||
-- Дисциплины
|
|
||||||
CREATE TABLE IF NOT EXISTS subjects (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
name VARCHAR(200) UNIQUE NOT NULL,
|
|
||||||
code VARCHAR(20),
|
|
||||||
description TEXT,
|
|
||||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
|
||||||
);
|
|
||||||
|
|
||||||
INSERT INTO subjects (name) VALUES
|
|
||||||
('Высшая математика'),
|
|
||||||
('Философия'),
|
|
||||||
('Информатика'),
|
|
||||||
('Базы данных'),
|
|
||||||
('Английский язык')
|
|
||||||
ON CONFLICT (name) DO NOTHING;
|
|
||||||
|
|
||||||
-- Типы занятий
|
|
||||||
CREATE TABLE IF NOT EXISTS lesson_types (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
name VARCHAR(50) UNIQUE NOT NULL,
|
|
||||||
color_code VARCHAR(7) DEFAULT '#3788d8', -- для цветовой индикации в календаре
|
|
||||||
duration_minutes INT DEFAULT 90
|
|
||||||
);
|
|
||||||
|
|
||||||
INSERT INTO lesson_types (name, color_code) VALUES
|
|
||||||
('Лекция', '#FF6B6B'),
|
|
||||||
('Практика', '#4ECDC4'),
|
|
||||||
('Лабораторная работа', '#45B7D1')
|
|
||||||
ON CONFLICT (name) DO NOTHING;
|
|
||||||
|
|
||||||
-- Оборудование
|
|
||||||
CREATE TABLE IF NOT EXISTS equipments (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
name VARCHAR(50) UNIQUE NOT NULL,
|
|
||||||
description TEXT,
|
|
||||||
inventory_number VARCHAR(50)
|
|
||||||
);
|
|
||||||
|
|
||||||
INSERT INTO equipments (name) VALUES
|
|
||||||
('Проектор'),
|
|
||||||
('ПК'),
|
|
||||||
('Лаборатория'),
|
|
||||||
('Интерактивная доска'),
|
|
||||||
('Документ-камера'),
|
|
||||||
('Аудиосистема')
|
|
||||||
ON CONFLICT (name) DO NOTHING;
|
|
||||||
|
|
||||||
-- Аудитории
|
|
||||||
CREATE TABLE IF NOT EXISTS classrooms (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
name VARCHAR(50) UNIQUE NOT NULL,
|
|
||||||
capacity INT NOT NULL CHECK (capacity > 0),
|
|
||||||
building VARCHAR(50),
|
|
||||||
floor INT,
|
|
||||||
is_available BOOLEAN DEFAULT TRUE,
|
|
||||||
description TEXT,
|
|
||||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
|
||||||
);
|
|
||||||
|
|
||||||
INSERT INTO classrooms (name, capacity, building, floor) VALUES
|
|
||||||
('101 Ленинская', 120, 'Главный корпус', 1),
|
|
||||||
('202 IT Lab', 20, 'Корпус IT', 2),
|
|
||||||
('303 Обычная', 30, 'Главный корпус', 3)
|
|
||||||
ON CONFLICT (name) DO NOTHING;
|
|
||||||
|
|
||||||
-- Привязка оборудования к аудиториям (Many-to-Many)
|
|
||||||
CREATE TABLE IF NOT EXISTS classroom_equipments (
|
|
||||||
classroom_id BIGINT NOT NULL REFERENCES classrooms(id) ON DELETE CASCADE,
|
|
||||||
equipment_id BIGINT NOT NULL REFERENCES equipments(id) ON DELETE CASCADE,
|
|
||||||
quantity INT DEFAULT 1 CHECK (quantity > 0),
|
|
||||||
notes TEXT,
|
|
||||||
PRIMARY KEY (classroom_id, equipment_id)
|
|
||||||
);
|
|
||||||
|
|
||||||
-- Заполнение привязок оборудования с использованием подзапросов
|
|
||||||
INSERT INTO classroom_equipments (classroom_id, equipment_id, quantity)
|
|
||||||
SELECT c.id, e.id,
|
|
||||||
CASE
|
|
||||||
WHEN e.name = 'ПК' AND c.name = '202 IT Lab' THEN 15
|
|
||||||
WHEN e.name = 'ПК' THEN 1
|
|
||||||
ELSE 1
|
|
||||||
END
|
|
||||||
FROM classrooms c, equipments e
|
|
||||||
WHERE
|
|
||||||
(c.name = '101 Ленинская' AND e.name IN ('Проектор', 'Интерактивная доска', 'Аудиосистема'))
|
|
||||||
OR (c.name = '202 IT Lab' AND e.name IN ('ПК', 'Проектор', 'Лаборатория', 'Интерактивная доска'))
|
|
||||||
OR (c.name = '303 Обычная' AND e.name IN ('Проектор'))
|
|
||||||
ON CONFLICT (classroom_id, equipment_id) DO NOTHING;
|
|
||||||
|
|
||||||
-- ==========================================
|
|
||||||
-- Связи для преподавателей
|
|
||||||
-- ==========================================
|
|
||||||
|
|
||||||
-- Привязка преподавателей к дисциплинам
|
|
||||||
CREATE TABLE IF NOT EXISTS teacher_subjects (
|
|
||||||
user_id BIGINT NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
|
||||||
subject_id BIGINT NOT NULL REFERENCES subjects(id) ON DELETE CASCADE,
|
|
||||||
qualification_level VARCHAR(50),
|
|
||||||
experience_years INT,
|
|
||||||
PRIMARY KEY(user_id, subject_id)
|
|
||||||
);
|
|
||||||
|
|
||||||
-- Какие типы занятий может вести преподаватель по дисциплине
|
|
||||||
CREATE TABLE IF NOT EXISTS teacher_lesson_types (
|
|
||||||
user_id BIGINT NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
|
||||||
subject_id BIGINT NOT NULL REFERENCES subjects(id) ON DELETE CASCADE,
|
|
||||||
lesson_type_id BIGINT NOT NULL REFERENCES lesson_types(id) ON DELETE CASCADE,
|
|
||||||
PRIMARY KEY (user_id, subject_id, lesson_type_id)
|
|
||||||
);
|
|
||||||
|
|
||||||
-- ==========================================
|
|
||||||
-- Основная таблица Расписания (Lessons)
|
|
||||||
-- ==========================================
|
|
||||||
CREATE TABLE IF NOT EXISTS lessons (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
teacher_id BIGINT NOT NULL REFERENCES users(id),
|
|
||||||
group_id BIGINT NOT NULL REFERENCES student_groups(id),
|
|
||||||
subject_id BIGINT NOT NULL REFERENCES subjects(id),
|
|
||||||
lesson_format VARCHAR(255) NOT NULL,
|
|
||||||
type_lesson VARCHAR(255) NOT NULL,
|
|
||||||
classroom_id BIGINT NOT NULL REFERENCES classrooms(id),
|
|
||||||
day VARCHAR(255) NOT NULL,
|
|
||||||
week VARCHAR(255) NOT NULL,
|
|
||||||
time VARCHAR(255) NOT NULL
|
|
||||||
);
|
|
||||||
|
|
||||||
INSERT INTO lessons (teacher_id, group_id, subject_id, lesson_format, type_lesson, classroom_id, day, week, time) VALUES
|
|
||||||
(2, 1, 1, 'Очно', 'Лекция', 1, 'Понедельник', 'Верхняя', '11:40 - 13:10'),
|
|
||||||
(1, 1, 2, 'Онлайн', 'Практическая работа', 2, 'Вторник', 'Нижняя', '15:00 - 16:30'),
|
|
||||||
(2, 1, 3, 'Очно', 'Лабораторная работа', 3, 'Среда', 'Верхняя', '8:00 - 9:30'),
|
|
||||||
(1, 1, 4, 'Онлайн', 'Лекция', 1, 'Четверг', 'Нижняя', '11:40 - 13:10'),
|
|
||||||
(2, 1, 5, 'Очно', 'Практическая работа', 2, 'Пятница', 'Верхняя', '15:00 - 16:30'),
|
|
||||||
(1, 1, 3, 'Онлайн', 'Лабораторная работа', 3, 'Суббота', 'Нижняя', '8:00 - 9:30');
|
|
||||||
|
|
||||||
-- ==========================================
|
|
||||||
-- Функция обновления timestamp
|
|
||||||
-- ==========================================
|
|
||||||
CREATE OR REPLACE FUNCTION update_updated_at_column()
|
|
||||||
RETURNS TRIGGER AS $$
|
|
||||||
BEGIN
|
|
||||||
NEW.updated_at = CURRENT_TIMESTAMP;
|
|
||||||
RETURN NEW;
|
|
||||||
END;
|
|
||||||
$$ LANGUAGE plpgsql;
|
|
||||||
|
|
||||||
-- Триггеры для обновления updated_at
|
|
||||||
CREATE TRIGGER update_users_updated_at
|
|
||||||
BEFORE UPDATE ON users
|
|
||||||
FOR EACH ROW
|
|
||||||
EXECUTE FUNCTION update_updated_at_column();
|
|
||||||
|
|
||||||
-- ==========================================
|
|
||||||
-- Комментарии к таблицам и полям (для документации)
|
|
||||||
-- ==========================================
|
|
||||||
COMMENT ON TABLE users IS 'Пользователи системы (студенты, преподаватели, администраторы)';
|
|
||||||
COMMENT ON TABLE lessons IS 'Основное расписание занятий';
|
|
||||||
@@ -1,2 +1,5 @@
|
|||||||
FROM httpd:alpine
|
FROM httpd:alpine
|
||||||
COPY . /usr/local/apache2/htdocs/
|
COPY . /usr/local/apache2/htdocs/
|
||||||
|
|
||||||
|
# Set appropriate permissions for the web server to serve static files
|
||||||
|
RUN chown -R www-data:www-data /usr/local/apache2/htdocs/
|
||||||
|
|||||||
235
frontend/admin/css/department.css
Normal file
235
frontend/admin/css/department.css
Normal file
@@ -0,0 +1,235 @@
|
|||||||
|
.wrap{
|
||||||
|
max-width: 900px;
|
||||||
|
margin: 0 auto;
|
||||||
|
background: var(--bg-card);
|
||||||
|
border: 1px solid var(--bg-card-border);
|
||||||
|
border-radius: 12px;
|
||||||
|
overflow: hidden;
|
||||||
|
box-shadow: 0 6px 20px rgba(0,0,0,.06);
|
||||||
|
}
|
||||||
|
|
||||||
|
.header{
|
||||||
|
padding: 14px 16px;
|
||||||
|
border-bottom: 1px solid var(--bg-card-border);
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
details.table-item{
|
||||||
|
border-top: 1px solid var(--bg-card-border);
|
||||||
|
}
|
||||||
|
details.table-item:first-of-type{ border-top:none; }
|
||||||
|
|
||||||
|
summary{
|
||||||
|
list-style: none;
|
||||||
|
cursor: pointer;
|
||||||
|
user-select: none;
|
||||||
|
padding: 12px 16px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 10px;
|
||||||
|
}
|
||||||
|
summary::-webkit-details-marker{ display:none; }
|
||||||
|
|
||||||
|
.chev{
|
||||||
|
width: 28px;
|
||||||
|
height: 28px;
|
||||||
|
border: 1px solid var(--bg-card-border);
|
||||||
|
border-radius: 10px;
|
||||||
|
display: grid;
|
||||||
|
place-items: center;
|
||||||
|
flex: 0 0 auto;
|
||||||
|
|
||||||
|
color: var(--text-secondary);
|
||||||
|
background: var(--bg-input);
|
||||||
|
|
||||||
|
transition: transform .18s ease, color .18s ease, border-color .18s ease, background .18s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chev-icon{
|
||||||
|
width: 16px;
|
||||||
|
height: 16px;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
summary:hover .chev{
|
||||||
|
background: var(--bg-hover);
|
||||||
|
border-color: color-mix(in srgb, var(--accent) 22%, var(--bg-card-border));
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
details[open] .chev{
|
||||||
|
transform: rotate(180deg);
|
||||||
|
color: var(--accent);
|
||||||
|
border-color: color-mix(in srgb, var(--accent) 35%, var(--bg-card-border));
|
||||||
|
background: color-mix(in srgb, var(--accent) 10%, var(--bg-input));
|
||||||
|
}
|
||||||
|
|
||||||
|
.meta{ color: var(--text-secondary); font-size: 12px; }
|
||||||
|
|
||||||
|
.content{ padding: 0 16px 16px 16px; }
|
||||||
|
|
||||||
|
.wrap table{
|
||||||
|
width: 100%;
|
||||||
|
border-collapse: collapse;
|
||||||
|
border: 1px solid var(--bg-card-border);
|
||||||
|
border-radius: 10px;
|
||||||
|
overflow: hidden;
|
||||||
|
background: var(--bg-card);
|
||||||
|
}
|
||||||
|
|
||||||
|
.wrap thead th{
|
||||||
|
text-align: left;
|
||||||
|
font-size: 13px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
background: var(--bg-input);
|
||||||
|
border-bottom: 1px solid var(--bg-card-border);
|
||||||
|
padding: 10px 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.wrap tbody td{
|
||||||
|
padding: 10px 12px;
|
||||||
|
border-bottom: 1px solid var(--bg-card-border);
|
||||||
|
font-size: 14px;
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.wrap tbody tr:hover{ background: var(--bg-hover); }
|
||||||
|
|
||||||
|
.title-multiline{
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 2px;
|
||||||
|
line-height: 1.2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.title-multiline .title-main{
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.title-multiline .title-sub{
|
||||||
|
font-weight: 500;
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.title-multiline b{
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* summary = 3 колонки: [chev] [title] [meta] */
|
||||||
|
details.table-item > summary{
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 28px 1fr auto;
|
||||||
|
gap: 12px;
|
||||||
|
align-items: start; /* важно: всё прижимаем к верху */
|
||||||
|
padding: 12px 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* чтобы текст нормально переносился и не растягивал мету */
|
||||||
|
details.table-item > summary .title{
|
||||||
|
min-width: 0; /* важно для grid, иначе может распирать */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* "2 записи" всегда справа и сверху, аккуратно */
|
||||||
|
details.table-item > summary .meta{
|
||||||
|
justify-self: end;
|
||||||
|
align-self: start;
|
||||||
|
white-space: nowrap;
|
||||||
|
padding-top: 4px; /* чуть опустить относительно первой строки */
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* стрелка тоже сверху */
|
||||||
|
details.table-item > summary .chev{
|
||||||
|
align-self: start;
|
||||||
|
margin-top: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.records-search{
|
||||||
|
width: min(360px, 60vw);
|
||||||
|
padding: 0.45rem 0.7rem;
|
||||||
|
background: var(--bg-input);
|
||||||
|
border: 1px solid var(--bg-card-border);
|
||||||
|
border-radius: var(--radius-sm);
|
||||||
|
color: var(--text-primary);
|
||||||
|
font-size: 0.9rem;
|
||||||
|
outline: none;
|
||||||
|
transition: border-color .2s ease, box-shadow .2s ease, background .2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.records-search::placeholder{ color: var(--text-placeholder); }
|
||||||
|
|
||||||
|
.records-search:focus{
|
||||||
|
background: var(--bg-input-focus);
|
||||||
|
border-color: var(--accent);
|
||||||
|
box-shadow: 0 0 0 3px var(--accent-glow);
|
||||||
|
}
|
||||||
|
/* Таблица внутри раскрывающегося блока */
|
||||||
|
details.table-item .content table{
|
||||||
|
width: 100%;
|
||||||
|
border-collapse: separate; /* нужно для красивых линий */
|
||||||
|
border-spacing: 0;
|
||||||
|
border: 1px solid var(--bg-card-border);
|
||||||
|
border-radius: 12px;
|
||||||
|
overflow: hidden;
|
||||||
|
background: var(--bg-card);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Шапка */
|
||||||
|
details.table-item .content thead th{
|
||||||
|
position: sticky; /* опционально: шапка прилипает при скролле */
|
||||||
|
top: 0;
|
||||||
|
z-index: 1;
|
||||||
|
|
||||||
|
background: var(--bg-input);
|
||||||
|
color: var(--text-secondary);
|
||||||
|
border-bottom: 1px solid var(--bg-card-border);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ячейки: одинаковые отступы */
|
||||||
|
details.table-item .content th,
|
||||||
|
details.table-item .content td{
|
||||||
|
padding: 0.75rem 0.85rem;
|
||||||
|
vertical-align: top;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Вертикальные разделители между колонками */
|
||||||
|
details.table-item .content th:not(:last-child),
|
||||||
|
details.table-item .content td:not(:last-child){
|
||||||
|
border-right: 1px solid var(--bg-card-border);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Горизонтальные разделители между строками */
|
||||||
|
details.table-item .content tbody td{
|
||||||
|
border-bottom: 1px solid var(--bg-card-border);
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* У последней строки нет нижней линии */
|
||||||
|
details.table-item .content tbody tr:last-child td{
|
||||||
|
border-bottom: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* "Зебра" для читабельности */
|
||||||
|
details.table-item .content tbody tr:nth-child(even){
|
||||||
|
background: color-mix(in srgb, var(--bg-card) 70%, var(--bg-hover));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ховер по строке */
|
||||||
|
details.table-item .content tbody tr:hover{
|
||||||
|
background: var(--bg-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (Опционально) Чтобы длинный текст не ломал ширину */
|
||||||
|
details.table-item .content td{
|
||||||
|
word-break: break-word;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* (Опционально) если таблица широкая — пусть скроллится горизонтально */
|
||||||
|
details.table-item .content{
|
||||||
|
overflow-x: auto;
|
||||||
|
}
|
||||||
@@ -14,6 +14,7 @@
|
|||||||
<link rel="stylesheet" href="css/layout.css">
|
<link rel="stylesheet" href="css/layout.css">
|
||||||
<link rel="stylesheet" href="css/components.css">
|
<link rel="stylesheet" href="css/components.css">
|
||||||
<link rel="stylesheet" href="css/modals.css">
|
<link rel="stylesheet" href="css/modals.css">
|
||||||
|
<link rel="stylesheet" href="css/department.css">
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
@@ -46,6 +47,17 @@
|
|||||||
</svg>
|
</svg>
|
||||||
Пользователи
|
Пользователи
|
||||||
</a>
|
</a>
|
||||||
|
<a href="#" class="nav-item" data-tab="department">
|
||||||
|
<svg width="20" height="20" viewBox="0 0 24 24" fill="none"
|
||||||
|
stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<path d="M4 21V5a2 2 0 0 1 2-2h12a2 2 0 0 1 2 2v16" />
|
||||||
|
<path d="M2 21h20" />
|
||||||
|
<path d="M8 7h0M12 7h0M16 7h0" />
|
||||||
|
<path d="M8 11h0M12 11h0M16 11h0" />
|
||||||
|
<path d="M10 21v-4h4v4" />
|
||||||
|
</svg>
|
||||||
|
Кафедра
|
||||||
|
</a>
|
||||||
<a href="#" class="nav-item" data-tab="groups">
|
<a href="#" class="nav-item" data-tab="groups">
|
||||||
<svg width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"
|
<svg width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"
|
||||||
stroke-linecap="round" stroke-linejoin="round">
|
stroke-linecap="round" stroke-linejoin="round">
|
||||||
@@ -96,6 +108,14 @@
|
|||||||
</svg>
|
</svg>
|
||||||
Расписание занятий
|
Расписание занятий
|
||||||
</a>
|
</a>
|
||||||
|
<a href="#" class="nav-item" data-tab="database">
|
||||||
|
<svg width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<ellipse cx="12" cy="5" rx="9" ry="3"></ellipse>
|
||||||
|
<path d="M21 12c0 1.66-4 3-9 3s-9-1.34-9-3"></path>
|
||||||
|
<path d="M3 5v14c0 1.66 4 3 9 3s9-1.34 9-3V5"></path>
|
||||||
|
</svg>
|
||||||
|
База данных
|
||||||
|
</a>
|
||||||
</nav>
|
</nav>
|
||||||
<div class="sidebar-footer">
|
<div class="sidebar-footer">
|
||||||
<button class="btn-logout" id="btn-logout">
|
<button class="btn-logout" id="btn-logout">
|
||||||
|
|||||||
@@ -8,6 +8,8 @@ import { initEquipments } from './views/equipments.js';
|
|||||||
import { initClassrooms } from './views/classrooms.js';
|
import { initClassrooms } from './views/classrooms.js';
|
||||||
import { initSubjects } from './views/subjects.js';
|
import { initSubjects } from './views/subjects.js';
|
||||||
import {initSchedule} from "./views/schedule.js";
|
import {initSchedule} from "./views/schedule.js";
|
||||||
|
import {initDatabase} from "./views/database.js";
|
||||||
|
import {initDepartment} from "./views/department.js";
|
||||||
|
|
||||||
// Configuration
|
// Configuration
|
||||||
const ROUTES = {
|
const ROUTES = {
|
||||||
@@ -17,8 +19,9 @@ const ROUTES = {
|
|||||||
equipments: { title: 'Оборудование', file: 'views/equipments.html', init: initEquipments },
|
equipments: { title: 'Оборудование', file: 'views/equipments.html', init: initEquipments },
|
||||||
classrooms: { title: 'Аудитории', file: 'views/classrooms.html', init: initClassrooms },
|
classrooms: { title: 'Аудитории', file: 'views/classrooms.html', init: initClassrooms },
|
||||||
subjects: { title: 'Дисциплины и преподаватели', file: 'views/subjects.html', init: initSubjects },
|
subjects: { title: 'Дисциплины и преподаватели', file: 'views/subjects.html', init: initSubjects },
|
||||||
// Новая вкладка
|
|
||||||
schedule: { title: 'Расписание занятий', file: 'views/schedule.html', init: initSchedule },
|
schedule: { title: 'Расписание занятий', file: 'views/schedule.html', init: initSchedule },
|
||||||
|
database: { title: 'База данных', file: 'views/database.html', init: initDatabase },
|
||||||
|
department: { title: 'Кафедры', file: 'views/department.html', init: initDepartment },
|
||||||
};
|
};
|
||||||
|
|
||||||
let currentTab = null;
|
let currentTab = null;
|
||||||
|
|||||||
157
frontend/admin/js/views/database.js
Executable file
157
frontend/admin/js/views/database.js
Executable file
@@ -0,0 +1,157 @@
|
|||||||
|
import { api } from '../api.js';
|
||||||
|
import { escapeHtml, showAlert, hideAlert } from '../utils.js';
|
||||||
|
|
||||||
|
export async function initDatabase() {
|
||||||
|
const tenantsTbody = document.getElementById('tenants-tbody');
|
||||||
|
const addTenantForm = document.getElementById('add-tenant-form');
|
||||||
|
const statusInfo = document.getElementById('db-status-info');
|
||||||
|
const btnTest = document.getElementById('btn-test-connection');
|
||||||
|
|
||||||
|
// === Загрузка статуса текущего подключения ===
|
||||||
|
async function loadStatus() {
|
||||||
|
try {
|
||||||
|
const data = await api.get('/api/database/status');
|
||||||
|
const statusBadge = data.connected
|
||||||
|
? '<span class="badge badge-available">Online</span>'
|
||||||
|
: '<span class="badge badge-unavailable">Offline</span>';
|
||||||
|
|
||||||
|
statusInfo.innerHTML = `
|
||||||
|
<div style="display: flex; align-items: center; gap: 1rem; flex-wrap: wrap;">
|
||||||
|
<div>
|
||||||
|
<span style="color: var(--text-secondary); font-size: 0.85rem;">Тенант:</span>
|
||||||
|
<strong>${escapeHtml(data.tenant || '—')}</strong>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<span style="color: var(--text-secondary); font-size: 0.85rem;">Название:</span>
|
||||||
|
<strong>${escapeHtml(data.name || '—')}</strong>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<span style="color: var(--text-secondary); font-size: 0.85rem;">Статус:</span>
|
||||||
|
${statusBadge}
|
||||||
|
</div>
|
||||||
|
${data.url ? `<div>
|
||||||
|
<span style="color: var(--text-secondary); font-size: 0.85rem;">URL:</span>
|
||||||
|
<code style="font-size: 0.85rem;">${escapeHtml(data.url)}</code>
|
||||||
|
</div>` : ''}
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
} catch (e) {
|
||||||
|
statusInfo.innerHTML = `<div class="form-alert error" style="display:block">Ошибка загрузки статуса: ${e.message}</div>`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// === Загрузка списка тенантов ===
|
||||||
|
async function loadTenants() {
|
||||||
|
try {
|
||||||
|
const tenants = await api.get('/api/database/tenants');
|
||||||
|
renderTenantsTable(tenants);
|
||||||
|
} catch (e) {
|
||||||
|
tenantsTbody.innerHTML = `<tr><td colspan="6" class="loading-row">Ошибка загрузки: ${e.message}</td></tr>`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function renderTenantsTable(tenants) {
|
||||||
|
if (!tenants || !tenants.length) {
|
||||||
|
tenantsTbody.innerHTML = '<tr><td colspan="6" class="loading-row">Нет подключённых тенантов</td></tr>';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
tenantsTbody.innerHTML = tenants.map(t => {
|
||||||
|
const statusBadge = t.connected
|
||||||
|
? '<span class="badge badge-available">Online</span>'
|
||||||
|
: '<span class="badge badge-unavailable">Offline</span>';
|
||||||
|
|
||||||
|
return `
|
||||||
|
<tr>
|
||||||
|
<td>${escapeHtml(t.name || '—')}</td>
|
||||||
|
<td><code>${escapeHtml(t.domain)}</code></td>
|
||||||
|
<td><code style="font-size: 0.82rem;">${escapeHtml(t.url)}</code></td>
|
||||||
|
<td>${escapeHtml(t.username || '—')}</td>
|
||||||
|
<td>${statusBadge}</td>
|
||||||
|
<td><button class="btn-delete" data-domain="${escapeHtml(t.domain)}">Удалить</button></td>
|
||||||
|
</tr>`;
|
||||||
|
}).join('');
|
||||||
|
}
|
||||||
|
|
||||||
|
// === Тест подключения ===
|
||||||
|
btnTest.addEventListener('click', async () => {
|
||||||
|
hideAlert('add-tenant-alert');
|
||||||
|
const url = document.getElementById('tenant-url').value.trim();
|
||||||
|
const username = document.getElementById('tenant-username').value.trim();
|
||||||
|
const password = document.getElementById('tenant-password').value;
|
||||||
|
|
||||||
|
if (!url) {
|
||||||
|
showAlert('add-tenant-alert', 'Введите JDBC URL', 'error');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
btnTest.textContent = '...';
|
||||||
|
btnTest.disabled = true;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await api.post('/api/database/test', { url, username, password });
|
||||||
|
if (result.success) {
|
||||||
|
showAlert('add-tenant-alert', '✓ Подключение успешно!', 'success');
|
||||||
|
} else {
|
||||||
|
showAlert('add-tenant-alert', `✗ ${result.message}`, 'error');
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
showAlert('add-tenant-alert', `Ошибка: ${e.message}`, 'error');
|
||||||
|
} finally {
|
||||||
|
btnTest.textContent = 'Тест';
|
||||||
|
btnTest.disabled = false;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// === Добавление тенанта ===
|
||||||
|
addTenantForm.addEventListener('submit', async (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
hideAlert('add-tenant-alert');
|
||||||
|
|
||||||
|
const name = document.getElementById('tenant-name').value.trim();
|
||||||
|
const domain = document.getElementById('tenant-domain').value.trim().toLowerCase();
|
||||||
|
const url = document.getElementById('tenant-url').value.trim();
|
||||||
|
const username = document.getElementById('tenant-username').value.trim();
|
||||||
|
const password = document.getElementById('tenant-password').value;
|
||||||
|
|
||||||
|
if (!name || !domain || !url) {
|
||||||
|
showAlert('add-tenant-alert', 'Заполните все обязательные поля', 'error');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const result = await api.post('/api/database/tenants', { name, domain, url, username, password });
|
||||||
|
if (result.success) {
|
||||||
|
showAlert('add-tenant-alert', `Тенант "${escapeHtml(domain)}" добавлен!`, 'success');
|
||||||
|
addTenantForm.reset();
|
||||||
|
loadTenants();
|
||||||
|
loadStatus();
|
||||||
|
} else {
|
||||||
|
showAlert('add-tenant-alert', result.message, 'error');
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
showAlert('add-tenant-alert', `Ошибка: ${e.message}`, 'error');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// === Удаление тенанта ===
|
||||||
|
tenantsTbody.addEventListener('click', async (e) => {
|
||||||
|
const btn = e.target.closest('.btn-delete');
|
||||||
|
if (!btn) return;
|
||||||
|
|
||||||
|
const domain = btn.dataset.domain;
|
||||||
|
if (!confirm(`Удалить тенант "${domain}"? Пул соединений будет закрыт.`)) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await api.delete(`/api/database/tenants/${domain}`);
|
||||||
|
loadTenants();
|
||||||
|
loadStatus();
|
||||||
|
} catch (e) {
|
||||||
|
alert(`Ошибка: ${e.message}`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// === Init ===
|
||||||
|
loadStatus();
|
||||||
|
loadTenants();
|
||||||
|
}
|
||||||
4
frontend/admin/js/views/department.js
Normal file
4
frontend/admin/js/views/department.js
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
import { api } from '../api.js';
|
||||||
|
import { escapeHtml } from '../utils.js';
|
||||||
|
|
||||||
|
export async function initDepartment() { }
|
||||||
@@ -17,7 +17,7 @@ export async function initGroups() {
|
|||||||
populateEfSelects(educationForms);
|
populateEfSelects(educationForms);
|
||||||
await loadGroups();
|
await loadGroups();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
groupsTbody.innerHTML = '<tr><td colspan="4" class="loading-row">Ошибка загрузки данных</td></tr>';
|
groupsTbody.innerHTML = '<tr><td colspan="7" class="loading-row">Ошибка загрузки данных</td></tr>';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -26,7 +26,7 @@ export async function initGroups() {
|
|||||||
allGroups = await api.get('/api/groups');
|
allGroups = await api.get('/api/groups');
|
||||||
applyGroupFilter();
|
applyGroupFilter();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
groupsTbody.innerHTML = '<tr><td colspan="4" class="loading-row">Ошибка загрузки</td></tr>';
|
groupsTbody.innerHTML = '<tr><td colspan="7" class="loading-row">Ошибка загрузки</td></tr>';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -61,7 +61,7 @@ export async function initGroups() {
|
|||||||
|
|
||||||
function renderGroups(groups) {
|
function renderGroups(groups) {
|
||||||
if (!groups || !groups.length) {
|
if (!groups || !groups.length) {
|
||||||
groupsTbody.innerHTML = '<tr><td colspan="4" class="loading-row">Нет групп</td></tr>';
|
groupsTbody.innerHTML = '<tr><td colspan="7" class="loading-row">Нет групп</td></tr>';
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
groupsTbody.innerHTML = groups.map(g => `
|
groupsTbody.innerHTML = groups.map(g => `
|
||||||
@@ -70,6 +70,8 @@ export async function initGroups() {
|
|||||||
<td>${escapeHtml(g.name)}</td>
|
<td>${escapeHtml(g.name)}</td>
|
||||||
<td>${escapeHtml(g.groupSize)}</td>
|
<td>${escapeHtml(g.groupSize)}</td>
|
||||||
<td><span class="badge badge-ef">${escapeHtml(g.educationFormName)}</span></td>
|
<td><span class="badge badge-ef">${escapeHtml(g.educationFormName)}</span></td>
|
||||||
|
<td>${g.departmentId || '-'}</td>
|
||||||
|
<td>${g.course || '-'}</td>
|
||||||
<td><button class="btn-delete" data-id="${g.id}">Удалить</button></td>
|
<td><button class="btn-delete" data-id="${g.id}">Удалить</button></td>
|
||||||
</tr>`).join('');
|
</tr>`).join('');
|
||||||
}
|
}
|
||||||
@@ -80,14 +82,24 @@ export async function initGroups() {
|
|||||||
const name = document.getElementById('new-group-name').value.trim();
|
const name = document.getElementById('new-group-name').value.trim();
|
||||||
const groupSize = document.getElementById('new-group-size').value;
|
const groupSize = document.getElementById('new-group-size').value;
|
||||||
const educationFormId = newGroupEfSelect.value;
|
const educationFormId = newGroupEfSelect.value;
|
||||||
|
const departmentId = document.getElementById('new-group-department').value;
|
||||||
|
const course = document.getElementById('new-group-course').value;
|
||||||
|
|
||||||
if (!name) { showAlert('create-group-alert', 'Введите название группы', 'error'); return; }
|
if (!name) { showAlert('create-group-alert', 'Введите название группы', 'error'); return; }
|
||||||
if (!groupSize) { showAlert('create-group-alert', 'Введите размер группы', 'error'); return; }
|
if (!groupSize) { showAlert('create-group-alert', 'Введите размер группы', 'error'); return; }
|
||||||
if (!educationFormId) { showAlert('create-group-alert', 'Выберите форму обучения', 'error'); return; }
|
if (!educationFormId) { showAlert('create-group-alert', 'Выберите форму обучения', 'error'); return; }
|
||||||
|
if (!departmentId) { showAlert('create-group-alert', 'Введите ID кафедры', 'error'); return; }
|
||||||
|
if (!course) { showAlert('create-group-alert', 'Введите курс', 'error'); return; }
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const data = await api.post('/api/groups', { name, groupSize, educationFormId: Number(educationFormId) });
|
const data = await api.post('/api/groups', {
|
||||||
showAlert('create-group-alert', `Группа "${escapeHtml(data.name)}" создана`, 'success');
|
name,
|
||||||
|
groupSize: Number(groupSize),
|
||||||
|
educationFormId: Number(educationFormId),
|
||||||
|
departmentId: Number(departmentId),
|
||||||
|
course: Number(course)
|
||||||
|
});
|
||||||
|
showAlert('create-group-alert', `Группа "${escapeHtml(data.name || name)}" создана`, 'success');
|
||||||
createGroupForm.reset();
|
createGroupForm.reset();
|
||||||
loadGroups();
|
loadGroups();
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
|||||||
@@ -24,19 +24,21 @@ export async function initSubjects() {
|
|||||||
renderSubjects(allSubjects);
|
renderSubjects(allSubjects);
|
||||||
populateSubjectSelect(allSubjects);
|
populateSubjectSelect(allSubjects);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (subjectsTbody) subjectsTbody.innerHTML = '<tr><td colspan="3" class="loading-row">Ошибка загрузки</td></tr>';
|
if (subjectsTbody) subjectsTbody.innerHTML = '<tr><td colspan="5" class="loading-row">Ошибка загрузки</td></tr>';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderSubjects(subjects) {
|
function renderSubjects(subjects) {
|
||||||
if (!subjects || !subjects.length) {
|
if (!subjects || !subjects.length) {
|
||||||
subjectsTbody.innerHTML = '<tr><td colspan="3" class="loading-row">Нет дисциплин</td></tr>';
|
subjectsTbody.innerHTML = '<tr><td colspan="5" class="loading-row">Нет дисциплин</td></tr>';
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
subjectsTbody.innerHTML = subjects.map(s => `
|
subjectsTbody.innerHTML = subjects.map(s => `
|
||||||
<tr>
|
<tr>
|
||||||
<td>${s.id}</td>
|
<td>${s.id}</td>
|
||||||
<td>${escapeHtml(s.name)}</td>
|
<td>${escapeHtml(s.name)}</td>
|
||||||
|
<td>${escapeHtml(s.code || '-')}</td>
|
||||||
|
<td>${s.departmentId || '-'}</td>
|
||||||
<td><button class="btn-delete" data-id="${s.id}">Удалить</button></td>
|
<td><button class="btn-delete" data-id="${s.id}">Удалить</button></td>
|
||||||
</tr>`).join('');
|
</tr>`).join('');
|
||||||
}
|
}
|
||||||
@@ -100,11 +102,19 @@ export async function initSubjects() {
|
|||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
hideAlert('create-subject-alert');
|
hideAlert('create-subject-alert');
|
||||||
const name = document.getElementById('new-subject-name').value.trim();
|
const name = document.getElementById('new-subject-name').value.trim();
|
||||||
|
const code = document.getElementById('new-subject-code').value.trim();
|
||||||
|
const departmentId = document.getElementById('new-subject-department').value;
|
||||||
if (!name) { showAlert('create-subject-alert', 'Введите название', 'error'); return; }
|
if (!name) { showAlert('create-subject-alert', 'Введите название', 'error'); return; }
|
||||||
|
if (!code) { showAlert('create-subject-alert', 'Введите код предмета', 'error'); return; }
|
||||||
|
if (!departmentId) { showAlert('create-subject-alert', 'Введите идентификатор кафедры', 'error'); return; }
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const data = await api.post('/api/subjects', { name });
|
const data = await api.post('/api/subjects', {
|
||||||
showAlert('create-subject-alert', `Дисциплина "${escapeHtml(data.name)}" добавлена`, 'success');
|
name,
|
||||||
|
code,
|
||||||
|
departmentId: Number(departmentId)
|
||||||
|
});
|
||||||
|
showAlert('create-subject-alert', `Дисциплина "${escapeHtml(data.name || name)}" добавлена`, 'success');
|
||||||
createSubjectForm.reset();
|
createSubjectForm.reset();
|
||||||
loadSubjects();
|
loadSubjects();
|
||||||
} catch (e) { showAlert('create-subject-alert', e.message || 'Ошибка создания', 'error'); }
|
} catch (e) { showAlert('create-subject-alert', e.message || 'Ошибка создания', 'error'); }
|
||||||
|
|||||||
@@ -196,14 +196,14 @@ export async function initUsers() {
|
|||||||
renderUsers(users);
|
renderUsers(users);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
usersTbody.innerHTML =
|
usersTbody.innerHTML =
|
||||||
'<tr><td colspan="4" class="loading-row">Ошибка загрузки: ' +
|
'<tr><td colspan="8" class="loading-row">Ошибка загрузки: ' +
|
||||||
escapeHtml(e.message) + '</td></tr>';
|
escapeHtml(e.message) + '</td></tr>';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderUsers(users) {
|
function renderUsers(users) {
|
||||||
if (!users || !users.length) {
|
if (!users || !users.length) {
|
||||||
usersTbody.innerHTML = '<tr><td colspan="4" class="loading-row">Нет пользователей</td></tr>';
|
usersTbody.innerHTML = '<tr><td colspan="8" class="loading-row">Нет пользователей</td></tr>';
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -211,6 +211,9 @@ export async function initUsers() {
|
|||||||
<tr>
|
<tr>
|
||||||
<td>${u.id}</td>
|
<td>${u.id}</td>
|
||||||
<td>${escapeHtml(u.username)}</td>
|
<td>${escapeHtml(u.username)}</td>
|
||||||
|
<td>${escapeHtml(u.fullName || '-')}</td>
|
||||||
|
<td>${escapeHtml(u.jobTitle || '-')}</td>
|
||||||
|
<td>${u.departmentId || '-'}</td>
|
||||||
<td><span class="badge ${ROLE_BADGE[u.role] || ''}">${ROLE_LABELS[u.role] || escapeHtml(u.role)}</span></td>
|
<td><span class="badge ${ROLE_BADGE[u.role] || ''}">${ROLE_LABELS[u.role] || escapeHtml(u.role)}</span></td>
|
||||||
<td>
|
<td>
|
||||||
<button class="btn-delete" data-id="${u.id}">Удалить</button>
|
<button class="btn-delete" data-id="${u.id}">Удалить</button>
|
||||||
@@ -378,14 +381,24 @@ export async function initUsers() {
|
|||||||
const username = document.getElementById('new-username').value.trim();
|
const username = document.getElementById('new-username').value.trim();
|
||||||
const password = document.getElementById('new-password').value;
|
const password = document.getElementById('new-password').value;
|
||||||
const role = document.getElementById('new-role').value;
|
const role = document.getElementById('new-role').value;
|
||||||
|
const fullName = document.getElementById('new-fullname').value.trim();
|
||||||
|
const jobTitle = document.getElementById('new-jobtitle').value.trim();
|
||||||
|
const departmentId = document.getElementById('new-department').value;
|
||||||
|
|
||||||
if (!username || !password) {
|
if (!username || !password || !fullName || !jobTitle || !departmentId) {
|
||||||
showAlert('create-alert', 'Заполните все поля', 'error');
|
showAlert('create-alert', 'Заполните все поля', 'error');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const data = await api.post('/api/users', { username, password, role });
|
const data = await api.post('/api/users', {
|
||||||
|
username,
|
||||||
|
password,
|
||||||
|
role,
|
||||||
|
fullName,
|
||||||
|
jobTitle,
|
||||||
|
departmentId: Number(departmentId)
|
||||||
|
});
|
||||||
showAlert('create-alert', `Пользователь "${escapeHtml(data.username)}" создан`, 'success');
|
showAlert('create-alert', `Пользователь "${escapeHtml(data.username)}" создан`, 'success');
|
||||||
createForm.reset();
|
createForm.reset();
|
||||||
loadUsers();
|
loadUsers();
|
||||||
|
|||||||
73
frontend/admin/views/database.html
Executable file
73
frontend/admin/views/database.html
Executable file
@@ -0,0 +1,73 @@
|
|||||||
|
<!-- ===== Database / Tenants Tab ===== -->
|
||||||
|
|
||||||
|
<!-- Текущее подключение -->
|
||||||
|
<div class="card">
|
||||||
|
<h2>Текущее подключение</h2>
|
||||||
|
<div id="db-status-info" class="db-status-card">
|
||||||
|
<div class="loading-row">Загрузка...</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Таблица тенантов -->
|
||||||
|
<div class="card">
|
||||||
|
<h2>Подключённые университеты</h2>
|
||||||
|
<div class="table-wrap">
|
||||||
|
<table id="tenants-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Название</th>
|
||||||
|
<th>Домен</th>
|
||||||
|
<th>JDBC URL</th>
|
||||||
|
<th>Пользователь</th>
|
||||||
|
<th>Статус</th>
|
||||||
|
<th>Действия</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody id="tenants-tbody">
|
||||||
|
<tr>
|
||||||
|
<td colspan="6" class="loading-row">Загрузка...</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Добавить тенант -->
|
||||||
|
<div class="card create-card">
|
||||||
|
<h2>Добавить подключение</h2>
|
||||||
|
<form id="add-tenant-form">
|
||||||
|
<div class="form-row">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="tenant-name">Название университета</label>
|
||||||
|
<input type="text" id="tenant-name" placeholder="ЮЗГУ" required>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="tenant-domain">Поддомен</label>
|
||||||
|
<input type="text" id="tenant-domain" placeholder="swsu" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-row" style="margin-top: 0.75rem;">
|
||||||
|
<div class="form-group" style="flex: 3;">
|
||||||
|
<label for="tenant-url">JDBC URL</label>
|
||||||
|
<input type="text" id="tenant-url" placeholder="jdbc:postgresql://192.168.1.50:5432/magistr_db" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-row" style="margin-top: 0.75rem;">
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="tenant-username">Пользователь</label>
|
||||||
|
<input type="text" id="tenant-username" placeholder="postgres" required>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="tenant-password">Пароль</label>
|
||||||
|
<input type="password" id="tenant-password" placeholder="••••••••" required>
|
||||||
|
</div>
|
||||||
|
<button type="button" class="btn-primary" id="btn-test-connection" style="height: fit-content;">
|
||||||
|
Тест
|
||||||
|
</button>
|
||||||
|
<button type="submit" class="btn-primary" style="height: fit-content;">
|
||||||
|
Добавить
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="form-alert" id="add-tenant-alert" role="alert"></div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
193
frontend/admin/views/department.html
Normal file
193
frontend/admin/views/department.html
Normal file
@@ -0,0 +1,193 @@
|
|||||||
|
<div class="card">
|
||||||
|
<h2>Кафедра</h2>
|
||||||
|
|
||||||
|
<div class="filter-row" style="gap:.75rem;">
|
||||||
|
<label for="recordsSearch">Поиск</label>
|
||||||
|
<input
|
||||||
|
id="recordsSearch"
|
||||||
|
class="records-search"
|
||||||
|
type="search"
|
||||||
|
placeholder="Группа, дисциплина, преподаватель…"
|
||||||
|
autocomplete="off"
|
||||||
|
/>
|
||||||
|
<button type="button" class="btn-delete" id="recordsSearchClear">Сброс</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<div class="table-wrap">
|
||||||
|
|
||||||
|
<!-- Таблица 1 -->
|
||||||
|
<details class="table-item">
|
||||||
|
<summary>
|
||||||
|
<div class="chev" aria-hidden="true">
|
||||||
|
<svg viewBox="0 0 20 20" class="chev-icon" focusable="false" aria-hidden="true">
|
||||||
|
<path d="M5.5 7.5L10 12l4.5-4.5" fill="none" stroke="currentColor" stroke-width="2"
|
||||||
|
stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
<div class="title title-multiline">
|
||||||
|
<span class="title-main">Данные к составлению расписания</span>
|
||||||
|
<span class="title-sub">Кафедра: <b>Информационная безопасность</b></span>
|
||||||
|
<span class="title-sub">Факультет: <b>ФиПИ</b></span>
|
||||||
|
<span class="title-sub">Семестр: <b>весенний</b></span>
|
||||||
|
<span class="title-sub">Уч. год: <b>2024/2025</b></span>
|
||||||
|
</div>
|
||||||
|
<div class="meta">3 записи</div>
|
||||||
|
</summary>
|
||||||
|
|
||||||
|
<div class="content">
|
||||||
|
<table>
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Специальность</th>
|
||||||
|
<th>Курс и семестр</th>
|
||||||
|
<th>Группа</th>
|
||||||
|
<th>Дисциплина</th>
|
||||||
|
<th>Вид занятий</th>
|
||||||
|
<th>Часов в неделю</th>
|
||||||
|
<th>Деление на подгруппы</th>
|
||||||
|
<th>Фамилия преподавателя</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
|
||||||
|
<tbody>
|
||||||
|
<!-- 1 строка = 1 запись HARDCODE -->
|
||||||
|
<tr>
|
||||||
|
<td>09.02.07</td>
|
||||||
|
<td>2 курс, 4 семестр</td>
|
||||||
|
<td>ИС-21</td>
|
||||||
|
<td>Базы данных</td>
|
||||||
|
<td>Лабораторная</td>
|
||||||
|
<td>2</td>
|
||||||
|
<td>Да</td>
|
||||||
|
<td>Иванов</td>
|
||||||
|
</tr>
|
||||||
|
|
||||||
|
<tr>
|
||||||
|
<td>09.02.07</td>
|
||||||
|
<td>2 курс, 4 семестр</td>
|
||||||
|
<td>ИС-22</td>
|
||||||
|
<td>Операционные системы</td>
|
||||||
|
<td>Практика</td>
|
||||||
|
<td>1</td>
|
||||||
|
<td>Нет</td>
|
||||||
|
<td>Смирнов</td>
|
||||||
|
</tr>
|
||||||
|
|
||||||
|
<tr>
|
||||||
|
<td>09.02.07</td>
|
||||||
|
<td>1 курс, 2 семестр</td>
|
||||||
|
<td>ИС-12</td>
|
||||||
|
<td>Алгоритмы</td>
|
||||||
|
<td>Лекция</td>
|
||||||
|
<td>2</td>
|
||||||
|
<td>Нет</td>
|
||||||
|
<td>Кузнецов</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</details>
|
||||||
|
|
||||||
|
<!-- Таблица 2 -->
|
||||||
|
<details class="table-item">
|
||||||
|
<summary>
|
||||||
|
<div class="chev" aria-hidden="true">
|
||||||
|
<svg viewBox="0 0 20 20" class="chev-icon" focusable="false" aria-hidden="true">
|
||||||
|
<path d="M5.5 7.5L10 12l4.5-4.5" fill="none" stroke="currentColor" stroke-width="2"
|
||||||
|
stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
<div class="title">orders</div>
|
||||||
|
<div class="meta">1 запись</div>
|
||||||
|
</summary>
|
||||||
|
|
||||||
|
<div class="content">
|
||||||
|
<table>
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Специальность</th>
|
||||||
|
<th>Курс и семестр</th>
|
||||||
|
<th>Группа</th>
|
||||||
|
<th>Дисциплина</th>
|
||||||
|
<th>Вид занятий</th>
|
||||||
|
<th>Часов в неделю</th>
|
||||||
|
<th>Деление на подгруппы</th>
|
||||||
|
<th>Фамилия преподавателя</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td>38.02.01</td>
|
||||||
|
<td>1 курс, 1 семестр</td>
|
||||||
|
<td>ЭК-11</td>
|
||||||
|
<td>Экономика</td>
|
||||||
|
<td>Лекция</td>
|
||||||
|
<td>1</td>
|
||||||
|
<td>Нет</td>
|
||||||
|
<td>Петров</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</details>
|
||||||
|
|
||||||
|
<!-- Таблица 3 -->
|
||||||
|
<details class="table-item">
|
||||||
|
<summary>
|
||||||
|
<div class="chev" aria-hidden="true">
|
||||||
|
<svg viewBox="0 0 20 20" class="chev-icon" focusable="false" aria-hidden="true">
|
||||||
|
<path d="M5.5 7.5L10 12l4.5-4.5" fill="none" stroke="currentColor" stroke-width="2"
|
||||||
|
stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
<div class="title">products</div>
|
||||||
|
<div class="meta">2 записи</div>
|
||||||
|
</summary>
|
||||||
|
|
||||||
|
<div class="content">
|
||||||
|
<table>
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Специальность</th>
|
||||||
|
<th>Курс и семестр</th>
|
||||||
|
<th>Группа</th>
|
||||||
|
<th>Дисциплина</th>
|
||||||
|
<th>Вид занятий</th>
|
||||||
|
<th>Часов в неделю</th>
|
||||||
|
<th>Деление на подгруппы</th>
|
||||||
|
<th>Фамилия преподавателя</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td>15.02.08</td>
|
||||||
|
<td>3 курс, 6 семестр</td>
|
||||||
|
<td>МС-31</td>
|
||||||
|
<td>Материаловедение</td>
|
||||||
|
<td>Практика</td>
|
||||||
|
<td>3</td>
|
||||||
|
<td>Да</td>
|
||||||
|
<td>Сидоров</td>
|
||||||
|
</tr>
|
||||||
|
|
||||||
|
<tr>
|
||||||
|
<td>15.02.08</td>
|
||||||
|
<td>3 курс, 6 семестр</td>
|
||||||
|
<td>МС-32</td>
|
||||||
|
<td>Технология металлов</td>
|
||||||
|
<td>Лабораторная</td>
|
||||||
|
<td>2</td>
|
||||||
|
<td>Да</td>
|
||||||
|
<td>Орлов</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</details>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
@@ -17,6 +17,14 @@
|
|||||||
<option value="">Загрузка...</option>
|
<option value="">Загрузка...</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="new-group-department">ID кафедры</label>
|
||||||
|
<input type="number" id="new-group-department" placeholder="ID" required>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="new-group-course">Курс</label>
|
||||||
|
<input type="number" id="new-group-course" placeholder="1-6" min="1" max="6" required>
|
||||||
|
</div>
|
||||||
<button type="submit" class="btn-primary">Создать</button>
|
<button type="submit" class="btn-primary">Создать</button>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-alert" id="create-group-alert" role="alert"></div>
|
<div class="form-alert" id="create-group-alert" role="alert"></div>
|
||||||
@@ -41,12 +49,14 @@
|
|||||||
<th>Название</th>
|
<th>Название</th>
|
||||||
<th>Численность (чел.)</th>
|
<th>Численность (чел.)</th>
|
||||||
<th>Форма обучения</th>
|
<th>Форма обучения</th>
|
||||||
|
<th>ID кафедры</th>
|
||||||
|
<th>Курс</th>
|
||||||
<th>Действия</th>
|
<th>Действия</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody id="groups-tbody">
|
<tbody id="groups-tbody">
|
||||||
<tr>
|
<tr>
|
||||||
<td colspan="4" class="loading-row">Загрузка...</td>
|
<td colspan="7" class="loading-row">Загрузка...</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|||||||
@@ -7,6 +7,14 @@
|
|||||||
<label for="new-subject-name">Название дисциплины</label>
|
<label for="new-subject-name">Название дисциплины</label>
|
||||||
<input type="text" id="new-subject-name" placeholder="Высшая математика" required>
|
<input type="text" id="new-subject-name" placeholder="Высшая математика" required>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="new-subject-code">Код предмета</label>
|
||||||
|
<input type="text" id="new-subject-code" placeholder="Например: MATH101" required>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="new-subject-department">Идентификатор кафедры</label>
|
||||||
|
<input type="number" id="new-subject-department" placeholder="ID кафедры" required>
|
||||||
|
</div>
|
||||||
<button type="submit" class="btn-primary">Добавить</button>
|
<button type="submit" class="btn-primary">Добавить</button>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-alert" id="create-subject-alert" role="alert"></div>
|
<div class="form-alert" id="create-subject-alert" role="alert"></div>
|
||||||
@@ -43,12 +51,14 @@
|
|||||||
<tr>
|
<tr>
|
||||||
<th>ID</th>
|
<th>ID</th>
|
||||||
<th>Название</th>
|
<th>Название</th>
|
||||||
|
<th>Код предмета</th>
|
||||||
|
<th>Кафедра (ID)</th>
|
||||||
<th>Действия</th>
|
<th>Действия</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody id="subjects-tbody">
|
<tbody id="subjects-tbody">
|
||||||
<tr>
|
<tr>
|
||||||
<td colspan="3" class="loading-row">Загрузка...</td>
|
<td colspan="5" class="loading-row">Загрузка...</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|||||||
@@ -19,6 +19,18 @@
|
|||||||
<option value="ADMIN">Администратор</option>
|
<option value="ADMIN">Администратор</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="new-fullname">ФИО пользователя</label>
|
||||||
|
<input type="text" id="new-fullname" placeholder="Иванов Иван Иванович" required>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="new-jobtitle">Должность</label>
|
||||||
|
<input type="text" id="new-jobtitle" placeholder="Студент / Доцент" required>
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="new-department">ID кафедры</label>
|
||||||
|
<input type="number" id="new-department" placeholder="ID" required>
|
||||||
|
</div>
|
||||||
<button type="submit" class="btn-primary">Создать</button>
|
<button type="submit" class="btn-primary">Создать</button>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-alert" id="create-alert" role="alert"></div>
|
<div class="form-alert" id="create-alert" role="alert"></div>
|
||||||
@@ -33,13 +45,16 @@
|
|||||||
<tr>
|
<tr>
|
||||||
<th>ID</th>
|
<th>ID</th>
|
||||||
<th>Имя пользователя</th>
|
<th>Имя пользователя</th>
|
||||||
|
<th>ФИО</th>
|
||||||
|
<th>Должность</th>
|
||||||
|
<th>ID кафедры</th>
|
||||||
<th>Роль</th>
|
<th>Роль</th>
|
||||||
<th>Действия</th>
|
<th colspan="2">Действия</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody id="users-tbody">
|
<tbody id="users-tbody">
|
||||||
<tr>
|
<tr>
|
||||||
<td colspan="4" class="loading-row">Загрузка...</td>
|
<td colspan="8" class="loading-row">Загрузка...</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|||||||
@@ -1,6 +1,32 @@
|
|||||||
(() => {
|
(() => {
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
// --- OpenTelemetry Frontend Instrumentation ---
|
||||||
|
// Загружаем OTel Web SDK динамически через esm.sh, чтобы не ломать старый Vanilla JS (без type="module")
|
||||||
|
import('https://esm.sh/@opentelemetry/sdk-trace-web').then(async ({ WebTracerProvider, BatchSpanProcessor }) => {
|
||||||
|
const { OTLPTraceExporter } = await import('https://esm.sh/@opentelemetry/exporter-trace-otlp-http');
|
||||||
|
const { getWebAutoInstrumentations } = await import('https://esm.sh/@opentelemetry/auto-instrumentations-web');
|
||||||
|
const { registerInstrumentations } = await import('https://esm.sh/@opentelemetry/instrumentation');
|
||||||
|
const { Resource } = await import('https://esm.sh/@opentelemetry/resources');
|
||||||
|
|
||||||
|
const exporter = new OTLPTraceExporter({
|
||||||
|
url: window.location.origin + '/otel/v1/traces' // Трафик пойдет через ваш Caddy Proxy
|
||||||
|
});
|
||||||
|
|
||||||
|
const provider = new WebTracerProvider({
|
||||||
|
resource: new Resource({ 'service.name': 'magistr-frontend' }),
|
||||||
|
});
|
||||||
|
|
||||||
|
provider.addSpanProcessor(new BatchSpanProcessor(exporter));
|
||||||
|
provider.register();
|
||||||
|
|
||||||
|
registerInstrumentations({
|
||||||
|
instrumentations: [getWebAutoInstrumentations()]
|
||||||
|
});
|
||||||
|
console.log("SigNoz (OpenTelemetry) инициализирован во фронтенде.");
|
||||||
|
}).catch(e => console.error("Ошибка загрузки OTel:", e));
|
||||||
|
// ----------------------------------------------
|
||||||
|
|
||||||
const form = document.getElementById('login-form');
|
const form = document.getElementById('login-form');
|
||||||
const usernameInput = document.getElementById('username');
|
const usernameInput = document.getElementById('username');
|
||||||
const passwordInput = document.getElementById('password');
|
const passwordInput = document.getElementById('password');
|
||||||
|
|||||||
Reference in New Issue
Block a user