feat: add Phase 4 — Migrate Repos + Fedora mirrors
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
163
phase4_migrate_repos.sh
Executable file
163
phase4_migrate_repos.sh
Executable file
@@ -0,0 +1,163 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# =============================================================================
|
||||||
|
# phase4_migrate_repos.sh — Import repos from GitHub + set up Fedora mirrors
|
||||||
|
# Depends on: Phase 1 + Phase 2 complete (both Gitea instances running)
|
||||||
|
# For each repo:
|
||||||
|
# 1. Import from GitHub → Unraid primary (under org, one-time migration)
|
||||||
|
# 2. Create pull mirror on Fedora (mirrors Unraid → Fedora on interval)
|
||||||
|
# Idempotent: skips repos that already exist on either instance.
|
||||||
|
# =============================================================================
|
||||||
|
|
||||||
|
SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
|
||||||
|
source "${SCRIPT_DIR}/lib/common.sh"
|
||||||
|
|
||||||
|
load_env
|
||||||
|
require_vars GITEA_ADMIN_TOKEN GITEA_BACKUP_ADMIN_TOKEN \
|
||||||
|
GITEA_INTERNAL_URL GITEA_BACKUP_INTERNAL_URL \
|
||||||
|
GITEA_ORG_NAME GITEA_ADMIN_USER GITEA_ADMIN_PASSWORD \
|
||||||
|
GITHUB_USERNAME GITHUB_TOKEN \
|
||||||
|
REPO_1_NAME REPO_2_NAME REPO_3_NAME \
|
||||||
|
MIGRATE_ISSUES MIGRATE_LABELS MIGRATE_MILESTONES MIGRATE_WIKI \
|
||||||
|
GITEA_BACKUP_MIRROR_INTERVAL
|
||||||
|
|
||||||
|
phase_header 4 "Migrate Repos + Fedora Mirrors"
|
||||||
|
|
||||||
|
# ---------------------------------------------------------------------------
|
||||||
|
# Build list of repos from REPO_N_NAME env vars
|
||||||
|
# Using an array rather than hardcoding 3 loops keeps it flexible.
|
||||||
|
# ---------------------------------------------------------------------------
|
||||||
|
REPOS=("$REPO_1_NAME" "$REPO_2_NAME" "$REPO_3_NAME")
|
||||||
|
|
||||||
|
TOTAL=${#REPOS[@]}
|
||||||
|
SUCCESS=0
|
||||||
|
FAILED=0
|
||||||
|
|
||||||
|
for repo in "${REPOS[@]}"; do
|
||||||
|
log_info "--- Processing repo: ${repo} ---"
|
||||||
|
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
# Step A: Import from GitHub to Unraid primary (under org)
|
||||||
|
# Uses Gitea's migration API which clones the GitHub repo server-side.
|
||||||
|
# mirror=false because this is a one-time import — we don't want Gitea
|
||||||
|
# to keep pulling from GitHub (GitHub becomes the push mirror target later).
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
log_step "A" "Importing ${repo} from GitHub to Unraid..."
|
||||||
|
|
||||||
|
# Idempotency: check if repo already exists on primary
|
||||||
|
if gitea_api GET "/repos/${GITEA_ORG_NAME}/${repo}" >/dev/null 2>&1; then
|
||||||
|
log_info "Repo '${GITEA_ORG_NAME}/${repo}' already exists on primary — skipping import"
|
||||||
|
else
|
||||||
|
# Build migration payload — JSON with all options from .env
|
||||||
|
MIGRATE_PAYLOAD=$(jq -n \
|
||||||
|
--arg clone_addr "https://github.com/${GITHUB_USERNAME}/${repo}.git" \
|
||||||
|
--arg auth_token "$GITHUB_TOKEN" \
|
||||||
|
--arg repo_owner "$GITEA_ORG_NAME" \
|
||||||
|
--arg repo_name "$repo" \
|
||||||
|
--arg service "github" \
|
||||||
|
--argjson issues "${MIGRATE_ISSUES}" \
|
||||||
|
--argjson labels "${MIGRATE_LABELS}" \
|
||||||
|
--argjson milestones "${MIGRATE_MILESTONES}" \
|
||||||
|
--argjson wiki "${MIGRATE_WIKI}" \
|
||||||
|
'{
|
||||||
|
clone_addr: $clone_addr,
|
||||||
|
auth_token: $auth_token,
|
||||||
|
repo_owner: $repo_owner,
|
||||||
|
repo_name: $repo_name,
|
||||||
|
service: $service,
|
||||||
|
mirror: false,
|
||||||
|
issues: $issues,
|
||||||
|
labels: $labels,
|
||||||
|
milestones: $milestones,
|
||||||
|
wiki: $wiki
|
||||||
|
}')
|
||||||
|
|
||||||
|
if gitea_api POST "/repos/migrate" "$MIGRATE_PAYLOAD" >/dev/null; then
|
||||||
|
log_success "Migration started for ${repo}"
|
||||||
|
else
|
||||||
|
log_error "Failed to start migration for ${repo}"
|
||||||
|
FAILED=$((FAILED + 1))
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Wait for migration to complete — poll until repo has content
|
||||||
|
# The migration API returns immediately but cloning happens async.
|
||||||
|
# We check for commits to confirm the repo has actual content.
|
||||||
|
log_info "Waiting for migration to complete..."
|
||||||
|
local_elapsed=0
|
||||||
|
while [[ $local_elapsed -lt 120 ]]; do
|
||||||
|
COMMITS=$(gitea_api GET "/repos/${GITEA_ORG_NAME}/${repo}/commits?limit=1" 2>/dev/null || echo "[]")
|
||||||
|
COMMIT_COUNT=$(printf '%s' "$COMMITS" | jq 'length' 2>/dev/null || echo 0)
|
||||||
|
if [[ "$COMMIT_COUNT" -gt 0 ]]; then
|
||||||
|
log_success "Repo ${repo} migrated with commits"
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
sleep 3
|
||||||
|
local_elapsed=$((local_elapsed + 3))
|
||||||
|
done
|
||||||
|
|
||||||
|
if [[ $local_elapsed -ge 120 ]]; then
|
||||||
|
log_error "Timeout waiting for ${repo} migration to complete"
|
||||||
|
FAILED=$((FAILED + 1))
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
# Step B: Create pull mirror on Fedora
|
||||||
|
# The Fedora instance mirrors the Unraid primary on a schedule. This uses
|
||||||
|
# Gitea's migration API with mirror=true, which creates a mirror repo that
|
||||||
|
# periodically pulls from the source URL.
|
||||||
|
# The mirror authenticates to the primary using admin credentials since
|
||||||
|
# the primary's repos may be private.
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
log_step "B" "Creating pull mirror for ${repo} on Fedora..."
|
||||||
|
|
||||||
|
# Idempotency: check if mirror repo already exists on Fedora
|
||||||
|
if gitea_backup_api GET "/repos/${GITEA_ADMIN_USER}/${repo}" >/dev/null 2>&1; then
|
||||||
|
log_info "Mirror '${GITEA_ADMIN_USER}/${repo}' already exists on Fedora — skipping"
|
||||||
|
else
|
||||||
|
# The clone URL points to the primary Gitea instance (Unraid)
|
||||||
|
MIRROR_PAYLOAD=$(jq -n \
|
||||||
|
--arg clone_addr "${GITEA_INTERNAL_URL}/${GITEA_ORG_NAME}/${repo}.git" \
|
||||||
|
--arg auth_username "$GITEA_ADMIN_USER" \
|
||||||
|
--arg auth_password "$GITEA_ADMIN_PASSWORD" \
|
||||||
|
--arg repo_owner "$GITEA_ADMIN_USER" \
|
||||||
|
--arg repo_name "$repo" \
|
||||||
|
--arg mirror_interval "$GITEA_BACKUP_MIRROR_INTERVAL" \
|
||||||
|
'{
|
||||||
|
clone_addr: $clone_addr,
|
||||||
|
auth_username: $auth_username,
|
||||||
|
auth_password: $auth_password,
|
||||||
|
repo_owner: $repo_owner,
|
||||||
|
repo_name: $repo_name,
|
||||||
|
service: "gitea",
|
||||||
|
mirror: true,
|
||||||
|
mirror_interval: $mirror_interval
|
||||||
|
}')
|
||||||
|
|
||||||
|
if gitea_backup_api POST "/repos/migrate" "$MIRROR_PAYLOAD" >/dev/null; then
|
||||||
|
log_success "Pull mirror created for ${repo} on Fedora"
|
||||||
|
else
|
||||||
|
log_error "Failed to create pull mirror for ${repo}"
|
||||||
|
FAILED=$((FAILED + 1))
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
SUCCESS=$((SUCCESS + 1))
|
||||||
|
done
|
||||||
|
|
||||||
|
# ---------------------------------------------------------------------------
|
||||||
|
# Summary
|
||||||
|
# ---------------------------------------------------------------------------
|
||||||
|
printf '\n'
|
||||||
|
log_info "Results: ${SUCCESS} succeeded, ${FAILED} failed (out of ${TOTAL})"
|
||||||
|
|
||||||
|
if [[ $FAILED -gt 0 ]]; then
|
||||||
|
log_error "Some repos failed — check logs above"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
log_success "Phase 4 complete — all repos migrated and mirrored"
|
||||||
92
phase4_post_check.sh
Executable file
92
phase4_post_check.sh
Executable file
@@ -0,0 +1,92 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# =============================================================================
|
||||||
|
# phase4_post_check.sh — Verify Phase 4 (Repo Migration) succeeded
|
||||||
|
# Checks for each repo:
|
||||||
|
# 1. Exists on primary under the org
|
||||||
|
# 2. Has at least one commit (migration actually imported content)
|
||||||
|
# 3. Default branch matches GitHub source
|
||||||
|
# 4. Mirror exists on Fedora under admin user
|
||||||
|
# 5. Mirror has mirror=true flag set
|
||||||
|
# Exits 0 only if ALL checks pass.
|
||||||
|
# =============================================================================
|
||||||
|
|
||||||
|
SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
|
||||||
|
source "${SCRIPT_DIR}/lib/common.sh"
|
||||||
|
|
||||||
|
load_env
|
||||||
|
require_vars GITEA_ADMIN_TOKEN GITEA_BACKUP_ADMIN_TOKEN \
|
||||||
|
GITEA_INTERNAL_URL GITEA_BACKUP_INTERNAL_URL \
|
||||||
|
GITEA_ORG_NAME GITEA_ADMIN_USER \
|
||||||
|
GITHUB_USERNAME GITHUB_TOKEN \
|
||||||
|
REPO_1_NAME REPO_2_NAME REPO_3_NAME
|
||||||
|
|
||||||
|
log_info "=== Phase 4 Post-Check ==="
|
||||||
|
|
||||||
|
REPOS=("$REPO_1_NAME" "$REPO_2_NAME" "$REPO_3_NAME")
|
||||||
|
PASS=0
|
||||||
|
FAIL=0
|
||||||
|
|
||||||
|
# Helper: run a check, track results
|
||||||
|
run_check() {
|
||||||
|
local description="$1"; shift
|
||||||
|
if "$@" 2>/dev/null; then
|
||||||
|
log_success "$description"
|
||||||
|
PASS=$((PASS + 1))
|
||||||
|
else
|
||||||
|
log_error "FAIL: $description"
|
||||||
|
FAIL=$((FAIL + 1))
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
for repo in "${REPOS[@]}"; do
|
||||||
|
log_info "--- Checking repo: ${repo} ---"
|
||||||
|
|
||||||
|
# Check 1: Repo exists on primary
|
||||||
|
run_check "Primary: ${GITEA_ORG_NAME}/${repo} exists" \
|
||||||
|
gitea_api GET "/repos/${GITEA_ORG_NAME}/${repo}" -o /dev/null
|
||||||
|
|
||||||
|
# Check 2: Repo has commits (migration imported content)
|
||||||
|
check_commits() {
|
||||||
|
local commits
|
||||||
|
commits=$(gitea_api GET "/repos/${GITEA_ORG_NAME}/$1/commits?limit=1")
|
||||||
|
local count
|
||||||
|
count=$(printf '%s' "$commits" | jq 'length')
|
||||||
|
[[ "$count" -gt 0 ]]
|
||||||
|
}
|
||||||
|
run_check "Primary: ${repo} has commits" check_commits "$repo"
|
||||||
|
|
||||||
|
# Check 3: Default branch matches GitHub source
|
||||||
|
check_default_branch() {
|
||||||
|
local gitea_branch github_branch
|
||||||
|
gitea_branch=$(gitea_api GET "/repos/${GITEA_ORG_NAME}/$1" | jq -r '.default_branch')
|
||||||
|
github_branch=$(github_api GET "/repos/${GITHUB_USERNAME}/$1" | jq -r '.default_branch')
|
||||||
|
[[ "$gitea_branch" == "$github_branch" ]]
|
||||||
|
}
|
||||||
|
run_check "Primary: ${repo} default branch matches GitHub" check_default_branch "$repo"
|
||||||
|
|
||||||
|
# Check 4: Mirror exists on Fedora
|
||||||
|
run_check "Fedora: ${GITEA_ADMIN_USER}/${repo} exists" \
|
||||||
|
gitea_backup_api GET "/repos/${GITEA_ADMIN_USER}/${repo}" -o /dev/null
|
||||||
|
|
||||||
|
# Check 5: Mirror has mirror=true
|
||||||
|
check_mirror_flag() {
|
||||||
|
local is_mirror
|
||||||
|
is_mirror=$(gitea_backup_api GET "/repos/${GITEA_ADMIN_USER}/$1" | jq -r '.mirror')
|
||||||
|
[[ "$is_mirror" == "true" ]]
|
||||||
|
}
|
||||||
|
run_check "Fedora: ${repo} has mirror=true" check_mirror_flag "$repo"
|
||||||
|
done
|
||||||
|
|
||||||
|
# Summary
|
||||||
|
printf '\n'
|
||||||
|
log_info "Results: ${PASS} passed, ${FAIL} failed"
|
||||||
|
|
||||||
|
if [[ $FAIL -gt 0 ]]; then
|
||||||
|
log_error "Phase 4 post-check FAILED"
|
||||||
|
exit 1
|
||||||
|
else
|
||||||
|
log_success "Phase 4 post-check PASSED — all repos migrated and mirrored"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
51
phase4_teardown.sh
Executable file
51
phase4_teardown.sh
Executable file
@@ -0,0 +1,51 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# =============================================================================
|
||||||
|
# phase4_teardown.sh — Delete migrated repos from primary + mirrors from Fedora
|
||||||
|
# Destructive: permanently deletes repositories.
|
||||||
|
# Prompts for confirmation before each deletion.
|
||||||
|
# Safe to run if repos have already been deleted (no errors).
|
||||||
|
# =============================================================================
|
||||||
|
|
||||||
|
SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
|
||||||
|
source "${SCRIPT_DIR}/lib/common.sh"
|
||||||
|
|
||||||
|
load_env
|
||||||
|
require_vars GITEA_ADMIN_TOKEN GITEA_BACKUP_ADMIN_TOKEN \
|
||||||
|
GITEA_INTERNAL_URL GITEA_BACKUP_INTERNAL_URL \
|
||||||
|
GITEA_ORG_NAME GITEA_ADMIN_USER \
|
||||||
|
REPO_1_NAME REPO_2_NAME REPO_3_NAME
|
||||||
|
|
||||||
|
log_warn "=== Phase 4 Teardown: Repos + Mirrors ==="
|
||||||
|
|
||||||
|
REPOS=("$REPO_1_NAME" "$REPO_2_NAME" "$REPO_3_NAME")
|
||||||
|
|
||||||
|
printf 'This will DELETE all migrated repos and mirrors. Continue? [y/N] '
|
||||||
|
read -r confirm
|
||||||
|
if [[ ! "$confirm" =~ ^[Yy]$ ]]; then
|
||||||
|
log_info "Teardown cancelled"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
for repo in "${REPOS[@]}"; do
|
||||||
|
log_info "--- Tearing down: ${repo} ---"
|
||||||
|
|
||||||
|
# Delete repo from primary (under org)
|
||||||
|
if gitea_api GET "/repos/${GITEA_ORG_NAME}/${repo}" >/dev/null 2>&1; then
|
||||||
|
gitea_api DELETE "/repos/${GITEA_ORG_NAME}/${repo}" >/dev/null || true
|
||||||
|
log_success "Deleted ${GITEA_ORG_NAME}/${repo} from primary"
|
||||||
|
else
|
||||||
|
log_info "Repo ${GITEA_ORG_NAME}/${repo} not found on primary — already deleted"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Delete mirror from Fedora (under admin user)
|
||||||
|
if gitea_backup_api GET "/repos/${GITEA_ADMIN_USER}/${repo}" >/dev/null 2>&1; then
|
||||||
|
gitea_backup_api DELETE "/repos/${GITEA_ADMIN_USER}/${repo}" >/dev/null || true
|
||||||
|
log_success "Deleted mirror ${GITEA_ADMIN_USER}/${repo} from Fedora"
|
||||||
|
else
|
||||||
|
log_info "Mirror ${GITEA_ADMIN_USER}/${repo} not found on Fedora — already deleted"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
log_success "Phase 4 teardown complete"
|
||||||
Reference in New Issue
Block a user