aboutsummaryrefslogtreecommitdiff
path: root/objectstore/packed/store_open_pack.go
blob: f101b62412486273015f8d5ddcbb581c6934af13 (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
92
93
94
95
96
97
98
99
100
package packed

import "fmt"

// openPack returns one opened and validated pack handle.
func (store *Store) openPack(name string) (*packFile, error) {
	store.stateMu.RLock()

	pack, ok := store.packs[name]
	if ok {
		store.stateMu.RUnlock()

		return pack, nil
	}

	store.stateMu.RUnlock()

	file, err := store.root.Open(name)
	if err != nil {
		return nil, err
	}

	info, err := file.Stat()
	if err != nil {
		_ = file.Close()

		return nil, err
	}

	pack, err = openPackFile(name, file, info.Size())
	if err != nil {
		_ = file.Close()

		return nil, err
	}

	err = store.verifyPackMatchesIndexes(pack)
	if err != nil {
		_ = pack.close()

		return nil, err
	}

	store.stateMu.Lock()

	existing, ok := store.packs[name]
	if ok {
		store.stateMu.Unlock()

		_ = pack.close()

		return existing, nil
	}

	store.packs[name] = pack
	store.stateMu.Unlock()

	return pack, nil
}

// verifyPackMatchesIndexes checks that one opened pack's trailer hash matches
// every loaded index that references the same pack name.
func (store *Store) verifyPackMatchesIndexes(pack *packFile) error {
	err := store.ensureCandidates()
	if err != nil {
		return err
	}

	candidate, ok := store.candidateForPack(pack.name)
	if !ok {
		return fmt.Errorf("objectstore/packed: missing index for pack %q", pack.name)
	}

	index, err := store.openIndex(candidate)
	if err != nil {
		return err
	}

	err = verifyMappedPackMatchesMappedIdx(pack.data, index.data, store.algo)
	if err != nil {
		return fmt.Errorf("objectstore/packed: pack %q does not match idx %q: %w", pack.name, index.idxName, err)
	}

	return nil
}

// entryMetaAt parses one pack entry header at location.
func (store *Store) entryMetaAt(loc location) (*packFile, entryMeta, error) {
	pack, err := store.openPack(loc.packName)
	if err != nil {
		return nil, entryMeta{}, err
	}

	meta, err := parseEntryMeta(pack, store.algo, loc.offset)
	if err != nil {
		return nil, entryMeta{}, err
	}

	return pack, meta, nil
}