223 lines
6.8 KiB
Bash
223 lines
6.8 KiB
Bash
#!/bin/bash
|
||
# ============================================================================
|
||
# 文件名: delete_all_projects.sh
|
||
# 描述: 删除 GCP 账户下的所有项目(危险操作)
|
||
# 作者: Cloud Tools Project
|
||
# 版本: 2.1.1(支持远程库加载)
|
||
# 警告: 此操作不可逆,所有项目数据将永久丢失
|
||
# ============================================================================
|
||
|
||
set -euo pipefail
|
||
|
||
# ============================================================================
|
||
# 远程库加载配置
|
||
# ============================================================================
|
||
|
||
# 远程仓库 URL(可通过环境变量覆盖)
|
||
readonly REMOTE_BASE_URL="${REMOTE_LIB_URL:-https://gitea.bcde.io/wangdefa/tools/raw/branch/main}"
|
||
|
||
# 获取脚本目录(用于本地加载)
|
||
readonly SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
||
readonly PROJECT_ROOT="$(dirname "$SCRIPT_DIR")"
|
||
|
||
#
|
||
# 智能加载公共库
|
||
#
|
||
# 加载策略:
|
||
# 1. 如果 FORCE_REMOTE=1,强制使用远程库
|
||
# 2. 否则尝试使用本地库
|
||
# 3. 本地库不存在时自动回退到远程库
|
||
#
|
||
load_common_libs() {
|
||
local use_remote=false
|
||
|
||
# 检查是否强制远程
|
||
if [[ "${FORCE_REMOTE:-0}" == "1" ]]; then
|
||
echo "[INFO] 强制使用远程库 (FORCE_REMOTE=1)" >&2
|
||
use_remote=true
|
||
# 检查本地库是否存在
|
||
elif [[ -f "${PROJECT_ROOT}/common/logging.sh" ]] && [[ -f "${PROJECT_ROOT}/common/error_handler.sh" ]]; then
|
||
# shellcheck disable=SC1091
|
||
source "${PROJECT_ROOT}/common/logging.sh"
|
||
# shellcheck disable=SC1091
|
||
source "${PROJECT_ROOT}/common/error_handler.sh"
|
||
return 0
|
||
else
|
||
echo "[WARN] 本地库不存在,使用远程库" >&2
|
||
use_remote=true
|
||
fi
|
||
|
||
# 使用远程库
|
||
if [[ "$use_remote" == "true" ]]; then
|
||
# 下载到临时文件(避免 process substitution 与 set -u 的交互问题)
|
||
local temp_loader
|
||
temp_loader=$(mktemp)
|
||
|
||
if command -v curl &>/dev/null; then
|
||
echo "[INFO] 使用 curl 下载远程库..." >&2
|
||
if curl -fsSL "${REMOTE_BASE_URL}/common/remote_loader.sh" -o "$temp_loader" 2>/dev/null; then
|
||
# shellcheck disable=SC1090
|
||
if source "$temp_loader"; then
|
||
rm -f "$temp_loader"
|
||
return 0
|
||
fi
|
||
fi
|
||
elif command -v wget &>/dev/null; then
|
||
echo "[INFO] 使用 wget 下载远程库..." >&2
|
||
if wget -qO "$temp_loader" "${REMOTE_BASE_URL}/common/remote_loader.sh" 2>/dev/null; then
|
||
# shellcheck disable=SC1090
|
||
if source "$temp_loader"; then
|
||
rm -f "$temp_loader"
|
||
return 0
|
||
fi
|
||
fi
|
||
fi
|
||
|
||
rm -f "$temp_loader"
|
||
echo "[ERROR] 无法加载公共库" >&2
|
||
echo "[ERROR] - 本地库不存在" >&2
|
||
echo "[ERROR] - 远程下载失败(需要 curl 或 wget)" >&2
|
||
echo "[ERROR] - 仓库 URL: ${REMOTE_BASE_URL}" >&2
|
||
exit 1
|
||
fi
|
||
}
|
||
|
||
# 加载公共库
|
||
load_common_libs
|
||
|
||
# 创建日志文件
|
||
readonly LOG_FILE="deleted_projects_$(date +%Y%m%d_%H%M%S).log"
|
||
log_set_file "$LOG_FILE"
|
||
|
||
#
|
||
# 获取所有项目 ID 列表
|
||
# 返回: 项目 ID 列表(每行一个),通过 stdout 返回
|
||
# 退出码: 0 成功,1 失败
|
||
#
|
||
get_projects() {
|
||
log_warning "获取项目列表..."
|
||
local projects
|
||
projects=$(gcloud projects list --format="value(projectId)")
|
||
|
||
if [ -z "$projects" ]; then
|
||
log_warning "没有找到任何项目"
|
||
return 0
|
||
fi
|
||
|
||
echo "$projects"
|
||
}
|
||
|
||
#
|
||
# 显示项目列表(带编号)
|
||
# 参数: $1 - 项目列表(换行符分隔)
|
||
# 输出: 到 stderr
|
||
#
|
||
display_projects() {
|
||
local projects="$1"
|
||
local project_count
|
||
project_count=$(echo "$projects" | wc -l | tr -d ' ')
|
||
|
||
log_warning "以下项目将被删除:"
|
||
echo "$projects" | nl >&2
|
||
echo >&2
|
||
log_warning "总计: $project_count 个项目"
|
||
}
|
||
|
||
#
|
||
# 删除单个项目
|
||
#
|
||
delete_project() {
|
||
local project_id=$1
|
||
local log_file=$2
|
||
|
||
log_info "正在删除项目: $project_id"
|
||
|
||
if gcloud projects delete "$project_id" --quiet 2>&1 | tee -a "$log_file"; then
|
||
log_success "成功删除项目: $project_id"
|
||
echo "$(date): 成功删除项目 $project_id" >> "$log_file"
|
||
return 0
|
||
else
|
||
log_error "删除项目 $project_id 失败"
|
||
echo "$(date): 删除项目 $project_id 失败" >> "$log_file"
|
||
return 1
|
||
fi
|
||
}
|
||
|
||
#
|
||
# 主函数 - 删除所有 GCP 项目
|
||
#
|
||
main() {
|
||
# 第一次警告和确认
|
||
log_error "################################################################################"
|
||
log_error "警告:此脚本将删除您 GCP 账户下的所有项目!"
|
||
log_error "这是一个不可逆的操作,所有项目数据将永久丢失!"
|
||
log_error "################################################################################"
|
||
echo
|
||
|
||
# 第一次确认
|
||
read -p "您确定要继续吗?输入 'YES' 继续: " confirm1
|
||
if [ "$confirm1" != "YES" ]; then
|
||
log_warning "操作已取消"
|
||
exit 0
|
||
fi
|
||
|
||
# 获取项目列表
|
||
local projects
|
||
projects=$(get_projects)
|
||
|
||
# 检查是否有项目
|
||
if [ -z "$projects" ]; then
|
||
log_warning "没有找到任何项目,退出"
|
||
exit 0
|
||
fi
|
||
|
||
# 显示项目列表
|
||
display_projects "$projects"
|
||
|
||
# 统计项目数量
|
||
local project_count
|
||
project_count=$(echo "$projects" | wc -l | tr -d ' ')
|
||
|
||
# 第二次确认
|
||
log_error "最后确认:您将要删除 $project_count 个项目!"
|
||
read -p "输入 'DELETE ALL' 确认删除所有项目: " confirm2
|
||
if [ "$confirm2" != "DELETE ALL" ]; then
|
||
log_warning "操作已取消"
|
||
exit 0
|
||
fi
|
||
|
||
# 开始删除项目
|
||
log_warning "开始删除项目..."
|
||
log_info "删除记录将保存到: $LOG_FILE"
|
||
echo "================== 删除开始于: $(date) ==================" >> "$LOG_FILE"
|
||
|
||
local deleted_count=0
|
||
local failed_count=0
|
||
|
||
for project_id in $projects; do
|
||
if delete_project "$project_id" "$LOG_FILE"; then
|
||
((deleted_count++)) || true
|
||
else
|
||
((failed_count++)) || true
|
||
fi
|
||
|
||
# 添加短暂延迟,避免 API 限制
|
||
sleep 2
|
||
done
|
||
|
||
# 显示结果
|
||
log_info "================== 执行完成 ==================="
|
||
log_success "成功删除: $deleted_count 个项目"
|
||
log_error "删除失败: $failed_count 个项目"
|
||
log_info "详细日志: $LOG_FILE"
|
||
echo "================== 删除完成于: $(date) ==================" >> "$LOG_FILE"
|
||
|
||
# 如果有删除失败的项目,返回错误状态
|
||
if [ "$failed_count" -gt 0 ]; then
|
||
log_error "有 $failed_count 个项目删除失败。请检查日志。"
|
||
exit 1
|
||
fi
|
||
}
|
||
|
||
# 执行主函数
|
||
main "$@" |