#!/bin/bash # APM软件包转换器 - 将DEB包转换为APM格式 log.warn() { echo -e "[\e[33mWARN\e[0m]: \e[1m$*\e[0m"; } log.error() { echo -e "[\e[31mERROR\e[0m]: \e[1m$*\e[0m"; } log.info() { echo -e "[\e[96mINFO\e[0m]: \e[1m$*\e[0m"; } log.debug() { echo -e "[\e[32mDEBUG\e[0m]: \e[1m$*\e[0m"; } SCRIPT_NAME=$(basename "$0") if ! command -v dpkg > /dev/null ; then log.error "若想使用APM软件包转换器,您需先安装dpkg" exit 1 fi # 显示用法信息 usage() { echo "用法: $SCRIPT_NAME [--manual] --base [--base ...] " echo " 或者在手动模式下不传入 DEB 文件: $SCRIPT_NAME --manual --base [--base ...]" echo "" echo "参数说明:" echo " --manual 启用手动模式:融合挂载后打开交互 shell,退出 shell 后脚本继续" echo " --basename 必填参数(非手动模式下),指定基础环境名称,可多次使用" echo " deb文件路径 要转换的DEB文件路径(非手动且非空模式下必填)" echo " --pkgname 可选参数,指定新包的包名(默认使用原DEB包名)" echo " --version 可选参数,指定新包的版本号(默认在原版本后追加'-apm')" echo "" echo "示例:" echo " $SCRIPT_NAME --base amber-pm-trixie /path/to/package.deb" echo " $SCRIPT_NAME --manual --base amber-pm-trixie # 只融合挂载并进入手动 shell" echo " $SCRIPT_NAME --manual --base a --pkgname newpkg --version 1.2.3 /path/to/package.deb" echo "" echo "说明: 最下层的base在最后面,从上到下写base" } # 解析参数 BASENAMES=() # 存放实际用于构建 overlay 的 base(可能会被递归添加) BASENAMES_ORIG=() # 存放用户原始输入的 base 列表(用于 control 中 Depends 等) DEB_PATH="" PKGNAME="" VERSION="" MANUAL_MODE=false # 简单参数解析(顺序敏感) while [ $# -gt 0 ]; do case "$1" in --base) if [ -z "$2" ]; then log.error "--base 后需要跟名称" usage exit 1 fi BASENAMES+=("$2") BASENAMES_ORIG+=("$2") shift 2 ;; --pkgname) PKGNAME="$2" shift 2 ;; --version) VERSION="$2" shift 2 ;; --manual) MANUAL_MODE=true shift ;; -*) log.error "未知选项: $1" usage exit 1 ;; *) # 非选项,视为 DEB 路径(只接受第一个非选项作为 DEB) if [ -z "$DEB_PATH" ]; then DEB_PATH="$1" shift else log.error "未知参数或多余的参数: $1" usage exit 1 fi ;; esac done # 基本参数验证: # 如果不是手动模式,则至少需要一个 --base 和一个 deb 文件 if [ "$MANUAL_MODE" = false ]; then if [ ${#BASENAMES[@]} -eq 0 ] || [ -z "$DEB_PATH" ]; then log.error "错误:非手动模式下至少需要一个 --base 参数 且 必须提供 DEB 文件路径" usage exit 1 fi else # 手动模式下允许没有 DEB_FILE,但仍然要有至少一个 base if [ ${#BASENAMES[@]} -eq 0 ]; then log.error "错误:手动模式下仍需提供至少一个 --base 参数" usage exit 1 fi fi # 如果传入了 DEB_PATH,检查文件是否存在 if [ -n "$DEB_PATH" ] && [ ! -f "$DEB_PATH" ]; then log.error "错误:DEB文件不存在: $DEB_PATH" exit 1 fi log.info "开始转换(手动模式: $MANUAL_MODE)" log.info "基础环境数量: ${#BASENAMES_ORIG[@]}" for i in "${!BASENAMES_ORIG[@]}"; do log.info " 原始基础环境 $((i+1)): ${BASENAMES_ORIG[$i]}" done if [ -n "$DEB_PATH" ]; then log.info "目标 DEB: $DEB_PATH" else log.info "未提供 DEB 文件,处于纯手动模式(手动修改/安装/打包)" fi # 1. 创建临时工作目录 CRAFT_DIR="$HOME/apm-craft-$$" log.info "创建临时工作目录: $CRAFT_DIR" mkdir -p "$CRAFT_DIR"/{core,work,mergedir,modified_deb,extract,new-pkg} export CRAFT_DIR # 检查是否已挂载,避免重复挂载 cleanup_mount() { if mountpoint -q "$CRAFT_DIR/mergedir"; then log.info "解除挂载: $CRAFT_DIR/mergedir" sudo umount "$CRAFT_DIR/mergedir" || true fi } # 清理函数 cleanup() { log.info "开始清理..." cleanup_mount if [ -d "$CRAFT_DIR" ]; then log.info "删除临时目录: $CRAFT_DIR" sudo rm -rf "$CRAFT_DIR" fi } # 设置退出时清理 trap cleanup EXIT # 递归获取info文件中的依赖 (会把新依赖追加到 BASENAMES 数组中) get_recursive_basenames() { local basename="$1" # 注意:根据之前脚本结构,info 存放在 /var/lib/apm/apm/files/ace-env/var/lib/apm//info local base_dir="/var/lib/apm/apm/files/ace-env/var/lib/apm/$basename" local info_file="$base_dir/info" if [ -f "$info_file" ]; then log.info "读取info文件: $info_file" while IFS= read -r base; do [[ -z "$base" ]] && continue # 如果依赖的 base 没有被记录过,则递归添加 local found=false for existing in "${BASENAMES[@]}"; do if [ "$existing" = "$base" ]; then found=true break fi done if [ "$found" = false ]; then BASENAMES+=("$base") get_recursive_basenames "$base" fi done < "$info_file" else log.info "未找到info文件,跳过: $info_file" fi } # 递归获取所有基础环境(从用户输入的 base 开始) for BASE in "${BASENAMES[@]}"; do get_recursive_basenames "$BASE" done # 如果用户传了 DEB,则读取原包信息(否则跳过) if [ -n "$DEB_PATH" ]; then log.info "检查原DEB包信息..." ORIG_PKGNAME=$(dpkg -f "$DEB_PATH" Package 2>/dev/null || echo "") ORIG_VERSION=$(dpkg -f "$DEB_PATH" Version 2>/dev/null || echo "") ORIG_ARCH=$(dpkg -f "$DEB_PATH" Architecture 2>/dev/null || echo "") log.info "原包名: ${ORIG_PKGNAME:-未知}" log.info "原版本: ${ORIG_VERSION:-未知}" log.info "原架构: ${ORIG_ARCH:-unknown}" else ORIG_PKGNAME="" ORIG_VERSION="" ORIG_ARCH="$(dpkg --print-architecture 2>/dev/null || echo "unknown")" fi # 设置新包名和版本(若手动模式且未指定,则稍后询问) NEW_PKGNAME="${PKGNAME:-${ORIG_PKGNAME}}" NEW_VERSION="${VERSION:-${ORIG_VERSION}-apm}" log.info "将使用的新包名: ${NEW_PKGNAME:-<未指定>}" log.info "将使用的新版本: ${NEW_VERSION:-<未指定>}" log.info "使用的架构: $ORIG_ARCH" # 2. 构建 lowerdir 路径(多个 base 按顺序叠放) log.info "构建 overlay lowerdir 路径..." LOWERDIRS=() for BASENAME in "${BASENAMES[@]}"; do ACE_ENV_PATH="/var/lib/apm/apm/files/ace-env/var/lib/apm/${BASENAME}/files/ace-env" CORE_PATH="/var/lib/apm/apm/files/ace-env/var/lib/apm/${BASENAME}/files/core" if [ -d "$ACE_ENV_PATH" ]; then log.info "使用 ace-env 路径: $ACE_ENV_PATH" LOWERDIRS+=("$ACE_ENV_PATH") elif [ -d "$CORE_PATH" ]; then log.info "使用 core 路径: $CORE_PATH" LOWERDIRS+=("$CORE_PATH") else log.error "错误:基础环境路径不存在: $BASENAME" log.error " 检查的路径: $ACE_ENV_PATH" log.error " 检查的路径: $CORE_PATH" exit 1 fi done # 将 lowerdirs 数组用冒号连接 LOWERDIR=$(IFS=:; echo "${LOWERDIRS[*]}") log.debug "最终 lowerdir: $LOWERDIR" # 3. 进行融合挂载 log.info "正在进行融合挂载..." sudo mount -t overlay overlay \ -o "lowerdir=$LOWERDIR,upperdir=$CRAFT_DIR/core/,workdir=$CRAFT_DIR/work/" \ "$CRAFT_DIR/mergedir" if ! mountpoint -q "$CRAFT_DIR/mergedir"; then log.error "错误:融合挂载失败" exit 1 fi log.info "挂载完成: $CRAFT_DIR/mergedir" # 导出 chrootEnvPath 以便 ace-run-pkg 使用(并在需要时传递给 sudo -E) export chrootEnvPath="$CRAFT_DIR/mergedir" log.debug "已导出 chrootEnvPath=$chrootEnvPath" # 如果在手动模式下,立即打开交互 shell 并在退出后继续脚本 if [ "$MANUAL_MODE" = true ]; then log.info "进入手动模式:将在融合挂载环境中打开交互 shell(使用 ace-run-pkg)。" log.info "在 shell 中,您可以手动修改、测试安装或进行其他操作。退出 shell 后脚本将继续。" # 启动交互 shell,保留环境变量(使用 sudo -E) sudo -E chrootEnvPath="$chrootEnvPath" /var/lib/apm/apm/files/ace-run-pkg bash --login || { log.warn "ace-run-pkg shell 退出或出现错误,继续脚本..." } log.info "用户已退出手动 shell,脚本将继续。" # 如果没有 DEB,询问是否要进行后续打包(允许返回 shell) if [ -z "$DEB_PATH" ]; then while true; do echo "" read -r -p "未提供 DEB 文件。是否现在进行新 APM 包的自动打包? (y = 打包, r = 返回 shell, n = 跳过打包) [y/r/n]: " yn case "$yn" in y|Y) # 如果缺少包名或版本,交互询问 if [ -z "$NEW_PKGNAME" ]; then read -r -p "请输入要创建的包名 (Package): " NEW_PKGNAME fi if [ -z "$NEW_VERSION" ] || [[ "$NEW_VERSION" == "-apm" ]]; then read -r -p "请输入要创建的版本 (Version): " NEW_VERSION fi break ;; r|R) log.info "返回交互 shell(使用 ace-run-pkg)。退出 shell 后再次询问。" sudo -E chrootEnvPath="$chrootEnvPath" /var/lib/apm/apm/files/ace-run-pkg bash --login || true ;; n|N) log.info "跳过自动打包。脚本结束。" exit 0 ;; *) echo "请输入 y, r, 或 n。" ;; esac done fi fi # 到这里:非手动模式或手动模式退出后继续(如果是非手动并且有 DEB,继续原本流程) # 4. 如果有 DEB 文件,进行自动化的检查、解包与修改 if [ -n "$DEB_PATH" ]; then # 在融合环境中更新包列表并做 dry-run 检查(如果 ace-run-pkg aptss 可用) log.info "在融合环境中测试安装 DEB 包(dry-run)..." sudo -E chrootEnvPath="$chrootEnvPath" /var/lib/apm/apm/files/ace-run-pkg aptss update || log.warn "aptss update 返回非零状态,继续但请注意" if ! sudo -E chrootEnvPath="$chrootEnvPath" /var/lib/apm/apm/files/ace-run-pkg aptss install "$DEB_PATH" --dry-run ; then log.error "错误:安装前检查失败,DEB包可能无法在基础环境中安装" log.error "请检查依赖关系或基础环境是否兼容" exit 1 fi log.info "安装前检查通过,准备进行提取与修改..." # 提取 DEB 包内容并准备修改 log.info "提取并修改原DEB包..." EXTRACT_DIR="$CRAFT_DIR/extract" MODIFIED_DEB_DIR="$CRAFT_DIR/modified_deb" mkdir -p "$EXTRACT_DIR" mkdir -p "$MODIFIED_DEB_DIR/DEBIAN" dpkg -x "$DEB_PATH" "$EXTRACT_DIR" dpkg -e "$DEB_PATH" "$MODIFIED_DEB_DIR/DEBIAN" # 处理 .desktop 文件:注意使用 while read 来避免 subshell 问题 DESKTOP_MODIFIED=false while IFS= read -r desktop_file; do [ -z "$desktop_file" ] && continue log.info "处理桌面文件: $desktop_file" # 尝试用 busybox dos2unix(若不存在则跳过转换) if command -v busybox >/dev/null 2>&1; then busybox dos2unix "$desktop_file" 2>/dev/null || true else dos2unix "$desktop_file" 2>/dev/null || true || true fi DESKTOP_MODIFIED=true # 处理 Exec 行:在原有命令前追加 apm run $NEW_PKGNAME if grep -q '^Exec=' "$desktop_file"; then sed -i "s|^Exec=\(.*\)$|Exec=apm run ${NEW_PKGNAME:-$ORIG_PKGNAME} \1|" "$desktop_file" fi # 删除 TryExec 行 if grep -q '^TryExec=' "$desktop_file"; then sed -i '/^TryExec=/d' "$desktop_file" log.info "已删除 TryExec 行" fi # 处理 Icon 路径(若以 / 开头) if grep -q '^Icon=/' "$desktop_file"; then sed -i "s|^Icon=/|Icon=/var/lib/apm/apm/files/ace-env/var/lib/apm/${NEW_PKGNAME:-$ORIG_PKGNAME}/files/core/|" "$desktop_file" fi # 添加 X-APM-APPID if ! grep -q "X-APM-APPID" "$desktop_file"; then echo "X-APM-APPID=${NEW_PKGNAME:-$ORIG_PKGNAME}" >> "$desktop_file" fi # 检查修改结果并打印调试 if grep -q "apm run ${NEW_PKGNAME:-$ORIG_PKGNAME}" "$desktop_file"; then log.info "桌面文件修改成功: $desktop_file" else log.warn "桌面文件可能未正确修改: $desktop_file" fi done < <(find "$EXTRACT_DIR" -name "*.desktop" -print) if [ "$DESKTOP_MODIFIED" = false ]; then log.info "未找到需要修改的 .desktop 文件" fi # 复制修改后的文件结构到打包目录并重新打包 modified deb(供本地测试/安装使用) MODIFIED_DEB_PATH="$CRAFT_DIR/modified_${ORIG_PKGNAME:-package}.deb" log.info "重新打包修改后的 DEB: $MODIFIED_DEB_PATH" mkdir -p "$MODIFIED_DEB_DIR/data" cp -r "$EXTRACT_DIR"/* "$MODIFIED_DEB_DIR/" 2>/dev/null || true (cd "$MODIFIED_DEB_DIR" && fakeroot dpkg-deb --build -Z none . "$MODIFIED_DEB_PATH") || { log.error "错误:重新打包 DEB 失败" exit 1 } if [ ! -f "$MODIFIED_DEB_PATH" ]; then log.error "错误:重新打包后的 DEB 未生成: $MODIFIED_DEB_PATH" exit 1 fi log.info "修改后的 DEB 包已生成: $MODIFIED_DEB_PATH" # 可选:在融合环境中实际安装修改后的包(默认使用 ssaudit 命令) if ! sudo -E chrootEnvPath="$chrootEnvPath" /var/lib/apm/apm/files/ace-run-pkg ssaudit "$MODIFIED_DEB_PATH" --native --no-create-desktop-entry ; then log.error "错误:修改后的 DEB 包安装失败(ssaudit)" exit 1 fi log.info "修改后的 DEB 包安装完成(ssaudit)" # 清理 apt 缓存 sudo -E chrootEnvPath="$chrootEnvPath" /var/lib/apm/apm/files/ace-run-pkg aptss clean || true sudo -E chrootEnvPath="$chrootEnvPath" /var/lib/apm/apm/files/ace-run-pkg rm -vfr /var/lib/apt/lists || true sudo -E chrootEnvPath="$chrootEnvPath" /var/lib/apm/apm/files/ace-run-pkg rm -vfr /var/lib/aptss/lists || true fi # 5. 创建新的 APM 包结构 log.info "创建新的APM包结构..." PKG_BUILD_DIR="$CRAFT_DIR/new-pkg" mkdir -p "$PKG_BUILD_DIR/DEBIAN" mkdir -p "$PKG_BUILD_DIR/var/lib/apm/$NEW_PKGNAME"/{entries,files} 2>/dev/null || true # info 和 info_debug:写入原始输入的 base 列表 和 递归展开后的 base 列表 log.info "创建 info 文件(包含原始输入的基础环境)..." : > "$PKG_BUILD_DIR/var/lib/apm/${NEW_PKGNAME}/info" 2>/dev/null || true for BASENAME in "${BASENAMES_ORIG[@]}"; do echo "$BASENAME" >> "$PKG_BUILD_DIR/var/lib/apm/${NEW_PKGNAME}/info" log.info " 写入: $BASENAME" done log.info "创建 info_debug 文件(包含所有递归依赖的基础环境)..." : > "$PKG_BUILD_DIR/var/lib/apm/${NEW_PKGNAME}/info_debug" 2>/dev/null || true for BASENAME in "${BASENAMES[@]}"; do echo "$BASENAME" >> "$PKG_BUILD_DIR/var/lib/apm/${NEW_PKGNAME}/info_debug" log.info " 写入: $BASENAME" done # 创建 postrm 脚本 cat > "$PKG_BUILD_DIR/DEBIAN/postrm" << 'EOF' #!/bin/bash PACKAGE_NAME="$DPKG_MAINTSCRIPT_PACKAGE" if [ "$1" = "remove" ] || [ "$1" = "purge" ]; then echo "清理卸载残留" rm -rf "/var/lib/apm/$PACKAGE_NAME" for username in $(ls /home); do if [ -d "/home/$username/.apm/$PACKAGE_NAME" ]; then rm -fr "/home/$username/.apm/$PACKAGE_NAME" fi done else echo "非卸载,跳过清理" fi EOF chmod +x "$PKG_BUILD_DIR/DEBIAN/postrm" # 6. 复制需要的文件到新的 APM 包 log.info "复制文件到新的APM包..." # 复制 /usr/share 内容到 entries if [ -d "$CRAFT_DIR/extract/usr/share" ]; then log.info "复制 /usr/share 内容..." mkdir -p "$PKG_BUILD_DIR/var/lib/apm/${NEW_PKGNAME}/entries" cp -r "$CRAFT_DIR/extract/usr/share/"* "$PKG_BUILD_DIR/var/lib/apm/${NEW_PKGNAME}/entries/" 2>/dev/null || true fi # 复制 /opt/apps//entries(如果存在) if [ -n "$ORIG_PKGNAME" ] && [ -d "$CRAFT_DIR/extract/opt/apps/$ORIG_PKGNAME/entries" ]; then log.info "复制 /opt/apps/$ORIG_PKGNAME/entries 内容..." cp -r "$CRAFT_DIR/extract/opt/apps/$ORIG_PKGNAME/entries/"* "$PKG_BUILD_DIR/var/lib/apm/${NEW_PKGNAME}/entries/" 2>/dev/null || true fi # 复制融合环境(core, work)到新的包内 files(以便运行时使用) log.info "复制融合环境文件..." sudo cp -r "$CRAFT_DIR"/core "$PKG_BUILD_DIR/var/lib/apm/${NEW_PKGNAME}/files/" 2>/dev/null || true sudo cp -r "$CRAFT_DIR"/work "$PKG_BUILD_DIR/var/lib/apm/${NEW_PKGNAME}/files/" 2>/dev/null || true # 设置文件权限 sudo chmod -R 755 "$PKG_BUILD_DIR/var/lib/apm/${NEW_PKGNAME}/files/" 2>/dev/null || true # 7. 解除挂载(如果尚未解除) log.info "解除挂载..." cleanup_mount # 8. 计算目录大小函数 calculate_directory_size() { local dir="$1" if [ -d "$dir" ]; then du -sk "$dir" | cut -f1 else echo "0" fi } # 构建依赖字符串 - 包含所有用户原始输入的 base(用于 control) DEPENDS_STR=$(IFS=,; echo "${BASENAMES_ORIG[*]}") # 若打包前没有 NEW_PKGNAME/NEW_VERSION,交互询问(一般出现在手动无DEB场景) if [ -z "$NEW_PKGNAME" ]; then read -r -p "请输入要创建的包名 (Package): " NEW_PKGNAME fi if [ -z "$NEW_VERSION" ] || [[ "$NEW_VERSION" == "-apm" ]]; then read -r -p "请输入要创建的版本 (Version): " NEW_VERSION fi # 创建 control 文件 cat > "${PKG_BUILD_DIR}/DEBIAN/control" << EOF Package: $NEW_PKGNAME Version: $NEW_VERSION Architecture: $ORIG_ARCH Maintainer: APM Converter Depends: $DEPENDS_STR Installed-Size: $(calculate_directory_size "$PKG_BUILD_DIR") Description: APM converted package from ${ORIG_PKGNAME:-original} This package was automatically converted from the original deb package. Based on: ${BASENAMES_ORIG[*]} EOF # 9. 打包并生成输出文件名 OUTPUT_DEB="${NEW_PKGNAME}_${NEW_VERSION}_${ORIG_ARCH}.deb" log.info "开始使用 fakeroot 打包: $OUTPUT_DEB" fakeroot dpkg-deb -Z xz --build "$PKG_BUILD_DIR" "$OUTPUT_DEB" || { log.error "错误:打包 APM 包失败" exit 1 } log.info "转换完成!" log.info "生成的APM包: $OUTPUT_DEB" log.info "包名: $NEW_PKGNAME" log.info "版本: $NEW_VERSION" log.info "架构: $ORIG_ARCH" log.info "依赖: $DEPENDS_STR" log.info "基础环境(原始输入): ${BASENAMES_ORIG[*]}" log.info "基础环境(递归展开): ${BASENAMES[*]}" log.info "注意:桌面文件如存在已被修改,添加了 apm run 前缀和 X-APM-APPID" # 退出(trap 会触发 cleanup) exit 0