随机
Enter 搜索 ↑↓ 切换 Esc 清空

linux_dir_file_backup_old

脚本

Linux 目录与文件备份 - 带时间戳的压缩备份

linux_dir_file_backup_old

Linux 目录与文件备份 - 带时间戳的压缩备份

一键脚本

bash <(curl -sL gitee.com/meimolihan/cmdbox/raw/master/sh/linux_dir_file_backup_old.sh) /源目录 /备份目录
bash <(curl -sL gitee.com/meimolihan/cmdbox/raw/master/sh/linux_dir_file_backup_old.sh) /源目录 /备份目录 tar.gz
bash <(curl -sL gitee.com/meimolihan/cmdbox/raw/master/sh/linux_dir_file_backup_old.sh) /源目录 /备份目录 zip
功能模式 执行命令 作用说明
传参快速执行(2参数) ./backup.sh /源目录 /备份目录 直接指定路径,免交互批量备份全部项目
传参指定格式(3参数) ./backup.sh /源目录 /备份目录 tar.gz 额外指定压缩格式(tar.gz / zip)
交互模式 ./backup.sh 不传参,手动输入目录可视化备份
批量全备份 交互界面输入 666 一键备份当前目录所有文件/文件夹
实用示例 ./backup.sh /data/www /mnt/backup tar.gz 备份 /data/www 内项目到 /mnt/backup,格式为 tar.gz
备份规则 - 带时间戳,备份文件按时间命名不覆盖,保留多版本
支持格式 - 支持 tar.gz / zip 两种格式
日志记录 - 自动在备份目录的 logs/ 下生成日志文件

备份目录结构图

./backup.sh /mnt/data /mnt/backup

# 源目录(待备份)
/mnt/data
├── project/
├── config/
└── readme.txt

# 备份后(带时间戳·保留多版本)
/mnt/backup
├── project_20250911_103000.tar.gz
├── config_20250911_103000.tar.gz
├── readme.txt_20250911_103000.tar.gz
└── logs/
    └── 2025-09-11-10-30-00.log

效果预览

执行脚本效果预览

补充说明

该脚本用于交互式或参数化备份目录和文件,自动在备份文件名中添加时间戳(YYYYMMDD_HHMMSS),每次备份生成唯一文件,不会覆盖历史备份,适合需要保留多个历史版本的场景。

功能特点

实例使用方法

# 交互式操作(会提示输入工作目录和备份保存目录)
bash <(curl -sL gitee.com/meimolihan/cmdbox/raw/master/sh/linux_dir_file_backup_old.sh)

# 直接传参:工作目录 + 备份保存目录 + 指定 tar.gz 格式
bash <(curl -sL gitee.com/meimolihan/cmdbox/raw/master/sh/linux_dir_file_backup_old.sh) /vol1/1000/compose /vol2/1000/file/myfile/compose/downloads tar.gz

# 直接传参:工作目录 + 备份保存目录 + 指定 zip 格式
bash <(curl -sL gitee.com/meimolihan/cmdbox/raw/master/sh/linux_dir_file_backup_old.sh) /vol1/1000/compose /vol2/1000/file/myfile/compose/downloads zip

注意事项

脚本源码

#!/bin/bash
set -uo pipefail

list_color_init() {
    export gl_hui=$'\033[38;5;59m'
    export gl_hong=$'\033[38;5;9m'
    export gl_lv=$'\033[38;5;10m'
    export gl_huang=$'\033[38;5;11m'
    export gl_lan=$'\033[38;5;32m'
    export gl_bai=$'\033[38;5;15m'
    export gl_zi=$'\033[38;5;13m'
    export gl_bufan=$'\033[38;5;14m'
    export reset=$'\033[0m'
}
list_color_init

log_info()  { echo -e "${gl_lan}[信息]${gl_bai} $*"; }
log_ok()    { echo -e "${gl_lv}[成功]${gl_bai} $*"; }
log_warn()  { echo -e "${gl_huang}[警告]${gl_bai} $*"; }
log_error() { echo -e "${gl_hong}[错误]${gl_bai} $*" >&2; }

write_log() {
    local log_file="$1"
    local log_content="$2"
    echo -e "$log_content" >> "$log_file"
}

init_log_file() {
    local backup_dest_dir="$1"
    local logs_dir="${backup_dest_dir}/logs"
    mkdir -p "$logs_dir"
    local timestamp=$(date +"%Y-%m-%d-%H-%M-%S")
    local log_file="${logs_dir}/${timestamp}.log"
    touch "$log_file"
    echo "$log_file"
}

break_end() {
    echo -e "${gl_lv}操作完成${gl_bai}"
    echo -e "${gl_bai}按任意键继续 ${gl_hong}.${gl_huang}.${gl_lv}.${gl_bai} \c"
    read -r -n 1 -s
    echo ""
    clear
}

sleep_fractional() {
    local seconds=$1
    if sleep "$seconds" 2>/dev/null; then return 0; fi
    if command -v perl >/dev/null 2>&1; then perl -e "select(undef, undef, undef, $seconds)"; return 0; fi
    if command -v python3 >/dev/null 2>&1; then python3 -c "import time; time.sleep($seconds)"; return 0; fi
    if command -v python >/dev/null 2>&1; then python -c "import time; time.sleep($seconds)"; return 0; fi
    local int_seconds=$(awk -v s="$seconds" 'BEGIN{print int(s+0.999)}')
    sleep "$int_seconds"
}

exit_script() {
    echo ""
    echo -ne "${gl_hong}感谢使用,再见! ${gl_hong}.${gl_huang}.${gl_lv}.${gl_bai}\c"
    sleep_fractional 0.5
    echo -ne "${gl_hong}.${gl_huang}.${gl_lv}.${gl_bai}\c"
    sleep_fractional 0.6
    clear
    exit 0
}

exit_animation() {
    echo -ne "\r${gl_lv}即将退出 ${gl_hong}.${gl_huang}.${gl_lv}.${gl_bai}\c"
    sleep_fractional 0.5
    echo -ne "${gl_hong}.${gl_huang}.${gl_lv}.${gl_bai}\c"
    sleep_fractional 0.6
    echo ""
}

install() {
    [[ $# -eq 0 ]] && {
        log_error "未提供软件包参数!"
        return 1
    }
    local pkg mgr ver cmd_ver installed=false
    for pkg in "$@"; do
        installed=false
        ver=""
        if command -v "$pkg" &>/dev/null; then
            cmd_ver=$("$pkg" --version 2>/dev/null | head -n1 | tr -cd '[:print:]' | grep -oE '[0-9]+(\.[0-9]+)+' | head -n1 || echo "")
            [[ -n "$cmd_ver" ]] && ver="$cmd_ver"
            installed=true
        fi
        if [[ "$pkg" == "7zip" || "$pkg" == "7z" ]]; then
            if command -v 7z &>/dev/null; then
                ver=$(7z 2>&1 | grep -oE '[0-9]+(\.[0-9]+)+' | head -n1 || echo "")
                [[ -n "$ver" ]] && installed=true
            fi
        fi
        if [[ "$installed" == false ]]; then
            if command -v opkg &>/dev/null; then
                if opkg list-installed | grep -q "^${pkg} "; then
                    installed=true
                    ver=$(opkg list-installed | grep "^${pkg} " | awk '{print $3}' 2>/dev/null || echo "")
                fi
            elif command -v dpkg-query &>/dev/null; then
                if dpkg-query -W -f='${Status}' "$pkg" 2>/dev/null | grep -q "install ok installed"; then
                    installed=true
                    ver=$(dpkg-query -W -f='${Version}' "$pkg" 2>/dev/null || echo "")
                fi
            elif command -v rpm &>/dev/null; then
                if rpm -q "$pkg" &>/dev/null; then
                    installed=true
                    ver=$(rpm -q --qf '%{VERSION}' "$pkg" 2>/dev/null || echo "")
                fi
            elif command -v apk &>/dev/null; then
                if apk info "$pkg" 2>/dev/null | grep -q "^installed"; then
                    installed=true
                    ver=$(apk info -a "$pkg" 2>/dev/null | grep -oE '[0-9]+(\.[0-9]+)+' | head -n1 || echo "")
                fi
            elif command -v pacman &>/dev/null; then
                if pacman -Qi "$pkg" &>/dev/null; then
                    installed=true
                    ver=$(pacman -Qi "$pkg" 2>/dev/null | grep -i "version" | grep -oE '[0-9]+(\.[0-9]+)+' | head -n1 || echo "")
                fi
            fi
        fi
        if [[ "$installed" == true ]]; then
            echo -e "${gl_huang}${pkg}${gl_bai} ${gl_lv}已安装${gl_bai}" "$([[ -n "$ver" ]] && echo "版本 ${gl_lv}${ver}${gl_bai}")"
            continue
        fi
        echo -e ""
        echo -e "${gl_huang}开始安装:${gl_bai}${pkg}"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        local install_success=false
        for mgr in opkg dnf yum apt apk pacman zypper pkg; do
            if ! command -v "$mgr" &>/dev/null; then
                continue
            fi
            case $mgr in
            opkg)
                echo -e "${gl_bai}使用包管理器: ${gl_zi}opkg (OpenWrt/iStoreOS)${gl_bai}"
                if [[ "$pkg" == "7zip" || "$pkg" == "7z" ]]; then
                    echo -e "${gl_bai}正在安装: ${gl_lv}p7zip${gl_bai}"
                    opkg update && opkg install p7zip && install_success=true
                else
                    opkg update && opkg install "$pkg" && install_success=true
                fi
                ;;
            dnf)
                echo -e "${gl_bai}使用包管理器: ${gl_zi}dnf (Fedora/RHEL)${gl_bai}"
                dnf -y update && dnf install -y "$pkg" && install_success=true
                ;;
            yum)
                echo -e "${gl_bai}使用包管理器: ${gl_zi}yum (CentOS/RHEL)${gl_bai}"
                yum -y update && yum install -y "$pkg" && install_success=true
                ;;
            apt)
                echo -e "${gl_bai}使用包管理器: ${gl_zi}apt (Debian/Ubuntu)${gl_bai}"
                apt update -y && apt install -y "$pkg" && install_success=true
                ;;
            apk)
                echo -e "${gl_bai}使用包管理器: ${gl_zi}apk (Alpine)${gl_bai}"
                apk update && apk add "$pkg" && install_success=true
                ;;
            pacman)
                echo -e "${gl_bai}使用包管理器: ${gl_zi}pacman (Arch/Manjaro)${gl_bai}"
                pacman -Syu --noconfirm && pacman -S --noconfirm "$pkg" && install_success=true
                ;;
            zypper)
                echo -e "${gl_bai}使用包管理器: ${gl_zi}zypper (openSUSE)${gl_bai}"
                zypper refresh && zypper install -y "$pkg" && install_success=true
                ;;
            pkg)
                echo -e "${gl_bai}使用包管理器: ${gl_zi}pkg (FreeBSD)${gl_bai}"
                pkg update && pkg install -y "$pkg" && install_success=true
                ;;
            esac
            [[ "$install_success" == true ]] && break
        done
        if [[ "$install_success" == true ]]; then
            echo -e "${gl_lv}${pkg} 安装成功${gl_bai}"
        else
            echo -e "${gl_hong}${pkg} 安装失败${gl_bai}"
        fi
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
    done
}

display_horizontal_list() {
    local items=("$@")
    local cols=4
    local count=0
    local max_len=0
    for item in "${items[@]}"; do
        local fname=$(basename "$item" 2>/dev/null || echo "$item")
        [[ ${#fname} -gt "$max_len" ]] && max_len=${#fname}
    done
    ((max_len += 2))
    for item in "${items[@]}"; do
        local fname=$(basename "$item" 2>/dev/null || echo "$item")
        printf "${gl_bufan}%2d.${gl_bai} ${gl_lv}%-*s${gl_bai} " $((count + 1)) "$max_len" "$fname"
        count=$((count + 1))
        if (( count % cols == 0 )); then
            echo
        fi
    done
    (( count % cols != 0 )) && echo
}

backup_single_compose_project() {
    local target="$1"
    local format="$2"
    local temp_dir="$3"
    local dest_dir="$4"
    local log_file="$5"
    local backup_name="${target}_$(date +%Y%m%d_%H%M%S).${format}"
    local backup_path="${dest_dir}/${backup_name}"
    local source_abs_path=$(cd "$(dirname "$target")" && pwd)/$(basename "$target")
    echo -e ""
    echo -e "${gl_huang}>>> 开始备份:${gl_lv}${target}${gl_bai}"
    echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
    log_info "备份格式:${gl_lan}${format}${gl_bai}"
    log_info "备份路径:${gl_huang}${backup_path}${gl_bai}"
    case "$format" in
        tar.gz)
            tar -zcvf "$backup_path" -C "$(dirname "$target")" "$(basename "$target")" >/dev/null 2>&1
            ;;
        zip)
            zip -rq "$backup_path" "$target" >/dev/null 2>&1
            ;;
    esac
    if [[ -f "$backup_path" ]]; then
        log_ok "备份成功!文件:${gl_lv}${backup_path}${gl_bai}"
        write_log "$log_file" "项目:${target}"
        write_log "$log_file" "源路径:${source_abs_path}"
        write_log "$log_file" "备份文件:${backup_path}"
        write_log "$log_file" "状态:成功"
        write_log "$log_file" ""
    else
        log_error "备份失败!"
        write_log "$log_file" "项目:${target}"
        write_log "$log_file" "源路径:${source_abs_path}"
        write_log "$log_file" "状态:失败"
        write_log "$log_file" ""
    fi
}

backup_all_compose_projects() {
    local log_file="$1"
    local fmt="$2"
    local temp_dir="$3"
    local dest_dir="$4"
    shift 4
    local projects=("$@")
    echo -e ""
    echo -e "${gl_zi}开始批量备份所有项目,共计:${gl_lv}${#projects[@]} ${gl_zi}${gl_bai}"
    echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
    write_log "$log_file" "===== 批量备份任务开始 ====="
    write_log "$log_file" "备份总数:${#projects[@]} 个"
    write_log "$log_file" "备份格式:${fmt}"
    write_log "$log_file" "备份目录:${dest_dir}"
    write_log "$log_file" ""
    for project in "${projects[@]}"; do
        backup_single_compose_project "$project" "$fmt" "$temp_dir" "$dest_dir" "$log_file"
    done
    write_log "$log_file" "===== 批量备份任务结束 ====="
    log_ok "全部项目备份完成!"
}

get_backup_list() {
    local work_dir="$1"
    local -n out_list="$2"
    out_list=()
    local compress_extensions=("zip" "7z" "tar" "tar.gz" "tar.bz2" "tar.xz" "tgz" "tbz2" "txz" "rar" "gz" "bz2" "xz")
    cd "$work_dir" || return
    local item
    for item in *; do
        [[ "$item" == .* ]] && continue
        [[ ! -e "$item" ]] && continue
        local is_compressed=0
        for ext in "${compress_extensions[@]}"; do
            if [[ "$item" == *".$ext" ]]; then
                is_compressed=1
                break
            fi
        done
        ((is_compressed == 0)) && out_list+=("$item")
    done
    for item in */; do
        if [[ -d "$item" && "$item" != .*/ && "$item" != "./" ]]; then
            item="${item%/}"
            local exists=0
            for existing in "${out_list[@]}"; do
                [[ "$existing" == "$item" ]] && exists=1 && break
            done
            ((exists == 0)) && out_list+=("$item")
        fi
    done
    if ((${#out_list[@]} > 0)); then
        IFS=$'\n' out_list=($(sort <<<"${out_list[*]}"))
        unset IFS
    fi
}

linux_dir_file_backup_old() {
    local DEFAULT_WORK_DIR="$(pwd)"
    local DEFAULT_BACKUP_DIR="/mnt/backup"
    local BACKUP_TEMP_DIR="/tmp"
    local WORK_DIR
    local BACKUP_DEST_DIR
    local LOG_FILE=""
    local SCRIPT_START_TIME=$(date +"%Y-%m-%d %H:%M:%S")

    if [[ $# -eq 3 ]]; then
        WORK_DIR="$1"
        BACKUP_DEST_DIR="$2"
        local BACKUP_FMT="$3"
        if [[ "$BACKUP_FMT" != "tar.gz" && "$BACKUP_FMT" != "zip" ]]; then
            log_error "格式错误,仅支持 tar.gz / zip"
            exit 1
        fi
        if [[ ! -d "$WORK_DIR" ]]; then
            log_error "工作目录不存在:${WORK_DIR}"
            exit 1
        fi
        mkdir -p "$BACKUP_DEST_DIR"
        install zip unzip tar
        local -a file_list
        get_backup_list "$WORK_DIR" file_list
        if ((${#file_list[@]} == 0)); then
            log_warn "目录内无可备份文件/目录,退出"
            exit 0
        fi
        LOG_FILE=$(init_log_file "$BACKUP_DEST_DIR")
        write_log "$LOG_FILE" "===== 免交互批量备份 ====="
        write_log "$LOG_FILE" "执行时间:${SCRIPT_START_TIME}"
        write_log "$LOG_FILE" "源目录:${WORK_DIR}"
        write_log "$LOG_FILE" "备份目录:${BACKUP_DEST_DIR}"
        write_log "$LOG_FILE" ""
        cd "$WORK_DIR" || {
            log_error "无法进入工作目录:${WORK_DIR}"
            exit 1
        }
        backup_all_compose_projects "$LOG_FILE" "$BACKUP_FMT" "$BACKUP_TEMP_DIR" "$BACKUP_DEST_DIR" "${file_list[@]}"
        local SCRIPT_END_TIME=$(date +"%Y-%m-%d %H:%M:%S")
        write_log "$LOG_FILE" ""
        write_log "$LOG_FILE" "执行结束时间:${SCRIPT_END_TIME}"
        log_ok "日志已保存至:${LOG_FILE}"
        exit 0
    fi

    if [[ $# -ge 2 ]]; then
        WORK_DIR="$1"
        BACKUP_DEST_DIR="$2"
        log_info "使用命令行参数:工作目录=${WORK_DIR} 备份目录=${BACKUP_DEST_DIR}"
    else
        install zip unzip tar
        clear
        echo -e ""
        echo -e "${gl_zi}>>> 文件/目录 批量备份工具【${gl_huang}带时间戳${gl_zi}${gl_bai}"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        echo -e "${gl_huang}未传参默认备份当前目录文件${gl_bai}"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        read -r -e -p "$(echo -e "${gl_bai}请输入工作目录 [回车默认: ${gl_lv}${DEFAULT_WORK_DIR}${gl_bai}] (${gl_hong}0${gl_bai}退出):")" input_work
        WORK_DIR="${input_work:-$DEFAULT_WORK_DIR}"
        if [[ "$input_work" == "0" ]]; then
            exit_script
        fi
        read -r -e -p "$(echo -e "${gl_bai}请输入备份目录 [回车默认: ${gl_lv}${DEFAULT_BACKUP_DIR}${gl_bai}] (${gl_hong}0${gl_bai}退出):")" input_backup
        BACKUP_DEST_DIR="${input_backup:-$DEFAULT_BACKUP_DIR}"
        if [[ "$input_backup" == "0" ]]; then
            exit_script
        fi
        log_info "已配置:工作目录=${WORK_DIR} 备份目录=${BACKUP_DEST_DIR}"
        sleep_fractional 1
    fi

    if [[ ! -d "$WORK_DIR" ]]; then
        log_error "工作目录不存在:${WORK_DIR}"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        break_end
        exit 0
    fi
    mkdir -p "$BACKUP_DEST_DIR" || {
        log_error "无法创建备份目录:${BACKUP_DEST_DIR}"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        break_end
        exit 0
    }

    cd "$WORK_DIR" || {
        log_error "无法进入工作目录:${WORK_DIR}"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        break_end
        exit 0
    }

    while true; do
        install zip unzip tar
        clear
        echo -e "${gl_huang}>>> 当前${gl_bai}(${gl_lv}文件${gl_hong}/${gl_zi}目录${gl_bai})${gl_huang}列表:${gl_lv}$(pwd)${gl_bai}"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        local list=() item i choice target fmt_idx format
        local compress_extensions=("zip" "7z" "tar" "tar.gz" "tar.bz2" "tar.xz" "tgz" "tbz2" "txz" "rar" "gz" "bz2" "xz")
        for item in *; do
            [[ "$item" == .* ]] && continue
            [[ ! -e "$item" ]] && continue
            local is_compressed=0
            for ext in "${compress_extensions[@]}"; do
                if [[ "$item" == *".$ext" ]]; then
                    is_compressed=1
                    break
                fi
            done
            ((is_compressed == 0)) && list+=("$item")
        done
        for item in */; do
            if [[ -d "$item" && "$item" != .*/ && "$item" != "./" ]]; then
                item="${item%/}"
                local exists=0
                for existing in "${list[@]}"; do
                    [[ "$existing" == "$item" ]] && exists=1 && break
                done
                ((exists == 0)) && list+=("$item")
            fi
        done
        if ((${#list[@]} > 0)); then
            IFS=$'\n' list=($(sort <<<"${list[*]}"))
            unset IFS
        fi
        if ((${#list[@]} == 0)); then
            echo -e "${gl_huang}当前目录无可压缩的文件/目录${gl_bai}"
            echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
            break_end
            exit 0
        fi
        display_horizontal_list "${list[@]}"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        echo -e ""
        echo -e "${gl_zi}>>> 文件/目录 批量备份工具【${gl_huang}带时间戳${gl_zi}${gl_bai}"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        echo -e "${gl_bai}临时目录: ${gl_lv}$BACKUP_TEMP_DIR${gl_bai}"
        echo -e "${gl_bai}工作目录: ${gl_lv}$WORK_DIR${gl_bai}"
        echo -e "${gl_bai}备份目标: ${gl_lv}$BACKUP_DEST_DIR${gl_bai}"
        echo -e "${gl_huang}温馨提示: 输入序号备份单个项目,输入 ${gl_hong}666${gl_huang} 备份全部项目${gl_bai}"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        read -r -e -p "$(echo -e "${gl_bai}请输入序号选择,或手动输入${gl_bai}(${gl_lv}文件${gl_hong}/${gl_zi}目录${gl_bai}) (${gl_hong}0${gl_bai}退出): ")" choice
        if [[ -z "$choice" ]]; then
            echo -ne "${gl_hong}输入为空请重新输入 ${gl_hong}.${gl_huang}.${gl_lv}.${gl_bai}\c"
            sleep_fractional 0.5
            echo -ne "${gl_hong}.${gl_huang}.${gl_lv}.${gl_bai}\c"
            sleep_fractional 0.6
            continue
        fi
        if [[ "$choice" == "0" ]]; then
            exit_script
        fi
        LOG_FILE=$(init_log_file "$BACKUP_DEST_DIR")
        if [[ "$choice" == "666" ]]; then
            echo -e ""
            echo -e "${gl_huang}请选择备份格式:${gl_bai}"
            echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
            echo -e "${gl_huang}1.${gl_bai} tar.gz (推荐)   ${gl_huang}2.${gl_bai} zip (通用)"
            echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
            read -r -e -p "请输入你的选择:" fmt_idx
            case "$fmt_idx" in
                1) format="tar.gz" ;;
                2) format="zip" ;;
                *)
                    echo -e "${gl_hong}无效序号!${gl_bai}"
                    read -r -n1 -s
                    continue
                    ;;
            esac
            backup_all_compose_projects "$LOG_FILE" "$format" "$BACKUP_TEMP_DIR" "$BACKUP_DEST_DIR" "${list[@]}"
            echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
            local SCRIPT_END_TIME=$(date +"%Y-%m-%d %H:%M:%S")
            write_log "$LOG_FILE" ""
            write_log "$LOG_FILE" "交互批量备份结束时间:${SCRIPT_END_TIME}"
            log_ok "日志已保存至:${LOG_FILE}"
            break_end
            continue
        fi
        if [[ "$choice" =~ ^[0-9]+$ ]] && ((choice >= 1 && choice <= ${#list[@]})); then
            target="${list[$((choice - 1))]}"
        else
            target="$choice"
        fi
        [[ -e "$target" ]] || {
            echo -e ""
            echo -e "${gl_hong}错误:'$target' 不存在!${gl_bai}"
            echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
            exit_animation
            continue
        }
        echo -e ""
        echo -e "${gl_huang}请选择备份格式:${gl_bai}"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        echo -e "${gl_huang}1.${gl_bai} tar.gz (推荐)   ${gl_huang}2.${gl_bai} zip (通用)"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        read -r -e -p "请输入你的选择:" fmt_idx
        case "$fmt_idx" in
        1) format="tar.gz" ;;
        2) format="zip" ;;
        *)
            echo -e "${gl_hong}无效序号!${gl_bai}"
            echo -e "${gl_bai}按任意键继续 ${gl_hong}.${gl_huang}.${gl_lv}.${gl_bai}"
            read -r -n1 -s
            continue
            ;;
        esac
        backup_single_compose_project "$target" "$format" "$BACKUP_TEMP_DIR" "$BACKUP_DEST_DIR" "$LOG_FILE"
        echo -e "${gl_bufan}————————————————————————————————————————————————${gl_bai}"
        local SCRIPT_END_TIME=$(date +"%Y-%m-%d %H:%M:%S")
        write_log "$LOG_FILE" ""
        write_log "$LOG_FILE" "单文件备份结束时间:${SCRIPT_END_TIME}"
        log_ok "日志已保存至:${LOG_FILE}"
        break_end
    done
}

linux_dir_file_backup_old "$@"

创建本地脚本

new_script="new_test.sh"

cat > "$new_script" << 'EOF'
#!/bin/bash

# 粘贴脚本源码

EOF

# 保留本地脚本,去掉 rm -f "$new_script"
chmod +x "$new_script" && ./"$new_script" && rm -f "$new_script"

相关命令