-
Notifications
You must be signed in to change notification settings - Fork 351
/
writer.go
184 lines (156 loc) · 4.78 KB
/
writer.go
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
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
package sstable
import (
"context"
"encoding/hex"
"fmt"
"hash"
"hash/fnv"
"strconv"
"github.com/cockroachdb/pebble/sstable"
"github.com/treeverse/lakefs/pkg/graveler"
"github.com/treeverse/lakefs/pkg/graveler/committed"
"github.com/treeverse/lakefs/pkg/ident"
"github.com/treeverse/lakefs/pkg/pyramid"
)
const (
MetadataFirstKey = "min_key"
MetadataLastKey = "max_key"
MetadataNumRecordsKey = "count"
MetadataEstimatedSizeKey = "estimated_size_bytes"
)
type DiskWriter struct {
ctx context.Context
w *sstable.Writer
props map[string]string
tierFS pyramid.FS
first committed.Key
last committed.Key
count int
hash hash.Hash
fh pyramid.StoredFile
closed bool
}
func NewDiskWriter(ctx context.Context, tierFS pyramid.FS, ns committed.Namespace, hash hash.Hash, metadata graveler.Metadata) (*DiskWriter, error) {
fh, err := tierFS.Create(ctx, string(ns))
if err != nil {
return nil, fmt.Errorf("opening file: %w", err)
}
props := make(map[string]string)
for k, v := range metadata {
props[k] = v
}
writer := sstable.NewWriter(fh, sstable.WriterOptions{
Compression: sstable.SnappyCompression,
TablePropertyCollectors: []func() sstable.TablePropertyCollector{NewStaticCollector(props)},
})
return &DiskWriter{
ctx: ctx,
w: writer,
props: props,
fh: fh,
tierFS: tierFS,
hash: hash,
}, nil
}
// SetMetadata associates metadata value (which will be stringified) with key.
// Keys and values are also calculated as part of the resulting range ID
func (dw *DiskWriter) SetMetadata(key, value string) {
dw.props[key] = value
}
func (dw *DiskWriter) GetFS() pyramid.FS {
return dw.tierFS
}
func (dw *DiskWriter) GetStoredFile() pyramid.StoredFile {
return dw.fh
}
func (dw *DiskWriter) WriteRecord(record committed.Record) error {
if err := dw.w.Set(record.Key, record.Value); err != nil {
return fmt.Errorf("setting key and value: %w", err)
}
// updating stats
if dw.count == 0 {
dw.first = make(committed.Key, len(record.Key))
copy(dw.first, record.Key)
}
dw.last = make(committed.Key, len(record.Key))
copy(dw.last, record.Key)
dw.count++
if err := dw.writeHashWithLen(record.Key); err != nil {
return err
}
return dw.writeHashWithLen(record.Value)
}
func (dw *DiskWriter) GetApproximateSize() uint64 {
return dw.w.EstimatedSize()
}
func (dw *DiskWriter) writeHashWithLen(buf []byte) error {
if _, err := dw.hash.Write([]byte(strconv.Itoa(len(buf)))); err != nil {
return err
}
if _, err := dw.hash.Write(buf); err != nil {
return err
}
if _, err := dw.hash.Write([]byte("|")); err != nil {
return err
}
return nil
}
func (dw *DiskWriter) Abort() error {
if dw.closed {
return nil
}
if err := dw.w.Close(); err != nil {
return fmt.Errorf("sstable file close: %w", err)
}
if err := dw.fh.Abort(dw.ctx); err != nil {
return fmt.Errorf("sstable file abort: %w", err)
}
return nil
}
func (dw *DiskWriter) Close() (*committed.WriteResult, error) {
// Before closing, we write all user supplied metadata keys and values to the hash
// This is done to avoid collisions, especially on empty sstables that might hash to the same value otherwise.
ident.MarshalStringMap(dw.hash, dw.props)
tableHash := dw.hash.Sum(nil)
sstableID := hex.EncodeToString(tableHash)
// Prepare metadata properties for Close to write. The map was already set in the
// sstable.Writer constructor and cannot be changed, but we can replace its values
// before writing it out.
first := dw.first
last := dw.last
estimatedSize := dw.w.EstimatedSize()
count := dw.count
dw.SetMetadata(MetadataFirstKey, string(first))
dw.SetMetadata(MetadataLastKey, string(last))
dw.SetMetadata(MetadataNumRecordsKey, strconv.Itoa(count))
dw.SetMetadata(MetadataEstimatedSizeKey, strconv.FormatUint(estimatedSize, 10))
if err := dw.w.Close(); err != nil {
return nil, fmt.Errorf("sstable close (%s): %w", sstableID, err)
}
if err := dw.fh.Store(dw.ctx, sstableID); err != nil {
return nil, fmt.Errorf("sstable store (%s): %w", sstableID, err)
}
dw.closed = true
return &committed.WriteResult{
RangeID: committed.ID(sstableID),
First: first,
Last: last,
Count: count,
EstimatedRangeSizeBytes: estimatedSize,
}, nil
}
// ShouldBreakAtKey returns true if it should break range after the given key
func (dw *DiskWriter) ShouldBreakAtKey(key graveler.Key, params *committed.Params) bool {
approximateSize := dw.GetApproximateSize()
if approximateSize < params.MinRangeSizeBytes {
return false
}
if approximateSize >= params.MaxRangeSizeBytes {
return true
}
h := fnv.New64a()
// FNV always reads all bytes and never fails; ignore its return values
_, _ = h.Write(key)
r := h.Sum64() % uint64(params.RangeSizeEntriesRaggedness)
return r == 0
}