返回 Skill 列表
extension
分类: 其它无需 API Key

从 Git 仓库历史中彻底删除文件

|Git删除了文件,但还会在Git 仓库历史中遗留。 - 想要从 Git 历史移除敏感信息(密码、密钥、token、私钥) - 想要删除意外提交的大文件 - 想要重写 Git 历史、移除特定文件的所有痕迹 - 提到 "force push"、"重写 commit 历史" - 需要从所有分支、所有 tag 中删除某个文件

person作者: user_137732f6hubcommunity

Git 历史中彻底删除文件

本技能指导用户使用 git filter-repo 工具从 Git 仓库的完整历史记录中彻底删除指定文件。

适用场景

  • 意外提交了敏感信息(密码、API 密钥、私钥、token)
  • 意外提交了大文件,导致仓库体积膨胀
  • 需要清理历史记录中的特定文件
  • 需要移除所有分支和标签中的某个文件

核心流程

1. 注意! 第一步, 先询问用户,确定用户的仓库路径,是否是当前工作目录。

询问用户是否需要备份仓库(强烈建议)

必须用户确认后,继续执行后续步骤。

2. 备份仓库(强烈建议)

在执行任何操作前,先备份仓库:

# 方式1:复制整个仓库目录
cp -r your-repo your-repo-backup

# 方式2:创建裸仓库备份
git clone --bare your-repo your-repo-backup.git

3. 安装 git filter-repo

根据操作系统选择安装方式:

macOS:

brew install git-filter-repo

Ubuntu / Debian:

pip install git-filter-repo

Windows:

pip install git-filter-repo

4. 确认当前仓库状态

进入仓库目录后检查:

git status
git remote -v

如果没有远程仓库,先添加:

git remote add origin git@github.com:用户名/仓库名.git

5. 使用 git filter-repo 删除文件

假设要删除的文件是 secrets.txt

git filter-repo --path secrets.txt --invert-paths

这一步的作用:

  • 从所有 commit 中移除该文件
  • 从所有分支中移除该文件
  • 从所有 tag 中移除该文件

删除多个文件:

git filter-repo --path file1.txt --path file2.txt --path secrets/ --invert-paths

删除整个目录:

git filter-repo --path 目录名/ --invert-paths

6. 验证文件是否已删除

查看历史中是否还有这个文件:

git log --all -- 文件名

如果没有输出,说明已经删干净了。

6. 强制推送到远程仓库

因为历史被重写了,必须强制推送:

# 推送所有分支
git push origin --force --all

# 推送所有标签
git push origin --force --tags

敏感信息删除后的额外操作

如果删除的是敏感信息(密钥、token、密码、私钥),除了删除历史,还要:

立即吊销旧密钥

  • 在对应的服务平台(GitHub、AWS、Azure 等)立即吊销/删除旧密钥
  • 重新生成新的密钥

检查其他位置

  • 检查是否有其他人 fork 了仓库
  • 检查 CI/CD 缓存
  • 检查本地 clone 是否还保留旧内容

重要提醒: 历史删掉了,不代表别人之前没复制走。一旦敏感信息被提交到公开仓库,就应当视为已泄露。

通知协作者

因为 commit 历史变了,其他人本地仓库会和远程不一致。需要通知他们:

  1. 重新 clone 仓库(推荐)

  2. 或者执行以下命令同步:

    git fetch origin
    git reset --hard origin/main  # 会丢失本地未推送的更改,谨慎使用
    
  3. 警告: 不要继续基于旧历史开发,否则再次推送时会重新引入已删除的文件

常见问题

Q: 提示 "fatal: not a git repository"?

确保你在 Git 仓库根目录下执行命令。

Q: 提示 "git filter-repo: command not found"?

先安装 git-filter-repo,参考上面的安装步骤。

Q: 删除后仓库体积没有变小?

需要执行垃圾回收:

git gc --aggressive --prune=now

Q: 只想修改最近一次 commit?

如果文件只在最近一次 commit 中,不需要用 filter-repo,用:

git rm --cached 文件名
git commit --amend

一句话版本

最核心的命令:

# 删除文件
 git filter-repo --path 文件名 --invert-paths

# 如果没有远程,先添加
git remote add origin <repo-url>

# 强制推送
git push origin --force --all
git push origin --force --tags