Изменение на 23.02.2026 22:40
This commit is contained in:
@@ -1,371 +0,0 @@
|
||||
# ══════════════════════════════════════════════════════════════════════════════
|
||||
# CI/CD Pipeline: Сборка и публикация релиза GenAudioBookInfo
|
||||
# ══════════════════════════════════════════════════════════════════════════════
|
||||
#
|
||||
# ЗАПУСК:
|
||||
# make release VERSION=2.1.0
|
||||
# Создаёт git-тег v2.1.0, пушит его → запускается этот workflow.
|
||||
#
|
||||
# ──────────────────────────────────────────────────────────────────────────────
|
||||
# ЕДИНЫЙ JOB (без передачи артефактов между jobs):
|
||||
#
|
||||
# 1. Качество кода: go vet + go test
|
||||
# 2. Кросс-компиляция: 16 бинарников для всех платформ
|
||||
# 3. Архивирование: Windows → ZIP, Unix → tar.gz, SHA256
|
||||
# 4. Описание релиза: авто-changelog из git-коммитов на русском языке
|
||||
# 5. Публикация: Gitea Release + загрузка всех файлов
|
||||
#
|
||||
# ──────────────────────────────────────────────────────────────────────────────
|
||||
# НЕОБХОДИМАЯ НАСТРОЙКА:
|
||||
#
|
||||
# В настройках репозитория Gitea → Settings → Secrets:
|
||||
# GIT_TOKEN (или GITEA_TOKEN) — токен с правами write:repository
|
||||
#
|
||||
# ══════════════════════════════════════════════════════════════════════════════
|
||||
|
||||
name: "Release CI/CD"
|
||||
|
||||
on:
|
||||
push:
|
||||
tags:
|
||||
- "v*"
|
||||
|
||||
env:
|
||||
GO_VERSION: "1.24"
|
||||
APP_NAME: genaudiobookinfo
|
||||
BUILD_DIR: build
|
||||
|
||||
# ══════════════════════════════════════════════════════════════════════════════
|
||||
jobs:
|
||||
|
||||
release:
|
||||
name: "Сборка и публикация релиза"
|
||||
runs-on: ubuntu-latest
|
||||
|
||||
steps:
|
||||
# ── Исходный код (полная история для changelog) ──────────────────────
|
||||
- name: "Получение исходного кода"
|
||||
uses: actions/checkout@v4
|
||||
with:
|
||||
fetch-depth: 0
|
||||
|
||||
- name: "Установка Go ${{ env.GO_VERSION }}"
|
||||
uses: actions/setup-go@v5
|
||||
with:
|
||||
go-version: ${{ env.GO_VERSION }}
|
||||
cache: true
|
||||
|
||||
# ── Определение версии ──────────────────────────────────────────────
|
||||
- name: "Определение версии и предыдущего тега"
|
||||
id: ver
|
||||
shell: bash
|
||||
run: |
|
||||
TAG="${GITHUB_REF_NAME}"
|
||||
VER="${TAG#v}"
|
||||
echo "version=${VER}" >> "$GITHUB_OUTPUT"
|
||||
echo "tag=${TAG}" >> "$GITHUB_OUTPUT"
|
||||
|
||||
PREV_TAG=$(git describe --tags --abbrev=0 "${TAG}^" 2>/dev/null || echo "")
|
||||
echo "prev_tag=${PREV_TAG}" >> "$GITHUB_OUTPUT"
|
||||
|
||||
echo "══════════════════════════════════════"
|
||||
echo " Версия: ${VER}"
|
||||
echo " Git-тег: ${TAG}"
|
||||
echo " Пред. тег: ${PREV_TAG:-<первый релиз>}"
|
||||
echo " Коммит: ${GITHUB_SHA:0:8}"
|
||||
echo "══════════════════════════════════════"
|
||||
|
||||
# ── Качество кода ───────────────────────────────────────────────────
|
||||
- name: "Проверка качества кода (vet + tests)"
|
||||
shell: bash
|
||||
run: |
|
||||
echo ">>> Загрузка зависимостей..."
|
||||
go mod download
|
||||
echo ">>> Статический анализ (go vet)..."
|
||||
go vet ./...
|
||||
echo ">>> Unit-тесты..."
|
||||
go test ./... -count=1 -timeout 5m
|
||||
echo ">>> Качество кода — ОК"
|
||||
|
||||
# ── Кросс-компиляция ────────────────────────────────────────────────
|
||||
- name: "Кросс-компиляция для всех платформ (16 бинарников)"
|
||||
shell: bash
|
||||
run: |
|
||||
make build-all VERSION=${{ steps.ver.outputs.version }}
|
||||
echo ""
|
||||
echo "Собранные бинарники:"
|
||||
ls -lh ${BUILD_DIR}/
|
||||
|
||||
# ── Архивы + контрольные суммы ──────────────────────────────────────
|
||||
- name: "Создание архивов и контрольных сумм"
|
||||
shell: bash
|
||||
run: |
|
||||
cd "${BUILD_DIR}"
|
||||
mkdir -p archives
|
||||
|
||||
echo "=== Создание архивов ==="
|
||||
for f in *; do
|
||||
[ -f "$f" ] || continue
|
||||
case "$f" in
|
||||
*.exe)
|
||||
ARCNAME="${f%.exe}.zip"
|
||||
echo " ${f} → archives/${ARCNAME}"
|
||||
zip "archives/${ARCNAME}" "$f"
|
||||
;;
|
||||
${APP_NAME}-*)
|
||||
ARCNAME="${f}.tar.gz"
|
||||
echo " ${f} → archives/${ARCNAME}"
|
||||
tar -czf "archives/${ARCNAME}" "$f"
|
||||
;;
|
||||
esac
|
||||
done
|
||||
|
||||
echo ""
|
||||
echo "=== Контрольные суммы SHA256 ==="
|
||||
cd archives
|
||||
sha256sum * > checksums-sha256.txt
|
||||
cat checksums-sha256.txt
|
||||
echo ""
|
||||
echo "Готово: $(ls | wc -l) файлов"
|
||||
|
||||
# ── Генерация описания релиза на русском языке ──────────────────────
|
||||
- name: "Генерация описания релиза"
|
||||
shell: bash
|
||||
run: |
|
||||
TAG="${{ steps.ver.outputs.tag }}"
|
||||
VER="${{ steps.ver.outputs.version }}"
|
||||
PREV_TAG="${{ steps.ver.outputs.prev_tag }}"
|
||||
COMMIT="${{ github.sha }}"
|
||||
DATE=$(date -u '+%d.%m.%Y %H:%M UTC')
|
||||
|
||||
# ── Changelog из git-коммитов ──────────────────────────────────
|
||||
if [ -n "${PREV_TAG}" ]; then
|
||||
RAW_LOG=$(git log --pretty=format:"%s" "${PREV_TAG}..${TAG}" --no-merges 2>/dev/null || echo "")
|
||||
else
|
||||
RAW_LOG=$(git log --pretty=format:"%s" --no-merges 2>/dev/null | head -50 || echo "")
|
||||
fi
|
||||
|
||||
# Категоризация коммитов
|
||||
FEATURES=""
|
||||
FIXES=""
|
||||
REFACTOR=""
|
||||
OTHER=""
|
||||
|
||||
while IFS= read -r line; do
|
||||
[ -z "$line" ] && continue
|
||||
line_lower=$(echo "$line" | tr '[:upper:]' '[:lower:]')
|
||||
case "$line_lower" in
|
||||
функция:*|feat:*|feature:*|добавлен*|реализован*|новое:*)
|
||||
FEATURES="${FEATURES}
|
||||
- ${line}"
|
||||
;;
|
||||
исправлен*|fix:*|bugfix:*|баг:*|ошибка:*)
|
||||
FIXES="${FIXES}
|
||||
- ${line}"
|
||||
;;
|
||||
рефакторинг:*|refactor:*|оптимизац*|улучшен*)
|
||||
REFACTOR="${REFACTOR}
|
||||
- ${line}"
|
||||
;;
|
||||
ci:*|docs:*|ci/*|build:*)
|
||||
OTHER="${OTHER}
|
||||
- ${line}"
|
||||
;;
|
||||
*)
|
||||
OTHER="${OTHER}
|
||||
- ${line}"
|
||||
;;
|
||||
esac
|
||||
done <<< "$RAW_LOG"
|
||||
|
||||
# ── Начинаем формировать тело релиза ───────────────────────────
|
||||
{
|
||||
echo "## GenAudioBookInfo ${TAG}"
|
||||
echo ""
|
||||
|
||||
# Пользовательские заметки (RELEASE_NOTES.md — приоритет)
|
||||
if [ -f "RELEASE_NOTES.md" ]; then
|
||||
cat RELEASE_NOTES.md
|
||||
echo ""
|
||||
fi
|
||||
|
||||
# Автоматический changelog
|
||||
HAS_CHANGES=false
|
||||
|
||||
if [ -n "${FEATURES}" ]; then
|
||||
HAS_CHANGES=true
|
||||
echo "### 🚀 Новые возможности"
|
||||
echo "${FEATURES}"
|
||||
echo ""
|
||||
fi
|
||||
|
||||
if [ -n "${FIXES}" ]; then
|
||||
HAS_CHANGES=true
|
||||
echo "### 🐛 Исправления"
|
||||
echo "${FIXES}"
|
||||
echo ""
|
||||
fi
|
||||
|
||||
if [ -n "${REFACTOR}" ]; then
|
||||
HAS_CHANGES=true
|
||||
echo "### ♻️ Рефакторинг и оптимизация"
|
||||
echo "${REFACTOR}"
|
||||
echo ""
|
||||
fi
|
||||
|
||||
if [ -n "${OTHER}" ]; then
|
||||
HAS_CHANGES=true
|
||||
echo "### 📝 Прочие изменения"
|
||||
echo "${OTHER}"
|
||||
echo ""
|
||||
fi
|
||||
|
||||
if [ "${HAS_CHANGES}" = "false" ]; then
|
||||
echo "- Первый релиз"
|
||||
echo ""
|
||||
fi
|
||||
|
||||
echo "---"
|
||||
echo ""
|
||||
echo "### Поддерживаемые платформы"
|
||||
echo ""
|
||||
echo "| ОС | Архитектура | Файл |"
|
||||
echo "|---|---|---|"
|
||||
echo "| Windows | amd64 (64-бит) | \`${APP_NAME}-windows-amd64.zip\` |"
|
||||
echo "| Windows | 386 (32-бит) | \`${APP_NAME}-windows-386.zip\` |"
|
||||
echo "| Windows | arm64 | \`${APP_NAME}-windows-arm64.zip\` |"
|
||||
echo "| Linux | amd64 | \`${APP_NAME}-linux-amd64.tar.gz\` |"
|
||||
echo "| Linux | arm64 | \`${APP_NAME}-linux-arm64.tar.gz\` |"
|
||||
echo "| Linux | armv7 | \`${APP_NAME}-linux-armv7.tar.gz\` |"
|
||||
echo "| Linux | 386 | \`${APP_NAME}-linux-386.tar.gz\` |"
|
||||
echo "| Linux | MIPS | \`${APP_NAME}-linux-mips.tar.gz\` |"
|
||||
echo "| Linux | MIPSle | \`${APP_NAME}-linux-mipsle.tar.gz\` |"
|
||||
echo "| Linux | RISC-V 64 | \`${APP_NAME}-linux-riscv64.tar.gz\` |"
|
||||
echo "| macOS | amd64 (Intel) | \`${APP_NAME}-darwin-amd64.tar.gz\` |"
|
||||
echo "| macOS | arm64 (Apple Silicon) | \`${APP_NAME}-darwin-arm64.tar.gz\` |"
|
||||
echo "| FreeBSD | amd64 | \`${APP_NAME}-freebsd-amd64.tar.gz\` |"
|
||||
echo "| FreeBSD | arm64 | \`${APP_NAME}-freebsd-arm64.tar.gz\` |"
|
||||
echo "| OpenBSD | amd64 | \`${APP_NAME}-openbsd-amd64.tar.gz\` |"
|
||||
echo "| NetBSD | amd64 | \`${APP_NAME}-netbsd-amd64.tar.gz\` |"
|
||||
echo ""
|
||||
echo "> 🔒 Контрольные суммы: \`checksums-sha256.txt\`"
|
||||
echo ""
|
||||
echo "### Информация о сборке"
|
||||
echo ""
|
||||
echo "| Параметр | Значение |"
|
||||
echo "|---|---|"
|
||||
echo "| Коммит | \`${COMMIT}\` |"
|
||||
echo "| Дата сборки | ${DATE} |"
|
||||
echo "| Go | ${{ env.GO_VERSION }} |"
|
||||
} > /tmp/release_body.md
|
||||
|
||||
echo ">>> Описание релиза сформировано:"
|
||||
echo "──────────────────────────────────────"
|
||||
cat /tmp/release_body.md
|
||||
echo "──────────────────────────────────────"
|
||||
|
||||
# ── Публикация в Gitea ──────────────────────────────────────────────
|
||||
- name: "Создание релиза в Gitea и загрузка файлов"
|
||||
shell: bash
|
||||
env:
|
||||
GIT_TOKEN: ${{ secrets.GIT_TOKEN }}
|
||||
GITEA_TOKEN: ${{ secrets.GITEA_TOKEN }}
|
||||
run: |
|
||||
TOKEN="${GIT_TOKEN:-$GITEA_TOKEN}"
|
||||
if [ -z "${TOKEN}" ]; then
|
||||
echo "ОШИБКА: не задан секрет GIT_TOKEN (или GITEA_TOKEN)."
|
||||
echo "Добавьте токен в Settings → Secrets репозитория."
|
||||
exit 1
|
||||
fi
|
||||
|
||||
TAG="${{ steps.ver.outputs.tag }}"
|
||||
COMMIT="${{ github.sha }}"
|
||||
GITEA_URL="${{ github.server_url }}"
|
||||
REPO="${{ github.repository }}"
|
||||
|
||||
echo "══════════════════════════════════════"
|
||||
echo " Сервер: ${GITEA_URL}"
|
||||
echo " Репо: ${REPO}"
|
||||
echo " Тег: ${TAG}"
|
||||
echo " Коммит: ${COMMIT:0:8}"
|
||||
echo "══════════════════════════════════════"
|
||||
|
||||
# ── JSON-кодирование тела релиза ───────────────────────────────
|
||||
BODY_JSON=$(python3 -c "
|
||||
import json
|
||||
with open('/tmp/release_body.md', 'r') as f:
|
||||
print(json.dumps(f.read()))
|
||||
")
|
||||
|
||||
# ── Создание релиза через Gitea API ────────────────────────────
|
||||
echo ""
|
||||
echo ">>> Создаём релиз ${TAG}..."
|
||||
|
||||
RELEASE_JSON=$(curl -sf -X POST \
|
||||
-H "Authorization: token ${TOKEN}" \
|
||||
-H "Content-Type: application/json" \
|
||||
"${GITEA_URL}/api/v1/repos/${REPO}/releases" \
|
||||
-d "{
|
||||
\"tag_name\": \"${TAG}\",
|
||||
\"target_commitish\": \"${COMMIT}\",
|
||||
\"name\": \"${APP_NAME} ${TAG}\",
|
||||
\"body\": ${BODY_JSON},
|
||||
\"draft\": false,
|
||||
\"prerelease\": false
|
||||
}")
|
||||
|
||||
RELEASE_ID=$(echo "$RELEASE_JSON" | \
|
||||
python3 -c "import sys,json; d=json.load(sys.stdin); print(d.get('id',''))" 2>/dev/null || true)
|
||||
|
||||
if [ -z "$RELEASE_ID" ] || [ "$RELEASE_ID" = "None" ]; then
|
||||
echo "ОШИБКА: не удалось создать релиз!"
|
||||
echo "Ответ API:"
|
||||
echo "$RELEASE_JSON" | python3 -m json.tool 2>/dev/null || echo "$RELEASE_JSON"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
echo ">>> Релиз создан: ID=${RELEASE_ID}"
|
||||
|
||||
# ── Загрузка артефактов ────────────────────────────────────────
|
||||
echo ""
|
||||
echo ">>> Загрузка артефактов..."
|
||||
UPLOAD_OK=0
|
||||
UPLOAD_FAIL=0
|
||||
|
||||
for FILE in ./${BUILD_DIR}/archives/*; do
|
||||
[ -f "$FILE" ] || continue
|
||||
FILENAME=$(basename "$FILE")
|
||||
FILESIZE=$(du -sh "$FILE" | cut -f1)
|
||||
|
||||
printf " %-55s [%5s] " "${FILENAME}" "${FILESIZE}"
|
||||
|
||||
HTTP_CODE=$(curl -s -o /dev/null -w "%{http_code}" \
|
||||
-X POST \
|
||||
-H "Authorization: token ${TOKEN}" \
|
||||
-H "Content-Type: application/octet-stream" \
|
||||
--data-binary @"${FILE}" \
|
||||
"${GITEA_URL}/api/v1/repos/${REPO}/releases/${RELEASE_ID}/assets?name=${FILENAME}")
|
||||
|
||||
if [ "$HTTP_CODE" -ge 200 ] && [ "$HTTP_CODE" -lt 300 ]; then
|
||||
echo "✓ (HTTP ${HTTP_CODE})"
|
||||
UPLOAD_OK=$((UPLOAD_OK + 1))
|
||||
else
|
||||
echo "✗ (HTTP ${HTTP_CODE})"
|
||||
UPLOAD_FAIL=$((UPLOAD_FAIL + 1))
|
||||
fi
|
||||
done
|
||||
|
||||
# ── Итог ───────────────────────────────────────────────────────
|
||||
echo ""
|
||||
echo "══════════════════════════════════════════════"
|
||||
echo " Загружено: ${UPLOAD_OK}"
|
||||
echo " Ошибок: ${UPLOAD_FAIL}"
|
||||
echo " Релиз: ${GITEA_URL}/${REPO}/releases/tag/${TAG}"
|
||||
echo "══════════════════════════════════════════════"
|
||||
|
||||
if [ "$UPLOAD_FAIL" -gt 0 ]; then
|
||||
echo "ОШИБКА: часть файлов не удалось загрузить!"
|
||||
exit 1
|
||||
fi
|
||||
|
||||
echo ">>> Релиз ${TAG} опубликован успешно."
|
||||
Reference in New Issue
Block a user