mirror of
https://github.com/Direct-Dev-Ru/go-lcg.git
synced 2025-11-16 17:49:55 +00:00
Compare commits
20 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| e54b99f6f4 | |||
| 4779c4bca4 | |||
| 1e2ce929b2 | |||
| 9aa5aefdad | |||
| 7455987c0f | |||
| 3e143ee7a1 | |||
| 96a8060afb | |||
| 99b1a74034 | |||
| 89d15bfdc9 | |||
| 5c672ecc39 | |||
| 164f32dbaf | |||
| 1545fe2508 | |||
| deb80f2b37 | |||
| 3c95eb85db | |||
| 5b78e775c1 | |||
| 9044b02d27 | |||
| e7c11879a1 | |||
| 6444c35bbb | |||
| 6ec41355d3 | |||
| 7a0d0746d4 |
32
.goreleaser.yaml
Normal file
32
.goreleaser.yaml
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
# Goreleaser configuration version 2
|
||||||
|
version: 2
|
||||||
|
|
||||||
|
builds:
|
||||||
|
- id: lcg
|
||||||
|
binary: "lcg_{{ .Version }}"
|
||||||
|
goos:
|
||||||
|
- linux
|
||||||
|
- windows
|
||||||
|
- darwin
|
||||||
|
goarch:
|
||||||
|
- amd64
|
||||||
|
- arm64
|
||||||
|
env:
|
||||||
|
- CGO_ENABLED=0
|
||||||
|
ldflags:
|
||||||
|
- -s -w
|
||||||
|
- -X main.version={{.Version}}
|
||||||
|
- -X main.commit={{.Commit}}
|
||||||
|
- -X main.date={{.Date}}
|
||||||
|
main: .
|
||||||
|
dir: .
|
||||||
|
|
||||||
|
archives:
|
||||||
|
- id: lcg
|
||||||
|
ids:
|
||||||
|
- lcg
|
||||||
|
formats:
|
||||||
|
- binary
|
||||||
|
name_template: "{{ .Binary }}_{{ .Os }}_{{ .Arch }}"
|
||||||
|
files:
|
||||||
|
- "lcg_{{ .Version }}"
|
||||||
@@ -1,6 +1,36 @@
|
|||||||
CHANGELOG
|
CHANGELOG
|
||||||
=========
|
=========
|
||||||
|
|
||||||
|
Версия 2.0.6 (2025-10-28)
|
||||||
|
=========================
|
||||||
|
|
||||||
|
## ✨ НОВОЕ И ИЗМЕНЕНО
|
||||||
|
|
||||||
|
- 🌐 Поддержка BasePath для всех веб‑роутов и шаблонов
|
||||||
|
- Новый параметр: `LCG_BASE_URL` (пример: `/lcg`) — префикс для всех страниц и API
|
||||||
|
- Обновлены редиректы и middleware с учетом BasePath
|
||||||
|
- 🧭 Кастомная страница 404 (красная тема), показывается для любого неизвестного пути под BasePath
|
||||||
|
- 📱 Улучшена мобильная верстка результатов — стиль карточек как в истории
|
||||||
|
- 🗂️ Человекочитаемые заголовки результатов: преобразование имени файла в «заголовок — дата время»
|
||||||
|
- 🗑️ Иконки удаления: единый бледно‑красный крест ✖ в результатах и истории
|
||||||
|
|
||||||
|
## 🐛 ИСПРАВЛЕНИЯ
|
||||||
|
|
||||||
|
- 🛡️ Исправлен просмотр/удаление файла при включенном BasePath (правильный разбор URL)
|
||||||
|
- 🧰 На старте сервера гарантируется создание `ResultFolder` и пустого `ResultHistory` (без 500)
|
||||||
|
- 🚧 Главная страница обрабатывается только по точному пути BasePath, а не по произвольным под‑путям
|
||||||
|
|
||||||
|
## ⚙️ КОНФИГУРАЦИЯ
|
||||||
|
|
||||||
|
- 🔍 Debug режим теперь включается и флагом `--debug`, и переменной `LCG_DEBUG=1|true`
|
||||||
|
- 🍪 Уточнена работа с `CookiePath`/`BasePath` в middleware
|
||||||
|
|
||||||
|
## 📚 ДОКУМЕНТАЦИЯ
|
||||||
|
|
||||||
|
- Обновлены `README.md`, `USAGE_GUIDE.md`, `API_GUIDE.md`, `REVERSE_PROXY_GUIDE.md` — добавлены примеры с BasePath и примечания к 404
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
Версия 2.0.1 (2025-10-22)
|
Версия 2.0.1 (2025-10-22)
|
||||||
=========================
|
=========================
|
||||||
|
|
||||||
|
|||||||
52
Dockerfiles/OllamaServer/.dockerignore
Normal file
52
Dockerfiles/OllamaServer/.dockerignore
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
# Git
|
||||||
|
.git
|
||||||
|
.gitignore
|
||||||
|
.gitattributes
|
||||||
|
|
||||||
|
# Build artifacts
|
||||||
|
dist/
|
||||||
|
*.exe
|
||||||
|
*.dll
|
||||||
|
*.so
|
||||||
|
*.dylib
|
||||||
|
lcg
|
||||||
|
go-lcg
|
||||||
|
|
||||||
|
# IDE
|
||||||
|
.idea/
|
||||||
|
.vscode/
|
||||||
|
*.swp
|
||||||
|
*.swo
|
||||||
|
*~
|
||||||
|
|
||||||
|
# Documentation
|
||||||
|
docs/
|
||||||
|
*.md
|
||||||
|
!README.md
|
||||||
|
|
||||||
|
# Tests
|
||||||
|
*_test.go
|
||||||
|
test_*.sh
|
||||||
|
|
||||||
|
# Deployment scripts
|
||||||
|
deploy/
|
||||||
|
shell-code/
|
||||||
|
kustomize/
|
||||||
|
|
||||||
|
# Temporary files
|
||||||
|
*.log
|
||||||
|
*.tmp
|
||||||
|
*.temp
|
||||||
|
|
||||||
|
# OS files
|
||||||
|
.DS_Store
|
||||||
|
Thumbs.db
|
||||||
|
|
||||||
|
# Go
|
||||||
|
vendor/
|
||||||
|
|
||||||
|
# Docker
|
||||||
|
Dockerfile*
|
||||||
|
docker-compose*.yml
|
||||||
|
.dockerignore
|
||||||
|
|
||||||
15
Dockerfiles/OllamaServer/.gitignore
vendored
Normal file
15
Dockerfiles/OllamaServer/.gitignore
vendored
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
# Временные файлы
|
||||||
|
*.log
|
||||||
|
*.tmp
|
||||||
|
*.temp
|
||||||
|
|
||||||
|
# IDE
|
||||||
|
.idea/
|
||||||
|
.vscode/
|
||||||
|
*.swp
|
||||||
|
*.swo
|
||||||
|
|
||||||
|
# OS files
|
||||||
|
.DS_Store
|
||||||
|
Thumbs.db
|
||||||
|
|
||||||
78
Dockerfiles/OllamaServer/Dockerfile
Normal file
78
Dockerfiles/OllamaServer/Dockerfile
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
# Multi-stage build для LCG с Ollama
|
||||||
|
FROM golang:1.24.6-alpine3.22 AS builder
|
||||||
|
|
||||||
|
WORKDIR /build
|
||||||
|
|
||||||
|
# Копируем файлы зависимостей
|
||||||
|
COPY go.mod go.sum ./
|
||||||
|
RUN go mod download
|
||||||
|
|
||||||
|
# Копируем исходный код
|
||||||
|
COPY . .
|
||||||
|
|
||||||
|
# Собираем бинарник
|
||||||
|
RUN CGO_ENABLED=0 GOOS=linux go build -ldflags="-w -s -buildid=" -trimpath -o /build/lcg .
|
||||||
|
|
||||||
|
# Финальный образ с Ollama
|
||||||
|
FROM alpine:3.22
|
||||||
|
|
||||||
|
# Устанавливаем необходимые пакеты
|
||||||
|
RUN apk add --no-cache \
|
||||||
|
curl \
|
||||||
|
ca-certificates \
|
||||||
|
bash \
|
||||||
|
&& rm -rf /var/cache/apk/*
|
||||||
|
|
||||||
|
# Устанавливаем Ollama 0.9.5 (поддержка разных архитектур)
|
||||||
|
ARG TARGETARCH
|
||||||
|
RUN case ${TARGETARCH} in \
|
||||||
|
amd64) OLLAMA_ARCH=amd64 ;; \
|
||||||
|
arm64) OLLAMA_ARCH=arm64 ;; \
|
||||||
|
arm) OLLAMA_ARCH=arm64 ;; \
|
||||||
|
*) echo "Unsupported architecture: ${TARGETARCH}" && exit 1 ;; \
|
||||||
|
esac && \
|
||||||
|
curl -L https://github.com/ollama/ollama/releases/download/v0.9.5/ollama-linux-${OLLAMA_ARCH} -o /usr/local/bin/ollama \
|
||||||
|
&& chmod +x /usr/local/bin/ollama
|
||||||
|
|
||||||
|
# Создаем пользователя для запуска сервисов
|
||||||
|
RUN addgroup -g 1000 ollama && \
|
||||||
|
adduser -D -u 1000 -G ollama ollama && \
|
||||||
|
mkdir -p /home/ollama/.ollama && \
|
||||||
|
chown -R ollama:ollama /home/ollama
|
||||||
|
|
||||||
|
# Копируем бинарник lcg
|
||||||
|
COPY --from=builder /build/lcg /usr/local/bin/lcg
|
||||||
|
RUN chmod +x /usr/local/bin/lcg
|
||||||
|
|
||||||
|
# Копируем entrypoint скрипт
|
||||||
|
COPY --chmod=755 Dockerfiles/OllamaServer/entrypoint.sh /entrypoint.sh
|
||||||
|
|
||||||
|
# Создаем директории для данных
|
||||||
|
RUN mkdir -p /app/data/results /app/data/prompts /app/data/config \
|
||||||
|
&& chown -R ollama:ollama /app/data
|
||||||
|
|
||||||
|
# Настройки по умолчанию
|
||||||
|
ENV LCG_PROVIDER=ollama
|
||||||
|
ENV LCG_HOST=http://127.0.0.1:11434/
|
||||||
|
ENV LCG_MODEL=codegeex4
|
||||||
|
ENV LCG_RESULT_FOLDER=/app/data/results
|
||||||
|
ENV LCG_PROMPT_FOLDER=/app/data/prompts
|
||||||
|
ENV LCG_CONFIG_FOLDER=/app/data/config
|
||||||
|
ENV LCG_SERVER_HOST=0.0.0.0
|
||||||
|
ENV LCG_SERVER_PORT=8080
|
||||||
|
ENV LCG_SERVER_ALLOW_HTTP=true
|
||||||
|
ENV OLLAMA_HOST=0.0.0.0
|
||||||
|
ENV OLLAMA_PORT=11434
|
||||||
|
|
||||||
|
# Expose порты
|
||||||
|
EXPOSE 8080 11434
|
||||||
|
|
||||||
|
# Переключаемся на пользователя ollama
|
||||||
|
USER ollama
|
||||||
|
|
||||||
|
WORKDIR /home/ollama
|
||||||
|
|
||||||
|
# Запускаем entrypoint
|
||||||
|
ENTRYPOINT ["/entrypoint.sh"]
|
||||||
|
CMD []
|
||||||
|
|
||||||
107
Dockerfiles/OllamaServer/Makefile
Normal file
107
Dockerfiles/OllamaServer/Makefile
Normal file
@@ -0,0 +1,107 @@
|
|||||||
|
.PHONY: build build-podman run run-podman stop stop-podman logs logs-podman clean help
|
||||||
|
|
||||||
|
# Переменные
|
||||||
|
IMAGE_NAME = lcg-ollama
|
||||||
|
IMAGE_TAG = latest
|
||||||
|
CONTAINER_NAME = lcg-ollama
|
||||||
|
DOCKERFILE = Dockerfile
|
||||||
|
CONTEXT = ../..
|
||||||
|
|
||||||
|
help: ## Показать справку
|
||||||
|
@echo "Доступные команды:"
|
||||||
|
@echo " make build - Собрать Docker образ"
|
||||||
|
@echo " make build-podman - Собрать Podman образ"
|
||||||
|
@echo " make run - Запустить контейнер (Docker)"
|
||||||
|
@echo " make run-podman - Запустить контейнер (Podman)"
|
||||||
|
@echo " make stop - Остановить контейнер (Docker)"
|
||||||
|
@echo " make stop-podman - Остановить контейнер (Podman)"
|
||||||
|
@echo " make logs - Показать логи (Docker)"
|
||||||
|
@echo " make logs-podman - Показать логи (Podman)"
|
||||||
|
@echo " make clean - Удалить контейнер и образ"
|
||||||
|
@echo " make compose-up - Запустить через docker-compose"
|
||||||
|
@echo " make compose-down - Остановить docker-compose"
|
||||||
|
@echo " make podman-compose-up - Запустить через podman-compose"
|
||||||
|
@echo " make podman-compose-down - Остановить podman-compose"
|
||||||
|
|
||||||
|
build: ## Собрать Docker образ
|
||||||
|
docker build -f $(DOCKERFILE) -t $(IMAGE_NAME):$(IMAGE_TAG) $(CONTEXT)
|
||||||
|
@echo "Образ $(IMAGE_NAME):$(IMAGE_TAG) успешно собран"
|
||||||
|
|
||||||
|
build-podman: ## Собрать Podman образ
|
||||||
|
podman build -f $(DOCKERFILE) -t $(IMAGE_NAME):$(IMAGE_TAG) $(CONTEXT)
|
||||||
|
@echo "Образ $(IMAGE_NAME):$(IMAGE_TAG) успешно собран"
|
||||||
|
|
||||||
|
run: ## Запустить контейнер (Docker)
|
||||||
|
docker run -d \
|
||||||
|
--name $(CONTAINER_NAME) \
|
||||||
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
|
-v ollama-data:/home/ollama/.ollama \
|
||||||
|
-v lcg-results:/app/data/results \
|
||||||
|
-v lcg-prompts:/app/data/prompts \
|
||||||
|
-v lcg-config:/app/data/config \
|
||||||
|
$(IMAGE_NAME):$(IMAGE_TAG)
|
||||||
|
@echo "Контейнер $(CONTAINER_NAME) запущен"
|
||||||
|
|
||||||
|
run-podman: ## Запустить контейнер (Podman)
|
||||||
|
podman run -d \
|
||||||
|
--name $(CONTAINER_NAME) \
|
||||||
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
|
-v ollama-data:/home/ollama/.ollama \
|
||||||
|
-v lcg-results:/app/data/results \
|
||||||
|
-v lcg-prompts:/app/data/prompts \
|
||||||
|
-v lcg-config:/app/data/config \
|
||||||
|
$(IMAGE_NAME):$(IMAGE_TAG)
|
||||||
|
@echo "Контейнер $(CONTAINER_NAME) запущен"
|
||||||
|
|
||||||
|
stop: ## Остановить контейнер (Docker)
|
||||||
|
docker stop $(CONTAINER_NAME) || true
|
||||||
|
docker rm $(CONTAINER_NAME) || true
|
||||||
|
@echo "Контейнер $(CONTAINER_NAME) остановлен и удален"
|
||||||
|
|
||||||
|
stop-podman: ## Остановить контейнер (Podman)
|
||||||
|
podman stop $(CONTAINER_NAME) || true
|
||||||
|
podman rm $(CONTAINER_NAME) || true
|
||||||
|
@echo "Контейнер $(CONTAINER_NAME) остановлен и удален"
|
||||||
|
|
||||||
|
logs: ## Показать логи (Docker)
|
||||||
|
docker logs -f $(CONTAINER_NAME)
|
||||||
|
|
||||||
|
logs-podman: ## Показать логи (Podman)
|
||||||
|
podman logs -f $(CONTAINER_NAME)
|
||||||
|
|
||||||
|
clean: ## Удалить контейнер и образ
|
||||||
|
docker stop $(CONTAINER_NAME) || true
|
||||||
|
docker rm $(CONTAINER_NAME) || true
|
||||||
|
docker rmi $(IMAGE_NAME):$(IMAGE_TAG) || true
|
||||||
|
@echo "Контейнер и образ удалены"
|
||||||
|
|
||||||
|
compose-up: ## Запустить через docker-compose
|
||||||
|
docker-compose up -d
|
||||||
|
@echo "Сервисы запущены через docker-compose"
|
||||||
|
|
||||||
|
compose-down: ## Остановить docker-compose
|
||||||
|
docker-compose down
|
||||||
|
@echo "Сервисы остановлены"
|
||||||
|
|
||||||
|
podman-compose-up: ## Запустить через podman-compose
|
||||||
|
podman-compose -f podman-compose.yml up -d
|
||||||
|
@echo "Сервисы запущены через podman-compose"
|
||||||
|
|
||||||
|
podman-compose-down: ## Остановить podman-compose
|
||||||
|
podman-compose -f podman-compose.yml down
|
||||||
|
@echo "Сервисы остановлены"
|
||||||
|
|
||||||
|
shell: ## Подключиться к контейнеру (Docker)
|
||||||
|
docker exec -it $(CONTAINER_NAME) sh
|
||||||
|
|
||||||
|
shell-podman: ## Подключиться к контейнеру (Podman)
|
||||||
|
podman exec -it $(CONTAINER_NAME) sh
|
||||||
|
|
||||||
|
pull-model: ## Загрузить модель codegeex4 (Docker)
|
||||||
|
docker exec $(CONTAINER_NAME) ollama pull codegeex4
|
||||||
|
|
||||||
|
pull-model-podman: ## Загрузить модель codegeex4 (Podman)
|
||||||
|
podman exec $(CONTAINER_NAME) ollama pull codegeex4
|
||||||
|
|
||||||
172
Dockerfiles/OllamaServer/QUICKSTART.md
Normal file
172
Dockerfiles/OllamaServer/QUICKSTART.md
Normal file
@@ -0,0 +1,172 @@
|
|||||||
|
# 🚀 Быстрый старт - LCG с Ollama
|
||||||
|
|
||||||
|
## Подготовка
|
||||||
|
|
||||||
|
1. Убедитесь, что у вас установлен Docker или Podman
|
||||||
|
2. Клонируйте репозиторий (если еще не сделали)
|
||||||
|
3. Перейдите в папку с Dockerfile
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd Dockerfiles/OllamaServer
|
||||||
|
```
|
||||||
|
|
||||||
|
## Запуск с Docker
|
||||||
|
|
||||||
|
### Вариант 1: Docker Compose (рекомендуется)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker-compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
### Вариант 2: Ручная сборка и запуск
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Сборка образа
|
||||||
|
docker build -f Dockerfile -t lcg-ollama:latest ../..
|
||||||
|
|
||||||
|
# Запуск контейнера
|
||||||
|
docker run -d \
|
||||||
|
--name lcg-ollama \
|
||||||
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
|
-v ollama-data:/home/ollama/.ollama \
|
||||||
|
-v lcg-results:/app/data/results \
|
||||||
|
lcg-ollama:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
## Запуск с Podman
|
||||||
|
|
||||||
|
### Вариант 1: Podman Compose
|
||||||
|
|
||||||
|
```bash
|
||||||
|
podman-compose -f podman-compose.yml up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
### Вариант 2: Ручная сборка и запуск
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Сборка образа
|
||||||
|
podman build -f Dockerfile -t lcg-ollama:latest ../..
|
||||||
|
|
||||||
|
# Запуск контейнера
|
||||||
|
podman run -d \
|
||||||
|
--name lcg-ollama \
|
||||||
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
|
-v ollama-data:/home/ollama/.ollama \
|
||||||
|
-v lcg-results:/app/data/results \
|
||||||
|
lcg-ollama:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
## Проверка запуска
|
||||||
|
|
||||||
|
### Проверка логов
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Docker
|
||||||
|
docker logs -f lcg-ollama
|
||||||
|
|
||||||
|
# Podman
|
||||||
|
podman logs -f lcg-ollama
|
||||||
|
```
|
||||||
|
|
||||||
|
Дождитесь сообщений:
|
||||||
|
- `Ollama сервер готов!`
|
||||||
|
- `LCG сервер запущен на http://0.0.0.0:8080`
|
||||||
|
|
||||||
|
### Проверка доступности
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Проверка Ollama
|
||||||
|
curl http://localhost:11434/api/tags
|
||||||
|
|
||||||
|
# Проверка LCG
|
||||||
|
curl http://localhost:8080/
|
||||||
|
```
|
||||||
|
|
||||||
|
## Загрузка модели
|
||||||
|
|
||||||
|
После запуска контейнера загрузите модель:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Docker
|
||||||
|
docker exec lcg-ollama ollama pull codegeex4
|
||||||
|
|
||||||
|
# Podman
|
||||||
|
podman exec lcg-ollama ollama pull codegeex4
|
||||||
|
```
|
||||||
|
|
||||||
|
Или используйте модель по умолчанию, указанную в переменных окружения.
|
||||||
|
|
||||||
|
## Доступ к веб-интерфейсу
|
||||||
|
|
||||||
|
Откройте в браузере: http://localhost:8080
|
||||||
|
|
||||||
|
## Остановка
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Docker
|
||||||
|
docker-compose down
|
||||||
|
|
||||||
|
# Podman
|
||||||
|
podman-compose -f podman-compose.yml down
|
||||||
|
```
|
||||||
|
|
||||||
|
Или для ручного запуска:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Docker
|
||||||
|
docker stop lcg-ollama
|
||||||
|
docker rm lcg-ollama
|
||||||
|
|
||||||
|
# Podman
|
||||||
|
podman stop lcg-ollama
|
||||||
|
podman rm lcg-ollama
|
||||||
|
```
|
||||||
|
|
||||||
|
## Решение проблем
|
||||||
|
|
||||||
|
### Порт занят
|
||||||
|
|
||||||
|
Измените порты в docker-compose.yml или команде run:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
-p 9000:8080 # LCG на порту 9000
|
||||||
|
-p 11435:11434 # Ollama на порту 11435
|
||||||
|
```
|
||||||
|
|
||||||
|
### Контейнер не запускается
|
||||||
|
|
||||||
|
Проверьте логи:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker logs lcg-ollama
|
||||||
|
# или
|
||||||
|
podman logs lcg-ollama
|
||||||
|
```
|
||||||
|
|
||||||
|
### Модель не загружена
|
||||||
|
|
||||||
|
Убедитесь, что модель существует:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker exec lcg-ollama ollama list
|
||||||
|
# или
|
||||||
|
podman exec lcg-ollama ollama list
|
||||||
|
```
|
||||||
|
|
||||||
|
Если модели нет, загрузите её:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker exec lcg-ollama ollama pull codegeex4
|
||||||
|
# или
|
||||||
|
podman exec lcg-ollama ollama pull codegeex4
|
||||||
|
```
|
||||||
|
|
||||||
|
## Следующие шаги
|
||||||
|
|
||||||
|
- Прочитайте полную документацию в [README.md](README.md)
|
||||||
|
- Настройте аутентификацию для продакшена
|
||||||
|
- Настройте reverse proxy для HTTPS
|
||||||
|
- Загрузите нужные модели Ollama
|
||||||
|
|
||||||
397
Dockerfiles/OllamaServer/README.md
Normal file
397
Dockerfiles/OllamaServer/README.md
Normal file
@@ -0,0 +1,397 @@
|
|||||||
|
# 🐳 LCG с Ollama Server - Docker/Podman контейнер
|
||||||
|
|
||||||
|
Этот образ содержит Linux Command GPT (LCG) и Ollama сервер, работающие вместе в одном контейнере.
|
||||||
|
|
||||||
|
Поддерживается запуск через Docker и Podman.
|
||||||
|
|
||||||
|
## 📋 Описание
|
||||||
|
|
||||||
|
Контейнер автоматически запускает:
|
||||||
|
1. **Ollama сервер** (v0.9.5) на порту 11434
|
||||||
|
2. **LCG веб-сервер** на порту 8080
|
||||||
|
|
||||||
|
Ollama используется как провайдер LLM для генерации Linux команд.
|
||||||
|
|
||||||
|
## 🚀 Быстрый старт
|
||||||
|
|
||||||
|
### Сборка образа
|
||||||
|
|
||||||
|
#### Docker
|
||||||
|
```bash
|
||||||
|
# Из корня проекта
|
||||||
|
docker build -f Dockerfiles/OllamaServer/Dockerfile -t lcg-ollama:latest .
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Podman
|
||||||
|
```bash
|
||||||
|
# Из корня проекта
|
||||||
|
podman build -f Dockerfiles/OllamaServer/Dockerfile -t lcg-ollama:latest .
|
||||||
|
```
|
||||||
|
|
||||||
|
### Запуск контейнера
|
||||||
|
|
||||||
|
#### Docker
|
||||||
|
```bash
|
||||||
|
docker run -d \
|
||||||
|
--name lcg-ollama \
|
||||||
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
|
lcg-ollama:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Podman
|
||||||
|
```bash
|
||||||
|
podman run -d \
|
||||||
|
--name lcg-ollama \
|
||||||
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
|
lcg-ollama:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
### Использование docker-compose / podman-compose
|
||||||
|
|
||||||
|
#### Docker Compose
|
||||||
|
```bash
|
||||||
|
cd Dockerfiles/OllamaServer
|
||||||
|
docker-compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Podman Compose
|
||||||
|
```bash
|
||||||
|
cd Dockerfiles/OllamaServer
|
||||||
|
podman-compose -f podman-compose.yml up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
Или используйте встроенную поддержку Podman:
|
||||||
|
```bash
|
||||||
|
cd Dockerfiles/OllamaServer
|
||||||
|
podman play kube podman-compose.yml
|
||||||
|
```
|
||||||
|
|
||||||
|
## 🌐 Доступ к сервисам
|
||||||
|
|
||||||
|
После запуска контейнера доступны:
|
||||||
|
|
||||||
|
- **LCG веб-интерфейс**: http://localhost:8080
|
||||||
|
- **Ollama API**: http://localhost:11434
|
||||||
|
|
||||||
|
## ⚙️ Переменные окружения
|
||||||
|
|
||||||
|
### Настройки LCG
|
||||||
|
|
||||||
|
| Переменная | Значение по умолчанию | Описание |
|
||||||
|
|------------|----------------------|----------|
|
||||||
|
| `LCG_PROVIDER` | `ollama` | Тип провайдера |
|
||||||
|
| `LCG_HOST` | `http://127.0.0.1:11434/` | URL Ollama API |
|
||||||
|
| `LCG_MODEL` | `codegeex4` | Модель для использования |
|
||||||
|
| `LCG_SERVER_HOST` | `0.0.0.0` | Хост LCG сервера |
|
||||||
|
| `LCG_SERVER_PORT` | `8080` | Порт LCG сервера |
|
||||||
|
| `LCG_SERVER_ALLOW_HTTP` | `true` | Разрешить HTTP |
|
||||||
|
| `LCG_RESULT_FOLDER` | `/app/data/results` | Папка для результатов |
|
||||||
|
| `LCG_PROMPT_FOLDER` | `/app/data/prompts` | Папка для промптов |
|
||||||
|
| `LCG_CONFIG_FOLDER` | `/app/data/config` | Папка для конфигурации |
|
||||||
|
|
||||||
|
### Настройки Ollama
|
||||||
|
|
||||||
|
| Переменная | Значение по умолчанию | Описание |
|
||||||
|
|------------|----------------------|----------|
|
||||||
|
| `OLLAMA_HOST` | `127.0.0.1` | Хост Ollama сервера |
|
||||||
|
| `OLLAMA_PORT` | `11434` | Порт Ollama сервера |
|
||||||
|
|
||||||
|
### Безопасность
|
||||||
|
|
||||||
|
| Переменная | Значение по умолчанию | Описание |
|
||||||
|
|------------|----------------------|----------|
|
||||||
|
| `LCG_SERVER_REQUIRE_AUTH` | `false` | Требовать аутентификацию |
|
||||||
|
| `LCG_SERVER_PASSWORD` | `admin#123456` | Пароль для аутентификации |
|
||||||
|
|
||||||
|
## 📦 Volumes
|
||||||
|
|
||||||
|
Рекомендуется монтировать volumes для персистентного хранения данных:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker run -d \
|
||||||
|
--name lcg-ollama \
|
||||||
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
|
-v ollama-data:/home/ollama/.ollama \
|
||||||
|
-v lcg-results:/app/data/results \
|
||||||
|
-v lcg-prompts:/app/data/prompts \
|
||||||
|
-v lcg-config:/app/data/config \
|
||||||
|
lcg-ollama:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
### Volumes описание
|
||||||
|
|
||||||
|
- `ollama-data`: Модели и данные Ollama
|
||||||
|
- `lcg-results`: Результаты генерации команд
|
||||||
|
- `lcg-prompts`: Системные промпты
|
||||||
|
- `lcg-config`: Конфигурация LCG
|
||||||
|
|
||||||
|
## 🔧 Примеры использования
|
||||||
|
|
||||||
|
### Запуск с кастомной моделью
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker run -d \
|
||||||
|
--name lcg-ollama \
|
||||||
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
|
-e LCG_MODEL=llama3:8b \
|
||||||
|
lcg-ollama:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
### Запуск с аутентификацией
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker run -d \
|
||||||
|
--name lcg-ollama \
|
||||||
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
|
-e LCG_SERVER_REQUIRE_AUTH=true \
|
||||||
|
-e LCG_SERVER_PASSWORD=my_secure_password \
|
||||||
|
lcg-ollama:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
### Запуск с кастомным портом
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker run -d \
|
||||||
|
--name lcg-ollama \
|
||||||
|
-p 9000:9000 \
|
||||||
|
-p 11434:11434 \
|
||||||
|
-e LCG_SERVER_PORT=9000 \
|
||||||
|
lcg-ollama:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
## 📥 Загрузка моделей Ollama
|
||||||
|
|
||||||
|
После запуска контейнера можно загрузить модели:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Подключиться к контейнеру
|
||||||
|
docker exec -it lcg-ollama sh
|
||||||
|
|
||||||
|
# Загрузить модель
|
||||||
|
ollama pull codegeex4
|
||||||
|
ollama pull llama3:8b
|
||||||
|
ollama pull qwen2.5:7b
|
||||||
|
```
|
||||||
|
|
||||||
|
Или извне контейнера:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Убедитесь, что Ollama доступен извне (OLLAMA_HOST=0.0.0.0)
|
||||||
|
docker exec lcg-ollama ollama pull codegeex4
|
||||||
|
```
|
||||||
|
|
||||||
|
## 🔍 Проверка работоспособности
|
||||||
|
|
||||||
|
### Проверка Ollama
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Проверка health
|
||||||
|
curl http://localhost:11434/api/tags
|
||||||
|
|
||||||
|
# Список моделей
|
||||||
|
curl http://localhost:11434/api/tags | jq '.models'
|
||||||
|
```
|
||||||
|
|
||||||
|
### Проверка LCG
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Проверка веб-интерфейса
|
||||||
|
curl http://localhost:8080/
|
||||||
|
|
||||||
|
# Проверка через API
|
||||||
|
curl -X POST http://localhost:8080/api/execute \
|
||||||
|
-H "Content-Type: application/json" \
|
||||||
|
-d '{"prompt": "создать директорию test"}'
|
||||||
|
```
|
||||||
|
|
||||||
|
## 🐧 Podman специфичные инструкции
|
||||||
|
|
||||||
|
### Запуск с Podman
|
||||||
|
|
||||||
|
Podman работает аналогично Docker, но есть несколько отличий:
|
||||||
|
|
||||||
|
#### Создание сетей (если нужно)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
podman network create lcg-network
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Запуск с сетью
|
||||||
|
|
||||||
|
```bash
|
||||||
|
podman run -d \
|
||||||
|
--name lcg-ollama \
|
||||||
|
--network lcg-network \
|
||||||
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
|
lcg-ollama:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Запуск в rootless режиме
|
||||||
|
|
||||||
|
Podman по умолчанию работает в rootless режиме, что повышает безопасность:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Не требует sudo
|
||||||
|
podman run -d \
|
||||||
|
--name lcg-ollama \
|
||||||
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
|
lcg-ollama:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Использование systemd для автозапуска
|
||||||
|
|
||||||
|
Создайте systemd unit файл:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Генерируем unit файл
|
||||||
|
podman generate systemd --name lcg-ollama --files
|
||||||
|
|
||||||
|
# Копируем в systemd
|
||||||
|
sudo cp container-lcg-ollama.service /etc/systemd/system/
|
||||||
|
|
||||||
|
# Включаем автозапуск
|
||||||
|
sudo systemctl enable container-lcg-ollama.service
|
||||||
|
sudo systemctl start container-lcg-ollama.service
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Проверка статуса
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Статус контейнера
|
||||||
|
podman ps
|
||||||
|
|
||||||
|
# Логи
|
||||||
|
podman logs lcg-ollama
|
||||||
|
|
||||||
|
# Логи в реальном времени
|
||||||
|
podman logs -f lcg-ollama
|
||||||
|
```
|
||||||
|
|
||||||
|
## 🐛 Отладка
|
||||||
|
|
||||||
|
### Просмотр логов
|
||||||
|
|
||||||
|
#### Docker
|
||||||
|
```bash
|
||||||
|
# Логи контейнера
|
||||||
|
docker logs lcg-ollama
|
||||||
|
|
||||||
|
# Логи в реальном времени
|
||||||
|
docker logs -f lcg-ollama
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Podman
|
||||||
|
```bash
|
||||||
|
# Логи контейнера
|
||||||
|
podman logs lcg-ollama
|
||||||
|
|
||||||
|
# Логи в реальном времени
|
||||||
|
podman logs -f lcg-ollama
|
||||||
|
```
|
||||||
|
|
||||||
|
### Подключение к контейнеру
|
||||||
|
|
||||||
|
#### Docker
|
||||||
|
```bash
|
||||||
|
docker exec -it lcg-ollama sh
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Podman
|
||||||
|
```bash
|
||||||
|
podman exec -it lcg-ollama sh
|
||||||
|
```
|
||||||
|
|
||||||
|
### Проверка процессов
|
||||||
|
|
||||||
|
#### Docker
|
||||||
|
```bash
|
||||||
|
docker exec lcg-ollama ps aux
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Podman
|
||||||
|
```bash
|
||||||
|
podman exec lcg-ollama ps aux
|
||||||
|
```
|
||||||
|
|
||||||
|
## 🔒 Безопасность
|
||||||
|
|
||||||
|
### Рекомендации для продакшена
|
||||||
|
|
||||||
|
1. **Используйте аутентификацию**:
|
||||||
|
```bash
|
||||||
|
-e LCG_SERVER_REQUIRE_AUTH=true
|
||||||
|
-e LCG_SERVER_PASSWORD=strong_password
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Ограничьте доступ к портам**:
|
||||||
|
- Используйте firewall правила
|
||||||
|
- Не экспортируйте порты на публичный интерфейс
|
||||||
|
|
||||||
|
3. **Используйте HTTPS**:
|
||||||
|
- Настройте reverse proxy (nginx, traefik)
|
||||||
|
- Используйте SSL сертификаты
|
||||||
|
|
||||||
|
4. **Ограничьте ресурсы**:
|
||||||
|
```bash
|
||||||
|
docker run -d \
|
||||||
|
--name lcg-ollama \
|
||||||
|
--memory="4g" \
|
||||||
|
--cpus="2" \
|
||||||
|
lcg-ollama:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
## 📊 Мониторинг
|
||||||
|
|
||||||
|
### Healthcheck
|
||||||
|
|
||||||
|
Контейнер включает healthcheck, который проверяет доступность LCG сервера:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker inspect lcg-ollama | jq '.[0].State.Health'
|
||||||
|
```
|
||||||
|
|
||||||
|
### Метрики
|
||||||
|
|
||||||
|
LCG предоставляет Prometheus метрики на `/metrics` endpoint (если включено).
|
||||||
|
|
||||||
|
## 🚀 Production Deployment
|
||||||
|
|
||||||
|
### С docker-compose
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd Dockerfiles/OllamaServer
|
||||||
|
docker-compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
### С Kubernetes
|
||||||
|
|
||||||
|
Используйте манифесты из папки `deploy/` или `kustomize/`.
|
||||||
|
|
||||||
|
## 📝 Примечания
|
||||||
|
|
||||||
|
- Ollama версия: 0.9.5
|
||||||
|
- LCG версия: см. VERSION.txt
|
||||||
|
- Минимальные требования: 2GB RAM, 2 CPU cores
|
||||||
|
- Рекомендуется: 4GB+ RAM для больших моделей
|
||||||
|
|
||||||
|
## 🔗 Полезные ссылки
|
||||||
|
|
||||||
|
- [Ollama документация](https://github.com/ollama/ollama)
|
||||||
|
- [LCG документация](../../docs/README.md)
|
||||||
|
- [LCG API Guide](../../docs/API_GUIDE.md)
|
||||||
|
|
||||||
|
## ❓ Поддержка
|
||||||
|
|
||||||
|
При возникновении проблем:
|
||||||
|
1. Проверьте логи: `docker logs lcg-ollama`
|
||||||
|
2. Проверьте переменные окружения
|
||||||
|
3. Убедитесь, что порты не заняты
|
||||||
|
4. Проверьте, что модели загружены в Ollama
|
||||||
|
|
||||||
111
Dockerfiles/OllamaServer/STRUCTURE.md
Normal file
111
Dockerfiles/OllamaServer/STRUCTURE.md
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
# 📁 Структура проекта OllamaServer
|
||||||
|
|
||||||
|
## Файлы
|
||||||
|
|
||||||
|
```
|
||||||
|
Dockerfiles/OllamaServer/
|
||||||
|
├── Dockerfile # Multi-stage Dockerfile для сборки образа
|
||||||
|
├── entrypoint.sh # Скрипт запуска LCG и Ollama серверов
|
||||||
|
├── docker-compose.yml # Docker Compose конфигурация
|
||||||
|
├── podman-compose.yml # Podman Compose конфигурация
|
||||||
|
├── Makefile # Команды для сборки и запуска
|
||||||
|
├── README.md # Полная документация
|
||||||
|
├── QUICKSTART.md # Быстрый старт
|
||||||
|
├── STRUCTURE.md # Этот файл
|
||||||
|
├── .dockerignore # Исключения для Docker build
|
||||||
|
└── .gitignore # Исключения для Git
|
||||||
|
```
|
||||||
|
|
||||||
|
## Описание файлов
|
||||||
|
|
||||||
|
### Dockerfile
|
||||||
|
Multi-stage Dockerfile, который:
|
||||||
|
1. Собирает бинарник LCG из исходного кода
|
||||||
|
2. Устанавливает Ollama 0.9.5
|
||||||
|
3. Создает пользователя ollama
|
||||||
|
4. Настраивает рабочее окружение
|
||||||
|
|
||||||
|
### entrypoint.sh
|
||||||
|
Скрипт запуска, который:
|
||||||
|
1. Запускает Ollama сервер в фоне
|
||||||
|
2. Ожидает готовности Ollama API
|
||||||
|
3. Запускает LCG сервер в фоне
|
||||||
|
4. Мониторит состояние процессов
|
||||||
|
5. Корректно обрабатывает сигналы завершения
|
||||||
|
|
||||||
|
### docker-compose.yml / podman-compose.yml
|
||||||
|
Конфигурация для запуска через compose:
|
||||||
|
- Настройки портов
|
||||||
|
- Переменные окружения
|
||||||
|
- Volumes для персистентного хранения
|
||||||
|
- Healthcheck
|
||||||
|
|
||||||
|
### Makefile
|
||||||
|
Удобные команды для:
|
||||||
|
- Сборки образа
|
||||||
|
- Запуска/остановки контейнера
|
||||||
|
- Просмотра логов
|
||||||
|
- Работы с compose
|
||||||
|
|
||||||
|
### README.md
|
||||||
|
Полная документация с:
|
||||||
|
- Описанием функциональности
|
||||||
|
- Инструкциями по установке
|
||||||
|
- Настройками переменных окружения
|
||||||
|
- Примерами использования
|
||||||
|
- Решением проблем
|
||||||
|
|
||||||
|
### QUICKSTART.md
|
||||||
|
Краткое руководство для быстрого старта.
|
||||||
|
|
||||||
|
## Порты
|
||||||
|
|
||||||
|
- **8080**: LCG веб-сервер
|
||||||
|
- **11434**: Ollama API
|
||||||
|
|
||||||
|
## Volumes
|
||||||
|
|
||||||
|
- `ollama-data`: Данные Ollama (модели)
|
||||||
|
- `lcg-results`: Результаты генерации команд
|
||||||
|
- `lcg-prompts`: Системные промпты
|
||||||
|
- `lcg-config`: Конфигурация LCG
|
||||||
|
|
||||||
|
## Переменные окружения
|
||||||
|
|
||||||
|
Основные переменные (см. README.md для полного списка):
|
||||||
|
- `LCG_PROVIDER=ollama`
|
||||||
|
- `LCG_HOST=http://127.0.0.1:11434/`
|
||||||
|
- `LCG_MODEL=codegeex4`
|
||||||
|
- `OLLAMA_HOST=0.0.0.0`
|
||||||
|
- `OLLAMA_PORT=11434`
|
||||||
|
|
||||||
|
## Запуск
|
||||||
|
|
||||||
|
### Docker
|
||||||
|
```bash
|
||||||
|
cd Dockerfiles/OllamaServer
|
||||||
|
docker-compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
### Podman
|
||||||
|
```bash
|
||||||
|
cd Dockerfiles/OllamaServer
|
||||||
|
podman-compose -f podman-compose.yml up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
### Make
|
||||||
|
```bash
|
||||||
|
cd Dockerfiles/OllamaServer
|
||||||
|
make compose-up
|
||||||
|
# или
|
||||||
|
make podman-compose-up
|
||||||
|
```
|
||||||
|
|
||||||
|
## Архитектура
|
||||||
|
|
||||||
|
Контейнер запускает два сервиса:
|
||||||
|
1. **Ollama** (порт 11434) - LLM сервер
|
||||||
|
2. **LCG** (порт 8080) - Веб-интерфейс и API
|
||||||
|
|
||||||
|
Оба сервиса работают в одном контейнере и общаются через localhost.
|
||||||
|
|
||||||
54
Dockerfiles/OllamaServer/docker-compose.yml
Normal file
54
Dockerfiles/OllamaServer/docker-compose.yml
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
version: '3.8'
|
||||||
|
|
||||||
|
services:
|
||||||
|
lcg-ollama:
|
||||||
|
build:
|
||||||
|
context: ../..
|
||||||
|
dockerfile: Dockerfiles/OllamaServer/Dockerfile
|
||||||
|
container_name: lcg-ollama
|
||||||
|
ports:
|
||||||
|
- "8080:8080" # LCG веб-сервер
|
||||||
|
- "11434:11434" # Ollama API
|
||||||
|
environment:
|
||||||
|
# Настройки LCG
|
||||||
|
- LCG_PROVIDER=ollama
|
||||||
|
- LCG_HOST=http://127.0.0.1:11434/
|
||||||
|
- LCG_MODEL=codegeex4
|
||||||
|
- LCG_RESULT_FOLDER=/app/data/results
|
||||||
|
- LCG_PROMPT_FOLDER=/app/data/prompts
|
||||||
|
- LCG_CONFIG_FOLDER=/app/data/config
|
||||||
|
- LCG_SERVER_HOST=0.0.0.0
|
||||||
|
- LCG_SERVER_PORT=8080
|
||||||
|
- LCG_SERVER_ALLOW_HTTP=true
|
||||||
|
# Настройки Ollama
|
||||||
|
- OLLAMA_HOST=0.0.0.0
|
||||||
|
- OLLAMA_PORT=11434
|
||||||
|
- OLLAMA_ORIGINS=*
|
||||||
|
# Опционально: настройки безопасности
|
||||||
|
- LCG_SERVER_REQUIRE_AUTH=false
|
||||||
|
- LCG_SERVER_PASSWORD=admin#123456
|
||||||
|
volumes:
|
||||||
|
# Персистентное хранилище для данных Ollama
|
||||||
|
- ollama-data:/home/ollama/.ollama
|
||||||
|
# Персистентное хранилище для результатов LCG
|
||||||
|
- lcg-results:/app/data/results
|
||||||
|
- lcg-prompts:/app/data/prompts
|
||||||
|
- lcg-config:/app/data/config
|
||||||
|
restart: unless-stopped
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD", "curl", "-f", "http://localhost:8080/"]
|
||||||
|
interval: 30s
|
||||||
|
timeout: 10s
|
||||||
|
retries: 3
|
||||||
|
start_period: 40s
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
ollama-data:
|
||||||
|
driver: local
|
||||||
|
lcg-results:
|
||||||
|
driver: local
|
||||||
|
lcg-prompts:
|
||||||
|
driver: local
|
||||||
|
lcg-config:
|
||||||
|
driver: local
|
||||||
|
|
||||||
171
Dockerfiles/OllamaServer/entrypoint.sh
Executable file
171
Dockerfiles/OllamaServer/entrypoint.sh
Executable file
@@ -0,0 +1,171 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
|
||||||
|
# Цвета для вывода
|
||||||
|
RED='\033[0;31m'
|
||||||
|
GREEN='\033[0;32m'
|
||||||
|
YELLOW='\033[1;33m'
|
||||||
|
BLUE='\033[0;34m'
|
||||||
|
NC='\033[0m' # No Color
|
||||||
|
|
||||||
|
# Функция для логирования
|
||||||
|
log() {
|
||||||
|
echo -e "${GREEN}[$(date +'%Y-%m-%d %H:%M:%S')]${NC} $1"
|
||||||
|
}
|
||||||
|
|
||||||
|
error() {
|
||||||
|
echo -e "${RED}[ERROR]${NC} $1" >&2
|
||||||
|
}
|
||||||
|
|
||||||
|
warn() {
|
||||||
|
echo -e "${YELLOW}[WARN]${NC} $1"
|
||||||
|
}
|
||||||
|
|
||||||
|
info() {
|
||||||
|
echo -e "${BLUE}[INFO]${NC} $1"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Обработка сигналов для корректного завершения
|
||||||
|
cleanup() {
|
||||||
|
log "Получен сигнал завершения, останавливаем сервисы..."
|
||||||
|
if [ ! -z "$LCG_PID" ]; then
|
||||||
|
kill $LCG_PID 2>/dev/null || true
|
||||||
|
wait $LCG_PID 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
if [ ! -z "$OLLAMA_PID" ]; then
|
||||||
|
kill $OLLAMA_PID 2>/dev/null || true
|
||||||
|
wait $OLLAMA_PID 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
log "Сервисы остановлены"
|
||||||
|
exit 0
|
||||||
|
}
|
||||||
|
|
||||||
|
trap cleanup SIGTERM SIGINT
|
||||||
|
|
||||||
|
# Проверка наличия бинарника lcg
|
||||||
|
if [ ! -f /usr/local/bin/lcg ]; then
|
||||||
|
error "Бинарник lcg не найден в /usr/local/bin/lcg"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Проверка наличия Ollama
|
||||||
|
if [ ! -f /usr/local/bin/ollama ]; then
|
||||||
|
error "Ollama не найден в /usr/local/bin/ollama"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Создаем необходимые директории
|
||||||
|
mkdir -p "${LCG_RESULT_FOLDER:-/app/data/results}"
|
||||||
|
mkdir -p "${LCG_PROMPT_FOLDER:-/app/data/prompts}"
|
||||||
|
mkdir -p "${LCG_CONFIG_FOLDER:-/app/data/config}"
|
||||||
|
|
||||||
|
# Настройка переменных окружения для Ollama
|
||||||
|
export OLLAMA_HOST="${OLLAMA_HOST:-0.0.0.0}"
|
||||||
|
export OLLAMA_PORT="${OLLAMA_PORT:-11434}"
|
||||||
|
export OLLAMA_ORIGINS="*"
|
||||||
|
|
||||||
|
# Настройка переменных окружения для LCG
|
||||||
|
export LCG_PROVIDER="${LCG_PROVIDER:-ollama}"
|
||||||
|
export LCG_HOST="${LCG_HOST:-http://127.0.0.1:11434/}"
|
||||||
|
export LCG_MODEL="${LCG_MODEL:-codegeex4}"
|
||||||
|
export LCG_RESULT_FOLDER="${LCG_RESULT_FOLDER:-/app/data/results}"
|
||||||
|
export LCG_PROMPT_FOLDER="${LCG_PROMPT_FOLDER:-/app/data/prompts}"
|
||||||
|
export LCG_CONFIG_FOLDER="${LCG_CONFIG_FOLDER:-/app/data/config}"
|
||||||
|
export LCG_SERVER_HOST="${LCG_SERVER_HOST:-0.0.0.0}"
|
||||||
|
export LCG_SERVER_PORT="${LCG_SERVER_PORT:-8080}"
|
||||||
|
export LCG_SERVER_ALLOW_HTTP="${LCG_SERVER_ALLOW_HTTP:-true}"
|
||||||
|
|
||||||
|
log "=========================================="
|
||||||
|
log "Запуск LCG с Ollama сервером"
|
||||||
|
log "=========================================="
|
||||||
|
info "LCG Provider: $LCG_PROVIDER"
|
||||||
|
info "LCG Host: $LCG_HOST"
|
||||||
|
info "LCG Model: $LCG_MODEL"
|
||||||
|
info "LCG Server: http://${LCG_SERVER_HOST}:${LCG_SERVER_PORT}"
|
||||||
|
info "Ollama Host: $OLLAMA_HOST:$OLLAMA_PORT"
|
||||||
|
log "=========================================="
|
||||||
|
|
||||||
|
# Запускаем Ollama сервер в фоне
|
||||||
|
log "Запуск Ollama сервера..."
|
||||||
|
/usr/local/bin/ollama serve &
|
||||||
|
OLLAMA_PID=$!
|
||||||
|
|
||||||
|
# Ждем, пока Ollama запустится
|
||||||
|
log "Ожидание запуска Ollama сервера..."
|
||||||
|
sleep 5
|
||||||
|
|
||||||
|
# Проверяем, что Ollama запущен
|
||||||
|
if ! kill -0 $OLLAMA_PID 2>/dev/null; then
|
||||||
|
error "Ollama сервер не запустился"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Проверяем доступность Ollama API
|
||||||
|
max_attempts=30
|
||||||
|
attempt=0
|
||||||
|
while [ $attempt -lt $max_attempts ]; do
|
||||||
|
# Проверяем через localhost, так как OLLAMA_HOST может быть 0.0.0.0
|
||||||
|
if curl -s -f "http://127.0.0.1:${OLLAMA_PORT}/api/tags" > /dev/null 2>&1; then
|
||||||
|
log "Ollama сервер готов!"
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
attempt=$((attempt + 1))
|
||||||
|
if [ $attempt -eq $max_attempts ]; then
|
||||||
|
error "Ollama сервер не отвечает после $max_attempts попыток"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
sleep 1
|
||||||
|
done
|
||||||
|
|
||||||
|
# Запускаем LCG сервер в фоне
|
||||||
|
log "Запуск LCG сервера..."
|
||||||
|
/usr/local/bin/lcg serve \
|
||||||
|
--host "${LCG_SERVER_HOST}" \
|
||||||
|
--port "${LCG_SERVER_PORT}" &
|
||||||
|
LCG_PID=$!
|
||||||
|
|
||||||
|
# Ждем, пока LCG запустится
|
||||||
|
sleep 3
|
||||||
|
|
||||||
|
# Проверяем, что LCG запущен
|
||||||
|
if ! kill -0 $LCG_PID 2>/dev/null; then
|
||||||
|
error "LCG сервер не запустился"
|
||||||
|
kill $OLLAMA_PID 2>/dev/null || true
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
log "LCG сервер запущен на http://${LCG_SERVER_HOST}:${LCG_SERVER_PORT}"
|
||||||
|
log "Ollama сервер доступен на http://${OLLAMA_HOST}:${OLLAMA_PORT}"
|
||||||
|
log "=========================================="
|
||||||
|
log "Сервисы запущены и готовы к работе!"
|
||||||
|
log "=========================================="
|
||||||
|
|
||||||
|
# Функция для проверки здоровья процессов
|
||||||
|
health_check() {
|
||||||
|
while true; do
|
||||||
|
# Проверяем Ollama
|
||||||
|
if ! kill -0 $OLLAMA_PID 2>/dev/null; then
|
||||||
|
error "Ollama процесс завершился неожиданно"
|
||||||
|
kill $LCG_PID 2>/dev/null || true
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Проверяем LCG
|
||||||
|
if ! kill -0 $LCG_PID 2>/dev/null; then
|
||||||
|
error "LCG процесс завершился неожиданно"
|
||||||
|
kill $OLLAMA_PID 2>/dev/null || true
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
sleep 10
|
||||||
|
done
|
||||||
|
}
|
||||||
|
|
||||||
|
# Запускаем проверку здоровья в фоне
|
||||||
|
health_check &
|
||||||
|
HEALTH_CHECK_PID=$!
|
||||||
|
|
||||||
|
# Ждем завершения процессов
|
||||||
|
wait $LCG_PID $OLLAMA_PID
|
||||||
|
kill $HEALTH_CHECK_PID 2>/dev/null || true
|
||||||
|
|
||||||
54
Dockerfiles/OllamaServer/podman-compose.yml
Normal file
54
Dockerfiles/OllamaServer/podman-compose.yml
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
version: '3.8'
|
||||||
|
|
||||||
|
services:
|
||||||
|
lcg-ollama:
|
||||||
|
build:
|
||||||
|
context: ../..
|
||||||
|
dockerfile: Dockerfiles/OllamaServer/Dockerfile
|
||||||
|
container_name: lcg-ollama
|
||||||
|
ports:
|
||||||
|
- "8080:8080" # LCG веб-сервер
|
||||||
|
- "11434:11434" # Ollama API
|
||||||
|
environment:
|
||||||
|
# Настройки LCG
|
||||||
|
- LCG_PROVIDER=ollama
|
||||||
|
- LCG_HOST=http://127.0.0.1:11434/
|
||||||
|
- LCG_MODEL=codegeex4
|
||||||
|
- LCG_RESULT_FOLDER=/app/data/results
|
||||||
|
- LCG_PROMPT_FOLDER=/app/data/prompts
|
||||||
|
- LCG_CONFIG_FOLDER=/app/data/config
|
||||||
|
- LCG_SERVER_HOST=0.0.0.0
|
||||||
|
- LCG_SERVER_PORT=8080
|
||||||
|
- LCG_SERVER_ALLOW_HTTP=true
|
||||||
|
# Настройки Ollama
|
||||||
|
- OLLAMA_HOST=0.0.0.0
|
||||||
|
- OLLAMA_PORT=11434
|
||||||
|
- OLLAMA_ORIGINS=*
|
||||||
|
# Опционально: настройки безопасности
|
||||||
|
- LCG_SERVER_REQUIRE_AUTH=false
|
||||||
|
- LCG_SERVER_PASSWORD=admin#123456
|
||||||
|
volumes:
|
||||||
|
# Персистентное хранилище для данных Ollama
|
||||||
|
- ollama-data:/home/ollama/.ollama
|
||||||
|
# Персистентное хранилище для результатов LCG
|
||||||
|
- lcg-results:/app/data/results
|
||||||
|
- lcg-prompts:/app/data/prompts
|
||||||
|
- lcg-config:/app/data/config
|
||||||
|
restart: unless-stopped
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD", "curl", "-f", "http://localhost:8080/"]
|
||||||
|
interval: 30s
|
||||||
|
timeout: 10s
|
||||||
|
retries: 3
|
||||||
|
start_period: 40s
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
ollama-data:
|
||||||
|
driver: local
|
||||||
|
lcg-results:
|
||||||
|
driver: local
|
||||||
|
lcg-prompts:
|
||||||
|
driver: local
|
||||||
|
lcg-config:
|
||||||
|
driver: local
|
||||||
|
|
||||||
@@ -1 +1 @@
|
|||||||
v2.0.3
|
v2.0.15
|
||||||
|
|||||||
@@ -68,6 +68,15 @@ type ValidationConfig struct {
|
|||||||
MaxExplanationLength int
|
MaxExplanationLength int
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func GetEnvBool(key string, defaultValue bool) bool {
|
||||||
|
if value, exists := os.LookupEnv(key); exists {
|
||||||
|
if boolValue, err := strconv.ParseBool(value); err == nil {
|
||||||
|
return boolValue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return defaultValue
|
||||||
|
}
|
||||||
|
|
||||||
func getEnv(key, defaultValue string) string {
|
func getEnv(key, defaultValue string) string {
|
||||||
if value, exists := os.LookupEnv(key); exists {
|
if value, exists := os.LookupEnv(key); exists {
|
||||||
return value
|
return value
|
||||||
|
|||||||
@@ -6,6 +6,8 @@ builds:
|
|||||||
binary: "lcg_{{ .Version }}"
|
binary: "lcg_{{ .Version }}"
|
||||||
goos:
|
goos:
|
||||||
- linux
|
- linux
|
||||||
|
- windows
|
||||||
|
- darwin
|
||||||
goarch:
|
goarch:
|
||||||
- amd64
|
- amd64
|
||||||
- arm64
|
- arm64
|
||||||
@@ -21,9 +23,10 @@ builds:
|
|||||||
|
|
||||||
archives:
|
archives:
|
||||||
- id: lcg
|
- id: lcg
|
||||||
builds:
|
ids:
|
||||||
- lcg
|
- lcg
|
||||||
format: binary
|
formats:
|
||||||
|
- binary
|
||||||
name_template: "{{ .Binary }}_{{ .Os }}_{{ .Arch }}"
|
name_template: "{{ .Binary }}_{{ .Os }}_{{ .Arch }}"
|
||||||
files:
|
files:
|
||||||
- "lcg_{{ .Version }}"
|
- "lcg_{{ .Version }}"
|
||||||
|
|||||||
@@ -16,7 +16,8 @@ data:
|
|||||||
LCG_CONFIG_FOLDER: "/app/data/config"
|
LCG_CONFIG_FOLDER: "/app/data/config"
|
||||||
LCG_NO_HISTORY: "false"
|
LCG_NO_HISTORY: "false"
|
||||||
LCG_ALLOW_EXECUTION: "false"
|
LCG_ALLOW_EXECUTION: "false"
|
||||||
LCG_DEBUG: "false"
|
LCG_DEBUG: "true"
|
||||||
|
LCG_PROVIDER: "proxy"
|
||||||
|
|
||||||
# Настройки аутентификации
|
# Настройки аутентификации
|
||||||
LCG_SERVER_REQUIRE_AUTH: "true"
|
LCG_SERVER_REQUIRE_AUTH: "true"
|
||||||
|
|||||||
@@ -15,7 +15,6 @@ spec:
|
|||||||
metadata:
|
metadata:
|
||||||
labels:
|
labels:
|
||||||
app: lcg
|
app: lcg
|
||||||
version: ${VERSION}
|
|
||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- name: lcg
|
- name: lcg
|
||||||
@@ -58,21 +57,18 @@ spec:
|
|||||||
readOnly: true
|
readOnly: true
|
||||||
# Health checks
|
# Health checks
|
||||||
startupProbe:
|
startupProbe:
|
||||||
httpGet:
|
tcpSocket:
|
||||||
path: /login
|
|
||||||
port: 8080
|
port: 8080
|
||||||
initialDelaySeconds: 10
|
initialDelaySeconds: 10
|
||||||
periodSeconds: 5
|
periodSeconds: 5
|
||||||
failureThreshold: 30
|
failureThreshold: 30
|
||||||
readinessProbe:
|
readinessProbe:
|
||||||
httpGet:
|
tcpSocket:
|
||||||
path: /login
|
|
||||||
port: 8080
|
port: 8080
|
||||||
initialDelaySeconds: 5
|
initialDelaySeconds: 5
|
||||||
periodSeconds: 10
|
periodSeconds: 10
|
||||||
livenessProbe:
|
livenessProbe:
|
||||||
httpGet:
|
tcpSocket:
|
||||||
path: /login
|
|
||||||
port: 8080
|
port: 8080
|
||||||
initialDelaySeconds: 10
|
initialDelaySeconds: 10
|
||||||
periodSeconds: 60
|
periodSeconds: 60
|
||||||
|
|||||||
@@ -127,15 +127,24 @@ fi
|
|||||||
if [ "$current_branch" != "main" ]; then
|
if [ "$current_branch" != "main" ]; then
|
||||||
git checkout main
|
git checkout main
|
||||||
git merge --no-ff -m "Merged branch '$current_branch' into main while building $VERSION" "$current_branch"
|
git merge --no-ff -m "Merged branch '$current_branch' into main while building $VERSION" "$current_branch"
|
||||||
|
git push origin main
|
||||||
elif [ "$current_branch" = "main" ]; then
|
elif [ "$current_branch" = "main" ]; then
|
||||||
log "🔄 Вы находитесь на ветке main. Слияние с release..."
|
log "🔄 Вы находитесь на ветке main. Слияние с release..."
|
||||||
git add .
|
git add .
|
||||||
git commit -m "Исправления в ветке $current_branch"
|
git commit -m "Исправления в ветке $current_branch"
|
||||||
|
git push origin main
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# переключиться на ветку release и слить с веткой main
|
# переключиться на ветку release и слить с веткой main
|
||||||
git checkout release
|
if git show-ref --quiet refs/heads/release; then
|
||||||
git merge --no-ff -m "Merged main into release while building $VERSION" main
|
log "ℹ️ Branch 'release' exists. Proceeding with merge."
|
||||||
|
git checkout release
|
||||||
|
git merge --no-ff -m "Merged main into release while building $VERSION" main
|
||||||
|
else
|
||||||
|
log "❌ Branch 'release' does not exist. Please create the branch before proceeding."
|
||||||
|
git checkout -b release
|
||||||
|
git merge --no-ff -m "Merged main into release while building $VERSION" main
|
||||||
|
fi
|
||||||
|
|
||||||
# если тег $VERSION существует, удалить его и принудительно запушить
|
# если тег $VERSION существует, удалить его и принудительно запушить
|
||||||
tag_exists=$(git tag -l "$VERSION")
|
tag_exists=$(git tag -l "$VERSION")
|
||||||
@@ -189,14 +198,7 @@ log "🔍 Проверка образа:"
|
|||||||
echo " docker run --rm $REPOSITORY:$VERSION /app/lcg --version"
|
echo " docker run --rm $REPOSITORY:$VERSION /app/lcg --version"
|
||||||
echo ""
|
echo ""
|
||||||
log "📝 Команды для использования:"
|
log "📝 Команды для использования:"
|
||||||
echo " kubectl apply -k kustomize"
|
|
||||||
echo " kubectl get pods"
|
echo " kubectl get pods"
|
||||||
echo " kubectl get services"
|
echo " kubectl get services"
|
||||||
echo " kubectl get ingress"
|
echo " kubectl get ingress"
|
||||||
echo " kubectl get hpa"
|
|
||||||
echo " kubectl get servicemonitor"
|
|
||||||
echo " kubectl get pods"
|
|
||||||
echo " kubectl get services"
|
|
||||||
echo " kubectl get ingress"
|
|
||||||
echo " kubectl get hpa"
|
|
||||||
echo " kubectl get servicemonitor"
|
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
v2.0.3
|
v2.0.15
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ metadata:
|
|||||||
namespace: lcg
|
namespace: lcg
|
||||||
labels:
|
labels:
|
||||||
app: lcg
|
app: lcg
|
||||||
version: ${VERSION}
|
|
||||||
spec:
|
spec:
|
||||||
entryPoints:
|
entryPoints:
|
||||||
- websecure
|
- websecure
|
||||||
|
|||||||
@@ -13,13 +13,13 @@ resources:
|
|||||||
- ingress-route.yaml
|
- ingress-route.yaml
|
||||||
|
|
||||||
# Common labels
|
# Common labels
|
||||||
commonLabels:
|
# commonLabels:
|
||||||
app: lcg
|
# app: lcg
|
||||||
version: ${VERSION}
|
# version: ${VERSION}
|
||||||
managed-by: kustomize
|
# managed-by: kustomize
|
||||||
|
|
||||||
# Images
|
# Images
|
||||||
images:
|
# images:
|
||||||
- name: lcg
|
# - name: lcg
|
||||||
newName: ${REPOSITORY}
|
# newName: ${REPOSITORY}
|
||||||
newTag: ${VERSION}
|
# newTag: ${VERSION}
|
||||||
|
|||||||
170
deploy/release-goreleaser.sh
Normal file
170
deploy/release-goreleaser.sh
Normal file
@@ -0,0 +1,170 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# release-goreleaser.sh
|
||||||
|
# Копирует deploy/.goreleaser.yaml в корень, запускает релиз и удаляет файл.
|
||||||
|
#
|
||||||
|
# Использование:
|
||||||
|
# deploy/release-goreleaser.sh # обычный релиз на GitHub (нужен GITHUB_TOKEN)
|
||||||
|
# deploy/release-goreleaser.sh --snapshot # локальный снепшот без публикации
|
||||||
|
|
||||||
|
ROOT_DIR="$(git rev-parse --show-toplevel 2>/dev/null || pwd)"
|
||||||
|
SRC_CFG="$ROOT_DIR/deploy/.goreleaser.yaml"
|
||||||
|
DST_CFG="$ROOT_DIR/.goreleaser.yaml"
|
||||||
|
|
||||||
|
log() { echo -e "\033[36m[release]\033[0m $*"; }
|
||||||
|
err() { echo -e "\033[31m[error]\033[0m $*" >&2; }
|
||||||
|
|
||||||
|
if ! command -v goreleaser >/dev/null 2>&1; then
|
||||||
|
err "goreleaser не найден. Установите: https://goreleaser.com/install/"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ ! -f "$SRC_CFG" ]]; then
|
||||||
|
err "Не найден файл конфигурации: $SRC_CFG"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
MODE="release"
|
||||||
|
if [[ "${1:-}" == "--snapshot" ]]; then
|
||||||
|
MODE="snapshot"
|
||||||
|
shift || true
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ -f "$DST_CFG" ]]; then
|
||||||
|
err "В корне уже существует .goreleaser.yaml. Удалите/переименуйте перед запуском."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
cleanup() {
|
||||||
|
if [[ -f "$DST_CFG" ]]; then
|
||||||
|
rm -f "$DST_CFG" || true
|
||||||
|
log "Удалил временный $DST_CFG"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
trap cleanup EXIT
|
||||||
|
|
||||||
|
log "Копирую конфиг: $SRC_CFG -> $DST_CFG"
|
||||||
|
cp "$SRC_CFG" "$DST_CFG"
|
||||||
|
|
||||||
|
pushd "$ROOT_DIR" >/dev/null
|
||||||
|
|
||||||
|
EXTRA_FLAGS=()
|
||||||
|
PREV_HEAD="$(git rev-parse HEAD 2>/dev/null || echo "")"
|
||||||
|
|
||||||
|
git add .
|
||||||
|
git commit --amend --no-edit || true
|
||||||
|
|
||||||
|
## Версию берём из deploy/VERSION.txt или VERSION.txt в корне
|
||||||
|
VERSION_FILE="$ROOT_DIR/deploy/VERSION.txt"
|
||||||
|
[[ -f "$VERSION_FILE" ]] || VERSION_FILE="$ROOT_DIR/VERSION.txt"
|
||||||
|
if [[ -f "$VERSION_FILE" ]]; then
|
||||||
|
VERSION_RAW="$(head -n1 "$VERSION_FILE" | tr -d ' \t\r\n')"
|
||||||
|
if [[ -n "$VERSION_RAW" ]]; then
|
||||||
|
TAG="$VERSION_RAW"
|
||||||
|
[[ "$TAG" == v* ]] || TAG="v$TAG"
|
||||||
|
export GORELEASER_CURRENT_TAG="$TAG"
|
||||||
|
log "Версия релиза: $TAG (из $(realpath --relative-to="$ROOT_DIR" "$VERSION_FILE" 2>/dev/null || echo "$VERSION_FILE"))"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
create_and_push_tag() {
|
||||||
|
local tag="$1"
|
||||||
|
if git rev-parse "$tag" >/dev/null 2>&1; then
|
||||||
|
log "Git tag уже существует: $tag"
|
||||||
|
else
|
||||||
|
log "Создаю git tag: $tag"
|
||||||
|
git tag -a "$tag" -m "Release $tag"
|
||||||
|
if [[ "${NO_GIT_PUSH:-false}" != "true" ]]; then
|
||||||
|
log "Пушу тег $tag на origin"
|
||||||
|
git push origin "$tag"
|
||||||
|
else
|
||||||
|
log "Пропущен пуш тега (NO_GIT_PUSH=true)"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
move_tag_to_head() {
|
||||||
|
local tag="$1"
|
||||||
|
if [[ -z "$tag" ]]; then
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
if git rev-parse "$tag" >/dev/null 2>&1; then
|
||||||
|
log "Переношу тег $tag на текущий коммит (HEAD)"
|
||||||
|
git tag -f "$tag" HEAD
|
||||||
|
if [[ "${NO_GIT_PUSH:-false}" != "true" ]]; then
|
||||||
|
log "Форс‑пуш тега $tag на origin"
|
||||||
|
git push -f origin "$tag"
|
||||||
|
else
|
||||||
|
log "Пропущен пуш тега (NO_GIT_PUSH=true)"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
log "Тега $tag нет — пропускаю перенос"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
fetch_token_from_k8s() {
|
||||||
|
export KUBECONFIG=/home/su/.kube/config_hlab
|
||||||
|
local ns="${K8S_NAMESPACE:-flux-system}"
|
||||||
|
local name="${K8S_SECRET_NAME:-git-secrets}"
|
||||||
|
# Предпочитаем jq (как в примере), при отсутствии используем jsonpath + base64 -d
|
||||||
|
if command -v jq >/dev/null 2>&1; then
|
||||||
|
kubectl get secret "$name" -n "$ns" -o json \
|
||||||
|
| jq -r '.data.password | @base64d'
|
||||||
|
else
|
||||||
|
kubectl get secret "$name" -n "$ns" -o jsonpath='{.data.password}' \
|
||||||
|
| base64 -d 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
if [[ "$MODE" == "snapshot" ]]; then
|
||||||
|
log "Запуск goreleaser (snapshot, без публикации)"
|
||||||
|
goreleaser release --snapshot --clean --config "$DST_CFG" "${EXTRA_FLAGS[@]}"
|
||||||
|
else
|
||||||
|
# Если версия определена и тега нет — создадим (goreleaser ориентируется на теги)
|
||||||
|
if [[ -n "${GORELEASER_CURRENT_TAG:-}" ]]; then
|
||||||
|
create_and_push_tag "$GORELEASER_CURRENT_TAG"
|
||||||
|
# Перемещаем тег на текущий HEAD (если существовал ранее, закрепим на последнем коммите)
|
||||||
|
move_tag_to_head "$GORELEASER_CURRENT_TAG"
|
||||||
|
else
|
||||||
|
# Если версия не задана, попробуем взять последний существующий тег и перенести его на HEAD
|
||||||
|
LAST_TAG="$(git describe --tags --abbrev=0 2>/dev/null || true)"
|
||||||
|
if [[ -n "$LAST_TAG" ]]; then
|
||||||
|
move_tag_to_head "$LAST_TAG"
|
||||||
|
export GORELEASER_CURRENT_TAG="$LAST_TAG"
|
||||||
|
log "Использую последний тег: $LAST_TAG"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
if [[ -z "${GITHUB_TOKEN:-}" ]]; then
|
||||||
|
log "GITHUB_TOKEN не задан — пробую получить из k8s секрета (${K8S_NAMESPACE:-flux-system}/${K8S_SECRET_NAME:-git-secrets}, ключ: password)"
|
||||||
|
if ! command -v kubectl >/dev/null 2>&1; then
|
||||||
|
err "kubectl не найден, а GITHUB_TOKEN не задан. Установите kubectl или экспортируйте GITHUB_TOKEN."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
TOKEN_FROM_K8S="$(fetch_token_from_k8s || true)"
|
||||||
|
if [[ -n "$TOKEN_FROM_K8S" && "$TOKEN_FROM_K8S" != "null" ]]; then
|
||||||
|
export GITHUB_TOKEN="$TOKEN_FROM_K8S"
|
||||||
|
log "GITHUB_TOKEN получен из секрета Kubernetes."
|
||||||
|
else
|
||||||
|
err "Не удалось получить GITHUB_TOKEN из секрета Kubernetes. Экспортируйте GITHUB_TOKEN и повторите."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
log "Запуск goreleaser (публикация на GitHub)"
|
||||||
|
goreleaser release --clean --config "$DST_CFG" "${EXTRA_FLAGS[@]}"
|
||||||
|
fi
|
||||||
|
|
||||||
|
popd >/dev/null
|
||||||
|
|
||||||
|
# Откатываем временный коммит, если он был
|
||||||
|
if [[ "${TEMP_COMMIT_DONE:-false}" == "true" && -n "$PREV_HEAD" ]]; then
|
||||||
|
if git reset --soft "$PREV_HEAD" >/dev/null 2>&1; then
|
||||||
|
log "Откатил временный коммит"
|
||||||
|
else
|
||||||
|
log "Не удалось откатить временный коммит — проверьте историю вручную"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
log "Готово."
|
||||||
|
|
||||||
|
|
||||||
@@ -5,7 +5,6 @@ metadata:
|
|||||||
namespace: lcg
|
namespace: lcg
|
||||||
labels:
|
labels:
|
||||||
app: lcg
|
app: lcg
|
||||||
version: ${VERSION}
|
|
||||||
spec:
|
spec:
|
||||||
type: ClusterIP
|
type: ClusterIP
|
||||||
ports:
|
ports:
|
||||||
@@ -15,4 +14,3 @@ spec:
|
|||||||
name: http
|
name: http
|
||||||
selector:
|
selector:
|
||||||
app: lcg
|
app: lcg
|
||||||
version: ${VERSION}
|
|
||||||
|
|||||||
@@ -69,8 +69,8 @@ lcg co -f
|
|||||||
"host": "localhost"
|
"host": "localhost"
|
||||||
},
|
},
|
||||||
"validation": {
|
"validation": {
|
||||||
"max_system_prompt_length": 1000,
|
"max_system_prompt_length": 2000,
|
||||||
"max_user_message_length": 2000,
|
"max_user_message_length": 4000,
|
||||||
"max_prompt_name_length": 2000,
|
"max_prompt_name_length": 2000,
|
||||||
"max_prompt_desc_length": 5000,
|
"max_prompt_desc_length": 5000,
|
||||||
"max_command_length": 8000,
|
"max_command_length": 8000,
|
||||||
@@ -116,12 +116,12 @@ lcg co -f
|
|||||||
|
|
||||||
### Настройки валидации (validation)
|
### Настройки валидации (validation)
|
||||||
|
|
||||||
- **max_system_prompt_length** - максимальная длина системного промпта
|
- **max_system_prompt_length** - максимальная длина системного промпта (по умолчанию: 2000)
|
||||||
- **max_user_message_length** - максимальная длина пользовательского сообщения
|
- **max_user_message_length** - максимальная длина пользовательского сообщения (по умолчанию: 4000)
|
||||||
- **max_prompt_name_length** - максимальная длина названия промпта
|
- **max_prompt_name_length** - максимальная длина названия промпта (по умолчанию: 2000)
|
||||||
- **max_prompt_desc_length** - максимальная длина описания промпта
|
- **max_prompt_desc_length** - максимальная длина описания промпта (по умолчанию: 5000)
|
||||||
- **max_command_length** - максимальная длина команды/ответа
|
- **max_command_length** - максимальная длина команды/ответа (по умолчанию: 8000)
|
||||||
- **max_explanation_length** - максимальная длина объяснения
|
- **max_explanation_length** - максимальная длина объяснения (по умолчанию: 20000)
|
||||||
|
|
||||||
## 🔒 Безопасность
|
## 🔒 Безопасность
|
||||||
|
|
||||||
|
|||||||
@@ -35,10 +35,17 @@ Explanations:
|
|||||||
|
|
||||||
Clipboard support requires `xclip` or `xsel`.
|
Clipboard support requires `xclip` or `xsel`.
|
||||||
|
|
||||||
## What's new in 2.0.1
|
## What's new in 2.0.14
|
||||||
|
|
||||||
- Mobile UI improvements: better responsiveness (buttons, fonts, spacing) and reduced motion support
|
- Authentication: JWT-based authentication with HTTP-only cookies
|
||||||
- Public REST endpoint: `POST /execute` (curl-only) for programmatic access — see `API_GUIDE.md`
|
- CSRF protection: Full CSRF protection with tokens and middleware
|
||||||
|
- Security: Enhanced security with token validation and sessions
|
||||||
|
- Kubernetes deployment: Full set of manifests for Kubernetes deployment with Traefik
|
||||||
|
- Reverse Proxy: Support for working behind reverse proxy with cookie configuration
|
||||||
|
- Web interface: Improved web interface with modern design
|
||||||
|
- Monitoring: Prometheus metrics and ServiceMonitor
|
||||||
|
- Scaling: HPA for automatic scaling
|
||||||
|
- Testing: CSRF protection testing tools
|
||||||
|
|
||||||
## Environment
|
## Environment
|
||||||
|
|
||||||
@@ -133,10 +140,15 @@ The `serve` command provides both a web interface and REST API:
|
|||||||
|
|
||||||
**Web Interface:**
|
**Web Interface:**
|
||||||
|
|
||||||
- Browse results at `http://localhost:8080/`
|
- Browse results at `http://localhost:8080/` (or `http://localhost:8080<BASE_PATH>/` if `LCG_BASE_URL` set)
|
||||||
- Execute requests at `http://localhost:8080/run`
|
- Execute requests at `.../run`
|
||||||
- Manage prompts at `http://localhost:8080/prompts`
|
- Manage prompts at `.../prompts`
|
||||||
- View history at `http://localhost:8080/history`
|
- View history at `.../history`
|
||||||
|
|
||||||
|
Notes:
|
||||||
|
- Base path: set `LCG_BASE_URL` (e.g. `/lcg`) to prefix all routes and API.
|
||||||
|
- Custom 404: unknown paths under base path render a modern 404 page.
|
||||||
|
- Debug: enable via flag `--debug` or env `LCG_DEBUG=1|true`.
|
||||||
|
|
||||||
**REST API:**
|
**REST API:**
|
||||||
|
|
||||||
|
|||||||
@@ -22,7 +22,7 @@ sudo apt-get install xsel
|
|||||||
|
|
||||||
```bash
|
```bash
|
||||||
|
|
||||||
git clone --depth 1 https://github.com/Direct-Dev-Ru/go-lcg.git ~/.linux-command-gpt
|
git clone --depth 1 https://github.com/Direct-Dev-Ru/linux-command-gpt.git ~/.linux-command-gpt
|
||||||
cd ~/.linux-command-gpt
|
cd ~/.linux-command-gpt
|
||||||
go build -o lcg
|
go build -o lcg
|
||||||
|
|
||||||
@@ -60,7 +60,7 @@ lcg --file /path/to/context.txt "хочу вывести список дирек
|
|||||||
Действия: (c)копировать, (s)сохранить, (r)перегенерировать, (e)выполнить, (v|vv|vvv)подробно, (n)ничего:
|
Действия: (c)копировать, (s)сохранить, (r)перегенерировать, (e)выполнить, (v|vv|vvv)подробно, (n)ничего:
|
||||||
```
|
```
|
||||||
|
|
||||||
### Что нового в 3.0.0
|
### Что нового в 2.0.14
|
||||||
|
|
||||||
- **Аутентификация**: Добавлена система аутентификации с JWT токенами и HTTP-only cookies
|
- **Аутентификация**: Добавлена система аутентификации с JWT токенами и HTTP-only cookies
|
||||||
- **CSRF защита**: Полная защита от CSRF атак с токенами и middleware
|
- **CSRF защита**: Полная защита от CSRF атак с токенами и middleware
|
||||||
|
|||||||
@@ -8,26 +8,26 @@
|
|||||||
|
|
||||||
| Переменная | Описание | По умолчанию |
|
| Переменная | Описание | По умолчанию |
|
||||||
|------------|----------|--------------|
|
|------------|----------|--------------|
|
||||||
| `LCG_MAX_SYSTEM_PROMPT_LENGTH` | Максимальная длина системного промпта | 1000 |
|
| `LCG_MAX_SYSTEM_PROMPT_LENGTH` | Максимальная длина системного промпта | 2000 |
|
||||||
| `LCG_MAX_USER_MESSAGE_LENGTH` | Максимальная длина пользовательского сообщения | 2000 |
|
| `LCG_MAX_USER_MESSAGE_LENGTH` | Максимальная длина пользовательского сообщения | 4000 |
|
||||||
| `LCG_MAX_PROMPT_NAME_LENGTH` | Максимальная длина названия промпта | 200 |
|
| `LCG_MAX_PROMPT_NAME_LENGTH` | Максимальная длина названия промпта | 2000 |
|
||||||
| `LCG_MAX_PROMPT_DESC_LENGTH` | Максимальная длина описания промпта | 500 |
|
| `LCG_MAX_PROMPT_DESC_LENGTH` | Максимальная длина описания промпта | 5000 |
|
||||||
| `LCG_MAX_COMMAND_LENGTH` | Максимальная длина команды/ответа | 2000 |
|
| `LCG_MAX_COMMAND_LENGTH` | Максимальная длина команды/ответа | 8000 |
|
||||||
| `LCG_MAX_EXPLANATION_LENGTH` | Максимальная длина объяснения | 2000 |
|
| `LCG_MAX_EXPLANATION_LENGTH` | Максимальная длина объяснения | 20000 |
|
||||||
|
|
||||||
## 🚀 Примеры использования
|
## 🚀 Примеры использования
|
||||||
|
|
||||||
### Установка через переменные окружения
|
### Установка через переменные окружения
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Увеличить лимит системного промпта до 2к символов
|
# Увеличить лимит системного промпта до 3к символов
|
||||||
export LCG_MAX_SYSTEM_PROMPT_LENGTH=2000
|
export LCG_MAX_SYSTEM_PROMPT_LENGTH=3000
|
||||||
|
|
||||||
# Уменьшить лимит пользовательского сообщения до 1к символов
|
# Уменьшить лимит пользовательского сообщения до 2к символов
|
||||||
export LCG_MAX_USER_MESSAGE_LENGTH=1000
|
export LCG_MAX_USER_MESSAGE_LENGTH=2000
|
||||||
|
|
||||||
# Увеличить лимит названия промпта до 500 символов
|
# Увеличить лимит названия промпта до 3000 символов
|
||||||
export LCG_MAX_PROMPT_NAME_LENGTH=500
|
export LCG_MAX_PROMPT_NAME_LENGTH=3000
|
||||||
```
|
```
|
||||||
|
|
||||||
### Установка в .env файле
|
### Установка в .env файле
|
||||||
@@ -35,11 +35,11 @@ export LCG_MAX_PROMPT_NAME_LENGTH=500
|
|||||||
```bash
|
```bash
|
||||||
# .env файл
|
# .env файл
|
||||||
LCG_MAX_SYSTEM_PROMPT_LENGTH=2000
|
LCG_MAX_SYSTEM_PROMPT_LENGTH=2000
|
||||||
LCG_MAX_USER_MESSAGE_LENGTH=1500
|
LCG_MAX_USER_MESSAGE_LENGTH=4000
|
||||||
LCG_MAX_PROMPT_NAME_LENGTH=300
|
LCG_MAX_PROMPT_NAME_LENGTH=2000
|
||||||
LCG_MAX_PROMPT_DESC_LENGTH=1000
|
LCG_MAX_PROMPT_DESC_LENGTH=5000
|
||||||
LCG_MAX_COMMAND_LENGTH=3000
|
LCG_MAX_COMMAND_LENGTH=8000
|
||||||
LCG_MAX_EXPLANATION_LENGTH=5000
|
LCG_MAX_EXPLANATION_LENGTH=20000
|
||||||
```
|
```
|
||||||
|
|
||||||
### Установка в systemd сервисе
|
### Установка в systemd сервисе
|
||||||
@@ -55,8 +55,8 @@ User=lcg
|
|||||||
WorkingDirectory=/opt/lcg
|
WorkingDirectory=/opt/lcg
|
||||||
ExecStart=/opt/lcg/lcg serve
|
ExecStart=/opt/lcg/lcg serve
|
||||||
Environment=LCG_MAX_SYSTEM_PROMPT_LENGTH=2000
|
Environment=LCG_MAX_SYSTEM_PROMPT_LENGTH=2000
|
||||||
Environment=LCG_MAX_USER_MESSAGE_LENGTH=1500
|
Environment=LCG_MAX_USER_MESSAGE_LENGTH=4000
|
||||||
Environment=LCG_MAX_PROMPT_NAME_LENGTH=300
|
Environment=LCG_MAX_PROMPT_NAME_LENGTH=2000
|
||||||
Restart=always
|
Restart=always
|
||||||
|
|
||||||
[Install]
|
[Install]
|
||||||
@@ -72,7 +72,7 @@ FROM golang:1.21-alpine AS builder
|
|||||||
FROM alpine:latest
|
FROM alpine:latest
|
||||||
COPY --from=builder /app/lcg /usr/local/bin/
|
COPY --from=builder /app/lcg /usr/local/bin/
|
||||||
ENV LCG_MAX_SYSTEM_PROMPT_LENGTH=2000
|
ENV LCG_MAX_SYSTEM_PROMPT_LENGTH=2000
|
||||||
ENV LCG_MAX_USER_MESSAGE_LENGTH=1500
|
ENV LCG_MAX_USER_MESSAGE_LENGTH=4000
|
||||||
CMD ["lcg", "serve"]
|
CMD ["lcg", "serve"]
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -84,8 +84,8 @@ services:
|
|||||||
image: lcg:latest
|
image: lcg:latest
|
||||||
environment:
|
environment:
|
||||||
- LCG_MAX_SYSTEM_PROMPT_LENGTH=2000
|
- LCG_MAX_SYSTEM_PROMPT_LENGTH=2000
|
||||||
- LCG_MAX_USER_MESSAGE_LENGTH=1500
|
- LCG_MAX_USER_MESSAGE_LENGTH=4000
|
||||||
- LCG_MAX_PROMPT_NAME_LENGTH=300
|
- LCG_MAX_PROMPT_NAME_LENGTH=2000
|
||||||
ports:
|
ports:
|
||||||
- "8080:8080"
|
- "8080:8080"
|
||||||
```
|
```
|
||||||
@@ -153,9 +153,9 @@ validation.FormatLengthInfo(systemPrompt, userMessage)
|
|||||||
## 📝 Примеры сообщений об ошибках
|
## 📝 Примеры сообщений об ошибках
|
||||||
|
|
||||||
```
|
```
|
||||||
❌ Ошибка: system_prompt: системный промпт слишком длинный: 1500 символов (максимум 1000)
|
❌ Ошибка: system_prompt: системный промпт слишком длинный: 2500 символов (максимум 2000)
|
||||||
❌ Ошибка: user_message: пользовательское сообщение слишком длинное: 2500 символов (максимум 2000)
|
❌ Ошибка: user_message: пользовательское сообщение слишком длинное: 4500 символов (максимум 4000)
|
||||||
❌ Ошибка: prompt_name: название промпта слишком длинное: 300 символов (максимум 200)
|
❌ Ошибка: prompt_name: название промпта слишком длинное: 2500 символов (максимум 2000)
|
||||||
```
|
```
|
||||||
|
|
||||||
## 🔄 Миграция с жестко заданных значений
|
## 🔄 Миграция с жестко заданных значений
|
||||||
@@ -179,25 +179,25 @@ if err := validation.ValidateSystemPrompt(prompt); err != nil {
|
|||||||
### Для разработки
|
### Для разработки
|
||||||
```bash
|
```bash
|
||||||
export LCG_MAX_SYSTEM_PROMPT_LENGTH=2000
|
export LCG_MAX_SYSTEM_PROMPT_LENGTH=2000
|
||||||
export LCG_MAX_USER_MESSAGE_LENGTH=2000
|
export LCG_MAX_USER_MESSAGE_LENGTH=4000
|
||||||
export LCG_MAX_PROMPT_NAME_LENGTH=200
|
export LCG_MAX_PROMPT_NAME_LENGTH=2000
|
||||||
export LCG_MAX_PROMPT_DESC_LENGTH=500
|
export LCG_MAX_PROMPT_DESC_LENGTH=5000
|
||||||
```
|
```
|
||||||
|
|
||||||
### Для продакшена
|
### Для продакшена
|
||||||
```bash
|
```bash
|
||||||
export LCG_MAX_SYSTEM_PROMPT_LENGTH=1000
|
export LCG_MAX_SYSTEM_PROMPT_LENGTH=2000
|
||||||
export LCG_MAX_USER_MESSAGE_LENGTH=1500
|
export LCG_MAX_USER_MESSAGE_LENGTH=4000
|
||||||
export LCG_MAX_PROMPT_NAME_LENGTH=100
|
export LCG_MAX_PROMPT_NAME_LENGTH=2000
|
||||||
export LCG_MAX_PROMPT_DESC_LENGTH=300
|
export LCG_MAX_PROMPT_DESC_LENGTH=5000
|
||||||
```
|
```
|
||||||
|
|
||||||
### Для высоконагруженных систем
|
### Для высоконагруженных систем
|
||||||
```bash
|
```bash
|
||||||
export LCG_MAX_SYSTEM_PROMPT_LENGTH=500
|
export LCG_MAX_SYSTEM_PROMPT_LENGTH=1000
|
||||||
export LCG_MAX_USER_MESSAGE_LENGTH=1000
|
export LCG_MAX_USER_MESSAGE_LENGTH=2000
|
||||||
export LCG_MAX_PROMPT_NAME_LENGTH=50
|
export LCG_MAX_PROMPT_NAME_LENGTH=1000
|
||||||
export LCG_MAX_PROMPT_DESC_LENGTH=200
|
export LCG_MAX_PROMPT_DESC_LENGTH=2500
|
||||||
```
|
```
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|||||||
@@ -124,6 +124,11 @@ func (p *ProxyAPIProvider) Chat(messages []Chat) (string, error) {
|
|||||||
req.Header.Set("Authorization", "Bearer "+p.JWTToken)
|
req.Header.Set("Authorization", "Bearer "+p.JWTToken)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if config.AppConfig.MainFlags.Debug {
|
||||||
|
fmt.Println("Chat URL: ", p.BaseURL+config.AppConfig.Server.ProxyUrl)
|
||||||
|
fmt.Println("ProxyChatRequest: ", req)
|
||||||
|
}
|
||||||
|
|
||||||
resp, err := p.HTTPClient.Do(req)
|
resp, err := p.HTTPClient.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", fmt.Errorf("ошибка выполнения запроса: %w", err)
|
return "", fmt.Errorf("ошибка выполнения запроса: %w", err)
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ metadata:
|
|||||||
namespace: lcg
|
namespace: lcg
|
||||||
data:
|
data:
|
||||||
# Основные настройки
|
# Основные настройки
|
||||||
LCG_VERSION: "v2.0.3"
|
LCG_VERSION: "v2.0.14"
|
||||||
LCG_BASE_PATH: "/lcg"
|
LCG_BASE_PATH: "/lcg"
|
||||||
LCG_SERVER_HOST: "0.0.0.0"
|
LCG_SERVER_HOST: "0.0.0.0"
|
||||||
LCG_SERVER_PORT: "8080"
|
LCG_SERVER_PORT: "8080"
|
||||||
@@ -16,7 +16,8 @@ data:
|
|||||||
LCG_CONFIG_FOLDER: "/app/data/config"
|
LCG_CONFIG_FOLDER: "/app/data/config"
|
||||||
LCG_NO_HISTORY: "false"
|
LCG_NO_HISTORY: "false"
|
||||||
LCG_ALLOW_EXECUTION: "false"
|
LCG_ALLOW_EXECUTION: "false"
|
||||||
LCG_DEBUG: "false"
|
LCG_DEBUG: "true"
|
||||||
|
LCG_PROVIDER: "proxy"
|
||||||
|
|
||||||
# Настройки аутентификации
|
# Настройки аутентификации
|
||||||
LCG_SERVER_REQUIRE_AUTH: "true"
|
LCG_SERVER_REQUIRE_AUTH: "true"
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ metadata:
|
|||||||
namespace: lcg
|
namespace: lcg
|
||||||
labels:
|
labels:
|
||||||
app: lcg
|
app: lcg
|
||||||
version: v2.0.3
|
version: v2.0.14
|
||||||
spec:
|
spec:
|
||||||
replicas: 1
|
replicas: 1
|
||||||
selector:
|
selector:
|
||||||
@@ -15,11 +15,10 @@ spec:
|
|||||||
metadata:
|
metadata:
|
||||||
labels:
|
labels:
|
||||||
app: lcg
|
app: lcg
|
||||||
version: v2.0.3
|
|
||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- name: lcg
|
- name: lcg
|
||||||
image: kuznetcovay/lcg:v2.0.3
|
image: kuznetcovay/lcg:v2.0.14
|
||||||
imagePullPolicy: Always
|
imagePullPolicy: Always
|
||||||
ports:
|
ports:
|
||||||
- containerPort: 8080
|
- containerPort: 8080
|
||||||
@@ -58,21 +57,18 @@ spec:
|
|||||||
readOnly: true
|
readOnly: true
|
||||||
# Health checks
|
# Health checks
|
||||||
startupProbe:
|
startupProbe:
|
||||||
httpGet:
|
tcpSocket:
|
||||||
path: /login
|
|
||||||
port: 8080
|
port: 8080
|
||||||
initialDelaySeconds: 10
|
initialDelaySeconds: 10
|
||||||
periodSeconds: 5
|
periodSeconds: 5
|
||||||
failureThreshold: 30
|
failureThreshold: 30
|
||||||
readinessProbe:
|
readinessProbe:
|
||||||
httpGet:
|
tcpSocket:
|
||||||
path: /login
|
|
||||||
port: 8080
|
port: 8080
|
||||||
initialDelaySeconds: 5
|
initialDelaySeconds: 5
|
||||||
periodSeconds: 10
|
periodSeconds: 10
|
||||||
livenessProbe:
|
livenessProbe:
|
||||||
httpGet:
|
tcpSocket:
|
||||||
path: /login
|
|
||||||
port: 8080
|
port: 8080
|
||||||
initialDelaySeconds: 10
|
initialDelaySeconds: 10
|
||||||
periodSeconds: 60
|
periodSeconds: 60
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ metadata:
|
|||||||
namespace: lcg
|
namespace: lcg
|
||||||
labels:
|
labels:
|
||||||
app: lcg
|
app: lcg
|
||||||
version: v2.0.3
|
|
||||||
spec:
|
spec:
|
||||||
entryPoints:
|
entryPoints:
|
||||||
- websecure
|
- websecure
|
||||||
|
|||||||
@@ -13,13 +13,13 @@ resources:
|
|||||||
- ingress-route.yaml
|
- ingress-route.yaml
|
||||||
|
|
||||||
# Common labels
|
# Common labels
|
||||||
commonLabels:
|
# commonLabels:
|
||||||
app: lcg
|
# app: lcg
|
||||||
version: v2.0.3
|
# version: v2.0.14
|
||||||
managed-by: kustomize
|
# managed-by: kustomize
|
||||||
|
|
||||||
# Images
|
# Images
|
||||||
images:
|
# images:
|
||||||
- name: lcg
|
# - name: lcg
|
||||||
newName: kuznetcovay/lcg
|
# newName: kuznetcovay/lcg
|
||||||
newTag: v2.0.3
|
# newTag: v2.0.14
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ metadata:
|
|||||||
namespace: lcg
|
namespace: lcg
|
||||||
labels:
|
labels:
|
||||||
app: lcg
|
app: lcg
|
||||||
version: v2.0.3
|
|
||||||
spec:
|
spec:
|
||||||
type: ClusterIP
|
type: ClusterIP
|
||||||
ports:
|
ports:
|
||||||
@@ -15,4 +14,3 @@ spec:
|
|||||||
name: http
|
name: http
|
||||||
selector:
|
selector:
|
||||||
app: lcg
|
app: lcg
|
||||||
version: v2.0.3
|
|
||||||
|
|||||||
62
main.go
62
main.go
@@ -60,7 +60,7 @@ func main() {
|
|||||||
CompileConditions.NoServe = false
|
CompileConditions.NoServe = false
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Println("Build conditions:", CompileConditions)
|
// fmt.Println("Build conditions:", CompileConditions)
|
||||||
|
|
||||||
_ = colorBlue
|
_ = colorBlue
|
||||||
|
|
||||||
@@ -87,12 +87,56 @@ lcg [опции] <описание команды>
|
|||||||
{{.AppName}} - инструмент для генерации Linux команд из описаний на естественном языке.
|
{{.AppName}} - инструмент для генерации Linux команд из описаний на естественном языке.
|
||||||
Поддерживает чтение частей промпта из файлов и позволяет сохранять, копировать или перегенерировать результаты.
|
Поддерживает чтение частей промпта из файлов и позволяет сохранять, копировать или перегенерировать результаты.
|
||||||
может задавать системный промпт или выбирать из предустановленных промптов.
|
может задавать системный промпт или выбирать из предустановленных промптов.
|
||||||
|
|
||||||
Переменные окружения:
|
Переменные окружения:
|
||||||
|
|
||||||
|
Основные настройки:
|
||||||
LCG_HOST Endpoint для LLM API (по умолчанию: http://192.168.87.108:11434/)
|
LCG_HOST Endpoint для LLM API (по умолчанию: http://192.168.87.108:11434/)
|
||||||
LCG_MODEL Название модели (по умолчанию: codegeex4)
|
LCG_MODEL Название модели (по умолчанию: hf.co/yandex/YandexGPT-5-Lite-8B-instruct-GGUF:Q4_K_M)
|
||||||
LCG_PROMPT Текст промпта по умолчанию
|
LCG_PROMPT Текст промпта по умолчанию
|
||||||
LCG_PROVIDER Тип провайдера: "ollama" или "proxy" (по умолчанию: ollama)
|
LCG_PROVIDER Тип провайдера: "ollama" или "proxy" (по умолчанию: ollama)
|
||||||
LCG_JWT_TOKEN JWT токен для proxy провайдера
|
LCG_JWT_TOKEN JWT токен для proxy провайдера
|
||||||
|
LCG_PROMPT_ID ID промпта по умолчанию (по умолчанию: 1)
|
||||||
|
LCG_TIMEOUT Таймаут запроса в секундах (по умолчанию: 300)
|
||||||
|
LCG_COMPLETIONS_PATH Путь к API для завершений (по умолчанию: api/chat)
|
||||||
|
LCG_PROXY_URL URL прокси для proxy провайдера (по умолчанию: /api/v1/protected/sberchat/chat)
|
||||||
|
LCG_API_KEY_FILE Файл с API ключом (по умолчанию: .openai_api_key)
|
||||||
|
LCG_APP_NAME Название приложения (по умолчанию: Linux Command GPT)
|
||||||
|
|
||||||
|
Настройки истории и выполнения:
|
||||||
|
LCG_NO_HISTORY Отключить запись истории ("1" или "true" = отключено, пусто = включено)
|
||||||
|
LCG_ALLOW_EXECUTION Разрешить выполнение команд ("1" или "true" = разрешено, пусто = запрещено)
|
||||||
|
LCG_RESULT_FOLDER Папка для сохранения результатов (по умолчанию: ~/.config/lcg/gpt_results)
|
||||||
|
LCG_RESULT_HISTORY Файл истории результатов (по умолчанию: <result_folder>/lcg_history.json)
|
||||||
|
LCG_PROMPT_FOLDER Папка для системных промптов (по умолчанию: ~/.config/lcg/gpt_sys_prompts)
|
||||||
|
LCG_CONFIG_FOLDER Папка для конфигурации (по умолчанию: ~/.config/lcg/config)
|
||||||
|
|
||||||
|
Настройки сервера (команда serve):
|
||||||
|
LCG_SERVER_PORT Порт сервера (по умолчанию: 8080)
|
||||||
|
LCG_SERVER_HOST Хост сервера (по умолчанию: localhost)
|
||||||
|
LCG_SERVER_ALLOW_HTTP Разрешить HTTP соединения ("true" для localhost, "false" для других хостов)
|
||||||
|
LCG_SERVER_REQUIRE_AUTH Требовать аутентификацию ("1" или "true" = требуется, пусто = не требуется)
|
||||||
|
LCG_SERVER_PASSWORD Пароль администратора (по умолчанию: admin#123456)
|
||||||
|
LCG_SERVER_SSL_CERT_FILE Путь к SSL сертификату
|
||||||
|
LCG_SERVER_SSL_KEY_FILE Путь к приватному ключу SSL
|
||||||
|
LCG_DOMAIN Домен для сервера (по умолчанию: значение LCG_SERVER_HOST)
|
||||||
|
LCG_COOKIE_SECURE Безопасные cookie ("1" или "true" = включено, пусто = выключено)
|
||||||
|
LCG_COOKIE_PATH Путь для cookie (по умолчанию: /lcg)
|
||||||
|
LCG_COOKIE_TTL_HOURS Время жизни cookie в часах (по умолчанию: 168)
|
||||||
|
LCG_BASE_URL Базовый URL приложения (по умолчанию: /lcg)
|
||||||
|
LCG_HEALTH_URL URL для проверки здоровья API (по умолчанию: /api/v1/protected/sberchat/health)
|
||||||
|
|
||||||
|
Настройки валидации:
|
||||||
|
LCG_MAX_SYSTEM_PROMPT_LENGTH Максимальная длина системного промпта (по умолчанию: 2000)
|
||||||
|
LCG_MAX_USER_MESSAGE_LENGTH Максимальная длина пользовательского сообщения (по умолчанию: 4000)
|
||||||
|
LCG_MAX_PROMPT_NAME_LENGTH Максимальная длина названия промпта (по умолчанию: 2000)
|
||||||
|
LCG_MAX_PROMPT_DESC_LENGTH Максимальная длина описания промпта (по умолчанию: 5000)
|
||||||
|
LCG_MAX_COMMAND_LENGTH Максимальная длина команды (по умолчанию: 8000)
|
||||||
|
LCG_MAX_EXPLANATION_LENGTH Максимальная длина объяснения (по умолчанию: 20000)
|
||||||
|
|
||||||
|
Отладка и браузер:
|
||||||
|
LCG_DEBUG Включить режим отладки ("1" или "true" = включено, пусто = выключено)
|
||||||
|
LCG_BROWSER_PATH Путь к браузеру для автоматического открытия (команда serve --browser)
|
||||||
`,
|
`,
|
||||||
Flags: []cli.Flag{
|
Flags: []cli.Flag{
|
||||||
&cli.StringFlag{
|
&cli.StringFlag{
|
||||||
@@ -156,6 +200,12 @@ lcg [опции] <описание команды>
|
|||||||
Debug: c.Bool("debug"),
|
Debug: c.Bool("debug"),
|
||||||
}
|
}
|
||||||
disableHistory = config.AppConfig.MainFlags.NoHistory || config.AppConfig.IsNoHistoryEnabled()
|
disableHistory = config.AppConfig.MainFlags.NoHistory || config.AppConfig.IsNoHistoryEnabled()
|
||||||
|
|
||||||
|
config.AppConfig.MainFlags.Debug = config.AppConfig.MainFlags.Debug || config.GetEnvBool("LCG_DEBUG", false)
|
||||||
|
|
||||||
|
// fmt.Println("Debug:", config.AppConfig.MainFlags.Debug)
|
||||||
|
// fmt.Println("LCG_DEBUG:", config.GetEnvBool("LCG_DEBUG", false))
|
||||||
|
|
||||||
args := c.Args().Slice()
|
args := c.Args().Slice()
|
||||||
|
|
||||||
if len(args) == 0 {
|
if len(args) == 0 {
|
||||||
@@ -566,11 +616,9 @@ func getCommands() []*cli.Command {
|
|||||||
host := c.String("host")
|
host := c.String("host")
|
||||||
openBrowser := c.Bool("browser")
|
openBrowser := c.Bool("browser")
|
||||||
|
|
||||||
// Пробрасываем глобальный флаг debug для web-сервера
|
// Пробрасываем debug: флаг или переменная окружения LCG_DEBUG
|
||||||
// Позволяет запускать: lcg -d serve -p ...
|
// Позволяет запускать: LCG_DEBUG=1 lcg serve ... или lcg -d serve ...
|
||||||
if c.Bool("debug") {
|
config.AppConfig.MainFlags.Debug = c.Bool("debug") || config.GetEnvBool("LCG_DEBUG", false)
|
||||||
config.AppConfig.MainFlags.Debug = true
|
|
||||||
}
|
|
||||||
|
|
||||||
// Обновляем конфигурацию сервера с новыми параметрами
|
// Обновляем конфигурацию сервера с новыми параметрами
|
||||||
config.AppConfig.Server.Host = host
|
config.AppConfig.Server.Host = host
|
||||||
|
|||||||
@@ -94,7 +94,7 @@ func validateJWTToken(tokenString string) (*JWTClaims, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Парсим токен
|
// Парсим токен
|
||||||
token, err := jwt.ParseWithClaims(tokenString, &JWTClaims{}, func(token *jwt.Token) (interface{}, error) {
|
token, err := jwt.ParseWithClaims(tokenString, &JWTClaims{}, func(token *jwt.Token) (any, error) {
|
||||||
// Проверяем метод подписи
|
// Проверяем метод подписи
|
||||||
if _, ok := token.Method.(*jwt.SigningMethodHMAC); !ok {
|
if _, ok := token.Method.(*jwt.SigningMethodHMAC); !ok {
|
||||||
return nil, fmt.Errorf("unexpected signing method: %v", token.Header["alg"])
|
return nil, fmt.Errorf("unexpected signing method: %v", token.Header["alg"])
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ import (
|
|||||||
func handleLoginPage(w http.ResponseWriter, r *http.Request) {
|
func handleLoginPage(w http.ResponseWriter, r *http.Request) {
|
||||||
// Если пользователь уже авторизован, перенаправляем на главную
|
// Если пользователь уже авторизован, перенаправляем на главную
|
||||||
if isAuthenticated(r) {
|
if isAuthenticated(r) {
|
||||||
http.Redirect(w, r, "/", http.StatusSeeOther)
|
http.Redirect(w, r, makePath("/"), http.StatusSeeOther)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -41,6 +41,7 @@ func handleLoginPage(w http.ResponseWriter, r *http.Request) {
|
|||||||
Message: "",
|
Message: "",
|
||||||
Error: "",
|
Error: "",
|
||||||
CSRFToken: csrfToken,
|
CSRFToken: csrfToken,
|
||||||
|
BasePath: getBasePath(),
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := RenderLoginPage(w, data); err != nil {
|
if err := RenderLoginPage(w, data); err != nil {
|
||||||
@@ -73,6 +74,7 @@ type LoginPageData struct {
|
|||||||
Message string
|
Message string
|
||||||
Error string
|
Error string
|
||||||
CSRFToken string
|
CSRFToken string
|
||||||
|
BasePath string
|
||||||
}
|
}
|
||||||
|
|
||||||
// RenderLoginPage рендерит страницу входа
|
// RenderLoginPage рендерит страницу входа
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package serve
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/direct-dev-ru/linux-command-gpt/config"
|
"github.com/direct-dev-ru/linux-command-gpt/config"
|
||||||
)
|
)
|
||||||
@@ -15,8 +16,8 @@ func AuthMiddleware(next http.HandlerFunc) http.HandlerFunc {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Исключаем страницу входа и API логина из проверки
|
// Исключаем страницу входа и API логина из проверки (с учетом BasePath)
|
||||||
if r.URL.Path == "/login" || r.URL.Path == "/api/login" || r.URL.Path == "/api/validate-token" {
|
if r.URL.Path == makePath("/login") || r.URL.Path == makePath("/api/login") || r.URL.Path == makePath("/api/validate-token") {
|
||||||
next(w, r)
|
next(w, r)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -31,8 +32,8 @@ func AuthMiddleware(next http.HandlerFunc) http.HandlerFunc {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Для веб-запросов перенаправляем на страницу входа
|
// Для веб-запросов перенаправляем на страницу входа (с учетом BasePath)
|
||||||
http.Redirect(w, r, "/login", http.StatusSeeOther)
|
http.Redirect(w, r, makePath("/login"), http.StatusSeeOther)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -50,8 +51,8 @@ func CSRFMiddleware(next http.HandlerFunc) http.HandlerFunc {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Исключаем некоторые API endpoints
|
// Исключаем некоторые API endpoints (с учетом BasePath)
|
||||||
if r.URL.Path == "/api/login" || r.URL.Path == "/api/logout" {
|
if r.URL.Path == makePath("/api/login") || r.URL.Path == makePath("/api/logout") {
|
||||||
next(w, r)
|
next(w, r)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -103,7 +104,8 @@ func CSRFMiddleware(next http.HandlerFunc) http.HandlerFunc {
|
|||||||
// isAPIRequest проверяет, является ли запрос API запросом
|
// isAPIRequest проверяет, является ли запрос API запросом
|
||||||
func isAPIRequest(r *http.Request) bool {
|
func isAPIRequest(r *http.Request) bool {
|
||||||
path := r.URL.Path
|
path := r.URL.Path
|
||||||
return len(path) >= 4 && path[:4] == "/api"
|
apiPrefix := makePath("/api")
|
||||||
|
return strings.HasPrefix(path, apiPrefix)
|
||||||
}
|
}
|
||||||
|
|
||||||
// RequireAuth обертка для requireAuth из auth.go
|
// RequireAuth обертка для requireAuth из auth.go
|
||||||
|
|||||||
@@ -40,6 +40,7 @@ func generateAbbreviation(appName string) string {
|
|||||||
// FileInfo содержит информацию о файле
|
// FileInfo содержит информацию о файле
|
||||||
type FileInfo struct {
|
type FileInfo struct {
|
||||||
Name string
|
Name string
|
||||||
|
DisplayName string
|
||||||
Size string
|
Size string
|
||||||
ModTime string
|
ModTime string
|
||||||
Preview template.HTML
|
Preview template.HTML
|
||||||
@@ -134,6 +135,7 @@ func getResultFiles() ([]FileInfo, error) {
|
|||||||
|
|
||||||
files = append(files, FileInfo{
|
files = append(files, FileInfo{
|
||||||
Name: entry.Name(),
|
Name: entry.Name(),
|
||||||
|
DisplayName: formatFileDisplayName(entry.Name()),
|
||||||
Size: formatFileSize(info.Size()),
|
Size: formatFileSize(info.Size()),
|
||||||
ModTime: info.ModTime().Format("02.01.2006 15:04"),
|
ModTime: info.ModTime().Format("02.01.2006 15:04"),
|
||||||
Preview: template.HTML(preview),
|
Preview: template.HTML(preview),
|
||||||
@@ -167,24 +169,91 @@ func formatFileSize(size int64) string {
|
|||||||
return fmt.Sprintf("%.1f %cB", float64(size)/float64(div), "KMGTPE"[exp])
|
return fmt.Sprintf("%.1f %cB", float64(size)/float64(div), "KMGTPE"[exp])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// formatFileDisplayName преобразует имя файла вида
|
||||||
|
// gpt_request_GigaChat-2-Max_2025-10-22_13-50-13.md
|
||||||
|
// в "Gpt Request GigaChat 2 Max — 2025-10-22 13:50:13"
|
||||||
|
func formatFileDisplayName(filename string) string {
|
||||||
|
name := strings.TrimSuffix(filename, ".md")
|
||||||
|
// Разделим на части по '_'
|
||||||
|
parts := strings.Split(name, "_")
|
||||||
|
if len(parts) == 0 {
|
||||||
|
return filename
|
||||||
|
}
|
||||||
|
|
||||||
|
// Первая часть может быть префиксом gpt/request — заменим '_' на пробел и приведем регистр
|
||||||
|
var words []string
|
||||||
|
for _, p := range parts {
|
||||||
|
if p == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// Заменяем '-' на пробел в словах модели/текста
|
||||||
|
p = strings.ReplaceAll(p, "-", " ")
|
||||||
|
// Разбиваем по пробелам и капитализуем каждое слово
|
||||||
|
for _, w := range strings.Fields(p) {
|
||||||
|
if w == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
r := []rune(w)
|
||||||
|
r[0] = unicode.ToUpper(r[0])
|
||||||
|
words = append(words, string(r))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Попробуем распознать хвост как дату и время
|
||||||
|
// Ищем шаблон YYYY-MM-DD_HH-MM-SS в исходном имени
|
||||||
|
var pretty string
|
||||||
|
// ожидаем последние две части — дата и время
|
||||||
|
if len(parts) >= 3 {
|
||||||
|
datePart := parts[len(parts)-2]
|
||||||
|
timePart := parts[len(parts)-1]
|
||||||
|
// заменить '-' в времени на ':'
|
||||||
|
timePretty := strings.ReplaceAll(timePart, "-", ":")
|
||||||
|
if len(datePart) == 10 && len(timePart) == 8 { // примитивная проверка
|
||||||
|
// Собираем текст до датных частей
|
||||||
|
text := strings.Join(words[:len(words)-2], " ")
|
||||||
|
pretty = strings.TrimSpace(text)
|
||||||
|
if pretty != "" {
|
||||||
|
pretty += " — " + datePart + " " + timePretty
|
||||||
|
} else {
|
||||||
|
pretty = datePart + " " + timePretty
|
||||||
|
}
|
||||||
|
return pretty
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(words) > 0 {
|
||||||
|
pretty = strings.Join(words, " ")
|
||||||
|
return pretty
|
||||||
|
}
|
||||||
|
return filename
|
||||||
|
}
|
||||||
|
|
||||||
// handleFileView обрабатывает просмотр конкретного файла
|
// handleFileView обрабатывает просмотр конкретного файла
|
||||||
func handleFileView(w http.ResponseWriter, r *http.Request) {
|
func handleFileView(w http.ResponseWriter, r *http.Request) {
|
||||||
filename := strings.TrimPrefix(r.URL.Path, "/file/")
|
// Учитываем BasePath при извлечении имени файла
|
||||||
|
basePath := config.AppConfig.Server.BasePath
|
||||||
|
var filename string
|
||||||
|
if basePath != "" && basePath != "/" {
|
||||||
|
basePath = strings.TrimSuffix(basePath, "/")
|
||||||
|
filename = strings.TrimPrefix(r.URL.Path, basePath+"/file/")
|
||||||
|
} else {
|
||||||
|
filename = strings.TrimPrefix(r.URL.Path, "/file/")
|
||||||
|
}
|
||||||
if filename == "" {
|
if filename == "" {
|
||||||
http.NotFound(w, r)
|
renderNotFound(w, "Файл не указан", getBasePath())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Проверяем, что файл существует и находится в папке результатов
|
// Проверяем, что файл существует и находится в папке результатов
|
||||||
filePath := filepath.Join(config.AppConfig.ResultFolder, filename)
|
filePath := filepath.Join(config.AppConfig.ResultFolder, filename)
|
||||||
if !strings.HasPrefix(filePath, config.AppConfig.ResultFolder) {
|
if !strings.HasPrefix(filePath, config.AppConfig.ResultFolder) {
|
||||||
http.NotFound(w, r)
|
renderNotFound(w, "Запрошенный файл недоступен", getBasePath())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
content, err := os.ReadFile(filePath)
|
content, err := os.ReadFile(filePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
http.NotFound(w, r)
|
renderNotFound(w, "Файл не найден или был удален", getBasePath())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -195,9 +264,11 @@ func handleFileView(w http.ResponseWriter, r *http.Request) {
|
|||||||
data := struct {
|
data := struct {
|
||||||
Filename string
|
Filename string
|
||||||
Content template.HTML
|
Content template.HTML
|
||||||
|
BasePath string
|
||||||
}{
|
}{
|
||||||
Filename: filename,
|
Filename: filename,
|
||||||
Content: template.HTML(htmlContent),
|
Content: template.HTML(htmlContent),
|
||||||
|
BasePath: getBasePath(),
|
||||||
}
|
}
|
||||||
|
|
||||||
// Парсим и выполняем шаблон
|
// Парсим и выполняем шаблон
|
||||||
@@ -221,22 +292,30 @@ func handleDeleteFile(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
filename := strings.TrimPrefix(r.URL.Path, "/delete/")
|
// Учитываем BasePath при извлечении имени файла
|
||||||
|
basePath := config.AppConfig.Server.BasePath
|
||||||
|
var filename string
|
||||||
|
if basePath != "" && basePath != "/" {
|
||||||
|
basePath = strings.TrimSuffix(basePath, "/")
|
||||||
|
filename = strings.TrimPrefix(r.URL.Path, basePath+"/delete/")
|
||||||
|
} else {
|
||||||
|
filename = strings.TrimPrefix(r.URL.Path, "/delete/")
|
||||||
|
}
|
||||||
if filename == "" {
|
if filename == "" {
|
||||||
http.NotFound(w, r)
|
renderNotFound(w, "Файл не указан", getBasePath())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Проверяем, что файл существует и находится в папке результатов
|
// Проверяем, что файл существует и находится в папке результатов
|
||||||
filePath := filepath.Join(config.AppConfig.ResultFolder, filename)
|
filePath := filepath.Join(config.AppConfig.ResultFolder, filename)
|
||||||
if !strings.HasPrefix(filePath, config.AppConfig.ResultFolder) {
|
if !strings.HasPrefix(filePath, config.AppConfig.ResultFolder) {
|
||||||
http.NotFound(w, r)
|
renderNotFound(w, "Запрошенный файл недоступен", getBasePath())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Проверяем, что файл существует
|
// Проверяем, что файл существует
|
||||||
if _, err := os.Stat(filePath); os.IsNotExist(err) {
|
if _, err := os.Stat(filePath); os.IsNotExist(err) {
|
||||||
http.NotFound(w, r)
|
renderNotFound(w, "Файл не найден или уже удален", getBasePath())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -3,11 +3,13 @@ package serve
|
|||||||
import (
|
import (
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"html/template"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/direct-dev-ru/linux-command-gpt/config"
|
"github.com/direct-dev-ru/linux-command-gpt/config"
|
||||||
|
"github.com/direct-dev-ru/linux-command-gpt/serve/templates"
|
||||||
"github.com/direct-dev-ru/linux-command-gpt/ssl"
|
"github.com/direct-dev-ru/linux-command-gpt/ssl"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -48,6 +50,18 @@ func StartResultServer(host, port string) error {
|
|||||||
return fmt.Errorf("failed to initialize CSRF manager: %v", err)
|
return fmt.Errorf("failed to initialize CSRF manager: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Гарантируем наличие папки результатов и файла истории
|
||||||
|
if _, err := os.Stat(config.AppConfig.ResultFolder); os.IsNotExist(err) {
|
||||||
|
if mkErr := os.MkdirAll(config.AppConfig.ResultFolder, 0755); mkErr != nil {
|
||||||
|
return fmt.Errorf("failed to create results folder: %v", mkErr)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if _, err := os.Stat(config.AppConfig.ResultHistory); os.IsNotExist(err) {
|
||||||
|
if writeErr := Write(config.AppConfig.ResultHistory, []HistoryEntry{}); writeErr != nil {
|
||||||
|
return fmt.Errorf("failed to create history file: %v", writeErr)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
addr := fmt.Sprintf("%s:%s", host, port)
|
addr := fmt.Sprintf("%s:%s", host, port)
|
||||||
|
|
||||||
// Проверяем, нужно ли использовать HTTPS
|
// Проверяем, нужно ли использовать HTTPS
|
||||||
@@ -138,15 +152,21 @@ func registerHTTPSRoutes() {
|
|||||||
// Регистрируем все маршруты кроме главной страницы
|
// Регистрируем все маршруты кроме главной страницы
|
||||||
registerRoutesExceptHome()
|
registerRoutesExceptHome()
|
||||||
|
|
||||||
// Регистрируем главную страницу с проверкой HTTPS
|
// Регистрируем главную страницу (строго по BasePath) с проверкой HTTPS
|
||||||
http.HandleFunc(makePath("/"), func(w http.ResponseWriter, r *http.Request) {
|
http.HandleFunc(makePath("/"), func(w http.ResponseWriter, r *http.Request) {
|
||||||
// Проверяем, пришел ли запрос по HTTP (не HTTPS)
|
// Проверяем, пришел ли запрос по HTTP (не HTTPS)
|
||||||
if r.TLS == nil {
|
if r.TLS == nil {
|
||||||
handleHTTPSRedirect(w, r)
|
handleHTTPSRedirect(w, r)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
// Если уже HTTPS, обрабатываем как обычно
|
// Обрабатываем только точные пути: BasePath или BasePath/
|
||||||
|
bp := getBasePath()
|
||||||
|
p := r.URL.Path
|
||||||
|
if (bp == "" && (p == "/" || p == "")) || (bp != "" && (p == bp || p == bp+"/")) {
|
||||||
AuthMiddleware(handleResultsPage)(w, r)
|
AuthMiddleware(handleResultsPage)(w, r)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
renderNotFound(w, "Страница не найдена", bp)
|
||||||
})
|
})
|
||||||
|
|
||||||
// Регистрируем главную страницу без слэша в конце для BasePath
|
// Регистрируем главную страницу без слэша в конце для BasePath
|
||||||
@@ -163,6 +183,13 @@ func registerHTTPSRoutes() {
|
|||||||
AuthMiddleware(handleResultsPage)(w, r)
|
AuthMiddleware(handleResultsPage)(w, r)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Catch-all 404 для любых незарегистрированных путей (только когда BasePath задан)
|
||||||
|
if getBasePath() != "" {
|
||||||
|
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
renderNotFound(w, "Страница не найдена", getBasePath())
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// registerRoutesExceptHome регистрирует все маршруты кроме главной страницы
|
// registerRoutesExceptHome регистрирует все маршруты кроме главной страницы
|
||||||
@@ -203,6 +230,13 @@ func registerRoutesExceptHome() {
|
|||||||
// API для сохранения результатов и истории
|
// API для сохранения результатов и истории
|
||||||
http.HandleFunc(makePath("/api/save-result"), AuthMiddleware(CSRFMiddleware(handleSaveResult)))
|
http.HandleFunc(makePath("/api/save-result"), AuthMiddleware(CSRFMiddleware(handleSaveResult)))
|
||||||
http.HandleFunc(makePath("/api/add-to-history"), AuthMiddleware(CSRFMiddleware(handleAddToHistory)))
|
http.HandleFunc(makePath("/api/add-to-history"), AuthMiddleware(CSRFMiddleware(handleAddToHistory)))
|
||||||
|
|
||||||
|
// Catch-all 404 для любых незарегистрированных путей (только когда BasePath задан)
|
||||||
|
// if getBasePath() != "" {
|
||||||
|
// http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
// renderNotFound(w, "Страница не найдена", getBasePath())
|
||||||
|
// })
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
|
|
||||||
// registerRoutes регистрирует все маршруты сервера
|
// registerRoutes регистрирует все маршруты сервера
|
||||||
@@ -215,8 +249,17 @@ func registerRoutes() {
|
|||||||
http.HandleFunc(makePath("/api/logout"), handleLogout)
|
http.HandleFunc(makePath("/api/logout"), handleLogout)
|
||||||
http.HandleFunc(makePath("/api/validate-token"), handleValidateToken)
|
http.HandleFunc(makePath("/api/validate-token"), handleValidateToken)
|
||||||
|
|
||||||
// Главная страница и файлы
|
// Главная страница (строго по BasePath) и файлы
|
||||||
http.HandleFunc(makePath("/"), AuthMiddleware(handleResultsPage))
|
http.HandleFunc(makePath("/"), func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
// Обрабатываем только точные пути: BasePath или BasePath/
|
||||||
|
bp := getBasePath()
|
||||||
|
p := r.URL.Path
|
||||||
|
if (bp == "" && (p == "/" || p == "")) || (bp != "" && (p == bp || p == bp+"/")) {
|
||||||
|
AuthMiddleware(handleResultsPage)(w, r)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
renderNotFound(w, "Страница не найдена", bp)
|
||||||
|
})
|
||||||
http.HandleFunc(makePath("/file/"), AuthMiddleware(handleFileView))
|
http.HandleFunc(makePath("/file/"), AuthMiddleware(handleFileView))
|
||||||
http.HandleFunc(makePath("/delete/"), AuthMiddleware(handleDeleteFile))
|
http.HandleFunc(makePath("/delete/"), AuthMiddleware(handleDeleteFile))
|
||||||
|
|
||||||
@@ -251,4 +294,30 @@ func registerRoutes() {
|
|||||||
basePath = strings.TrimSuffix(basePath, "/")
|
basePath = strings.TrimSuffix(basePath, "/")
|
||||||
http.HandleFunc(basePath, AuthMiddleware(handleResultsPage))
|
http.HandleFunc(basePath, AuthMiddleware(handleResultsPage))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Catch-all 404 для любых незарегистрированных путей (только когда BasePath задан)
|
||||||
|
if getBasePath() != "" {
|
||||||
|
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
renderNotFound(w, "Страница не найдена", getBasePath())
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// renderNotFound рендерит кастомную страницу 404
|
||||||
|
func renderNotFound(w http.ResponseWriter, message, basePath string) {
|
||||||
|
w.WriteHeader(http.StatusNotFound)
|
||||||
|
data := struct {
|
||||||
|
Message string
|
||||||
|
BasePath string
|
||||||
|
}{
|
||||||
|
Message: message,
|
||||||
|
BasePath: basePath,
|
||||||
|
}
|
||||||
|
tmpl, err := template.New("not_found").Parse(templates.NotFoundTemplate)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, "404 Not Found", http.StatusNotFound)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.Header().Set("Content-Type", "text/html; charset=utf-8")
|
||||||
|
_ = tmpl.Execute(w, data)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -126,7 +126,7 @@ const FileViewTemplate = `
|
|||||||
<div class="container">
|
<div class="container">
|
||||||
<div class="header">
|
<div class="header">
|
||||||
<h1>📄 {{.Filename}}</h1>
|
<h1>📄 {{.Filename}}</h1>
|
||||||
<a href="/" class="back-btn">← Назад к списку</a>
|
<a href="{{.BasePath}}/" class="back-btn">← Назад к списку</a>
|
||||||
</div>
|
</div>
|
||||||
<div class="content">
|
<div class="content">
|
||||||
{{.Content}}
|
{{.Content}}
|
||||||
|
|||||||
@@ -111,19 +111,26 @@ const HistoryPageTemplate = `
|
|||||||
font-size: 0.9em;
|
font-size: 0.9em;
|
||||||
color: #2d5016;
|
color: #2d5016;
|
||||||
border-left: 3px solid #2d5016;
|
border-left: 3px solid #2d5016;
|
||||||
|
max-height: 72px; /* ~4 строки */
|
||||||
|
overflow: hidden;
|
||||||
|
display: -webkit-box;
|
||||||
|
-webkit-line-clamp: 4;
|
||||||
|
-webkit-box-orient: vertical;
|
||||||
}
|
}
|
||||||
.delete-btn {
|
.delete-btn {
|
||||||
background: #e74c3c;
|
background: transparent;
|
||||||
color: white;
|
color: #ef9a9a; /* бледно-красный */
|
||||||
border: none;
|
border: none;
|
||||||
padding: 6px 12px;
|
padding: 2px 6px;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-size: 0.8em;
|
font-size: 18px;
|
||||||
transition: background 0.3s ease;
|
line-height: 1;
|
||||||
|
transition: color 0.2s ease, transform 0.2s ease;
|
||||||
}
|
}
|
||||||
.delete-btn:hover {
|
.delete-btn:hover {
|
||||||
background: #c0392b;
|
color: rgb(171, 27, 24); /* ярче при ховере */
|
||||||
|
transform: translateY(-1px);
|
||||||
}
|
}
|
||||||
.empty-state {
|
.empty-state {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
@@ -180,7 +187,7 @@ const HistoryPageTemplate = `
|
|||||||
<span class="history-index">#{{.Index}}</span>
|
<span class="history-index">#{{.Index}}</span>
|
||||||
<span class="history-timestamp">{{.Timestamp}}</span>
|
<span class="history-timestamp">{{.Timestamp}}</span>
|
||||||
</div>
|
</div>
|
||||||
<button class="delete-btn" onclick="event.stopPropagation(); deleteHistoryEntry({{.Index}})">🗑️ Удалить</button>
|
<button class="delete-btn" onclick="event.stopPropagation(); deleteHistoryEntry({{.Index}})">✖</button>
|
||||||
</div>
|
</div>
|
||||||
<div class="history-command">{{.Command}}</div>
|
<div class="history-command">{{.Command}}</div>
|
||||||
<div class="history-response">{{.Response}}</div>
|
<div class="history-response">{{.Response}}</div>
|
||||||
|
|||||||
@@ -285,7 +285,7 @@ const LoginPageTemplate = `
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const csrfToken = document.getElementById('csrf_token').value;
|
const csrfToken = document.getElementById('csrf_token').value;
|
||||||
const response = await fetch('/api/login', {
|
const response = await fetch('{{.BasePath}}/api/login', {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
headers: {
|
headers: {
|
||||||
'Content-Type': 'application/json',
|
'Content-Type': 'application/json',
|
||||||
@@ -302,7 +302,7 @@ const LoginPageTemplate = `
|
|||||||
|
|
||||||
if (data.success) {
|
if (data.success) {
|
||||||
// Успешная авторизация, перенаправляем на главную страницу
|
// Успешная авторизация, перенаправляем на главную страницу
|
||||||
window.location.href = '/';
|
window.location.href = '{{.BasePath}}/';
|
||||||
} else {
|
} else {
|
||||||
// Ошибка авторизации
|
// Ошибка авторизации
|
||||||
showMessage(data.error || 'Ошибка авторизации', 'error');
|
showMessage(data.error || 'Ошибка авторизации', 'error');
|
||||||
|
|||||||
157
serve/templates/not_found.go
Normal file
157
serve/templates/not_found.go
Normal file
@@ -0,0 +1,157 @@
|
|||||||
|
package templates
|
||||||
|
|
||||||
|
// NotFoundTemplate современная страница 404
|
||||||
|
const NotFoundTemplate = `
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="ru">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8" />
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
|
<title>Страница не найдена — 404</title>
|
||||||
|
<style>
|
||||||
|
:root {
|
||||||
|
--bg: #1a0b0b; /* глубокий темно-красный фон */
|
||||||
|
--bg2: #2a0f0f; /* второй оттенок фона */
|
||||||
|
--fg: #ffeaea; /* светлый текст с красным оттенком */
|
||||||
|
--accent: #ff3b30; /* основной красный (iOS red) */
|
||||||
|
--accent2: #ff6f61; /* дополнительный коралловый */
|
||||||
|
--accentGlow: rgba(255,59,48,0.35);
|
||||||
|
--accentGlow2: rgba(255,111,97,0.30);
|
||||||
|
}
|
||||||
|
* { box-sizing: border-box; }
|
||||||
|
body {
|
||||||
|
margin: 0;
|
||||||
|
min-height: 100vh;
|
||||||
|
display: grid;
|
||||||
|
place-items: center;
|
||||||
|
background:
|
||||||
|
radial-gradient(1200px 600px at 10% 10%, rgba(255,59,48,0.12), transparent),
|
||||||
|
radial-gradient(1200px 600px at 90% 90%, rgba(255,111,97,0.12), transparent),
|
||||||
|
linear-gradient(135deg, var(--bg), var(--bg2));
|
||||||
|
color: var(--fg);
|
||||||
|
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Inter, sans-serif;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
.glow {
|
||||||
|
position: absolute;
|
||||||
|
inset: -20%;
|
||||||
|
background:
|
||||||
|
radial-gradient(700px 340px at 20% 30%, rgba(255,59,48,0.22), transparent 60%),
|
||||||
|
radial-gradient(700px 340px at 80% 70%, rgba(255,111,97,0.20), transparent 60%);
|
||||||
|
filter: blur(40px);
|
||||||
|
z-index: 0;
|
||||||
|
}
|
||||||
|
.card {
|
||||||
|
position: relative;
|
||||||
|
z-index: 1;
|
||||||
|
width: min(720px, 92vw);
|
||||||
|
padding: 32px;
|
||||||
|
border-radius: 20px;
|
||||||
|
background: rgba(255,255,255,0.03);
|
||||||
|
border: 1px solid rgba(255,255,255,0.08);
|
||||||
|
box-shadow: 0 10px 40px rgba(80,0,0,0.45), inset 0 0 0 1px rgba(255,255,255,0.03);
|
||||||
|
backdrop-filter: blur(10px);
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
@keyframes pulse {
|
||||||
|
0%, 100% {
|
||||||
|
transform: scale(1);
|
||||||
|
text-shadow: 0 8px 40px var(--accentGlow);
|
||||||
|
}
|
||||||
|
50% {
|
||||||
|
transform: scale(1.15);
|
||||||
|
text-shadow: 0 12px 60px var(--accentGlow), 0 0 30px var(--accentGlow2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.code {
|
||||||
|
font-size: clamp(48px, 12vw, 120px);
|
||||||
|
line-height: 0.9;
|
||||||
|
font-weight: 800;
|
||||||
|
letter-spacing: -2px;
|
||||||
|
background: linear-gradient(90deg, var(--accent), var(--accent2));
|
||||||
|
-webkit-background-clip: text;
|
||||||
|
background-clip: text;
|
||||||
|
color: transparent;
|
||||||
|
margin: 8px 0 12px 0;
|
||||||
|
text-shadow: 0 8px 40px var(--accentGlow);
|
||||||
|
animation: pulse 2.5s ease-in-out infinite;
|
||||||
|
transform-origin: center;
|
||||||
|
}
|
||||||
|
.title {
|
||||||
|
font-size: clamp(18px, 3.2vw, 28px);
|
||||||
|
font-weight: 600;
|
||||||
|
opacity: 0.95;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
.desc {
|
||||||
|
font-size: 15px;
|
||||||
|
opacity: 0.75;
|
||||||
|
margin: 0 auto 20px auto;
|
||||||
|
max-width: 60ch;
|
||||||
|
}
|
||||||
|
.btns {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 12px;
|
||||||
|
justify-content: center;
|
||||||
|
margin-top: 8px;
|
||||||
|
}
|
||||||
|
.btn {
|
||||||
|
appearance: none;
|
||||||
|
border: none;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 12px 18px;
|
||||||
|
border-radius: 12px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: white;
|
||||||
|
background: linear-gradient(135deg, var(--accent), #c62828);
|
||||||
|
box-shadow: 0 6px 18px var(--accentGlow);
|
||||||
|
transition: transform .2s ease, box-shadow .2s ease, filter .2s ease;
|
||||||
|
text-decoration: none;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
.btn:hover { transform: translateY(-2px); filter: brightness(1.05); }
|
||||||
|
.btn.secondary { background: linear-gradient(135deg, #e65100, var(--accent2)); box-shadow: 0 6px 18px var(--accentGlow2); }
|
||||||
|
.hint { margin-top: 16px; font-size: 13px; opacity: 0.6; }
|
||||||
|
</style>
|
||||||
|
<script>
|
||||||
|
function goHome() {
|
||||||
|
window.location.href = '{{.BasePath}}/';
|
||||||
|
}
|
||||||
|
function bindEsc() {
|
||||||
|
const handler = (e) => { if (e.key === 'Escape' || e.key === 'Esc') { e.preventDefault(); goHome(); } };
|
||||||
|
window.addEventListener('keydown', handler);
|
||||||
|
document.addEventListener('keydown', handler);
|
||||||
|
// фокус на body для гарантии получения клавиш
|
||||||
|
if (document && document.body) {
|
||||||
|
document.body.setAttribute('tabindex', '-1');
|
||||||
|
document.body.focus({ preventScroll: true });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (document.readyState === 'loading') {
|
||||||
|
document.addEventListener('DOMContentLoaded', bindEsc);
|
||||||
|
} else {
|
||||||
|
bindEsc();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<meta http-equiv="refresh" content="30">
|
||||||
|
<meta name="robots" content="noindex">
|
||||||
|
<link rel="icon" href="data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 64 64'><text y='50%' x='50%' dominant-baseline='middle' text-anchor='middle' font-size='42'>🚫</text></svg>">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="glow"></div>
|
||||||
|
<div class="card">
|
||||||
|
<div class="code">404</div>
|
||||||
|
<div class="title">Страница не найдена</div>
|
||||||
|
<p class="desc">Такой страницы не существует. Вы можете вернуться на главную страницу или выполнить команду.</p>
|
||||||
|
<div class="btns">
|
||||||
|
<a class="btn" href="{{.BasePath}}/">🏠 На главную</a>
|
||||||
|
<a class="btn secondary" href="{{.BasePath}}/run">🚀 К выполнению</a>
|
||||||
|
</div>
|
||||||
|
<div class="hint">Нажмите Esc, чтобы вернуться на главную</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`
|
||||||
@@ -73,8 +73,10 @@ const ResultsPageTemplate = `
|
|||||||
}
|
}
|
||||||
.files-grid {
|
.files-grid {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
|
grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
|
||||||
gap: 20px;
|
gap: 20px;
|
||||||
|
align-items: stretch;
|
||||||
|
grid-auto-rows: auto;
|
||||||
}
|
}
|
||||||
.file-card {
|
.file-card {
|
||||||
background: white;
|
background: white;
|
||||||
@@ -91,32 +93,36 @@ const ResultsPageTemplate = `
|
|||||||
}
|
}
|
||||||
.file-card-content {
|
.file-card-content {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
padding-left: 28px;
|
||||||
}
|
}
|
||||||
.file-actions {
|
.file-actions {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 10px;
|
top: 10px;
|
||||||
right: 10px;
|
left: 10px;
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
}
|
}
|
||||||
.delete-btn {
|
.delete-btn {
|
||||||
background: #e74c3c;
|
background: transparent;
|
||||||
color: white;
|
color: #ef9a9a; /* бледно-красный */
|
||||||
border: none;
|
border: none;
|
||||||
padding: 6px 12px;
|
padding: 4px 8px;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-size: 0.8em;
|
font-size: 18px;
|
||||||
transition: background 0.3s ease;
|
line-height: 1;
|
||||||
|
transition: color 0.2s ease, transform 0.2s ease;
|
||||||
}
|
}
|
||||||
.delete-btn:hover {
|
.delete-btn:hover {
|
||||||
background: #c0392b;
|
color:rgb(171, 27, 24); /* чуть ярче при ховере */
|
||||||
|
transform: translateY(-1px);
|
||||||
}
|
}
|
||||||
.file-name {
|
.file-name {
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
color: #333;
|
color: #333;
|
||||||
margin-bottom: 8px;
|
margin-bottom: 8px;
|
||||||
font-size: 1.1em;
|
font-size: 1.1em;
|
||||||
|
padding-right: 10px;
|
||||||
}
|
}
|
||||||
.file-info {
|
.file-info {
|
||||||
color: #666;
|
color: #666;
|
||||||
@@ -165,16 +171,25 @@ const ResultsPageTemplate = `
|
|||||||
body { padding: 10px; }
|
body { padding: 10px; }
|
||||||
.container { margin: 0; border-radius: 8px; box-shadow: 0 10px 20px rgba(0,0,0,0.1); }
|
.container { margin: 0; border-radius: 8px; box-shadow: 0 10px 20px rgba(0,0,0,0.1); }
|
||||||
.header { padding: 20px; }
|
.header { padding: 20px; }
|
||||||
.header h1 { font-size: 2em; }
|
.header h1 { font-size: 1.9em; }
|
||||||
.content { padding: 20px; }
|
.content { padding: 20px; }
|
||||||
.files-grid { grid-template-columns: 1fr; }
|
.files-grid { dummy-attr: none; }
|
||||||
|
/* Стили карточек как в истории */
|
||||||
|
.file-card { background: #f0f8f0; border: 1px solid #a8e6cf; padding: 15px; }
|
||||||
|
.file-card:hover { border-color: #2d5016; box-shadow: 0 8px 25px rgba(45,80,22,0.2); transform: translateY(-2px); }
|
||||||
|
.file-name { color: #333; margin-bottom: 8px; }
|
||||||
|
.file-info { color: #666; font-size: 0.9em; }
|
||||||
|
.file-preview { background: #f8f9fa; border-left: 3px solid #2d5016; font-size: 0.85em; }
|
||||||
|
.file-actions { top: 8px; left: 8px; }
|
||||||
|
.delete-btn { padding: 2px 6px; font-size: 16px; }
|
||||||
.stats { grid-template-columns: 1fr 1fr; }
|
.stats { grid-template-columns: 1fr 1fr; }
|
||||||
.nav-buttons { flex-direction: column; gap: 8px; }
|
.nav-buttons { flex-direction: column; gap: 8px; }
|
||||||
.nav-btn, .nav-button { text-align: center; padding: 12px 16px; font-size: 14px; }
|
.nav-btn, .nav-button { text-align: center; padding: 12px 16px; font-size: 14px; }
|
||||||
.search-container input { font-size: 16px; width: 96% !important; }
|
.search-container input { font-size: 16px; width: 96% !important; }
|
||||||
}
|
}
|
||||||
@media (max-width: 480px) {
|
@media (max-width: 480px) {
|
||||||
.header h1 { font-size: 1.8em; }
|
.header h1 { font-size: 1.6em; }
|
||||||
|
.content { padding: 16px; }
|
||||||
.stats { grid-template-columns: 1fr; }
|
.stats { grid-template-columns: 1fr; }
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
@@ -216,10 +231,10 @@ const ResultsPageTemplate = `
|
|||||||
{{range .Files}}
|
{{range .Files}}
|
||||||
<div class="file-card" data-content="{{.Content}}">
|
<div class="file-card" data-content="{{.Content}}">
|
||||||
<div class="file-actions">
|
<div class="file-actions">
|
||||||
<button class="delete-btn" onclick="deleteFile('{{.Name}}')" title="Удалить файл">🗑️</button>
|
<button class="delete-btn" onclick="deleteFile('{{.Name}}')" title="Удалить файл">✖</button>
|
||||||
</div>
|
</div>
|
||||||
<div class="file-card-content" onclick="window.location.href='{{$.BasePath}}/file/{{.Name}}'">
|
<div class="file-card-content" onclick="window.location.href='{{$.BasePath}}/file/{{.Name}}'">
|
||||||
<div class="file-name">{{.Name}}</div>
|
<div class="file-name">{{.DisplayName}}</div>
|
||||||
<div class="file-info">
|
<div class="file-info">
|
||||||
📅 {{.ModTime}} | 📏 {{.Size}}
|
📅 {{.ModTime}} | 📏 {{.Size}}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
18
ssl/ssl.go
18
ssl/ssl.go
@@ -1,6 +1,7 @@
|
|||||||
package ssl
|
package ssl
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"slices"
|
||||||
"crypto/rand"
|
"crypto/rand"
|
||||||
"crypto/rsa"
|
"crypto/rsa"
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
@@ -139,26 +140,23 @@ func LoadOrGenerateCert(host string) (*tls.Certificate, error) {
|
|||||||
// IsSecureHost проверяет, является ли хост безопасным для HTTP
|
// IsSecureHost проверяет, является ли хост безопасным для HTTP
|
||||||
func IsSecureHost(host string) bool {
|
func IsSecureHost(host string) bool {
|
||||||
secureHosts := []string{"localhost", "127.0.0.1", "::1"}
|
secureHosts := []string{"localhost", "127.0.0.1", "::1"}
|
||||||
for _, secureHost := range secureHosts {
|
return slices.Contains(secureHosts, host)
|
||||||
if host == secureHost {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ShouldUseHTTPS определяет, нужно ли использовать HTTPS
|
// ShouldUseHTTPS определяет, нужно ли использовать HTTPS
|
||||||
func ShouldUseHTTPS(host string) bool {
|
func ShouldUseHTTPS(host string) bool {
|
||||||
// Если хост не localhost/127.0.0.1, принуждаем к HTTPS
|
|
||||||
if !IsSecureHost(host) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
// Если явно разрешен HTTP, используем HTTP
|
// Если явно разрешен HTTP, используем HTTP
|
||||||
if config.AppConfig.Server.AllowHTTP {
|
if config.AppConfig.Server.AllowHTTP {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Если хост не localhost/127.0.0.1, принуждаем к HTTPS
|
||||||
|
if !IsSecureHost(host) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// По умолчанию для localhost используем HTTP
|
// По умолчанию для localhost используем HTTP
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user