Some checks failed
Git Repository Backup (Pre-scan) / backup (push) Failing after 1h41m49s
647 lines
30 KiB
YAML
647 lines
30 KiB
YAML
name: Git Repository Backup (Pre-scan)
|
|
|
|
on:
|
|
workflow_dispatch:
|
|
inputs:
|
|
branches:
|
|
description: 'Comma-separated list of sourceRepo/branch entries to mirror (e.g., dev_Net8.git/Develop_Net8,dev.git/develop). If empty, all entries from branch_list file will be mirrored.'
|
|
required: false
|
|
default: ''
|
|
schedule:
|
|
- cron: '0 17 * * 1-5' # UTC 17:00 == KST 02:00, Mon-Fri only
|
|
|
|
jobs:
|
|
backup:
|
|
runs-on: [internal]
|
|
timeout-minutes: 500
|
|
steps:
|
|
- name: Checkout
|
|
uses: actions/checkout@v4
|
|
|
|
- name: Set up SSH
|
|
env:
|
|
SSH_PRIVATE_KEY: ${{ secrets.SSH_PRIVATE_KEY }}
|
|
run: |
|
|
mkdir -p ~/.ssh
|
|
echo "${SSH_PRIVATE_KEY}" > ~/.ssh/id_rsa
|
|
chmod 600 ~/.ssh/id_rsa
|
|
ssh-keyscan -p 22 172.16.10.191 >> ~/.ssh/known_hosts
|
|
|
|
- name: Refresh source mirror repositories (git fetch --mirror)
|
|
run: |
|
|
set -euo pipefail
|
|
echo "Refreshing mirror repos on 172.16.10.191 ..."
|
|
ssh engdev@172.16.10.191 'set -euo pipefail; shopt -s nullglob; for repo in *.git; do [ -d "${repo}" ] || continue; echo "Updating ${repo}"; (cd "${repo}" && git fetch --prune --prune-tags origin "+refs/*:refs/*"); done'
|
|
|
|
- name: Backup Branches (pre-scan → decision → execution)
|
|
continue-on-error: true
|
|
env:
|
|
BASE_GITEA_TOKEN: ${{ secrets.BASE_GITEA_TOKEN }}
|
|
BASE_GITEA_URL: ${{ vars.BASE_GITEA_URL }} # e.g., https://gitea.example.com
|
|
BASE_GITEA_USER: ${{ vars.BASE_GITEA_USER }} # The user who owns the token
|
|
INPUT_BRANCHES: ${{ github.event.inputs.branches }}
|
|
NOTIFY_WEBHOOK: ${{ vars.NOTIFY_WEBHOOK }} # Optional chat webhook
|
|
SYNC_TAGS: ${{ vars.SYNC_TAGS }} # Optional, "false" to skip tag sync
|
|
TARGET_SEED_DEPTH: ${{ vars.TARGET_SEED_DEPTH }} # Optional, fallback seed depth (default 50)
|
|
USE_SHALLOW_EXCLUDE: ${{ vars.USE_SHALLOW_EXCLUDE }} # Optional, "false" to disable shallow-exclude
|
|
GIT_PROTOCOL: version=2
|
|
run: |
|
|
set -euo pipefail
|
|
|
|
CENTER_ORG="center_dev"
|
|
AUTH_HEADER="Authorization: token ${BASE_GITEA_TOKEN}"
|
|
SOURCE_SSH_HOST="engdev@172.16.10.191"
|
|
ROOT_DIR="$(pwd)"
|
|
NOTIFY_WEBHOOK="${NOTIFY_WEBHOOK:-}"
|
|
SYNC_TAGS="${SYNC_TAGS:-true}"
|
|
TARGET_SEED_DEPTH="${TARGET_SEED_DEPTH:-50}"
|
|
USE_SHALLOW_EXCLUDE="${USE_SHALLOW_EXCLUDE:-true}"
|
|
if [[ "${USE_SHALLOW_EXCLUDE}" == "false" ]]; then
|
|
echo "::notice::Shallow-exclude disabled by configuration."
|
|
fi
|
|
if ! [[ "${TARGET_SEED_DEPTH}" =~ ^[0-9]+$ ]] || (( TARGET_SEED_DEPTH <= 0 )); then
|
|
echo "::warning::TARGET_SEED_DEPTH(${TARGET_SEED_DEPTH}) is invalid; resetting to 50"
|
|
TARGET_SEED_DEPTH=50
|
|
fi
|
|
TOTAL_SUCCESS=0
|
|
TOTAL_SKIP=0
|
|
TOTAL_ERROR=0
|
|
TS_KST=$(TZ=Asia/Seoul date '+%Y%m%d_%H%M%S')
|
|
REPORT_DIR="${ROOT_DIR}/backup_reports"
|
|
mkdir -p "${REPORT_DIR}"
|
|
DECISIONS_LOG="${REPORT_DIR}/decisions_${TS_KST}.log"
|
|
TIMINGS_LOG="${REPORT_DIR}/timings_${TS_KST}.log"
|
|
REPORT_MD="${REPORT_DIR}/report_${TS_KST}.md"
|
|
echo "REPORT_TS=${TS_KST}" >> "${GITHUB_ENV}"
|
|
echo "REPORT_DIR=${REPORT_DIR}" >> "${GITHUB_ENV}"
|
|
echo "SOURCE_SSH_HOST=${SOURCE_SSH_HOST}" >> "${GITHUB_ENV}"
|
|
: > "${DECISIONS_LOG}"
|
|
: > "${TIMINGS_LOG}"
|
|
|
|
notify_status() {
|
|
local status="$1" repo="$2" branch="$3" mode="$4" start_epoch="$5" reason="${6:-}" details="${7:-}"
|
|
local ts end_epoch duration text payload
|
|
case "${status}" in
|
|
success) ((++TOTAL_SUCCESS)) ;;
|
|
skip) ((++TOTAL_SKIP)) ;;
|
|
error) ((++TOTAL_ERROR)) ;;
|
|
esac
|
|
[[ -z "${NOTIFY_WEBHOOK}" ]] && return
|
|
ts=$(TZ=Asia/Seoul date '+%Y-%m-%d %H:%M:%S %Z')
|
|
case "${status}" in
|
|
start)
|
|
text="센터Git ${repo} 백업을 ${ts}에 시작합니다. (branch: ${branch}, mode: ${mode})"
|
|
;;
|
|
skip)
|
|
end_epoch=$(date +%s)
|
|
duration=$((end_epoch - start_epoch))
|
|
text="센터Git ${repo} 백업을 건너뜁니다. (branch: ${branch} -> main, mode: ${mode}, duration: ${duration}s, 시각: ${ts})"
|
|
;;
|
|
success)
|
|
end_epoch=$(date +%s)
|
|
duration=$((end_epoch - start_epoch))
|
|
text="센터Git ${repo} 백업을 완료했습니다. (branch: ${branch} -> main, mode: ${mode}, duration: ${duration}s, 완료시각: ${ts})"
|
|
;;
|
|
error)
|
|
end_epoch=$(date +%s)
|
|
duration=$((end_epoch - start_epoch))
|
|
text="센터Git ${repo} 백업이 실패했습니다. (branch: ${branch}, mode: ${mode}, duration: ${duration}s, 이유: ${reason}, 시각: ${ts})"
|
|
;;
|
|
*)
|
|
text="센터Git ${repo} 상태: ${status} (${ts})"
|
|
;;
|
|
esac
|
|
if [[ -n "${details}" ]]; then
|
|
text="${text} (heads: ${details})"
|
|
fi
|
|
payload=${text//\"/\\\"}
|
|
curl -sS -i -X POST \
|
|
-H "Content-Type: application/json" \
|
|
-d "{\"username\":\"Gitea\",\"icon_url\":\"https://gitea.hmac.kr/assets/img/logo.svg\",\"text\":\"${payload}\"}" \
|
|
"${NOTIFY_WEBHOOK}" >/dev/null || echo "::warning::Notification failed for ${repo} (${status})"
|
|
}
|
|
|
|
set_default_branch_main() {
|
|
local repo_name="$1"
|
|
local response http_status body attempt max_api_retry
|
|
max_api_retry=3
|
|
attempt=1
|
|
while (( attempt<=max_api_retry )); do
|
|
response=$(curl -s -w "\n%{http_code}" -X PATCH -H "Content-Type: application/json" -H "${AUTH_HEADER}" -d "{\"default_branch\":\"main\"}" "${BASE_GITEA_URL}/api/v1/repos/${CENTER_ORG}/${repo_name}") || response=$'\n000'
|
|
http_status=$(echo "${response}" | tail -n1)
|
|
body=$(echo "${response}" | sed '$d')
|
|
if [[ "${http_status}" == "000" || "${http_status}" =~ ^5 || "${http_status}" == "429" ]]; then
|
|
echo "::warning::Retrying default branch set (HTTP ${http_status}) for ${repo_name} (${attempt}/3)..."
|
|
sleep 5
|
|
((attempt++))
|
|
continue
|
|
fi
|
|
break
|
|
done
|
|
|
|
if [[ "${http_status}" != "200" ]]; then
|
|
echo "::warning::Failed to set default branch to 'main' for ${CENTER_ORG}/${repo_name} (status ${http_status})"
|
|
if [[ -n "${body}" ]]; then
|
|
echo "${body}"
|
|
fi
|
|
else
|
|
echo "Default branch set to 'main' for ${CENTER_ORG}/${repo_name}"
|
|
fi
|
|
}
|
|
|
|
append_decision_row() {
|
|
local note="$1"
|
|
printf "%s|%s|%s|%s|%s|%s|%s|%s\n" "${source_repo}" "${branch_name}" "${alias_name}" "${repo_name}" "${source_commit}" "${target_commit}" "${decision}" "${note}" >> "${DECISIONS_LOG}"
|
|
}
|
|
|
|
append_timing_row() {
|
|
local status="$1" note="$2"
|
|
local end_ts="${exec_end_epoch:-$(date +%s)}"
|
|
local dec_ts="${decision_epoch:-$start_epoch}"
|
|
local exec_start_ts="${exec_start_epoch:-}"
|
|
local total_dur=$(( end_ts - start_epoch ))
|
|
local exec_dur=0
|
|
if [[ -n "${exec_start_ts}" ]]; then
|
|
exec_dur=$(( end_ts - exec_start_ts ))
|
|
fi
|
|
printf "%s|%s|%s|%s|%s|%s|%s|%s|%s|%s|%s|%s|%s|%s|%s\n" \
|
|
"${source_repo}" "${branch_name}" "${alias_name}" "${repo_name}" \
|
|
"${source_commit}" "${target_commit}" "${decision}" "${status}" \
|
|
"${start_epoch}" "${dec_ts}" "${exec_start_ts}" "${end_ts}" \
|
|
"${total_dur}" "${exec_dur}" "${note}" >> "${TIMINGS_LOG}"
|
|
}
|
|
|
|
map_repo_name() {
|
|
local branch_name="$1" alias_name="$2" resolved
|
|
resolved="${branch_name}"
|
|
if [[ "${branch_name}" == "Develop_Net8" ]]; then
|
|
resolved="base"
|
|
elif [[ "${branch_name}" == "Develop_Net8_"* ]]; then
|
|
resolved="${branch_name#Develop_Net8_}"
|
|
elif [[ "${branch_name}" == "Develop_"* ]]; then
|
|
resolved="${branch_name#Develop_}"
|
|
elif [[ "${branch_name}" == "develop_"* ]]; then
|
|
resolved="${branch_name#develop_}"
|
|
fi
|
|
if [[ -n "${alias_name}" ]]; then
|
|
resolved="${alias_name}"
|
|
fi
|
|
echo "${resolved}"
|
|
}
|
|
|
|
declare -A SOURCE_HEADS
|
|
declare -A ENTRY_SOURCE
|
|
declare -A ENTRY_BRANCH
|
|
declare -A ENTRY_ALIAS
|
|
declare -A ENTRY_REPO
|
|
declare -A SOURCE_REPOS
|
|
declare -A SOURCE_CACHE_PATH
|
|
declare -a ENTRY_KEYS=()
|
|
|
|
add_entry() {
|
|
local entry_raw="$1" entry alias_name source_repo branch_name repo_name idx
|
|
entry="${entry_raw%%#*}"
|
|
entry="$(echo "$entry" | xargs)"
|
|
[[ -z "${entry}" ]] && return
|
|
|
|
alias_name=""
|
|
if [[ "${entry}" == *","* ]]; then
|
|
IFS=',' read -r entry alias_name <<< "${entry}"
|
|
entry="$(echo "$entry" | xargs)"
|
|
alias_name="$(echo "$alias_name" | xargs)"
|
|
fi
|
|
|
|
if [[ "${entry}" != */* ]]; then
|
|
echo "::warning::Entry '${entry}' is missing sourceRepo/branch format. Skipping."
|
|
return
|
|
fi
|
|
|
|
source_repo="${entry%%/*}"
|
|
branch_name="${entry#*/}"
|
|
if [[ -z "${source_repo}" || -z "${branch_name}" ]]; then
|
|
echo "::warning::Invalid entry '${entry}'. Skipping."
|
|
return
|
|
fi
|
|
|
|
repo_name="$(map_repo_name "${branch_name}" "${alias_name}")"
|
|
idx="${#ENTRY_KEYS[@]}"
|
|
ENTRY_KEYS+=("${idx}")
|
|
ENTRY_SOURCE["${idx}"]="${source_repo}"
|
|
ENTRY_BRANCH["${idx}"]="${branch_name}"
|
|
ENTRY_ALIAS["${idx}"]="${alias_name}"
|
|
ENTRY_REPO["${idx}"]="${repo_name}"
|
|
SOURCE_REPOS["${source_repo}"]=1
|
|
}
|
|
|
|
if [[ -n "${INPUT_BRANCHES}" ]]; then
|
|
echo "Processing manually specified branches (pre-scan mode): ${INPUT_BRANCHES}"
|
|
IFS=',' read -r -a branches_to_process <<< "${INPUT_BRANCHES}"
|
|
for branch in "${branches_to_process[@]}"; do
|
|
add_entry "${branch}"
|
|
done
|
|
else
|
|
echo "Processing branch_list in pre-scan mode."
|
|
while IFS= read -r branch_line || [[ -n "${branch_line}" ]]; do
|
|
add_entry "${branch_line}"
|
|
done < branch_list
|
|
fi
|
|
|
|
if (( ${#ENTRY_KEYS[@]} == 0 )); then
|
|
echo "::error::No valid branch entries to process."
|
|
exit 1
|
|
fi
|
|
|
|
echo "Step 1) 소스 브랜치 해시 스캔"
|
|
for source_repo in "${!SOURCE_REPOS[@]}"; do
|
|
source_repo_url="${SOURCE_SSH_HOST}:${source_repo}"
|
|
echo " - ${source_repo_url}"
|
|
if ! remote_output=$(git ls-remote --heads "${source_repo_url}"); then
|
|
echo "::warning::Failed to ls-remote ${source_repo_url}. Entries for this repo may fail."
|
|
continue
|
|
fi
|
|
if [[ -z "${remote_output}" ]]; then
|
|
echo "::warning::No branches found for ${source_repo_url}"
|
|
continue
|
|
fi
|
|
while IFS=$'\t' read -r commit ref || [[ -n "${commit}" ]]; do
|
|
[[ -z "${commit}" || -z "${ref}" ]] && continue
|
|
branch="${ref#refs/heads/}"
|
|
SOURCE_HEADS["${source_repo}|${branch}"]="${commit}"
|
|
done <<< "${remote_output}"
|
|
done
|
|
|
|
echo "Step 2) 타겟 해시 스캔 및 실행 계획 수립"
|
|
GITEA_HOSTNAME=$(echo "${BASE_GITEA_URL}" | sed -e 's~^https*://~~' -e 's~/$~~')
|
|
|
|
for idx in "${ENTRY_KEYS[@]}"; do
|
|
source_repo="${ENTRY_SOURCE[${idx}]}"
|
|
branch_name="${ENTRY_BRANCH[${idx}]}"
|
|
alias_name="${ENTRY_ALIAS[${idx}]}"
|
|
repo_name="${ENTRY_REPO[${idx}]}"
|
|
start_epoch=$(date +%s)
|
|
backup_mode="미정"
|
|
note=""
|
|
target_commit=""
|
|
heads_detail="src=unknown tgt=unknown"
|
|
decision_epoch=""
|
|
exec_start_epoch=""
|
|
exec_end_epoch=""
|
|
|
|
source_commit="${SOURCE_HEADS[${source_repo}|${branch_name}]:-}"
|
|
if [[ -z "${source_commit}" ]]; then
|
|
decision="사전 스캔 실패"
|
|
decision_epoch=$(date +%s)
|
|
note="source branch 없음/조회 실패 (repo=${source_repo}, branch=${branch_name})"
|
|
append_decision_row "${note}"
|
|
heads_detail="src=none tgt=unknown"
|
|
notify_status "error" "${repo_name}" "${branch_name}" "사전 스캔 실패" "${start_epoch}" "${note}" "${heads_detail}"
|
|
append_timing_row "error" "${note}"
|
|
echo "Skipping ${branch_name} (${repo_name}) - ${note}"
|
|
continue
|
|
fi
|
|
|
|
GITEA_REMOTE="https://${BASE_GITEA_USER}:${BASE_GITEA_TOKEN}@${GITEA_HOSTNAME}/${CENTER_ORG}/${repo_name}.git"
|
|
repo_exists=false
|
|
just_created=false
|
|
max_api_retry=3
|
|
attempt=1
|
|
while (( attempt<=max_api_retry )); do
|
|
http_status=$(curl -s -o /dev/null -w "%{http_code}" -H "${AUTH_HEADER}" "${BASE_GITEA_URL}/api/v1/repos/${CENTER_ORG}/${repo_name}") || http_status="000"
|
|
if [[ "${http_status}" == "000" || "${http_status}" =~ ^5 || "${http_status}" == "429" ]]; then
|
|
echo "::warning::Repo check HTTP ${http_status} for ${repo_name} (${attempt}/3); retrying in 5s..."
|
|
sleep 5
|
|
((attempt++))
|
|
continue
|
|
fi
|
|
break
|
|
done
|
|
|
|
if [[ "${http_status}" == "200" ]]; then
|
|
repo_exists=true
|
|
elif [[ "${http_status}" == "404" ]]; then
|
|
repo_exists=false
|
|
else
|
|
decision="사전 조회 실패"
|
|
decision_epoch=$(date +%s)
|
|
note="repo 조회 실패 (HTTP ${http_status})"
|
|
echo "::warning::${note}"
|
|
heads_detail="src=${source_commit} tgt=unknown"
|
|
notify_status "error" "${repo_name}" "${branch_name}" "사전 조회 실패" "${start_epoch}" "${note}" "${heads_detail}"
|
|
append_decision_row "${note}"
|
|
append_timing_row "error" "${note}"
|
|
continue
|
|
fi
|
|
|
|
target_commit=""
|
|
if ${repo_exists}; then
|
|
target_main_ref=$(git ls-remote "${GITEA_REMOTE}" "refs/heads/main" || true)
|
|
target_commit=$(echo "${target_main_ref}" | awk '{print $1}')
|
|
fi
|
|
heads_detail="src=${source_commit} tgt=${target_commit:-none}"
|
|
exists_text=$(${repo_exists} && echo "true" || echo "false")
|
|
|
|
decision=""
|
|
if ! ${repo_exists}; then
|
|
decision="신규 백업"
|
|
note="타겟 저장소 없음"
|
|
elif [[ -z "${target_commit}" ]]; then
|
|
decision="신규 백업"
|
|
note="타겟 main 없음"
|
|
elif [[ "${source_commit}" == "${target_commit}" ]]; then
|
|
decision="동일 커밋 - 건너뜀"
|
|
note="동일 커밋"
|
|
else
|
|
decision="증분 백업"
|
|
note="커밋 상이"
|
|
fi
|
|
|
|
decision_epoch=$(date +%s)
|
|
append_decision_row "${note}"
|
|
|
|
if [[ "${decision}" == "동일 커밋 - 건너뜀" ]]; then
|
|
notify_status "skip" "${repo_name}" "${branch_name}" "${decision}" "${start_epoch}" "" "${heads_detail}"
|
|
echo "Pre-scan marked ${branch_name} (${repo_name}) as skip (same commit)."
|
|
append_timing_row "skip" "${note}"
|
|
continue
|
|
fi
|
|
|
|
if [[ "${decision}" == "신규 백업" ]]; then
|
|
backup_mode="신규 백업"
|
|
echo "Repository 'center_dev/${repo_name}' does not exist or main is missing. Will create/push."
|
|
attempt=1
|
|
create_status=""
|
|
if ! ${repo_exists}; then
|
|
while (( attempt<=max_api_retry )); do
|
|
create_status=$(curl -s -o /dev/null -w "%{http_code}" -X POST -H "Content-Type: application/json" -H "${AUTH_HEADER}" -d "{\"name\":\"${repo_name}\",\"private\":true,\"default_branch\":\"main\"}" "${BASE_GITEA_URL}/api/v1/orgs/${CENTER_ORG}/repos") || create_status="000"
|
|
if [[ "${create_status}" == "000" || "${create_status}" =~ ^5 || "${create_status}" == "429" ]]; then
|
|
echo "::warning::Repo create HTTP ${create_status} for ${repo_name} (${attempt}/3); retrying in 5s..."
|
|
sleep 5
|
|
((attempt++))
|
|
continue
|
|
fi
|
|
break
|
|
done
|
|
if [[ "${create_status}" != "201" ]]; then
|
|
echo "::error::Failed to create repository. HTTP ${create_status}"
|
|
notify_status "error" "${repo_name}" "${branch_name}" "${backup_mode}" "${start_epoch}" "repo 생성 실패 (HTTP ${create_status})" "${heads_detail}"
|
|
exec_end_epoch=$(date +%s)
|
|
append_timing_row "error" "repo 생성 실패 (HTTP ${create_status})"
|
|
continue
|
|
fi
|
|
echo "Repository created successfully."
|
|
repo_exists=true
|
|
just_created=true
|
|
fi
|
|
else
|
|
backup_mode="증분 백업"
|
|
just_created=false
|
|
fi
|
|
|
|
notify_status "start" "${repo_name}" "${branch_name}" "${backup_mode}" "${start_epoch}" "" "${heads_detail}"
|
|
|
|
exec_start_epoch=$(date +%s)
|
|
SOURCE_FETCH_REMOTE="${SOURCE_SSH_HOST}:${source_repo}"
|
|
shallow_exclude_args=()
|
|
if ${repo_exists} && [[ -n "${target_commit:-}" ]] && [[ "${USE_SHALLOW_EXCLUDE}" != "false" ]]; then
|
|
commit_on_source=$(git ls-remote "${SOURCE_FETCH_REMOTE}" "${target_commit}" || true)
|
|
if [[ -n "${commit_on_source}" ]]; then
|
|
shallow_exclude_args=(--shallow-exclude="${target_commit}")
|
|
else
|
|
echo "::notice::Target commit ${target_commit} not found on source; skipping shallow-exclude."
|
|
fi
|
|
fi
|
|
|
|
CLONE_DIR=$(mktemp -d)
|
|
echo "Working directory: ${CLONE_DIR}"
|
|
if ${just_created}; then
|
|
echo "Target repo newly created; cloning source branch for initial push..."
|
|
if ! git clone --bare --no-tags --single-branch --branch "${branch_name}" "${SOURCE_FETCH_REMOTE}" "${CLONE_DIR}"; then
|
|
echo "::error::Failed to clone source repository ${SOURCE_FETCH_REMOTE}"
|
|
notify_status "error" "${repo_name}" "${branch_name}" "${backup_mode}" "${start_epoch}" "source clone 실패" "${heads_detail}"
|
|
exec_end_epoch=$(date +%s)
|
|
append_timing_row "error" "source clone 실패"
|
|
rm -rf "${CLONE_DIR}"
|
|
continue
|
|
fi
|
|
cd "${CLONE_DIR}"
|
|
git remote rename origin source
|
|
git remote add origin "${GITEA_REMOTE}"
|
|
else
|
|
cd "${CLONE_DIR}"
|
|
if ! git init --bare; then
|
|
echo "::error::Failed to init bare repository"
|
|
notify_status "error" "${repo_name}" "${branch_name}" "${backup_mode}" "${start_epoch}" "bare init 실패" "${heads_detail}"
|
|
exec_end_epoch=$(date +%s)
|
|
append_timing_row "error" "bare init 실패"
|
|
cd "${ROOT_DIR}"
|
|
rm -rf "${CLONE_DIR}"
|
|
continue
|
|
fi
|
|
git remote add origin "${GITEA_REMOTE}"
|
|
git remote add source "${SOURCE_FETCH_REMOTE}"
|
|
fi
|
|
|
|
FETCH_REMOTE="${SOURCE_FETCH_REMOTE:-${SOURCE_SSH_HOST}:${source_repo}}"
|
|
prep_elapsed=$(( $(date +%s) - start_epoch ))
|
|
echo "Prep completed in ${prep_elapsed}s. Fetching latest branch '${branch_name}' from source ${FETCH_REMOTE} ..."
|
|
FETCH_LOG="${CLONE_DIR}/fetch_shallow.log"
|
|
if ! git fetch --no-tags "${shallow_exclude_args[@]}" source "+refs/heads/${branch_name}:refs/heads/${branch_name}" 2> >(tee "${FETCH_LOG}" >&2); then
|
|
if [[ "${#shallow_exclude_args[@]}" -gt 0 ]]; then
|
|
echo "::warning::shallow-exclude fetch failed (likely unsupported). Log: ${FETCH_LOG}"
|
|
if [[ -s "${FETCH_LOG}" ]]; then
|
|
echo "[shallow-exclude stderr tail]"
|
|
tail -n 40 "${FETCH_LOG}"
|
|
fi
|
|
echo "[fallback] Seeding target main depth=${TARGET_SEED_DEPTH} then retrying full fetch without shallow-exclude"
|
|
SEED_LOG="${CLONE_DIR}/fetch_seed.log"
|
|
git fetch --no-tags --depth="${TARGET_SEED_DEPTH}" origin "refs/heads/main:refs/heads/main" 2> >(tee "${SEED_LOG}" >&2) || echo "::warning::Seeding from target main skipped (fetch failed or branch missing)"
|
|
backup_mode="증분 백업 (폴백: shallow-exclude 미지원)"
|
|
FULL_FETCH_LOG="${CLONE_DIR}/fetch_full.log"
|
|
if ! git fetch --no-tags source "+refs/heads/${branch_name}:refs/heads/${branch_name}" 2> >(tee "${FULL_FETCH_LOG}" >&2); then
|
|
if [[ -s "${SEED_LOG:-}" ]]; then
|
|
echo "[seed stderr tail]"
|
|
tail -n 40 "${SEED_LOG}"
|
|
fi
|
|
if [[ -s "${FULL_FETCH_LOG}" ]]; then
|
|
echo "[fallback fetch stderr tail]"
|
|
tail -n 40 "${FULL_FETCH_LOG}"
|
|
fi
|
|
echo "::error::Failed to fetch branch '${branch_name}' from source repo (fallback without shallow-exclude)"
|
|
notify_status "error" "${repo_name}" "${branch_name}" "${backup_mode}" "${start_epoch}" "source fetch 오류(폴백)" "${heads_detail}"
|
|
exec_end_epoch=$(date +%s)
|
|
append_timing_row "error" "source fetch 오류(폴백)"
|
|
cd "${ROOT_DIR}"
|
|
rm -rf "${CLONE_DIR}"
|
|
continue
|
|
fi
|
|
else
|
|
echo "::error::Failed to fetch branch '${branch_name}' from source repo"
|
|
notify_status "error" "${repo_name}" "${branch_name}" "${backup_mode}" "${start_epoch}" "source fetch 오류" "${heads_detail}"
|
|
exec_end_epoch=$(date +%s)
|
|
append_timing_row "error" "source fetch 오류"
|
|
cd "${ROOT_DIR}"
|
|
rm -rf "${CLONE_DIR}"
|
|
continue
|
|
fi
|
|
fi
|
|
|
|
if [[ "${SYNC_TAGS}" == "true" ]]; then
|
|
echo "Fetching tags from source..."
|
|
if ! git fetch --prune --prune-tags --no-tags source "refs/tags/*:refs/tags/*"; then
|
|
echo "::error::Failed to fetch tags from source repo"
|
|
notify_status "error" "${repo_name}" "${branch_name}" "${backup_mode}" "${start_epoch}" "tag fetch 오류" "${heads_detail}"
|
|
exec_end_epoch=$(date +%s)
|
|
append_timing_row "error" "tag fetch 오류"
|
|
cd "${ROOT_DIR}"
|
|
rm -rf "${CLONE_DIR}"
|
|
continue
|
|
fi
|
|
fi
|
|
|
|
echo "Pushing '${branch_name}' to Gitea repository '${repo_name}'..."
|
|
if ! git push --progress --force origin "refs/heads/${branch_name}:refs/heads/main"; then
|
|
echo "::error::Failed to push branch '${branch_name}' to target repository"
|
|
notify_status "error" "${repo_name}" "${branch_name}" "${backup_mode}" "${start_epoch}" "push 오류 (-> main)" "${heads_detail}"
|
|
exec_end_epoch=$(date +%s)
|
|
append_timing_row "error" "push 오류 (-> main)"
|
|
cd "${ROOT_DIR}"
|
|
rm -rf "${CLONE_DIR}"
|
|
continue
|
|
fi
|
|
if [[ "${SYNC_TAGS}" == "true" ]]; then
|
|
echo "Pushing tags to Gitea repository '${repo_name}'..."
|
|
if ! git push --force --prune origin "refs/tags/*:refs/tags/*"; then
|
|
echo "::warning::Failed to push tags to target repository"
|
|
notify_status "error" "${repo_name}" "${branch_name}" "${backup_mode}" "${start_epoch}" "tag push 오류" "${heads_detail}"
|
|
exec_end_epoch=$(date +%s)
|
|
append_timing_row "error" "tag push 오류"
|
|
fi
|
|
fi
|
|
|
|
cd "${ROOT_DIR}"
|
|
rm -rf "${CLONE_DIR}"
|
|
|
|
echo "Successfully mirrored ${branch_name} to center_dev/${repo_name}"
|
|
set_default_branch_main "${repo_name}"
|
|
final_heads_detail="src=${source_commit} tgt=${source_commit}"
|
|
exec_end_epoch=$(date +%s)
|
|
append_timing_row "success" "${backup_mode}"
|
|
notify_status "success" "${repo_name}" "${branch_name}" "${backup_mode}" "${start_epoch}" "" "${final_heads_detail}"
|
|
echo "================================================="
|
|
echo ""
|
|
done
|
|
|
|
TOTAL_PROCESSED=$((TOTAL_SUCCESS + TOTAL_SKIP + TOTAL_ERROR))
|
|
SUMMARY_TS=$(TZ=Asia/Seoul date '+%Y-%m-%d %H:%M:%S %Z')
|
|
SUMMARY_TEXT="브랜치 동기화 완료: 총 ${TOTAL_PROCESSED}개 (성공 ${TOTAL_SUCCESS}, 동일로 건너뜀 ${TOTAL_SKIP}, 오류 ${TOTAL_ERROR}) - ${SUMMARY_TS}"
|
|
echo "${SUMMARY_TEXT}"
|
|
echo "보고서(마크다운)는 이후 스텝에서 생성됩니다."
|
|
echo "${TOTAL_ERROR}" > "${REPORT_DIR}/exit_code"
|
|
if [[ -n "${NOTIFY_WEBHOOK}" ]]; then
|
|
SUMMARY_PAYLOAD=${SUMMARY_TEXT//\"/\\\"}
|
|
curl -sS -i -X POST \
|
|
-H "Content-Type: application/json" \
|
|
-d "{\"username\":\"Gitea\",\"icon_url\":\"https://gitea.hmac.kr/assets/img/logo.svg\",\"text\":\"${SUMMARY_PAYLOAD}\"}" \
|
|
"${NOTIFY_WEBHOOK}" >/dev/null || echo "::warning::Summary notification failed"
|
|
fi
|
|
|
|
if (( TOTAL_ERROR > 0 )); then
|
|
echo "::warning::One or more branches failed (${TOTAL_ERROR}). Proceeding to report generation..."
|
|
fi
|
|
|
|
- name: Build markdown report
|
|
if: always()
|
|
run: |
|
|
set -euo pipefail
|
|
REPORT_DIR="${REPORT_DIR:-${GITHUB_WORKSPACE}/backup_reports}"
|
|
TS="${REPORT_TS:-}"
|
|
if [[ -z "${TS}" ]]; then
|
|
shopt -s nullglob
|
|
decision_files=(${REPORT_DIR}/decisions_*.log)
|
|
if (( ${#decision_files[@]} == 0 )); then
|
|
echo "No decision logs found in ${REPORT_DIR}; skipping report generation."
|
|
exit 0
|
|
fi
|
|
IFS=$'\n' decision_files_sorted=($(ls -1t "${decision_files[@]}"))
|
|
shopt -u nullglob
|
|
TS=$(basename "${decision_files_sorted[0]}" | sed -E 's/decisions_([0-9_]+)\.log/\1/')
|
|
fi
|
|
if [[ -z "${TS}" ]]; then
|
|
echo "Unable to determine report timestamp; skipping report generation."
|
|
exit 0
|
|
fi
|
|
DECISIONS_LOG="${REPORT_DIR}/decisions_${TS}.log"
|
|
TIMINGS_LOG="${REPORT_DIR}/timings_${TS}.log"
|
|
REPORT_MD="${REPORT_DIR}/report_${TS}.md"
|
|
SOURCE_SSH_HOST="${SOURCE_SSH_HOST:-}"
|
|
|
|
SUMMARY_TS=$(TZ=Asia/Seoul date '+%Y-%m-%d %H:%M:%S %Z')
|
|
total=0; success=0; skip=0; error=0
|
|
if [[ -f "${TIMINGS_LOG}" ]]; then
|
|
while IFS='|' read -r sr br al rr sc tc dc st se de ee xe dt dexe nt; do
|
|
[[ -z "${sr}" ]] && continue
|
|
((total++))
|
|
case "${st}" in
|
|
success) ((success++)) ;;
|
|
skip) ((skip++)) ;;
|
|
error) ((error++)) ;;
|
|
esac
|
|
done < "${TIMINGS_LOG}"
|
|
fi
|
|
|
|
{
|
|
echo "# Git Backup Report"
|
|
echo ""
|
|
echo "- Generated: ${SUMMARY_TS}"
|
|
echo "- Run ID: ${GITHUB_RUN_ID:-local}"
|
|
[[ -n "${SOURCE_SSH_HOST}" ]] && echo "- Source host: ${SOURCE_SSH_HOST}"
|
|
echo "- Total: ${total} (success ${success}, skip ${skip}, error ${error})"
|
|
echo ""
|
|
echo "## Decisions (all)"
|
|
echo "|source_repo|branch|alias|resolved_repo|source_commit|target_commit|decision|note|"
|
|
echo "|---|---|---|---|---|---|---|---|"
|
|
if [[ -f "${DECISIONS_LOG}" ]]; then
|
|
while IFS='|' read -r sr br al rr sc tc dc nt; do
|
|
[[ -z "${sr}" ]] && continue
|
|
printf "|%s|%s|%s|%s|%s|%s|%s|%s|\n" "$sr" "$br" "$al" "$rr" "$sc" "$tc" "$dc" "$nt"
|
|
done < "${DECISIONS_LOG}"
|
|
else
|
|
echo "|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|"
|
|
fi
|
|
echo ""
|
|
echo "## Timings (all)"
|
|
echo "|source_repo|branch|alias|resolved_repo|source_commit|target_commit|decision|status|start_epoch|decision_epoch|exec_start_epoch|exec_end_epoch|duration_total(s)|duration_exec(s)|note|"
|
|
echo "|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|"
|
|
if [[ -f "${TIMINGS_LOG}" ]]; then
|
|
while IFS='|' read -r sr br al rr sc tc dc st se de ee xe dt dexe nt; do
|
|
[[ -z "${sr}" ]] && continue
|
|
printf "|%s|%s|%s|%s|%s|%s|%s|%s|%s|%s|%s|%s|%s|%s|%s|\n" "$sr" "$br" "$al" "$rr" "$sc" "$tc" "$dc" "$st" "$se" "$de" "$ee" "$xe" "$dt" "$dexe" "$nt"
|
|
done < "${TIMINGS_LOG}"
|
|
else
|
|
echo "|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|(missing)|"
|
|
fi
|
|
echo ""
|
|
} > "${REPORT_MD}"
|
|
echo "Report generated at ${REPORT_MD}"
|
|
|
|
- name: Upload backup reports
|
|
if: always()
|
|
uses: actions/upload-artifact@v4
|
|
with:
|
|
name: backup_reports_${{ github.run_id }}
|
|
path: backup_reports
|
|
|
|
- name: Fail if backup had errors
|
|
if: always()
|
|
run: |
|
|
EXIT_CODE=0
|
|
if [[ -f backup_reports/exit_code ]]; then
|
|
EXIT_CODE=$(cat backup_reports/exit_code)
|
|
fi
|
|
if [[ "${EXIT_CODE}" != "0" ]]; then
|
|
echo "::error::Backup reported ${EXIT_CODE} errors."
|
|
exit 1
|
|
else
|
|
echo "Backup completed without errors."
|
|
fi
|