WSL中使用Beyond Compare 3/4/5作为difftool

Linux中进行diff比较目录时,有时一边的文件是软链接,在Windows中就不支持,使用AI辅助写了一个脚本将临时目录中的软链接替换成真实文件,WSL环境下就可以继续使用bc作为difftool了。

git-difftool-bc5.sh

bash 复制代码
#!/bin/bash

BC_PATH="/mnt/c/Program Files/Beyond Compare 5/BComp.exe"

# 核心函数:递归将目录下的软链接替换为真实文件/目录的拷贝
flatten_symlinks() {
    local target_path="$1"
    
    # 如果传入的不是目录,直接返回
    if [[ ! -d "$target_path" ]]; then
        return
    fi
    
    # 使用 find 查找目录下所有的软链接 (包括文件和目录链接)
    # -print0 和 read -d '' 配合,完美处理文件名中的空格
    while IFS= read -r -d '' link; do
        local real_target
        real_target=$(readlink -f "$link")
        
        if [[ -e "$real_target" ]]; then
            # 删除软链接本身 (注意: rm 不加 -r,只删链接不删目标)
            rm -f "$link"
            
            # 将真实的文件或目录拷贝到原软链接的位置
            if [[ -d "$real_target" ]]; then
                cp -r "$real_target" "$link"
            else
                cp "$real_target" "$link"
            fi
        fi
    done < <(find "$target_path" -type l -print0)
}

# 路径转换函数
to_win_path() {
    local path="$1"
    if [[ -e "$path" ]] || [[ -L "$path" ]]; then
        wslpath -w "$(readlink -f "$path")"
    else
        wslpath -w "$path"
    fi
}

if [[ "$#" -eq 2 ]]; then
    LOCAL_PATH="$1"
    REMOTE_PATH="$2"
    
    # ★ 在调用 BC5 前,把临时目录里的软链接全部替换为真实文件
    flatten_symlinks "$LOCAL_PATH"
    flatten_symlinks "$REMOTE_PATH"
    
    WIN_LOCAL=$(to_win_path "$LOCAL_PATH")
    WIN_REMOTE=$(to_win_path "$REMOTE_PATH")
    
    "$BC_PATH" "$WIN_LOCAL" "$WIN_REMOTE"
    
elif [[ "$#" -eq 4 ]]; then
    BASE_PATH="$1"
    LOCAL_PATH="$2"
    REMOTE_PATH="$3"
    MERGED_PATH="$4"
    
    # Merge 时通常传的是文件而非目录,但以防万一也处理一下
    flatten_symlinks "$BASE_PATH"
    flatten_symlinks "$LOCAL_PATH"
    flatten_symlinks "$REMOTE_PATH"
    
    WIN_BASE=$(to_win_path "$BASE_PATH")
    WIN_LOCAL=$(to_win_path "$LOCAL_PATH")
    WIN_REMOTE=$(to_win_path "$REMOTE_PATH")
    WIN_MERGED=$(to_win_path "$MERGED_PATH")
    
    "$BC_PATH" "$WIN_BASE" "$WIN_LOCAL" "$WIN_REMOTE" "$WIN_MERGED"
else
    echo "Error: Invalid number of arguments. Expected 2 for diff, 4 for merge."
    exit 1
fi

相应的git config:

复制代码
[diff]
    tool = bc5
[difftool "bc5"]
    cmd = ~/git-difftool-bc5.sh $LOCAL $REMOTE
[difftool]
    prompt = false
    symlinks = false
相关推荐
A_Lonely_Cat16 小时前
记一次 GitHub 幽灵协作者大清洗:强制重写 Git 历史与穿透 CDN 缓存实践
git·github
和你看星星3 天前
Git rerere:让重复冲突只解决一次
git
嘻嘻仙人6 天前
Ubuntu中 git上传自己的项目和二次上传一般流程
git·github
Patrick_Wilson6 天前
Squash Merge 的血缘陷阱:为什么删掉的代码又活了过来
前端·git·程序员
沉浸学习的匿名网友6 天前
什么是 .gitignore?为什么每个 Git 项目几乎都离不开它?
前端·git
深海鱼在掘金7 天前
Git 完全指南 —— 第3章:理解工作区、暂存区、版本库三个核心
git
江华森7 天前
Git 基础筑基:从原理到团队协作的全栈实战
git
JakeJiang8 天前
Git 必备命令指南:从日常高频到项目开发实战
git
叫我少年8 天前
Windows 中安装 git
git
深海鱼在掘金14 天前
Git 完全指南 —— 第1章:Git 概览与版本控制演进
git