145M = .git/objects/pack/
I wrote a script to add up the sizes of differences of each commit and the commit before it going backwards from the tip of each branch. I get 129MB, which is without compression and without accounting for same files across branches and common history among branches.
Git takes all those things into account so I would expect much much smaller repository. So why is .git so big?
I've done:
git fsck --full
git gc --prune=today --aggressive
git repack
To answer about how many files/commits, I have 19 branches about 40 files in each. 287 commits, found using:
git log --oneline --all|wc -l
It should not be taking 10's of megabytes to store information about this.
Are you sure you are counting just the .pack files and not the .idx files? They are in the same directory as the .pack files, but do not have any of the repository data (as the extension indicates, they are nothing more than indexes for the corresponding pack — in fact, if you know the correct command, you can easily recreate them from the pack file, and git itself does it when cloning, as only a pack file is transferred using the native git protocol).
As a representative sample, I took a look at my local clone of the linux-2.6 repository:
Which indicates an expansion of around 7% should be common.
There are also the files outside
objects/
; in my personal experience, of themindex
andgitk.cache
tend to be the biggest ones (totaling 11M in my clone of the linux-2.6 repository).Some scripts I use:
git-fatfiles
If you want more lines, see also Perl version in a neighbouring answer: https://stackoverflow.com/a/45366030/266720
git-eradicate (for
video/parasite.avi
):Note: the second script is designed to remove info from Git completely (including all info from reflogs). Use with caution.
If you want to find what files are taking up space in your git repository, run
git verify-pack -v .git/objects/pack/*.idx | sort -k 3 -n | tail -5
Then, extract the blob reference that takes up the most space (the last line), and check the filename that is taking so much space
git rev-list --objects --all | grep <reference>
This might even be a file that you removed with
git rm
, but git remembers it because there are still references to it, such as tags, remotes and reflog.Once you know what file you want to get rid of, I recommend using
git forget-blob
https://ownyourbits.com/2017/01/18/completely-remove-a-file-from-a-git-repository-with-git-forget-blob/
It is easy to use, just do
git forget-blob file-to-forget
This will remove every reference from git, remove the blob from every commit in history, and run garbage collection to free up the space.
Just FYI, the biggest reason why you may end up with unwanted objects being kept around is that git maintains a reflog.
The reflog is there to save your butt when you accidentally delete your master branch or somehow otherwise catastrophically damage your repository.
The easiest way to fix this is to truncate your reflogs before compressing (just make sure that you never want to go back to any of the commits in the reflog).
This is different from
git gc --prune=today
in that it expires the entire reflog immediately.The git-fatfiles script from Vi's answer is lovely if you want to see the size of all your blobs, but it's so slow as to be unusable. I removed the 40-line output limit, and it tried to use all my computer's RAM instead of finishing. So I rewrote it: this is thousands of times faster, has added features (optional), and some strange bug was removed--the old version would give inaccurate counts if you sum the output to see the total space used by a file.
Name this git-fatfiles.pl and run it. To see the disk space used by all revisions of a file, use the
--sum
option. To see the same thing, but for files within each directory, use the--directories
option. If you install the Number::Bytes::Human cpan module (run "cpan Number::Bytes::Human"), the sizes will be formatted: "21M /path/to/file.mp4".before doing git filter-branch & git gc you should review tags that are present in your repo. Any real system which has automatic tagging for things like continuous integration and deployments will make unwated objects still refrenced by these tags , hence gc cant remove them and you will still keep wondering why the size of repo is still so big.
The best way to get rid of all un-wanted stuff is to run git-filter & git gc and then push master to a new bare repo. The new bare repo will have the cleaned up tree.