批量删除OpenStack实例

在Linux终端实现批量删除OpenStack实例,支持并发删除、安全确认、重试机制、优先清理运行中实例

bash 复制代码
#!/bin/bash
# ======================================================================
# 增强版 OpenStack 删除实例脚本
# 功能:支持并发删除、安全确认、重试机制、优先清理运行中实例
# 更新:2025年4月30日
# ======================================================================

#######################################
# 配置区(用户可自定义参数)
#######################################
delete_InstanceMAX_RETRY=20     # 最大尝试次数(删除失败时尝试再次删除实例)
CHECK_INTERVAL=5                # 实例状态检测间隔(单位:秒)
no_InstanceMAX_RETRY_TIME=20    # 实例列表为空,连续监测实例列表都是空的次数
CONCURRENCY=2                   # 并发删除线程数(建议不超过 API 限制)
MAX_WAIT_TIME=3600              # 最大等待超时时间(单位:秒,60 分钟)

LOG_FILE="./log_instance_delete_$(date +%Y%m%d).log" # 日志文件路径(自动按日期生成)

# 统计变量
TOTAL_DELETED_INSTANCES=0        # 已删除实例总数统计
DELETED_INSTANCES_FILE="/tmp/deleted_instances.txt" # 已删除实例ID临时记录文件

# 颜色定义(终端输出美化)
RED='\033[0;31m'    # 红色 - 错误/危险操作
GREEN='\033[0;32m'  # 绿色 - 成功/正常状态
YELLOW='\033[0;33m' # 黄色 - 警告/等待状态
BLUE='\033[0;34m'   # 蓝色 - 执行中/进度信息
NC='\033[0m'        # 无颜色 - 重置终端颜色

#######################################
# 函数:带重试机制的实例删除
# 参数:
#   $1 : 实例ID(必填)
# 返回值:
#   0 - 删除成功 | 1 - 删除失败
# 描述:
#   使用指数退避策略重试删除,最多尝试 delete_InstanceMAX_RETRY 次删除实例
#######################################
retry_delete() {
    local instance_id=$1
    for ((i=1; i<=delete_InstanceMAX_RETRY; i++)); do
        log "${BLUE}尝试删除实例 [ID: $instance_id] (第 $i 次尝试)...${NC}"
        if nova force-delete "$instance_id"; then
            log "${GREEN}删除实例成功: $instance_id${NC}"
            echo "$instance_id" >> "$DELETED_INSTANCES_FILE"
            return 0
        fi
        sleep $((i*2)) # 指数退避等待
    done
    log "${RED}错误:删除实例 $instance_id 失败,已达最大重试次数(${delete_InstanceMAX_RETRY} 次)${NC}"
    return 1
}

#######################################
# 函数:安全获取实例列表
# 参数:
#   $1 : 实例状态(默认 ACTIVE)
# 返回值:
#   实例ID列表(格式:每行一个ID)
# 描述:
#   过滤无效字符并格式化输出,避免脏数据干扰
#######################################
safe_get_instances() {
    local status=${1:-"ACTIVE"}
    nova list --status $status | awk -F '|' '
        /[0-9a-f]{8}-/ {
            gsub(/ /, "", $2);
            if ($4 ~ status) print $2
        }' status="$status"
}

#######################################
# 函数:日志记录
# 参数:
#   $1 : 日志信息(支持颜色代码)
# 描述:
#   自动添加时间戳,并同时输出到终端和日志文件
#######################################
log() {
    local msg="$1"
    echo -e "$(date '+%Y-%m-%d %T') - $msg" | tee -a "$LOG_FILE"
}

# 导出函数到子 Shell(确保并发调用时可用)
export -f log retry_delete

#######################################
# 主流程
#######################################

# 解析命令行参数
while getopts "s:i:c:l:" opt; do
    case $opt in
        s) STATUS=$OPTARG;;    # 指定目标实例状态(如 ACTIVE/BUILD)
        i) CHECK_INTERVAL=$OPTARG;;  # 自定义检测间隔时间
        c) CONCURRENCY=$OPTARG;;     # 调整并发线程数
        l) LOG_FILE=$OPTARG;;        # 指定日志文件路径
        *) echo -e "${RED}用法: $0 [-s 状态] [-i 间隔] [-c 并发数] [-l 日志路径]${NC}"
           exit 1;;
    esac
done

# 初始化 OpenStack 环境
source /openstack/admin-openrc.sh || { 
    log "${RED}错误:环境变量加载失败,请检查 admin-openrc.sh 文件权限或路径${NC}"
    exit 1
}

# 检查日志目录可写性
if [ ! -w "$(dirname "$LOG_FILE")" ]; then
    echo -e "${RED}错误:日志目录不可写 [路径: $(dirname "$LOG_FILE")]${NC}" >&2
    exit 1
fi

# 初始化临时文件(记录已删除实例)
> "$DELETED_INSTANCES_FILE"

# 进程锁检查(防止重复运行)
LOCK_FILE="/tmp/os_clean.lock"
if [ -e "$LOCK_FILE" ]; then
    log "${YELLOW}警告:检测到已有清理进程运行中 [PID: $(cat $LOCK_FILE)]${NC}"
    exit 1
fi
echo $$ > "$LOCK_FILE"
trap 'rm -f "$LOCK_FILE"; 
      TOTAL_DELETED_INSTANCES=$(wc -l < "$DELETED_INSTANCES_FILE"); 
      log "${GREEN}===== 脚本进程中断退出,已删除实例数: $TOTAL_DELETED_INSTANCES ====="; 
      rm -f "$DELETED_INSTANCES_FILE"' EXIT

# 安全确认(确保转义符被解析)
if [[ -t 0 ]]; then
    read -p "$(printf "${YELLOW}危险操作:是否确认执行删除实例?(y/N): ${NC}")" -n 1 confirm
    [[ $confirm != "y" ]] && { log "操作已取消"; exit; }
    echo
fi

# 主循环(状态检测与删除)
log "\n${GREEN}===== 开始删除实例 [时间: $(date +%T)] ====="
start_time=$(date +%s)
empty_count=0  # 连续空检测计数器

while :; do
    # 超时检查(防止无限等待)
    if (( $(date +%s) - start_time > MAX_WAIT_TIME )); then
        log "${RED}错误:操作超时,已超过最大等待时间(${MAX_WAIT_TIME} 秒)${NC}"
        exit 1
    fi

    # 获取目标实例列表
    instances=$(safe_get_instances "${STATUS:-ACTIVE}")
    creating_instances=$(safe_get_instances "BUILD")

    # 优先删除运行中的实例
    if [ -n "$instances" ]; then
        log "检测到 ${#instances[@]} 个运行中实例,启动并发删除..."
        printf "%s\n" "${instances[@]}" | xargs -P $CONCURRENCY -I {} bash -c "
            delete_InstanceMAX_RETRY='$delete_InstanceMAX_RETRY'
            LOG_FILE='$LOG_FILE'
            DELETED_INSTANCES_FILE='$DELETED_INSTANCES_FILE'
            $(declare -f log retry_delete)
            retry_delete \"{}\"
        "
        sleep $((CHECK_INTERVAL/2))  # 降低 API 请求频率
        continue
    fi

    # 处理创建中的实例(等待后重试)
    if [ -n "$creating_instances" ]; then
        log "${YELLOW}提示:检测到 ${#creating_instances[@]} 个创建中实例,等待 ${CHECK_INTERVAL} 秒...${NC}"
        sleep $CHECK_INTERVAL
        continue
    fi

    # 空状态处理(连续检测退出机制)
    if [ -z "$instances" ]; then
        log "${YELLOW}提示:未发现可操作实例,等待 ${CHECK_INTERVAL} 秒后重试,至多重试${no_InstanceMAX_RETRY_TIME}次...${NC}"
        sleep $CHECK_INTERVAL
        # 连续${no_InstanceMAX_RETRY_TIME} 次空检测则退出
        if (( empty_count++ >= ${no_InstanceMAX_RETRY_TIME} )); then
            TOTAL_DELETED_INSTANCES=$(wc -l < "$DELETED_INSTANCES_FILE")
            log "${GREEN}===== 脚本运行清理完成,已删除实例总数: ${TOTAL_DELETED_INSTANCES} ====="
            exit 0
        fi
        continue
    fi

    empty_count=0  # 重置空检测计数器
done
相关推荐
Ops菜鸟(Xu JieHao)19 分钟前
Linux Nginx网站服务【完整版】
linux·运维·服务器·nginx·网站
ikun·2 小时前
nginx 核心功能
linux·服务器·nginx
带娃的IT创业者2 小时前
《Python Web部署应知应会》Flask网站隐藏或改变浏览器URL:从Nginx反向代理到URL重写技术
前端·python·flask
Json____2 小时前
使用vue2 开发一个纯静态的校园二手交易平台-前端项目练习
前端·vue2·前端模板·vue脚手架·校园二手交易平台·项目项目练习
鱼嘻2 小时前
数据结构------C语言经典题目(6)
linux·c语言·开发语言·数据结构·算法
小二·2 小时前
前端技巧——性能优化篇
前端·性能优化
x-cmd3 小时前
[250430] Kali Linux 存储库密钥丢失导致所有用户无法正常更新 APT
linux·运维·服务器·kali linux
agenIT3 小时前
micro-app前端微服务原理解析
前端·微服务·架构
小宁爱Python3 小时前
深入理解CSS显示模式与盒子模型
前端·css
星雨流星天的笔记本3 小时前
3、Linux操作系统下,linux的技术手册使用(man)
linux·服务器