mirror of
https://github.com/Direct-Dev-Ru/go-lcg.git
synced 2025-11-17 18:19:56 +00:00
Compare commits
13 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 114146f4d2 | |||
| b4b902cb4c | |||
| 7933abe62d | |||
| d213de7a95 | |||
| 81b01d74ae | |||
| 1fbdd237a3 | |||
| 2d82b91090 | |||
| 5d3829d1fe | |||
| edadedcf80 | |||
| 5ff6d4e072 | |||
| ffc2d6ba0a | |||
| dab94df7d2 | |||
| 281f7f877a |
@@ -1,32 +0,0 @@
|
|||||||
# 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,36 +1,6 @@
|
|||||||
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)
|
||||||
=========================
|
=========================
|
||||||
|
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
v2.0.14
|
v2.0.10
|
||||||
|
|||||||
@@ -6,8 +6,6 @@ builds:
|
|||||||
binary: "lcg_{{ .Version }}"
|
binary: "lcg_{{ .Version }}"
|
||||||
goos:
|
goos:
|
||||||
- linux
|
- linux
|
||||||
- windows
|
|
||||||
- darwin
|
|
||||||
goarch:
|
goarch:
|
||||||
- amd64
|
- amd64
|
||||||
- arm64
|
- arm64
|
||||||
@@ -23,10 +21,9 @@ builds:
|
|||||||
|
|
||||||
archives:
|
archives:
|
||||||
- id: lcg
|
- id: lcg
|
||||||
ids:
|
builds:
|
||||||
- lcg
|
- lcg
|
||||||
formats:
|
format: binary
|
||||||
- binary
|
|
||||||
name_template: "{{ .Binary }}_{{ .Os }}_{{ .Arch }}"
|
name_template: "{{ .Binary }}_{{ .Os }}_{{ .Arch }}"
|
||||||
files:
|
files:
|
||||||
- "lcg_{{ .Version }}"
|
- "lcg_{{ .Version }}"
|
||||||
|
|||||||
@@ -198,7 +198,14 @@ 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.14
|
v2.0.10
|
||||||
|
|||||||
@@ -1,170 +0,0 @@
|
|||||||
#!/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 "Готово."
|
|
||||||
|
|
||||||
|
|
||||||
@@ -133,15 +133,10 @@ The `serve` command provides both a web interface and REST API:
|
|||||||
|
|
||||||
**Web Interface:**
|
**Web Interface:**
|
||||||
|
|
||||||
- Browse results at `http://localhost:8080/` (or `http://localhost:8080<BASE_PATH>/` if `LCG_BASE_URL` set)
|
- Browse results at `http://localhost:8080/`
|
||||||
- Execute requests at `.../run`
|
- Execute requests at `http://localhost:8080/run`
|
||||||
- Manage prompts at `.../prompts`
|
- Manage prompts at `http://localhost:8080/prompts`
|
||||||
- View history at `.../history`
|
- View history at `http://localhost:8080/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:**
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ metadata:
|
|||||||
namespace: lcg
|
namespace: lcg
|
||||||
data:
|
data:
|
||||||
# Основные настройки
|
# Основные настройки
|
||||||
LCG_VERSION: "v2.0.14"
|
LCG_VERSION: "v2.0.10"
|
||||||
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: v2.0.14
|
version: v2.0.10
|
||||||
spec:
|
spec:
|
||||||
replicas: 1
|
replicas: 1
|
||||||
selector:
|
selector:
|
||||||
@@ -18,7 +18,7 @@ spec:
|
|||||||
spec:
|
spec:
|
||||||
containers:
|
containers:
|
||||||
- name: lcg
|
- name: lcg
|
||||||
image: kuznetcovay/lcg:v2.0.14
|
image: kuznetcovay/lcg:v2.0.10
|
||||||
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: v2.0.14
|
# version: v2.0.10
|
||||||
# managed-by: kustomize
|
# managed-by: kustomize
|
||||||
|
|
||||||
# Images
|
# Images
|
||||||
# images:
|
# images:
|
||||||
# - name: lcg
|
# - name: lcg
|
||||||
# newName: kuznetcovay/lcg
|
# newName: kuznetcovay/lcg
|
||||||
# newTag: v2.0.14
|
# newTag: v2.0.10
|
||||||
|
|||||||
@@ -39,12 +39,11 @@ 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
|
Content string // Полное содержимое для поиска
|
||||||
Content string // Полное содержимое для поиска
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// handleResultsPage обрабатывает главную страницу со списком файлов
|
// handleResultsPage обрабатывает главную страницу со списком файлов
|
||||||
@@ -134,12 +133,11 @@ 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),
|
Content: fullContent,
|
||||||
Content: fullContent,
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -169,65 +167,6 @@ 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) {
|
||||||
// Учитываем BasePath при извлечении имени файла
|
// Учитываем BasePath при извлечении имени файла
|
||||||
|
|||||||
@@ -111,26 +111,19 @@ 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: transparent;
|
background: #e74c3c;
|
||||||
color: #ef9a9a; /* бледно-красный */
|
color: white;
|
||||||
border: none;
|
border: none;
|
||||||
padding: 2px 6px;
|
padding: 6px 12px;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-size: 18px;
|
font-size: 0.8em;
|
||||||
line-height: 1;
|
transition: background 0.3s ease;
|
||||||
transition: color 0.2s ease, transform 0.2s ease;
|
|
||||||
}
|
}
|
||||||
.delete-btn:hover {
|
.delete-btn:hover {
|
||||||
color: rgb(171, 27, 24); /* ярче при ховере */
|
background: #c0392b;
|
||||||
transform: translateY(-1px);
|
|
||||||
}
|
}
|
||||||
.empty-state {
|
.empty-state {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
@@ -187,7 +180,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>
|
||||||
|
|||||||
@@ -73,10 +73,8 @@ const ResultsPageTemplate = `
|
|||||||
}
|
}
|
||||||
.files-grid {
|
.files-grid {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
|
grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
|
||||||
gap: 20px;
|
gap: 20px;
|
||||||
align-items: stretch;
|
|
||||||
grid-auto-rows: auto;
|
|
||||||
}
|
}
|
||||||
.file-card {
|
.file-card {
|
||||||
background: white;
|
background: white;
|
||||||
@@ -93,36 +91,32 @@ 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;
|
||||||
left: 10px;
|
right: 10px;
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
}
|
}
|
||||||
.delete-btn {
|
.delete-btn {
|
||||||
background: transparent;
|
background: #e74c3c;
|
||||||
color: #ef9a9a; /* бледно-красный */
|
color: white;
|
||||||
border: none;
|
border: none;
|
||||||
padding: 4px 8px;
|
padding: 6px 12px;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-size: 18px;
|
font-size: 0.8em;
|
||||||
line-height: 1;
|
transition: background 0.3s ease;
|
||||||
transition: color 0.2s ease, transform 0.2s ease;
|
|
||||||
}
|
}
|
||||||
.delete-btn:hover {
|
.delete-btn:hover {
|
||||||
color:rgb(171, 27, 24); /* чуть ярче при ховере */
|
background: #c0392b;
|
||||||
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;
|
||||||
@@ -171,25 +165,16 @@ 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: 1.9em; }
|
.header h1 { font-size: 2em; }
|
||||||
.content { padding: 20px; }
|
.content { padding: 20px; }
|
||||||
.files-grid { dummy-attr: none; }
|
.files-grid { grid-template-columns: 1fr; }
|
||||||
/* Стили карточек как в истории */
|
|
||||||
.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.6em; }
|
.header h1 { font-size: 1.8em; }
|
||||||
.content { padding: 16px; }
|
|
||||||
.stats { grid-template-columns: 1fr; }
|
.stats { grid-template-columns: 1fr; }
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
@@ -231,10 +216,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">{{.DisplayName}}</div>
|
<div class="file-name">{{.Name}}</div>
|
||||||
<div class="file-info">
|
<div class="file-info">
|
||||||
📅 {{.ModTime}} | 📏 {{.Size}}
|
📅 {{.ModTime}} | 📏 {{.Size}}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
Reference in New Issue
Block a user