On Wed, Feb 03, 2021 at 10:59:25PM -0500, Taylor Blau wrote: > Often it is useful to both: > > - have relatively few packfiles in a repository, and > > - avoid having so few packfiles in a repository that we repack its > entire contents regularly > > This patch implements a '--geometric=<n>' option in 'git repack'. This > allows the caller to specify that they would like each pack to be at > least a factor times as large as the previous largest pack (by object > count). > > Concretely, say that a repository has 'n' packfiles, labeled P1, P2, > ..., up to Pn. Each packfile has an object count equal to 'objects(Pn)'. > With a geometric factor of 'r', it should be that: > > objects(Pi) > r*objects(P(i-1)) > > for all i in [1, n], where the packs are sorted by > > objects(P1) <= objects(P2) <= ... <= objects(Pn). Just devil's advocating for a moment. I think in this kind of geometric roll-up strategy, you want to imagine that you are rolling up recent pushes but leaving untouched a good "base" pack that you previously created. And that will usually be true if you are doing the rollup based on number of objects (or size, etc). But it won't always be (e.g., for some reason somebody makes a very large push relative to the current repository size). What happens when this assumption is violated? In some ways, it is a good thing to drift away from this "base pack" view of the world. If we're trying to amortize the per-object work done, then we are better off rolling up the small things into the large, regardless of where they came from. But the base pack may also have other properties we want to retain. Two I can think of: - it may have a .bitmap that we'll be throwing away, without generating a new one. I know that your end-game involves writing a midx with bitmaps that covers all of the packs, so this would become a non-issue in that strategy. - it may have been more carefully packed (e.g., with a larger window size, using "-f", etc) than the packs we got from pushes. We do _mostly_ retain the deltas when we roll up the packs, so it probably only has a small impact in practice (I'd expect in a few cases we'd throw away deltas because a pushed pack contains a duplicate of its base object that we added via --fix-thin). So I suspect it's probably OK in practice. These cases would happen rarely, and the impact would not be all that big. The bitmap thing I'd worry the most about. As part of a larger strategy involving a midx it is taken care of, but people using just this new feature may not realize that. The bitmaps of course are "just" an optimization, but it's hard to say how dire things are when they don't exist. For many situations, probably not very dire. But I know that on our servers, when repos lack bitmaps, people notice the performance degradation. On the other hand, by definition this happens in a case where there are more objects that have just been pushed (and are therefore not bitmapped) than existed already. So you _already_ have a performance problem either way until you get bitmap coverage of those new objects. > --- a/Documentation/git-repack.txt > +++ b/Documentation/git-repack.txt > @@ -165,6 +165,17 @@ depth is 4095. > Pass the `--delta-islands` option to `git-pack-objects`, see > linkgit:git-pack-objects[1]. > > +-g=<factor>:: > +--geometric=<factor>:: > + Arrange resulting pack structure so that each successive pack > + contains at least `<factor>` times the number of objects as the > + next-largest pack. > ++ > +`git repack` ensures this by determining a "cut" of packfiles that need to be > +repacked into one in order to ensure a geometric progression. It picks the > +smallest set of packfiles such that as many of the larger packfiles (by count of > +objects contained in that pack) may be left intact. I think we might need to make clear in the documentation how this differs from other repacks, in that it is not considering reachability at all. I like the term "roll up" to describe what is happening, but we probably need to define that term clearly, as well. Especially important, I think, is that we talk about what's happening with loose objects, which are part of the rollup here. And IMHO we should make clear that for now we include them all, without consideration of their reachability, but that this may change in the future. Likewise, are there any options that are incompatible with "-g"? I have to imagine that "--write-bitmap-index" would not work very well. I don't know that we need to enumerate them all, but I'm wondering if a blanket "this may not play well with other options" warning may be advisable. > +static void split_pack_geometry(struct pack_geometry *geometry, int factor) > [...] I'll admit I didn't carefully think about the math of the progression here. IMHO the exact split is the least interesting part of this whole series (compared to the general idea of "rolling up some packs" versus a whole repack). Between the comments and the tests, I'll assume it's generally behaving as advertised. (I of course did look for any obvious coding errors, but didn't see any). -Peff