mirror of
https://github.com/Direct-Dev-Ru/go-lcg.git
synced 2025-11-17 01:59:55 +00:00
Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| e54b99f6f4 |
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,83 +1,74 @@
|
|||||||
# Используем готовый образ Ollama
|
# Multi-stage build для LCG с Ollama
|
||||||
FROM localhost/ollama_packed:latest
|
FROM golang:1.24.6-alpine3.22 AS builder
|
||||||
|
|
||||||
# Устанавливаем bash если его нет (базовый образ ollama может быть на разных дистрибутивах)
|
WORKDIR /build
|
||||||
RUN if ! command -v bash >/dev/null 2>&1; then \
|
|
||||||
if command -v apk >/dev/null 2>&1; then \
|
|
||||||
apk add --no-cache bash; \
|
|
||||||
elif command -v apt-get >/dev/null 2>&1; then \
|
|
||||||
apt-get update && apt-get install -y --no-install-recommends bash && rm -rf /var/lib/apt/lists/*; \
|
|
||||||
fi; \
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Определяем архитектуру для копирования правильного бинарника
|
# Копируем файлы зависимостей
|
||||||
|
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
|
ARG TARGETARCH
|
||||||
ARG TARGETOS=linux
|
RUN case ${TARGETARCH} in \
|
||||||
|
amd64) OLLAMA_ARCH=amd64 ;; \
|
||||||
# Копируем папку dist с бинарниками
|
arm64) OLLAMA_ARCH=arm64 ;; \
|
||||||
# Структура: dist/lcg_linux_amd64_v1/lcg_* или dist/lcg_linux_arm64_v8.0/lcg_*
|
arm) OLLAMA_ARCH=arm64 ;; \
|
||||||
COPY dist/ /tmp/dist/
|
*) echo "Unsupported architecture: ${TARGETARCH}" && exit 1 ;; \
|
||||||
|
|
||||||
# Выбираем правильный бинарник в зависимости от архитектуры
|
|
||||||
# Если TARGETARCH не установлен, определяем архитектуру через uname
|
|
||||||
RUN ARCH="${TARGETARCH:-$(uname -m)}" && \
|
|
||||||
case "${ARCH}" in \
|
|
||||||
amd64|x86_64) \
|
|
||||||
BIN_FILE=$(find /tmp/dist/lcg_linux_amd64_v* -name "lcg_*" -type f 2>/dev/null | head -1) && \
|
|
||||||
if [ -n "$BIN_FILE" ]; then \
|
|
||||||
cp "$BIN_FILE" /usr/local/bin/lcg && \
|
|
||||||
echo "Установлен бинарник для amd64: $BIN_FILE"; \
|
|
||||||
else \
|
|
||||||
echo "Бинарник для amd64 не найден в /tmp/dist/" && exit 1; \
|
|
||||||
fi ;; \
|
|
||||||
arm64|aarch64|arm) \
|
|
||||||
BIN_FILE=$(find /tmp/dist/lcg_linux_arm64_v* -name "lcg_*" -type f 2>/dev/null | head -1) && \
|
|
||||||
if [ -n "$BIN_FILE" ]; then \
|
|
||||||
cp "$BIN_FILE" /usr/local/bin/lcg && \
|
|
||||||
echo "Установлен бинарник для arm64: $BIN_FILE"; \
|
|
||||||
else \
|
|
||||||
echo "Бинарник для arm64 не найден в /tmp/dist/" && exit 1; \
|
|
||||||
fi ;; \
|
|
||||||
*) \
|
|
||||||
echo "Unsupported architecture: ${ARCH}" && \
|
|
||||||
echo "Доступные бинарники:" && \
|
|
||||||
find /tmp/dist -name "lcg_*" -type f 2>/dev/null && \
|
|
||||||
exit 1 ;; \
|
|
||||||
esac && \
|
esac && \
|
||||||
chmod +x /usr/local/bin/lcg && \
|
curl -L https://github.com/ollama/ollama/releases/download/v0.9.5/ollama-linux-${OLLAMA_ARCH} -o /usr/local/bin/ollama \
|
||||||
rm -rf /tmp/dist && \
|
&& chmod +x /usr/local/bin/ollama
|
||||||
(lcg --version || echo "Бинарник lcg установлен")
|
|
||||||
|
# Создаем пользователя для запуска сервисов
|
||||||
|
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 скрипт
|
# Копируем entrypoint скрипт
|
||||||
COPY --chmod=755 Dockerfiles/OllamaServer/entrypoint.sh /entrypoint.sh
|
COPY --chmod=755 Dockerfiles/OllamaServer/entrypoint.sh /entrypoint.sh
|
||||||
|
|
||||||
# Создаем директории для данных LCG
|
# Создаем директории для данных
|
||||||
# В базовом образе ollama уже есть пользователь ollama
|
RUN mkdir -p /app/data/results /app/data/prompts /app/data/config \
|
||||||
RUN mkdir -p /app/data/results /app/data/prompts /app/data/config
|
&& chown -R ollama:ollama /app/data
|
||||||
|
|
||||||
# Устанавливаем права доступа (пользователь ollama должен существовать в базовом образе)
|
|
||||||
RUN chown -R ollama:ollama /app/data 2>/dev/null || \
|
|
||||||
(chown -R 1000:1000 /app/data 2>/dev/null || true)
|
|
||||||
|
|
||||||
# Настройки по умолчанию
|
# Настройки по умолчанию
|
||||||
ENV TZ='Asia/Omsk'
|
|
||||||
RUN ln -snf /usr/share/zoneinfo/$TZ /etc/localtime && echo $TZ > /etc/timezone
|
|
||||||
ENV LCG_PROVIDER=ollama
|
ENV LCG_PROVIDER=ollama
|
||||||
ENV LCG_HOST=http://127.0.0.1:11434/
|
ENV LCG_HOST=http://127.0.0.1:11434/
|
||||||
ENV LCG_MODEL=qwen2.5-coder:1.5b
|
ENV LCG_MODEL=codegeex4
|
||||||
ENV LCG_RESULT_FOLDER=/app/data/results
|
ENV LCG_RESULT_FOLDER=/app/data/results
|
||||||
ENV LCG_PROMPT_FOLDER=/app/data/prompts
|
ENV LCG_PROMPT_FOLDER=/app/data/prompts
|
||||||
ENV LCG_CONFIG_FOLDER=/app/data/config
|
ENV LCG_CONFIG_FOLDER=/app/data/config
|
||||||
ENV LCG_SERVER_HOST=0.0.0.0
|
ENV LCG_SERVER_HOST=0.0.0.0
|
||||||
ENV LCG_SERVER_PORT=8080
|
ENV LCG_SERVER_PORT=8080
|
||||||
ENV LCG_DOMAIN="remote.ollama-server.ru"
|
ENV LCG_SERVER_ALLOW_HTTP=true
|
||||||
ENV LCG_COOKIE_PATH="/lcg"
|
ENV OLLAMA_HOST=0.0.0.0
|
||||||
# ENV LCG_SERVER_ALLOW_HTTP=true
|
ENV OLLAMA_PORT=11434
|
||||||
# ENV OLLAMA_HOST=127.0.0.1
|
|
||||||
# ENV OLLAMA_PORT=11434
|
|
||||||
|
|
||||||
# Expose порты
|
# Expose порты
|
||||||
EXPOSE 8080
|
EXPOSE 8080 11434
|
||||||
|
|
||||||
|
# Переключаемся на пользователя ollama
|
||||||
|
USER ollama
|
||||||
|
|
||||||
WORKDIR /home/ollama
|
WORKDIR /home/ollama
|
||||||
|
|
||||||
|
|||||||
@@ -23,54 +23,37 @@ help: ## Показать справку
|
|||||||
@echo " make podman-compose-up - Запустить через podman-compose"
|
@echo " make podman-compose-up - Запустить через podman-compose"
|
||||||
@echo " make podman-compose-down - Остановить podman-compose"
|
@echo " make podman-compose-down - Остановить podman-compose"
|
||||||
|
|
||||||
build: ## Собрать Docker образ (требует собранных бинарников в dist/)
|
build: ## Собрать Docker образ
|
||||||
@echo "⚠️ Убедитесь, что бинарники собраны: goreleaser build --snapshot --clean"
|
|
||||||
docker build -f $(DOCKERFILE) -t $(IMAGE_NAME):$(IMAGE_TAG) $(CONTEXT)
|
docker build -f $(DOCKERFILE) -t $(IMAGE_NAME):$(IMAGE_TAG) $(CONTEXT)
|
||||||
@echo "Образ $(IMAGE_NAME):$(IMAGE_TAG) успешно собран"
|
@echo "Образ $(IMAGE_NAME):$(IMAGE_TAG) успешно собран"
|
||||||
|
|
||||||
build-podman: ## Собрать Podman образ (требует собранных бинарников в dist/)
|
build-podman: ## Собрать Podman образ
|
||||||
@echo "⚠️ Убедитесь, что бинарники собраны: goreleaser build --snapshot --clean"
|
|
||||||
podman build -f $(DOCKERFILE) -t $(IMAGE_NAME):$(IMAGE_TAG) $(CONTEXT)
|
podman build -f $(DOCKERFILE) -t $(IMAGE_NAME):$(IMAGE_TAG) $(CONTEXT)
|
||||||
@echo "Образ $(IMAGE_NAME):$(IMAGE_TAG) успешно собран"
|
@echo "Образ $(IMAGE_NAME):$(IMAGE_TAG) успешно собран"
|
||||||
|
|
||||||
build-binaries: ## Собрать бинарники перед сборкой образа
|
|
||||||
@echo "Сборка бинарников..."
|
|
||||||
cd $(CONTEXT) && goreleaser build --snapshot --clean
|
|
||||||
@echo "Бинарники собраны в $(CONTEXT)/dist/"
|
|
||||||
|
|
||||||
build-all: build-binaries build ## Собрать бинарники и Docker образ
|
|
||||||
@echo "✅ Все готово!"
|
|
||||||
|
|
||||||
build-all-podman: build-binaries build-podman ## Собрать бинарники и Podman образ
|
|
||||||
@echo "✅ Все готово!"
|
|
||||||
|
|
||||||
run: ## Запустить контейнер (Docker)
|
run: ## Запустить контейнер (Docker)
|
||||||
docker run -d \
|
docker run -d \
|
||||||
--name ${CONTAINER_NAME} \
|
--name $(CONTAINER_NAME) \
|
||||||
-p 8989:8080 \
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
-v ollama-data:/home/ollama/.ollama \
|
-v ollama-data:/home/ollama/.ollama \
|
||||||
-v lcg-results:/app/data/results \
|
-v lcg-results:/app/data/results \
|
||||||
-v lcg-prompts:/app/data/prompts \
|
-v lcg-prompts:/app/data/prompts \
|
||||||
-v lcg-config:/app/data/config \
|
-v lcg-config:/app/data/config \
|
||||||
${IMAGE_NAME}:${IMAGE_TAG}
|
$(IMAGE_NAME):$(IMAGE_TAG)
|
||||||
@echo "Контейнер ${CONTAINER_NAME} запущен"
|
@echo "Контейнер $(CONTAINER_NAME) запущен"
|
||||||
|
|
||||||
run-podman: ## Запустить контейнер (Podman)
|
run-podman: ## Запустить контейнер (Podman)
|
||||||
echo "Запустить контейнер ${CONTAINER_NAME}"
|
|
||||||
echo "IMAGE_NAME: ${IMAGE_NAME}"
|
|
||||||
echo "IMAGE_TAG: ${IMAGE_TAG}"
|
|
||||||
echo "CONTAINER_NAME: ${CONTAINER_NAME}"
|
|
||||||
|
|
||||||
podman run -d \
|
podman run -d \
|
||||||
--name ${CONTAINER_NAME} \
|
--name $(CONTAINER_NAME) \
|
||||||
--restart always \
|
-p 8080:8080 \
|
||||||
-p 8989:8080 \
|
-p 11434:11434 \
|
||||||
-v ollama-data:/home/ollama/.ollama \
|
-v ollama-data:/home/ollama/.ollama \
|
||||||
-v lcg-results:/app/data/results \
|
-v lcg-results:/app/data/results \
|
||||||
-v lcg-prompts:/app/data/prompts \
|
-v lcg-prompts:/app/data/prompts \
|
||||||
-v lcg-config:/app/data/config \
|
-v lcg-config:/app/data/config \
|
||||||
${IMAGE_NAME}:${IMAGE_TAG}
|
$(IMAGE_NAME):$(IMAGE_TAG)
|
||||||
@echo "Контейнер ${CONTAINER_NAME} запущен"
|
@echo "Контейнер $(CONTAINER_NAME) запущен"
|
||||||
|
|
||||||
stop: ## Остановить контейнер (Docker)
|
stop: ## Остановить контейнер (Docker)
|
||||||
docker stop $(CONTAINER_NAME) || true
|
docker stop $(CONTAINER_NAME) || true
|
||||||
|
|||||||
@@ -4,17 +4,7 @@
|
|||||||
|
|
||||||
1. Убедитесь, что у вас установлен Docker или Podman
|
1. Убедитесь, что у вас установлен Docker или Podman
|
||||||
2. Клонируйте репозиторий (если еще не сделали)
|
2. Клонируйте репозиторий (если еще не сделали)
|
||||||
3. Соберите бинарники (требуется перед сборкой образа)
|
3. Перейдите в папку с Dockerfile
|
||||||
|
|
||||||
```bash
|
|
||||||
# Из корня проекта
|
|
||||||
goreleaser build --snapshot --clean
|
|
||||||
|
|
||||||
# Или используйте скрипт
|
|
||||||
./deploy/4.build-binaries.sh v2.0.15
|
|
||||||
```
|
|
||||||
|
|
||||||
4. Перейдите в папку с Dockerfile
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd Dockerfiles/OllamaServer
|
cd Dockerfiles/OllamaServer
|
||||||
@@ -25,16 +15,14 @@ cd Dockerfiles/OllamaServer
|
|||||||
### Вариант 1: Docker Compose (рекомендуется)
|
### Вариант 1: Docker Compose (рекомендуется)
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Важно: убедитесь, что бинарники собраны в ../../dist/
|
|
||||||
docker-compose up -d
|
docker-compose up -d
|
||||||
```
|
```
|
||||||
|
|
||||||
### Вариант 2: Ручная сборка и запуск
|
### Вариант 2: Ручная сборка и запуск
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Сборка образа (контекст должен быть корень проекта)
|
# Сборка образа
|
||||||
cd ../.. # Переходим в корень проекта
|
docker build -f Dockerfile -t lcg-ollama:latest ../..
|
||||||
docker build -f Dockerfiles/OllamaServer/Dockerfile -t lcg-ollama:latest .
|
|
||||||
|
|
||||||
# Запуск контейнера
|
# Запуск контейнера
|
||||||
docker run -d \
|
docker run -d \
|
||||||
@@ -57,9 +45,8 @@ podman-compose -f podman-compose.yml up -d
|
|||||||
### Вариант 2: Ручная сборка и запуск
|
### Вариант 2: Ручная сборка и запуск
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Сборка образа (контекст должен быть корень проекта)
|
# Сборка образа
|
||||||
cd ../.. # Переходим в корень проекта
|
podman build -f Dockerfile -t lcg-ollama:latest ../..
|
||||||
podman build -f Dockerfiles/OllamaServer/Dockerfile -t lcg-ollama:latest .
|
|
||||||
|
|
||||||
# Запуск контейнера
|
# Запуск контейнера
|
||||||
podman run -d \
|
podman run -d \
|
||||||
|
|||||||
@@ -7,111 +7,62 @@
|
|||||||
## 📋 Описание
|
## 📋 Описание
|
||||||
|
|
||||||
Контейнер автоматически запускает:
|
Контейнер автоматически запускает:
|
||||||
|
1. **Ollama сервер** (v0.9.5) на порту 11434
|
||||||
1 **Ollama сервер** (v0.9.5) на порту 11434
|
2. **LCG веб-сервер** на порту 8080
|
||||||
|
|
||||||
2 **LCG веб-сервер** на порту 8080
|
|
||||||
|
|
||||||
Ollama используется как провайдер LLM для генерации Linux команд.
|
Ollama используется как провайдер LLM для генерации Linux команд.
|
||||||
|
|
||||||
## 🚀 Быстрый старт
|
## 🚀 Быстрый старт
|
||||||
|
|
||||||
### Предварительные требования
|
|
||||||
|
|
||||||
Перед сборкой Docker образа необходимо собрать бинарники:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Из корня проекта
|
|
||||||
# Используйте goreleaser для сборки бинарников
|
|
||||||
goreleaser build --snapshot --clean
|
|
||||||
|
|
||||||
# Или используйте скрипт сборки
|
|
||||||
./deploy/4.build-binaries.sh v2.0.15
|
|
||||||
```
|
|
||||||
|
|
||||||
Убедитесь, что в папке `dist/` есть бинарники:
|
|
||||||
|
|
||||||
- `dist/lcg_linux_amd64_v1/lcg_*` для amd64
|
|
||||||
- `dist/lcg_linux_arm64_v8.0/lcg_*` для arm64
|
|
||||||
|
|
||||||
### Сборка образа
|
### Сборка образа
|
||||||
|
|
||||||
#### Docker
|
#### Docker
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Из корня проекта (важно: контекст должен быть корень проекта)
|
# Из корня проекта
|
||||||
docker build -f Dockerfiles/OllamaServer/Dockerfile -t lcg-ollama:latest .
|
docker build -f Dockerfiles/OllamaServer/Dockerfile -t lcg-ollama:latest .
|
||||||
|
|
||||||
# Или с указанием архитектуры
|
|
||||||
docker buildx build \
|
|
||||||
--platform linux/amd64,linux/arm64 \
|
|
||||||
-f Dockerfiles/OllamaServer/Dockerfile \
|
|
||||||
-t lcg-ollama:latest .
|
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Podman
|
#### Podman
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Из корня проекта
|
# Из корня проекта
|
||||||
podman build -f Dockerfiles/OllamaServer/Dockerfile -t lcg-ollama:latest .
|
podman build -f Dockerfiles/OllamaServer/Dockerfile -t lcg-ollama:latest .
|
||||||
|
|
||||||
# Или с указанием архитектуры
|
|
||||||
podman build \
|
|
||||||
--platform linux/amd64,linux/arm64 \
|
|
||||||
-f Dockerfiles/OllamaServer/Dockerfile \
|
|
||||||
-t lcg-ollama:latest .
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### Запуск контейнера
|
### Запуск контейнера
|
||||||
|
|
||||||
#### Docker run
|
#### Docker
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
docker run -d \
|
docker run -d \
|
||||||
--name lcg-ollama \
|
--name lcg-ollama \
|
||||||
-p 8080:8080 \
|
-p 8080:8080 \
|
||||||
|
-p 11434:11434 \
|
||||||
lcg-ollama:latest
|
lcg-ollama:latest
|
||||||
ollama serve
|
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Podman run
|
#### Podman
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
podman run -d \
|
podman run -d \
|
||||||
--name lcg-ollama \
|
--name lcg-ollama \
|
||||||
-p 8989:8080 \
|
-p 8080:8080 \
|
||||||
--restart always \
|
-p 11434:11434 \
|
||||||
lcg-ollama:latest \
|
lcg-ollama:latest
|
||||||
ollama serve
|
|
||||||
```
|
```
|
||||||
|
|
||||||
когда контейнер запущен на удаленном хосте - можете воспользоваться консольными возможностями утилиты lcg следующим образом
|
|
||||||
|
|
||||||
``` bash
|
|
||||||
ssh user@[host_where_contaier_running] 'podman exec -it $(podman ps -q --filter \"ancestor=localhost/lcg-ollama:latest\") lcg [your query]
|
|
||||||
```
|
|
||||||
|
|
||||||
``` bash
|
|
||||||
ssh user@[host_where_contaier_running] 'podman exec -it $(podman ps -q --filter "ancestor=localhost/lcg-ollama:latest") /bin/sh -c "export LCG_MODEL=qwen3:0.6b && lcg config --full"'
|
|
||||||
|
|
||||||
### Использование docker-compose / podman-compose
|
### Использование docker-compose / podman-compose
|
||||||
|
|
||||||
#### Docker Compose
|
#### Docker Compose
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd Dockerfiles/OllamaServer
|
cd Dockerfiles/OllamaServer
|
||||||
docker-compose up -d
|
docker-compose up -d
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Podman Compose
|
#### Podman Compose
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd Dockerfiles/OllamaServer
|
cd Dockerfiles/OllamaServer
|
||||||
podman-compose -f podman-compose.yml up -d
|
podman-compose -f podman-compose.yml up -d
|
||||||
```
|
```
|
||||||
|
|
||||||
Или используйте встроенную поддержку Podman:
|
Или используйте встроенную поддержку Podman:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd Dockerfiles/OllamaServer
|
cd Dockerfiles/OllamaServer
|
||||||
podman play kube podman-compose.yml
|
podman play kube podman-compose.yml
|
||||||
@@ -121,8 +72,8 @@ podman play kube podman-compose.yml
|
|||||||
|
|
||||||
После запуска контейнера доступны:
|
После запуска контейнера доступны:
|
||||||
|
|
||||||
- **LCG веб-интерфейс**: <http://localhost:8080>
|
- **LCG веб-интерфейс**: http://localhost:8080
|
||||||
- **Ollama API**: <http://localhost:11434>
|
- **Ollama API**: http://localhost:11434
|
||||||
|
|
||||||
## ⚙️ Переменные окружения
|
## ⚙️ Переменные окружения
|
||||||
|
|
||||||
@@ -327,8 +278,7 @@ podman logs -f lcg-ollama
|
|||||||
|
|
||||||
### Просмотр логов
|
### Просмотр логов
|
||||||
|
|
||||||
#### Docker log
|
#### Docker
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Логи контейнера
|
# Логи контейнера
|
||||||
docker logs lcg-ollama
|
docker logs lcg-ollama
|
||||||
@@ -337,8 +287,7 @@ docker logs lcg-ollama
|
|||||||
docker logs -f lcg-ollama
|
docker logs -f lcg-ollama
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Podman log
|
#### Podman
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Логи контейнера
|
# Логи контейнера
|
||||||
podman logs lcg-ollama
|
podman logs lcg-ollama
|
||||||
@@ -349,28 +298,24 @@ podman logs -f lcg-ollama
|
|||||||
|
|
||||||
### Подключение к контейнеру
|
### Подключение к контейнеру
|
||||||
|
|
||||||
#### Docker exec
|
#### Docker
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
docker exec -it lcg-ollama sh
|
docker exec -it lcg-ollama sh
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Podman exec
|
#### Podman
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
podman exec -it lcg-ollama sh
|
podman exec -it lcg-ollama sh
|
||||||
```
|
```
|
||||||
|
|
||||||
### Проверка процессов
|
### Проверка процессов
|
||||||
|
|
||||||
#### Docker check ps
|
#### Docker
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
docker exec lcg-ollama ps aux
|
docker exec lcg-ollama ps aux
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Podman check ps
|
#### Podman
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
podman exec lcg-ollama ps aux
|
podman exec lcg-ollama ps aux
|
||||||
```
|
```
|
||||||
@@ -380,7 +325,6 @@ podman exec lcg-ollama ps aux
|
|||||||
### Рекомендации для продакшена
|
### Рекомендации для продакшена
|
||||||
|
|
||||||
1. **Используйте аутентификацию**:
|
1. **Используйте аутентификацию**:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
-e LCG_SERVER_REQUIRE_AUTH=true
|
-e LCG_SERVER_REQUIRE_AUTH=true
|
||||||
-e LCG_SERVER_PASSWORD=strong_password
|
-e LCG_SERVER_PASSWORD=strong_password
|
||||||
@@ -395,7 +339,6 @@ podman exec lcg-ollama ps aux
|
|||||||
- Используйте SSL сертификаты
|
- Используйте SSL сертификаты
|
||||||
|
|
||||||
4. **Ограничьте ресурсы**:
|
4. **Ограничьте ресурсы**:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
docker run -d \
|
docker run -d \
|
||||||
--name lcg-ollama \
|
--name lcg-ollama \
|
||||||
@@ -447,8 +390,8 @@ docker-compose up -d
|
|||||||
## ❓ Поддержка
|
## ❓ Поддержка
|
||||||
|
|
||||||
При возникновении проблем:
|
При возникновении проблем:
|
||||||
|
|
||||||
1. Проверьте логи: `docker logs lcg-ollama`
|
1. Проверьте логи: `docker logs lcg-ollama`
|
||||||
2. Проверьте переменные окружения
|
2. Проверьте переменные окружения
|
||||||
3. Убедитесь, что порты не заняты
|
3. Убедитесь, что порты не заняты
|
||||||
4. Проверьте, что модели загружены в Ollama
|
4. Проверьте, что модели загружены в Ollama
|
||||||
|
|
||||||
|
|||||||
@@ -19,21 +19,14 @@ Dockerfiles/OllamaServer/
|
|||||||
## Описание файлов
|
## Описание файлов
|
||||||
|
|
||||||
### Dockerfile
|
### Dockerfile
|
||||||
|
Multi-stage Dockerfile, который:
|
||||||
Dockerfile, который:
|
1. Собирает бинарник LCG из исходного кода
|
||||||
|
2. Устанавливает Ollama 0.9.5
|
||||||
1. Использует готовый образ `ollama/ollama:0.9.5` как базовый
|
3. Создает пользователя ollama
|
||||||
2. Копирует предварительно собранный бинарник LCG из папки `dist/`
|
4. Настраивает рабочее окружение
|
||||||
3. Выбирает правильный бинарник в зависимости от архитектуры (amd64/arm64)
|
|
||||||
4. Устанавливает entrypoint.sh для запуска обоих сервисов
|
|
||||||
5. Настраивает рабочее окружение и переменные окружения
|
|
||||||
|
|
||||||
**Важно**: Перед сборкой образа необходимо собрать бинарники с помощью `goreleaser build --snapshot --clean`
|
|
||||||
|
|
||||||
### entrypoint.sh
|
### entrypoint.sh
|
||||||
|
|
||||||
Скрипт запуска, который:
|
Скрипт запуска, который:
|
||||||
|
|
||||||
1. Запускает Ollama сервер в фоне
|
1. Запускает Ollama сервер в фоне
|
||||||
2. Ожидает готовности Ollama API
|
2. Ожидает готовности Ollama API
|
||||||
3. Запускает LCG сервер в фоне
|
3. Запускает LCG сервер в фоне
|
||||||
@@ -41,27 +34,21 @@ Dockerfile, который:
|
|||||||
5. Корректно обрабатывает сигналы завершения
|
5. Корректно обрабатывает сигналы завершения
|
||||||
|
|
||||||
### docker-compose.yml / podman-compose.yml
|
### docker-compose.yml / podman-compose.yml
|
||||||
|
|
||||||
Конфигурация для запуска через compose:
|
Конфигурация для запуска через compose:
|
||||||
|
|
||||||
- Настройки портов
|
- Настройки портов
|
||||||
- Переменные окружения
|
- Переменные окружения
|
||||||
- Volumes для персистентного хранения
|
- Volumes для персистентного хранения
|
||||||
- Healthcheck
|
- Healthcheck
|
||||||
|
|
||||||
### Makefile
|
### Makefile
|
||||||
|
|
||||||
Удобные команды для:
|
Удобные команды для:
|
||||||
|
|
||||||
- Сборки образа
|
- Сборки образа
|
||||||
- Запуска/остановки контейнера
|
- Запуска/остановки контейнера
|
||||||
- Просмотра логов
|
- Просмотра логов
|
||||||
- Работы с compose
|
- Работы с compose
|
||||||
|
|
||||||
### README.md
|
### README.md
|
||||||
|
|
||||||
Полная документация с:
|
Полная документация с:
|
||||||
|
|
||||||
- Описанием функциональности
|
- Описанием функциональности
|
||||||
- Инструкциями по установке
|
- Инструкциями по установке
|
||||||
- Настройками переменных окружения
|
- Настройками переменных окружения
|
||||||
@@ -69,7 +56,6 @@ Dockerfile, который:
|
|||||||
- Решением проблем
|
- Решением проблем
|
||||||
|
|
||||||
### QUICKSTART.md
|
### QUICKSTART.md
|
||||||
|
|
||||||
Краткое руководство для быстрого старта.
|
Краткое руководство для быстрого старта.
|
||||||
|
|
||||||
## Порты
|
## Порты
|
||||||
@@ -87,7 +73,6 @@ Dockerfile, который:
|
|||||||
## Переменные окружения
|
## Переменные окружения
|
||||||
|
|
||||||
Основные переменные (см. README.md для полного списка):
|
Основные переменные (см. README.md для полного списка):
|
||||||
|
|
||||||
- `LCG_PROVIDER=ollama`
|
- `LCG_PROVIDER=ollama`
|
||||||
- `LCG_HOST=http://127.0.0.1:11434/`
|
- `LCG_HOST=http://127.0.0.1:11434/`
|
||||||
- `LCG_MODEL=codegeex4`
|
- `LCG_MODEL=codegeex4`
|
||||||
@@ -96,48 +81,31 @@ Dockerfile, который:
|
|||||||
|
|
||||||
## Запуск
|
## Запуск
|
||||||
|
|
||||||
### Предварительная подготовка
|
|
||||||
|
|
||||||
Перед сборкой образа необходимо собрать бинарники:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Из корня проекта
|
|
||||||
goreleaser build --snapshot --clean
|
|
||||||
```
|
|
||||||
|
|
||||||
Убедитесь, что в папке `dist/` есть бинарники для нужных архитектур.
|
|
||||||
|
|
||||||
### Docker
|
### Docker
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd Dockerfiles/OllamaServer
|
cd Dockerfiles/OllamaServer
|
||||||
docker-compose up -d
|
docker-compose up -d
|
||||||
```
|
```
|
||||||
|
|
||||||
### Podman
|
### Podman
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd Dockerfiles/OllamaServer
|
cd Dockerfiles/OllamaServer
|
||||||
podman-compose -f podman-compose.yml up -d
|
podman-compose -f podman-compose.yml up -d
|
||||||
```
|
```
|
||||||
|
|
||||||
### Make
|
### Make
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd Dockerfiles/OllamaServer
|
cd Dockerfiles/OllamaServer
|
||||||
make build-all # Собрать бинарники и Docker образ
|
make compose-up
|
||||||
make compose-up # Запустить через docker-compose
|
# или
|
||||||
|
|
||||||
# Или для Podman
|
|
||||||
make build-all-podman
|
|
||||||
make podman-compose-up
|
make podman-compose-up
|
||||||
```
|
```
|
||||||
|
|
||||||
## Архитектура
|
## Архитектура
|
||||||
|
|
||||||
Контейнер запускает два сервиса:
|
Контейнер запускает два сервиса:
|
||||||
|
|
||||||
1. **Ollama** (порт 11434) - LLM сервер
|
1. **Ollama** (порт 11434) - LLM сервер
|
||||||
2. **LCG** (порт 8080) - Веб-интерфейс и API
|
2. **LCG** (порт 8080) - Веб-интерфейс и API
|
||||||
|
|
||||||
Оба сервиса работают в одном контейнере и общаются через localhost.
|
Оба сервиса работают в одном контейнере и общаются через localhost.
|
||||||
|
|
||||||
|
|||||||
@@ -3,9 +3,8 @@ version: '3.8'
|
|||||||
services:
|
services:
|
||||||
lcg-ollama:
|
lcg-ollama:
|
||||||
build:
|
build:
|
||||||
context: ../.. # Контекст сборки - корень проекта (для доступа к dist/)
|
context: ../..
|
||||||
dockerfile: Dockerfiles/OllamaServer/Dockerfile
|
dockerfile: Dockerfiles/OllamaServer/Dockerfile
|
||||||
# TARGETARCH определяется автоматически Docker на основе платформы хоста
|
|
||||||
container_name: lcg-ollama
|
container_name: lcg-ollama
|
||||||
ports:
|
ports:
|
||||||
- "8080:8080" # LCG веб-сервер
|
- "8080:8080" # LCG веб-сервер
|
||||||
|
|||||||
@@ -32,6 +32,10 @@ cleanup() {
|
|||||||
kill $LCG_PID 2>/dev/null || true
|
kill $LCG_PID 2>/dev/null || true
|
||||||
wait $LCG_PID 2>/dev/null || true
|
wait $LCG_PID 2>/dev/null || true
|
||||||
fi
|
fi
|
||||||
|
if [ ! -z "$OLLAMA_PID" ]; then
|
||||||
|
kill $OLLAMA_PID 2>/dev/null || true
|
||||||
|
wait $OLLAMA_PID 2>/dev/null || true
|
||||||
|
fi
|
||||||
log "Сервисы остановлены"
|
log "Сервисы остановлены"
|
||||||
exit 0
|
exit 0
|
||||||
}
|
}
|
||||||
@@ -44,6 +48,11 @@ if [ ! -f /usr/local/bin/lcg ]; then
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
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_RESULT_FOLDER:-/app/data/results}"
|
||||||
@@ -57,8 +66,8 @@ export OLLAMA_ORIGINS="*"
|
|||||||
|
|
||||||
# Настройка переменных окружения для LCG
|
# Настройка переменных окружения для LCG
|
||||||
export LCG_PROVIDER="${LCG_PROVIDER:-ollama}"
|
export LCG_PROVIDER="${LCG_PROVIDER:-ollama}"
|
||||||
export LCG_HOST="${LCG_HOST:-http://0.0.0.0:11434/}"
|
export LCG_HOST="${LCG_HOST:-http://127.0.0.1:11434/}"
|
||||||
export LCG_MODEL="${LCG_MODEL:-qwen2.5-coder:1.5b}"
|
export LCG_MODEL="${LCG_MODEL:-codegeex4}"
|
||||||
export LCG_RESULT_FOLDER="${LCG_RESULT_FOLDER:-/app/data/results}"
|
export LCG_RESULT_FOLDER="${LCG_RESULT_FOLDER:-/app/data/results}"
|
||||||
export LCG_PROMPT_FOLDER="${LCG_PROMPT_FOLDER:-/app/data/prompts}"
|
export LCG_PROMPT_FOLDER="${LCG_PROMPT_FOLDER:-/app/data/prompts}"
|
||||||
export LCG_CONFIG_FOLDER="${LCG_CONFIG_FOLDER:-/app/data/config}"
|
export LCG_CONFIG_FOLDER="${LCG_CONFIG_FOLDER:-/app/data/config}"
|
||||||
@@ -76,7 +85,39 @@ info "LCG Server: http://${LCG_SERVER_HOST}:${LCG_SERVER_PORT}"
|
|||||||
info "Ollama Host: $OLLAMA_HOST:$OLLAMA_PORT"
|
info "Ollama Host: $OLLAMA_HOST:$OLLAMA_PORT"
|
||||||
log "=========================================="
|
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 сервера..."
|
log "Запуск LCG сервера..."
|
||||||
/usr/local/bin/lcg serve \
|
/usr/local/bin/lcg serve \
|
||||||
--host "${LCG_SERVER_HOST}" \
|
--host "${LCG_SERVER_HOST}" \
|
||||||
@@ -94,6 +135,37 @@ if ! kill -0 $LCG_PID 2>/dev/null; then
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
log "LCG сервер запущен на http://${LCG_SERVER_HOST}:${LCG_SERVER_PORT}"
|
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
|
||||||
|
|
||||||
# Запускаем переданные аргументы
|
|
||||||
exec "$@"
|
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
v.2.0.19
|
v2.0.15
|
||||||
|
|||||||
@@ -27,7 +27,6 @@ type Config struct {
|
|||||||
ResultHistory string
|
ResultHistory string
|
||||||
NoHistoryEnv string
|
NoHistoryEnv string
|
||||||
AllowExecution bool
|
AllowExecution bool
|
||||||
Query string
|
|
||||||
MainFlags MainFlags
|
MainFlags MainFlags
|
||||||
Server ServerConfig
|
Server ServerConfig
|
||||||
Validation ValidationConfig
|
Validation ValidationConfig
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
v.2.0.19
|
v2.0.15
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ metadata:
|
|||||||
namespace: lcg
|
namespace: lcg
|
||||||
data:
|
data:
|
||||||
# Основные настройки
|
# Основные настройки
|
||||||
LCG_VERSION: "v.2.0.19"
|
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"
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ metadata:
|
|||||||
namespace: lcg
|
namespace: lcg
|
||||||
labels:
|
labels:
|
||||||
app: lcg
|
app: lcg
|
||||||
version: v.2.0.19
|
version: v2.0.14
|
||||||
spec:
|
spec:
|
||||||
replicas: 1
|
replicas: 1
|
||||||
selector:
|
selector:
|
||||||
@@ -18,7 +18,7 @@ spec:
|
|||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- name: lcg
|
- name: lcg
|
||||||
image: kuznetcovay/lcg:v.2.0.19
|
image: kuznetcovay/lcg:v2.0.14
|
||||||
imagePullPolicy: Always
|
imagePullPolicy: Always
|
||||||
ports:
|
ports:
|
||||||
- containerPort: 8080
|
- containerPort: 8080
|
||||||
|
|||||||
@@ -15,11 +15,11 @@ resources:
|
|||||||
# Common labels
|
# Common labels
|
||||||
# commonLabels:
|
# commonLabels:
|
||||||
# app: lcg
|
# app: lcg
|
||||||
# version: v.2.0.19
|
# 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: v.2.0.19
|
# newTag: v2.0.14
|
||||||
|
|||||||
80
main.go
80
main.go
@@ -9,7 +9,6 @@ import (
|
|||||||
"os/exec"
|
"os/exec"
|
||||||
"os/user"
|
"os/user"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"slices"
|
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
@@ -77,12 +76,6 @@ func main() {
|
|||||||
Usage: config.AppConfig.AppName + " - Генерация Linux команд из описаний",
|
Usage: config.AppConfig.AppName + " - Генерация Linux команд из описаний",
|
||||||
Version: Version,
|
Version: Version,
|
||||||
Commands: getCommands(),
|
Commands: getCommands(),
|
||||||
Before: func(c *cli.Context) error {
|
|
||||||
// Применяем флаги приложения к конфигурации перед выполнением любой команды
|
|
||||||
// Это гарантирует, что флаги будут применены даже для команд, которые не используют основной Action
|
|
||||||
applyAppFlagsToConfig(c)
|
|
||||||
return nil
|
|
||||||
},
|
|
||||||
UsageText: `
|
UsageText: `
|
||||||
lcg [опции] <описание команды>
|
lcg [опции] <описание команды>
|
||||||
|
|
||||||
@@ -151,25 +144,12 @@ lcg [опции] <описание команды>
|
|||||||
Aliases: []string{"f"},
|
Aliases: []string{"f"},
|
||||||
Usage: "Read part of the command from a file",
|
Usage: "Read part of the command from a file",
|
||||||
},
|
},
|
||||||
&cli.StringFlag{
|
|
||||||
Name: "model",
|
|
||||||
Aliases: []string{"M"},
|
|
||||||
DefaultText: "Use model from LCG_MODEL or default model",
|
|
||||||
Usage: "Model to use",
|
|
||||||
},
|
|
||||||
&cli.BoolFlag{
|
&cli.BoolFlag{
|
||||||
Name: "no-history",
|
Name: "no-history",
|
||||||
Aliases: []string{"nh"},
|
Aliases: []string{"nh"},
|
||||||
Usage: "Disable writing/updating command history (overrides LCG_NO_HISTORY)",
|
Usage: "Disable writing/updating command history (overrides LCG_NO_HISTORY)",
|
||||||
Value: false,
|
Value: false,
|
||||||
},
|
},
|
||||||
&cli.StringFlag{
|
|
||||||
Name: "query",
|
|
||||||
Aliases: []string{"Q"},
|
|
||||||
Usage: "Query to send to the model",
|
|
||||||
DefaultText: "Hello? what day is it today?",
|
|
||||||
Value: "Hello? what day is it today?",
|
|
||||||
},
|
|
||||||
&cli.StringFlag{
|
&cli.StringFlag{
|
||||||
Name: "sys",
|
Name: "sys",
|
||||||
Aliases: []string{"s"},
|
Aliases: []string{"s"},
|
||||||
@@ -201,25 +181,16 @@ lcg [опции] <описание команды>
|
|||||||
Action: func(c *cli.Context) error {
|
Action: func(c *cli.Context) error {
|
||||||
file := c.String("file")
|
file := c.String("file")
|
||||||
system := c.String("sys")
|
system := c.String("sys")
|
||||||
model := c.String("model")
|
|
||||||
query := c.String("query")
|
|
||||||
// обновляем конфиг на основе флагов
|
// обновляем конфиг на основе флагов
|
||||||
if c.IsSet("sys") && system != "" {
|
if system != "" {
|
||||||
config.AppConfig.Prompt = system
|
config.AppConfig.Prompt = system
|
||||||
}
|
}
|
||||||
if c.IsSet("query") && query != "" {
|
|
||||||
config.AppConfig.Query = query
|
|
||||||
}
|
|
||||||
if c.IsSet("timeout") {
|
if c.IsSet("timeout") {
|
||||||
config.AppConfig.Timeout = fmt.Sprintf("%d", c.Int("timeout"))
|
config.AppConfig.Timeout = fmt.Sprintf("%d", c.Int("timeout"))
|
||||||
}
|
}
|
||||||
if c.IsSet("model") {
|
|
||||||
config.AppConfig.Model = model
|
|
||||||
}
|
|
||||||
|
|
||||||
promptID := c.Int("prompt-id")
|
promptID := c.Int("prompt-id")
|
||||||
timeout := c.Int("timeout")
|
timeout := c.Int("timeout")
|
||||||
|
// сохраняем конкретные значения флагов
|
||||||
config.AppConfig.MainFlags = config.MainFlags{
|
config.AppConfig.MainFlags = config.MainFlags{
|
||||||
File: file,
|
File: file,
|
||||||
NoHistory: c.Bool("no-history"),
|
NoHistory: c.Bool("no-history"),
|
||||||
@@ -232,9 +203,12 @@ lcg [опции] <описание команды>
|
|||||||
|
|
||||||
config.AppConfig.MainFlags.Debug = config.AppConfig.MainFlags.Debug || config.GetEnvBool("LCG_DEBUG", false)
|
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 && config.AppConfig.Query == "" {
|
if len(args) == 0 {
|
||||||
cli.ShowAppHelp(c)
|
cli.ShowAppHelp(c)
|
||||||
showTips()
|
showTips()
|
||||||
return nil
|
return nil
|
||||||
@@ -257,12 +231,6 @@ lcg [опции] <описание команды>
|
|||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if config.AppConfig.Query != "" {
|
|
||||||
executeMain(file, system, config.AppConfig.Query, timeout)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
executeMain(file, system, strings.Join(args, " "), timeout)
|
executeMain(file, system, strings.Join(args, " "), timeout)
|
||||||
return nil
|
return nil
|
||||||
},
|
},
|
||||||
@@ -283,31 +251,6 @@ lcg [опции] <описание команды>
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// applyAppFlagsToConfig применяет флаги приложения к конфигурации
|
|
||||||
// Работает как для основного Action, так и для команд
|
|
||||||
func applyAppFlagsToConfig(c *cli.Context) {
|
|
||||||
// Применяем флаг model - проверяем и через IsSet, и значение напрямую
|
|
||||||
// так как IsSet может не работать для флагов без значения по умолчанию
|
|
||||||
if model := c.String("model"); model != "" {
|
|
||||||
config.AppConfig.Model = model
|
|
||||||
}
|
|
||||||
|
|
||||||
// Применяем флаг sys
|
|
||||||
if sys := c.String("sys"); sys != "" {
|
|
||||||
config.AppConfig.Prompt = sys
|
|
||||||
}
|
|
||||||
|
|
||||||
// Применяем флаг timeout (только если явно установлен)
|
|
||||||
if c.IsSet("timeout") {
|
|
||||||
config.AppConfig.Timeout = fmt.Sprintf("%d", c.Int("timeout"))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Применяем флаг query (игнорируем значение по умолчанию)
|
|
||||||
if query := c.String("query"); query != "" && query != "Hello? what day is it today?" {
|
|
||||||
config.AppConfig.Query = query
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func getCommands() []*cli.Command {
|
func getCommands() []*cli.Command {
|
||||||
commands := []*cli.Command{
|
commands := []*cli.Command{
|
||||||
{
|
{
|
||||||
@@ -447,10 +390,6 @@ func getCommands() []*cli.Command {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
Action: func(c *cli.Context) error {
|
Action: func(c *cli.Context) error {
|
||||||
// Флаги приложения уже применены через глобальный Before hook
|
|
||||||
// Но применяем их еще раз на случай, если глобальный Before не сработал
|
|
||||||
applyAppFlagsToConfig(c)
|
|
||||||
|
|
||||||
if c.Bool("full") {
|
if c.Bool("full") {
|
||||||
// Выводим полную конфигурацию в JSON формате
|
// Выводим полную конфигурацию в JSON формате
|
||||||
showFullConfig()
|
showFullConfig()
|
||||||
@@ -1088,7 +1027,12 @@ func getServerAllowHTTPForHost(host string) bool {
|
|||||||
// 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"}
|
||||||
return slices.Contains(secureHosts, host)
|
for _, secureHost := range secureHosts {
|
||||||
|
if host == secureHost {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// showShortConfig показывает краткую конфигурацию
|
// showShortConfig показывает краткую конфигурацию
|
||||||
|
|||||||
@@ -127,6 +127,7 @@ func getTokenFromCookie(r *http.Request) (string, error) {
|
|||||||
func setAuthCookie(w http.ResponseWriter, token string) {
|
func setAuthCookie(w http.ResponseWriter, token string) {
|
||||||
cookie := &http.Cookie{
|
cookie := &http.Cookie{
|
||||||
Name: "auth_token",
|
Name: "auth_token",
|
||||||
|
Domain: config.AppConfig.Server.Domain,
|
||||||
Value: token,
|
Value: token,
|
||||||
Path: config.AppConfig.Server.CookiePath,
|
Path: config.AppConfig.Server.CookiePath,
|
||||||
HttpOnly: true,
|
HttpOnly: true,
|
||||||
|
|||||||
@@ -13,13 +13,6 @@ import (
|
|||||||
"github.com/direct-dev-ru/linux-command-gpt/config"
|
"github.com/direct-dev-ru/linux-command-gpt/config"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
|
||||||
// CSRFTokenLifetimeHours минимальное время жизни CSRF токена в часах (не менее 12 часов)
|
|
||||||
CSRFTokenLifetimeHours = 12
|
|
||||||
// CSRFTokenLifetimeSeconds минимальное время жизни CSRF токена в секундах
|
|
||||||
CSRFTokenLifetimeSeconds = CSRFTokenLifetimeHours * 60 * 60
|
|
||||||
)
|
|
||||||
|
|
||||||
// CSRFManager управляет CSRF токенами
|
// CSRFManager управляет CSRF токенами
|
||||||
type CSRFManager struct {
|
type CSRFManager struct {
|
||||||
secretKey []byte
|
secretKey []byte
|
||||||
@@ -120,14 +113,13 @@ func (c *CSRFManager) ValidateToken(token, userID string) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// Проверяем время жизни токена (минимум 12 часов)
|
// Проверяем время жизни токена (24 часа)
|
||||||
timestamp, err := parseInt64(timestampStr)
|
timestamp, err := parseInt64(timestampStr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// Минимальное время жизни токена: 12 часов (не менее 12 часов согласно требованиям)
|
if time.Now().Unix()-timestamp > 24*60*60 {
|
||||||
if time.Now().Unix()-timestamp > CSRFTokenLifetimeSeconds {
|
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -161,15 +153,14 @@ func GetCSRFTokenFromCookie(r *http.Request) string {
|
|||||||
|
|
||||||
// setCSRFCookie устанавливает CSRF токен в cookie
|
// setCSRFCookie устанавливает CSRF токен в cookie
|
||||||
func setCSRFCookie(w http.ResponseWriter, token string) {
|
func setCSRFCookie(w http.ResponseWriter, token string) {
|
||||||
// Минимальное время жизни токена: 12 часов (не менее 12 часов согласно требованиям)
|
|
||||||
cookie := &http.Cookie{
|
cookie := &http.Cookie{
|
||||||
Name: "csrf_token",
|
Name: "csrf_token",
|
||||||
Value: token,
|
Value: token,
|
||||||
Path: config.AppConfig.Server.CookiePath,
|
Path: config.AppConfig.Server.CookiePath,
|
||||||
HttpOnly: true,
|
HttpOnly: true,
|
||||||
Secure: config.AppConfig.Server.CookieSecure,
|
Secure: config.AppConfig.Server.CookieSecure,
|
||||||
SameSite: http.SameSiteLaxMode, // Более мягкий режим для reverse proxy
|
SameSite: http.SameSiteLaxMode, // Более мягкий режим для reverse proxy
|
||||||
MaxAge: CSRFTokenLifetimeSeconds, // Минимум 12 часов в секундах
|
MaxAge: 1 * 60 * 60,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Добавляем домен если указан
|
// Добавляем домен если указан
|
||||||
|
|||||||
@@ -72,9 +72,8 @@ var ExecutePageCSSTemplate = template.Must(template.New("execute_css").Parse(`
|
|||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
.header h1 {
|
.header h1 {
|
||||||
margin: 0;
|
|
||||||
font-size: 2.5em;
|
font-size: 2.5em;
|
||||||
font-weight: 300;
|
margin-bottom: 10px;
|
||||||
}
|
}
|
||||||
.header p {
|
.header p {
|
||||||
opacity: 0.9;
|
opacity: 0.9;
|
||||||
|
|||||||
Reference in New Issue
Block a user