aboutsummaryrefslogtreecommitdiff
path: root/format/pack/ingest/thin_fix.go
blob: f97c07768b552f2a2e6f61a106376484eb460c24 (about) (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
package ingest

import (
	"fmt"

	"codeberg.org/lindenii/furgit/internal/intconv"
	"codeberg.org/lindenii/furgit/internal/utils"
)

// maybeFixThin appends missing bases and rewrites pack header/trailer when needed.
func maybeFixThin(state *ingestState) error {
	if len(state.unresolvedRefDeltas) == 0 {
		return nil
	}

	utils.FprintfBestEffort(
		state.opts.Progress,
		"fixing thin pack: %d unresolved bases\r",
		len(state.unresolvedRefDeltas),
	)

	if !state.opts.FixThin {
		return &ThinPackUnresolvedError{Count: len(state.unresolvedRefDeltas)}
	}

	if state.opts.Base == nil {
		return &ThinPackUnresolvedError{Count: len(state.unresolvedRefDeltas)}
	}

	hashSize := int64(state.algo.Size())

	info, err := state.packFile.Stat()
	if err != nil {
		return err
	}

	size := info.Size()
	if size < hashSize {
		return fmt.Errorf("format/pack/ingest: pack too short to trim trailer")
	}

	newEnd := size - hashSize

	err = state.packFile.Truncate(newEnd)
	if err != nil {
		return err
	}

	consumed, err := intconv.Int64ToUint64(newEnd)
	if err != nil {
		return err
	}

	state.stream.consumed = consumed

	baseIDs := unresolvedThinBaseIDs(state)

	total := len(baseIDs)
	if total > 0 {
		utils.FprintfBestEffort(state.opts.Progress, "fixing thin pack:   0%% (0/%d)\r", total)
	}

	for i, id := range baseIDs {
		ty, content, err := state.opts.Base.ReadBytesContent(id)
		if err != nil {
			continue
		}

		_, err = appendBaseObject(state, id, ty, content)
		if err != nil {
			return err
		}

		state.thinFixed = true

		done := i + 1
		percent := done * 100 / total
		utils.FprintfBestEffort(state.opts.Progress, "fixing thin pack: %3d%% (%d/%d)\r", percent, done, total)
	}

	err = rewritePackHeaderAndTrailer(state)
	if err != nil {
		return err
	}

	if state.thinFixed {
		utils.FprintfBestEffort(state.opts.Progress, "fixing thin pack: 100%% (%d/%d), done.\n", total, total)
	}

	return nil
}