751 lines
27 KiB
Bash
Executable File
751 lines
27 KiB
Bash
Executable File
#!/usr/bin/env bash
|
||
|
||
#====================================================
|
||
# OpenWrt 构建管理脚本
|
||
# 功能:提供交互式菜单,可选择执行特定操作
|
||
# 更新:2025-04-05
|
||
#====================================================
|
||
|
||
#---------------基础配置---------------#
|
||
# 日志目录设置
|
||
LOGS_ROOT="openwrt_logs"
|
||
SESSION_DIR="${LOGS_ROOT}/$(date +%Y%m%d_%H%M%S)"
|
||
LOG_FILE="${SESSION_DIR}/main_build_script.log" # 主脚本日志文件名更改,更清晰
|
||
|
||
#---------------清理配置---------------#
|
||
# 需要移除的目录列表 (示例,根据你的需要添加)
|
||
declare -a REMOVE_DIRS=(
|
||
# "feeds/packages/net/v2ray-geodata" # 示例
|
||
)
|
||
|
||
# 需要移除的包列表 (示例,根据你的需要添加)
|
||
declare -a REMOVE_PACKAGES=(
|
||
# "package/feeds/luci/luci-app-example" # 示例
|
||
)
|
||
|
||
#---------------自定义包配置---------------#
|
||
# 定义自定义包配置,格式:[包名]=[目标目录]=[仓库URL]=[分支]=[是否使用--depth 1]
|
||
declare -A CUSTOM_PACKAGES=(
|
||
["partexp"]="package/luci-app-partexp=https://github.com/sirpdboy/luci-app-partexp=main=false"
|
||
["openclash"]="package/openclash=https://github.com/vernesong/OpenClash=dev=true"
|
||
["zerotier"]="package/luci-app-zerotier=https://github.com/Firsgith/luci-app-zerotier.git=main=false"
|
||
["5gsupport"]="package/5G-Modem-Support=https://nanako.site/gitea/Nanako/openwrt-app-actions.git=main=false"
|
||
["pcat"]="package/pcat-manager=https://github.com/photonicat/rockchip_rk3568_pcat_manager.git=master=true"
|
||
["adguardhome"]="package/adguardhome=https://github.com/xiaoxiao29/luci-app-adguardhome.git=master=false"
|
||
)
|
||
|
||
#---------------日志系统---------------#
|
||
# 保存原始的标准输出和错误输出文件描述符
|
||
exec {STDOUT_ORIG}>&1
|
||
exec {STDERR_ORIG}>&2
|
||
|
||
# 全局变量标记日志是否启用
|
||
LOGGING_ENABLED=false # 初始为 false,由 init_logger 启用
|
||
|
||
# 初始化日志系统
|
||
init_logger() {
|
||
# 创建日志目录结构
|
||
mkdir -p "$SESSION_DIR" || {
|
||
echo "错误: 无法创建日志目录: $SESSION_DIR" >&2
|
||
exit 1
|
||
}
|
||
|
||
LOGGING_ENABLED=true
|
||
echo "日志将保存在: $SESSION_DIR"
|
||
echo "主脚本日志: $LOG_FILE"
|
||
|
||
# 重定向主脚本的 stdout 和 stderr 到 tee,同时输出到控制台和主日志文件
|
||
# 使用 process substitution 和 tee
|
||
exec > >(tee -a "$LOG_FILE")
|
||
exec 2> >(tee -a "$LOG_FILE" >&2)
|
||
|
||
# 记录初始信息到日志
|
||
echo "[$(date "+%Y-%m-%d %H:%M:%S")] [INFO]: 日志系统初始化完成。主日志文件: $LOG_FILE"
|
||
}
|
||
|
||
# 创建新的特定构建阶段的日志文件
|
||
create_build_log() {
|
||
local build_type=$1
|
||
local timestamp=$(date +%Y%m%d_%H%M%S)
|
||
# 使用更明确的文件名
|
||
BUILD_LOG_FILE="${SESSION_DIR}/${build_type}_build_${timestamp}.log"
|
||
# 确保文件被创建,即使命令没有输出
|
||
touch "$BUILD_LOG_FILE" || {
|
||
log "ERROR" "无法创建构建日志文件: $BUILD_LOG_FILE"
|
||
return 1
|
||
}
|
||
# 返回创建的日志文件名
|
||
echo "$BUILD_LOG_FILE"
|
||
return 0
|
||
}
|
||
|
||
# 禁用日志重定向 (用于 menuconfig 等交互场景)
|
||
disable_logging() {
|
||
if [ "$LOGGING_ENABLED" = true ]; then
|
||
log "DEBUG" "禁用日志重定向,恢复原始标准输出/错误。"
|
||
exec >&$STDOUT_ORIG
|
||
exec 2>&$STDERR_ORIG
|
||
LOGGING_ENABLED=false # 标记为禁用
|
||
fi
|
||
}
|
||
|
||
# 启用日志重定向 (恢复)
|
||
enable_logging() {
|
||
# 仅在之前被禁用的情况下才重新启用
|
||
if [ "$LOGGING_ENABLED" = false ]; then
|
||
# 重新应用重定向到主日志
|
||
exec > >(tee -a "$LOG_FILE")
|
||
exec 2> >(tee -a "$LOG_FILE" >&2)
|
||
LOGGING_ENABLED=true # 标记为启用
|
||
log "DEBUG" "重新启用日志重定向到主日志。"
|
||
fi
|
||
}
|
||
|
||
# 设置终端颜色
|
||
setup_colors() {
|
||
if [[ -t 1 ]]; then # 检查标准输出是否连接到终端
|
||
RED='\033[0;31m'
|
||
GREEN='\033[0;32m'
|
||
YELLOW='\033[1;33m'
|
||
BLUE='\033[0;34m'
|
||
NC='\033[0m' # No Color
|
||
else
|
||
RED='' GREEN='' YELLOW='' BLUE='' NC=''
|
||
fi
|
||
}
|
||
|
||
# 记录日志函数 - 使用此函数来明确日志级别
|
||
# 注意:此函数输出会经过 init_logger 设置的全局重定向
|
||
log() {
|
||
local level=$1
|
||
local message=$2
|
||
local timestamp
|
||
timestamp=$(date "+%Y-%m-%d %H:%M:%S")
|
||
local color="" level_str=""
|
||
|
||
case $level in
|
||
"INFO") color="$GREEN"; level_str="INFO " ;; # 加空格对齐
|
||
"WARN") color="$YELLOW"; level_str="WARN " ;;
|
||
"ERROR") color="$RED"; level_str="ERROR" ;;
|
||
"DEBUG") color="$BLUE"; level_str="DEBUG" ;;
|
||
*) color=""; level_str="LOG " ;; # 默认级别
|
||
esac
|
||
|
||
# 使用 printf 保证原子写入,减少并发问题
|
||
printf "[%s] %b[%s]%b: %s\n" "$timestamp" "$color" "$level_str" "$NC" "$message"
|
||
}
|
||
|
||
# 同时记录消息到主日志和特定的构建日志
|
||
log_to_both() {
|
||
local level=$1
|
||
local message=$2
|
||
local build_log_file=$3 # 明确参数名
|
||
|
||
# 1. 输出到主日志(通过全局重定向和 log 函数)
|
||
log "$level" "$message"
|
||
|
||
# 2. 追加到指定的构建日志文件
|
||
local timestamp
|
||
timestamp=$(date "+%Y-%m-%d %H:%M:%S")
|
||
# 直接追加,不需要颜色代码
|
||
printf "[%s] [%s]: %s\n" "$timestamp" "$level" "$message" >> "$build_log_file"
|
||
}
|
||
|
||
|
||
#---------------自定义包管理函数---------------#
|
||
# 更新单个自定义包
|
||
update_single_package() {
|
||
local pkg_name=$1
|
||
local config=$2
|
||
|
||
# 解析配置 (保持 cut 的方式,假设路径/URL/分支名不含 '=')
|
||
# 如果遇到问题,可以切换到 IFS/read
|
||
local target_dir repo_url branch use_depth
|
||
target_dir=$(echo "$config" | cut -d= -f1)
|
||
repo_url=$(echo "$config" | cut -d= -f2)
|
||
branch=$(echo "$config" | cut -d= -f3)
|
||
use_depth=$(echo "$config" | cut -d= -f4)
|
||
|
||
# 基础验证
|
||
if [ -z "$target_dir" ] || [ -z "$repo_url" ]; then
|
||
log "ERROR" "包 '$pkg_name' 的配置无效: $config"
|
||
return 1
|
||
fi
|
||
|
||
log "INFO" "更新自定义包: $pkg_name..."
|
||
|
||
# 检查目标目录的父目录是否存在
|
||
local parent_dir
|
||
parent_dir=$(dirname "$target_dir")
|
||
if [ ! -d "$parent_dir" ]; then
|
||
log "INFO" "创建父目录: $parent_dir"
|
||
mkdir -p "$parent_dir" || {
|
||
log "ERROR" "无法创建父目录: $parent_dir"
|
||
return 1
|
||
}
|
||
fi
|
||
|
||
# 移除现有目录
|
||
if [ -d "$target_dir" ]; then
|
||
log "INFO" "移除现有目录: $target_dir"
|
||
rm -rf "$target_dir" || {
|
||
log "ERROR" "无法删除目录: $target_dir"
|
||
return 1
|
||
}
|
||
fi
|
||
|
||
# 准备 git clone 命令参数数组 (移除 eval)
|
||
local git_options=("--progress") # 总是显示进度
|
||
# 处理分支
|
||
if [ -n "$branch" ] && [ "$branch" != "master" ]; then
|
||
git_options+=("-b" "$branch")
|
||
fi
|
||
# 处理 depth
|
||
if [ "$use_depth" = "true" ]; then
|
||
git_options+=("--depth" "1")
|
||
fi
|
||
|
||
# 执行克隆
|
||
log "INFO" "执行: git clone ${git_options[*]} $repo_url $target_dir"
|
||
if git clone "${git_options[@]}" "$repo_url" "$target_dir"; then
|
||
log "INFO" "包 '$pkg_name' 更新成功"
|
||
return 0
|
||
else
|
||
log "ERROR" "克隆包 '$pkg_name' 失败 (仓库: $repo_url)"
|
||
# 尝试给出更具体的错误提示
|
||
log "ERROR" "请检查网络连接、仓库 URL、分支名称以及目标目录权限。"
|
||
# 保留可能存在的空目录或部分克隆,让用户检查
|
||
# rm -rf "$target_dir" # 不再自动删除失败的克隆
|
||
return 1
|
||
fi
|
||
}
|
||
|
||
# 更新所有自定义包的主函数
|
||
update_custom_package() {
|
||
local failed_packages=()
|
||
local success_count=0
|
||
local total_packages=${#CUSTOM_PACKAGES[@]}
|
||
|
||
if [ "$total_packages" -eq 0 ]; then
|
||
log "INFO" "没有配置自定义包,跳过更新。"
|
||
return 0
|
||
fi
|
||
|
||
log "INFO" "开始更新 $total_packages 个自定义包..."
|
||
|
||
for pkg_name in "${!CUSTOM_PACKAGES[@]}"; do
|
||
# 使用 subshell 执行,避免污染当前环境的变量,但这里影响不大
|
||
if update_single_package "$pkg_name" "${CUSTOM_PACKAGES[$pkg_name]}"; then
|
||
((success_count++))
|
||
else
|
||
failed_packages+=("$pkg_name")
|
||
fi
|
||
done
|
||
|
||
# 显示更新结果
|
||
if [ ${#failed_packages[@]} -eq 0 ]; then
|
||
log "INFO" "所有自定义包更新成功 ($success_count/$total_packages)"
|
||
return 0
|
||
else
|
||
log "WARN" "部分自定义包更新失败 (成功: $success_count/$total_packages)"
|
||
log "ERROR" "更新失败的包: ${failed_packages[*]}"
|
||
return 1 # 返回失败状态码
|
||
fi
|
||
}
|
||
|
||
# 更新特定自定义包
|
||
update_specific_package() {
|
||
local pkg_name=$1
|
||
|
||
if [ -z "$pkg_name" ]; then
|
||
log "ERROR" "未指定包名"
|
||
return 1
|
||
fi
|
||
|
||
if [ -z "${CUSTOM_PACKAGES[$pkg_name]}" ]; then
|
||
log "ERROR" "未找到包配置:'$pkg_name'"
|
||
log "INFO" "可用的包名: ${!CUSTOM_PACKAGES[*]}"
|
||
return 1
|
||
fi
|
||
|
||
update_single_package "$pkg_name" "${CUSTOM_PACKAGES[$pkg_name]}"
|
||
}
|
||
|
||
#---------------基础功能函数---------------#
|
||
# 清理工作区 (make clean 的替代方案,更彻底)
|
||
clean_workspace() {
|
||
log "INFO" "开始彻底清理工作区..."
|
||
|
||
# 清理 feeds (可选,如果 feeds 内容经常变动或出问题可以启用)
|
||
# log "INFO" "执行 ./scripts/feeds clean ..."
|
||
# ./scripts/feeds clean || log "WARN" "./scripts/feeds clean 执行时遇到问题。"
|
||
|
||
# 清理编译产物和临时文件
|
||
log "INFO" "执行 rm -rf ./tmp ./logs ./staging_dir ./build_dir ./bin ./dl ..."
|
||
# 注意:这里移除了 logs 目录,但我们的脚本日志在 openwrt_logs 下,不会被删
|
||
# 如果 OpenWrt 自身有 logs 目录且需要保留,则从下面移除 ./logs
|
||
rm -rf ./tmp ./logs ./staging_dir ./build_dir ./bin ./dl || {
|
||
log "ERROR" "删除部分工作区目录失败,请检查权限或文件占用。"
|
||
return 1
|
||
}
|
||
|
||
log "INFO" "工作区清理完成。"
|
||
return 0
|
||
}
|
||
|
||
# 清理编译临时文件 (make clean 的快速版)
|
||
clean_build_temp() {
|
||
log "INFO" "开始清理编译临时文件..."
|
||
|
||
log "INFO" "执行 rm -rf ./tmp ./staging_dir ./build_dir ..."
|
||
rm -rf ./tmp ./staging_dir ./build_dir || {
|
||
log "ERROR" "删除部分编译临时目录失败。"
|
||
return 1
|
||
}
|
||
|
||
log "INFO" "编译临时文件清理完成。"
|
||
return 0
|
||
}
|
||
|
||
# 更新源码
|
||
update_source() {
|
||
log "INFO" "开始更新 OpenWrt 源码 (git pull)..."
|
||
if git pull --rebase; then # 使用 rebase 避免不必要的 merge commit
|
||
log "INFO" "源码更新完成"
|
||
return 0
|
||
else
|
||
log "ERROR" "源码更新失败 (git pull --rebase)"
|
||
log "WARN" "请检查本地是否有未提交的修改,或尝试手动解决 Git 冲突。"
|
||
return 1
|
||
fi
|
||
}
|
||
|
||
# 更新并安装 feeds
|
||
update_feeds() {
|
||
log "INFO" "开始更新 Feeds (update -a)..."
|
||
if ./scripts/feeds update -a; then
|
||
log "INFO" "Feeds 更新成功,开始安装 (install -a)..."
|
||
if ./scripts/feeds install -a; then
|
||
log "INFO" "Feeds 安装成功"
|
||
return 0
|
||
else
|
||
log "ERROR" "Feeds 安装失败 (install -a)"
|
||
return 1
|
||
fi
|
||
else
|
||
log "ERROR" "Feeds 更新失败 (update -a)"
|
||
return 1
|
||
fi
|
||
}
|
||
|
||
# 执行构建相关命令,并将 stdout 重定向到日志,stderr 同时输出到控制台和日志
|
||
run_build_command() {
|
||
local cmd=$1 # 要执行的命令字符串
|
||
local log_file=$2 # 目标日志文件
|
||
local description=$3 # 操作描述
|
||
|
||
log_to_both "INFO" "开始执行: $description" "$log_file"
|
||
log "DEBUG" "命令详情: $cmd" # 只记录到主日志
|
||
|
||
# 执行命令
|
||
# stdout (>>) 追加到日志文件
|
||
# stderr (2>) 通过管道给 tee
|
||
# tee -a "$log_file" 将 stderr 追加到日志文件
|
||
# tee 默认将其标准输入复制到标准输出,这里即脚本的 stderr,因此会显示在控制台
|
||
local start_time end_time duration
|
||
start_time=$(date +%s)
|
||
if eval "$cmd" >> "$log_file" 2> >(tee -a "$log_file" >&2); then
|
||
end_time=$(date +%s)
|
||
duration=$((end_time - start_time))
|
||
log_to_both "INFO" "$description 完成 (耗时: ${duration}s)" "$log_file"
|
||
return 0 # 成功返回 0
|
||
else
|
||
end_time=$(date +%s)
|
||
duration=$((end_time - start_time))
|
||
local exit_code=$? # 获取命令的退出码
|
||
# 错误信息已经通过 stderr 的 tee 输出到控制台了
|
||
log_to_both "ERROR" "$description 失败 (退出码: $exit_code, 耗时: ${duration}s)" "$log_file"
|
||
log "ERROR" "详细错误请查看控制台输出以及日志文件: $log_file"
|
||
return $exit_code # 失败返回命令的退出码
|
||
fi
|
||
}
|
||
|
||
# 编译固件
|
||
build_firmware() {
|
||
# 创建新的固件编译日志文件
|
||
local current_log
|
||
current_log=$(create_build_log "firmware") || return 1 # 创建失败则退出
|
||
|
||
log "INFO" "开始编译固件..."
|
||
log "INFO" "详细编译日志将保存在: $current_log"
|
||
|
||
# 1. 下载依赖包
|
||
local download_cmd="make download -j$(nproc) V=s"
|
||
run_build_command "$download_cmd" "$current_log" "下载依赖包" || {
|
||
log "ERROR" "依赖包下载失败,编译中止。"
|
||
# 即使下载失败,日志文件也已包含错误信息
|
||
return 1
|
||
}
|
||
|
||
# 2. 多线程编译
|
||
local cpu_cores
|
||
cpu_cores=$(nproc)
|
||
log "INFO" "检测到 $cpu_cores 个 CPU 核心,开始多线程编译..."
|
||
local build_cmd="make -j${cpu_cores} V=s"
|
||
|
||
if run_build_command "$build_cmd" "$current_log" "固件编译 (多线程)"; then
|
||
log "INFO" "${GREEN}固件编译成功完成!${NC}"
|
||
log "INFO" "固件输出目录: ./bin/targets/"
|
||
return 0 # 编译成功
|
||
else
|
||
# 多线程编译失败,询问用户是否重试
|
||
log "WARN" "多线程编译失败。"
|
||
local retry_choice
|
||
# -r: raw input, -p: prompt, -n 1: read only one char (optional but nice), -t 15: timeout (optional)
|
||
# Default to No
|
||
read -rp "$(echo -e ${YELLOW}"是否尝试使用单线程重新编译? (y/N): "${NC})" -n 1 -t 30 retry_choice
|
||
echo # Add a newline after read -n
|
||
|
||
if [[ "$retry_choice" =~ ^[Yy]$ ]]; then
|
||
log "INFO" "用户选择使用单线程重试编译..."
|
||
|
||
# 3. 单线程编译重试
|
||
local build_cmd_single="make -j1 V=s"
|
||
if run_build_command "$build_cmd_single" "$current_log" "固件编译 (单线程重试)"; then
|
||
log "INFO" "${GREEN}单线程编译成功完成!${NC}"
|
||
log "INFO" "固件输出目录: ./bin/targets/"
|
||
return 0 # 单线程编译成功
|
||
else
|
||
log "ERROR" "${RED}单线程编译也失败了。${NC}"
|
||
log "ERROR" "请仔细检查控制台错误输出以及日志文件: $current_log"
|
||
return 1 # 最终编译失败
|
||
fi
|
||
else
|
||
log "INFO" "用户选择不进行单线程重试编译。"
|
||
log "ERROR" "${RED}编译失败,未进行单线程重试。${NC}"
|
||
log "ERROR" "请仔细检查控制台错误输出以及日志文件: $current_log"
|
||
return 1 # 用户取消重试,编译失败
|
||
fi
|
||
fi
|
||
}
|
||
|
||
# 清理配置文件中指定的旧包/目录
|
||
clean_packages() {
|
||
log "INFO" "开始清理配置文件中指定的旧包和目录..."
|
||
local removed_count=0
|
||
local failed_count=0
|
||
local total_to_remove=$((${#REMOVE_DIRS[@]} + ${#REMOVE_PACKAGES[@]}))
|
||
|
||
if [ "$total_to_remove" -eq 0 ]; then
|
||
log "INFO" "配置文件中未指定需要移除的包或目录。"
|
||
return 0
|
||
fi
|
||
|
||
# 清理目录
|
||
for dir in "${REMOVE_DIRS[@]}"; do
|
||
if [ -e "$dir" ]; then # 使用 -e 检查文件或目录是否存在
|
||
log "INFO" "正在删除目录: $dir"
|
||
if rm -rf "$dir"; then
|
||
log "INFO" "已删除: $dir"
|
||
((removed_count++))
|
||
else
|
||
log "ERROR" "删除失败: $dir (请检查权限或是否被占用)"
|
||
((failed_count++))
|
||
fi
|
||
else
|
||
log "WARN" "指定的目录不存在,跳过: $dir"
|
||
fi
|
||
done
|
||
|
||
# 清理包 (通常包也是目录)
|
||
for pkg in "${REMOVE_PACKAGES[@]}"; do
|
||
if [ -e "$pkg" ]; then
|
||
log "INFO" "正在删除包/目录: $pkg"
|
||
if rm -rf "$pkg"; then
|
||
log "INFO" "已删除: $pkg"
|
||
((removed_count++))
|
||
else
|
||
log "ERROR" "删除失败: $pkg (请检查权限或是否被占用)"
|
||
((failed_count++))
|
||
fi
|
||
else
|
||
log "WARN" "指定的包/目录不存在,跳过: $pkg"
|
||
fi
|
||
done
|
||
|
||
if [ $failed_count -eq 0 ]; then
|
||
log "INFO" "旧包/目录清理完成,共处理 $removed_count 个项目。"
|
||
return 0
|
||
else
|
||
log "WARN" "旧包/目录清理部分完成 (成功: $removed_count, 失败: $failed_count)"
|
||
return 1 # 返回失败状态码
|
||
fi
|
||
}
|
||
|
||
# 更新所有组件 (源码 -> 清理旧包 -> 自定义包 -> feeds)
|
||
update_all_components() {
|
||
log "INFO" "开始更新所有组件..."
|
||
local success=true # 假设成功
|
||
|
||
# 1. 更新源码
|
||
log "INFO" "[1/4] 更新 OpenWrt 源码..."
|
||
update_source || success=false
|
||
|
||
# 2. 清理旧包 (即使上一步失败也尝试清理)
|
||
log "INFO" "[2/4] 清理配置文件指定的旧包/目录..."
|
||
clean_packages || log "WARN" "清理旧包/目录时遇到问题,继续执行..." # 清理失败通常不阻止后续
|
||
|
||
# 3. 更新自定义包 (只有在源码更新成功时才执行较好)
|
||
if [ "$success" = true ]; then
|
||
log "INFO" "[3/4] 更新自定义包..."
|
||
update_custom_package || success=false
|
||
else
|
||
log "WARN" "[3/4] 跳过更新自定义包,因为源码更新失败。"
|
||
fi
|
||
|
||
# 4. 更新 Feeds (只有在源码和自定义包都成功时执行较好)
|
||
if [ "$success" = true ]; then
|
||
log "INFO" "[4/4] 更新 Feeds..."
|
||
update_feeds || success=false
|
||
else
|
||
log "WARN" "[4/4] 跳过更新 Feeds,因为之前的步骤失败。"
|
||
fi
|
||
|
||
# 总结
|
||
if [ "$success" = true ]; then
|
||
log "INFO" "所有组件更新完成!"
|
||
return 0
|
||
else
|
||
log "ERROR" "部分或全部组件更新失败,请检查上面的日志。"
|
||
return 1
|
||
fi
|
||
}
|
||
|
||
# 菜单配置 (make menuconfig)
|
||
menuconfig() {
|
||
log "INFO" "准备进入菜单配置 (make menuconfig)..."
|
||
log "INFO" "注意:日志重定向将暂时禁用以进行交互。"
|
||
|
||
# 禁用日志重定向,否则 menuconfig 界面可能显示不正常
|
||
disable_logging
|
||
|
||
# 执行 menuconfig
|
||
if make menuconfig; then
|
||
# 配置成功后重新启用日志
|
||
enable_logging
|
||
log "INFO" "菜单配置完成。新配置已保存到 .config"
|
||
# 询问是否保存 defconfig (可选)
|
||
# read -p "是否将当前配置保存为 defconfig? (y/N): " save_defconfig
|
||
# if [[ "$save_defconfig" =~ ^[Yy]$ ]]; then
|
||
# ./scripts/diffconfig.sh > my_defconfig
|
||
# log "INFO" "当前配置已保存到 my_defconfig"
|
||
# fi
|
||
return 0
|
||
else
|
||
# 配置失败或被取消,也要恢复日志
|
||
enable_logging
|
||
log "WARN" "菜单配置未完成或被取消。"
|
||
return 1 # 返回失败状态码
|
||
fi
|
||
}
|
||
|
||
#---------------组合功能函数---------------#
|
||
# 完整构建流程
|
||
full_build() {
|
||
log "INFO" "启动完整构建流程..."
|
||
local step=1
|
||
local total_steps=6 # 预估步骤数
|
||
|
||
# 1. 清理工作区
|
||
log "INFO" "[${step}/${total_steps}] 清理工作区..."
|
||
clean_workspace || { log "WARN" "工作区清理可能不完整,尝试继续..."; } # 清理失败通常不致命
|
||
((step++))
|
||
|
||
# 2. 更新源码
|
||
log "INFO" "[${step}/${total_steps}] 更新源码..."
|
||
update_source || { log "ERROR" "源码更新失败,终止构建流程。"; return 1; }
|
||
((step++))
|
||
|
||
# 3. 清理旧包
|
||
log "INFO" "[${step}/${total_steps}] 清理配置文件指定的旧包..."
|
||
clean_packages || log "WARN" "清理旧包时遇到问题,继续执行..."
|
||
((step++))
|
||
|
||
# 4. 更新自定义包
|
||
log "INFO" "[${step}/${total_steps}] 更新自定义包..."
|
||
update_custom_package || { log "ERROR" "自定义包更新失败,终止构建流程。"; return 1; }
|
||
((step++))
|
||
|
||
# 5. 更新 Feeds
|
||
log "INFO" "[${step}/${total_steps}] 更新 Feeds..."
|
||
update_feeds || { log "ERROR" "Feeds 更新失败,终止构建流程。"; return 1; }
|
||
((step++))
|
||
|
||
# 6. 菜单配置
|
||
log "INFO" "[${step}/${total_steps}] 进入菜单配置 (make menuconfig)..."
|
||
menuconfig || { log "WARN" "菜单配置未完成或取消,但继续尝试编译..."; } # 允许使用现有配置编译
|
||
((step++)) # 无论成功与否,都算一步
|
||
|
||
# 7. 编译固件 (独立步骤,不计入前面的 total_steps)
|
||
log "INFO" "所有准备工作完成,开始编译固件..."
|
||
if build_firmware; then
|
||
log "INFO" "${GREEN}完整构建流程成功完成!${NC}"
|
||
return 0
|
||
else
|
||
log "ERROR" "${RED}完整构建流程失败于编译阶段。${NC}"
|
||
return 1
|
||
fi
|
||
}
|
||
|
||
#---------------菜单函数---------------#
|
||
# 打印主菜单
|
||
print_menu() {
|
||
echo -e "\n${BLUE}======= OpenWrt 构建管理菜单 =======${NC}"
|
||
echo -e " ${YELLOW}会话日志目录: ${SESSION_DIR}${NC}"
|
||
echo -e "${GREEN} 1.${NC} ${BOLD}完整构建流程${NC} (清理 > 更新源码/包/feeds > 配置 > 编译)"
|
||
echo -e "${GREEN} 2.${NC} 清理工作区 (删除 tmp, bin, build_dir 等)"
|
||
echo -e "${GREEN} 3.${NC} 更新所有组件 (源码 + 旧包清理 + 自定义包 + feeds)"
|
||
echo -e "${GREEN} 4.${NC} 清理配置文件指定的旧包/目录"
|
||
echo -e "${GREEN} 5.${NC} ${BOLD}编译固件${NC} (make V=s)"
|
||
echo -e "${GREEN} 6.${NC} 调整配置 (make menuconfig)"
|
||
echo -e "${GREEN} 7.${NC} 更新所有自定义包"
|
||
echo -e "${GREEN} 8.${NC} 更新特定的自定义包"
|
||
echo -e "${GREEN} 9.${NC} 更新 OpenWrt 源码 (git pull)"
|
||
echo -e "${GREEN}10.${NC} 更新并安装 Feeds"
|
||
echo -e "-------------------------------------"
|
||
echo -e "${GREEN} 0.${NC} 退出脚本"
|
||
echo -e "${BLUE}====================================${NC}"
|
||
# 使用 read -p 提供提示
|
||
read -rp "请输入选项 [0-10]: " choice
|
||
}
|
||
|
||
# 打印并选择特定自定义包进行更新
|
||
select_specific_package() {
|
||
local pkg_names=("${!CUSTOM_PACKAGES[@]}") # 获取所有包名
|
||
local num_packages=${#pkg_names[@]}
|
||
|
||
if [ "$num_packages" -eq 0 ]; then
|
||
log "WARN" "没有配置任何自定义包。"
|
||
return 1
|
||
fi
|
||
|
||
echo -e "\n${BLUE}--- 选择要更新的自定义包 ---${NC}"
|
||
local i=1
|
||
# 使用 sort 命令让包名按字母排序显示
|
||
local sorted_pkg_names=($(printf "%s\n" "${pkg_names[@]}" | sort))
|
||
|
||
for pkg_name in "${sorted_pkg_names[@]}"; do
|
||
echo -e " ${GREEN}$i.${NC} $pkg_name"
|
||
((i++))
|
||
done
|
||
echo -e " ${GREEN}0.${NC} 返回主菜单"
|
||
echo -e "${BLUE}----------------------------${NC}"
|
||
|
||
local choice_pkg
|
||
read -rp "请输入包的序号 [0-$num_packages]: " choice_pkg
|
||
|
||
# 验证选择
|
||
if [[ ! "$choice_pkg" =~ ^[0-9]+$ ]]; then
|
||
log "ERROR" "无效输入,请输入数字。"
|
||
return 1
|
||
fi
|
||
|
||
if [ "$choice_pkg" -eq 0 ]; then
|
||
log "INFO" "返回主菜单。"
|
||
return 0 # 返回成功,表示正常退出选择
|
||
elif [ "$choice_pkg" -ge 1 ] && [ "$choice_pkg" -le "$num_packages" ]; then
|
||
local selected_pkg=${sorted_pkg_names[$((choice_pkg-1))]}
|
||
log "INFO" "选择更新包: $selected_pkg"
|
||
update_specific_package "$selected_pkg" # 调用更新函数
|
||
return $? # 返回更新函数的执行结果
|
||
else
|
||
log "ERROR" "无效的选择序号: $choice_pkg"
|
||
return 1
|
||
fi
|
||
}
|
||
|
||
#---------------主函数---------------#
|
||
main() {
|
||
# 0. 设置颜色变量
|
||
setup_colors
|
||
|
||
# 1. 初始化日志系统 (会进行第一次输出重定向)
|
||
init_logger || exit 1 # 初始化失败则退出
|
||
|
||
# 2. 检查运行环境和依赖
|
||
log "INFO" "检查运行环境..."
|
||
local dep_missing=false
|
||
if [ ! -f "Makefile" ] || [ ! -d "scripts" ] || [ ! -d ".git" ]; then
|
||
log "ERROR" "脚本必须在 OpenWrt 源码的根目录下运行 (需要存在 Makefile, scripts, .git)。"
|
||
dep_missing=true
|
||
fi
|
||
for cmd in git make nproc tee cut dirname mkdir rm git date printf sort read; do
|
||
if ! command -v "$cmd" &> /dev/null; then
|
||
log "ERROR" "缺少必要的系统命令: $cmd"
|
||
dep_missing=true
|
||
fi
|
||
done
|
||
if [ "$dep_missing" = true ]; then
|
||
log "ERROR" "环境或依赖检查失败,请修正后重试。"
|
||
# 恢复原始输出,避免后续可能的错误信息无法显示
|
||
exec >&$STDOUT_ORIG
|
||
exec 2>&$STDERR_ORIG
|
||
exit 1
|
||
fi
|
||
log "INFO" "环境检查通过。"
|
||
|
||
# 记录脚本启动信息
|
||
log "INFO" "${BOLD}OpenWrt 构建管理脚本已启动${NC}"
|
||
log "INFO" "PID: $$"
|
||
log "INFO" "会话日志目录: $SESSION_DIR"
|
||
|
||
# 3. 主菜单循环
|
||
local choice # 提前声明 choice
|
||
while true; do
|
||
print_menu # 显示菜单并获取输入到 choice 变量
|
||
log "DEBUG" "用户选择了: $choice" # 记录用户选择
|
||
|
||
# 根据选择执行操作
|
||
case $choice in
|
||
1) full_build ;;
|
||
2) clean_workspace ;;
|
||
3) update_all_components ;;
|
||
4) clean_packages ;;
|
||
5) build_firmware ;;
|
||
6) menuconfig ;;
|
||
7) update_custom_package ;;
|
||
8) select_specific_package ;;
|
||
9) update_source ;;
|
||
10) update_feeds ;;
|
||
0)
|
||
log "INFO" "收到退出命令,正在退出脚本..."
|
||
# 恢复原始输出流 (好习惯)
|
||
exec >&$STDOUT_ORIG
|
||
exec 2>&$STDERR_ORIG
|
||
echo -e "${GREEN}脚本已退出。${NC}"
|
||
exit 0
|
||
;;
|
||
*)
|
||
log "WARN" "无效的选项: '$choice',请输入 0 到 10 之间的数字。"
|
||
;;
|
||
esac
|
||
|
||
local last_exit_code=$? # 获取上一个命令的退出状态码
|
||
# 操作完成后暂停,给用户查看结果的时间
|
||
if [[ $choice != 0 ]]; then
|
||
if [ $last_exit_code -eq 0 ]; then
|
||
echo -e "\n${GREEN}操作 '$choice' 执行完成。${NC}"
|
||
else
|
||
echo -e "\n${RED}操作 '$choice' 执行时遇到错误 (退出码: $last_exit_code)。${NC}"
|
||
fi
|
||
echo -e "${YELLOW}按 Enter 键返回主菜单...${NC}"
|
||
read -r # 等待用户按回车
|
||
fi
|
||
done
|
||
}
|
||
|
||
#---------------脚本入口---------------#
|
||
# 将所有主逻辑放入 main 函数,并在最后调用它
|
||
# "$@" 将所有命令行参数传递给 main 函数 (虽然本脚本目前未使用)
|
||
main "$@"
|