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