forked from hardcore-os/corekv
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdb.go
406 lines (369 loc) · 9.04 KB
/
db.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
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
package corekv
import (
"expvar"
"fmt"
"math"
"sync"
"sync/atomic"
"time"
"github.com/hardcore-os/corekv/lsm"
"github.com/hardcore-os/corekv/utils"
"github.com/pkg/errors"
)
type (
// coreKV对外提供的功能集合
CoreAPI interface {
Set(data *utils.Entry) error
Get(key []byte) (*utils.Entry, error)
Del(key []byte) error
NewIterator(opt *utils.Options) utils.Iterator
Info() *Stats
Close() error
}
// DB 对外暴露的接口对象 全局唯一,持有各种资源句柄
DB struct {
sync.RWMutex
opt *Options
lsm *lsm.LSM
vlog *valueLog
stats *Stats
flushChan chan flushTask // For flushing memtables.
writeCh chan *request
blockWrites int32
vhead *utils.ValuePtr
logRotates int32
}
)
var (
head = []byte("!corekv!head") // For storing value offset for replay.
)
/**
SSTableMaxSz: 1024,
MemTableSize: 1024,
BlockSize: 1024,
BloomFalsePositive: 0,
BaseLevelSize: 10 << 20,
LevelSizeMultiplier: 10,
BaseTableSize: 2 << 20,
TableSizeMultiplier: 2,
NumLevelZeroTables: 15,
MaxLevelNum: 7,
NumCompactors: 3,
*/
// Open DB
// TODO 这里是不是要上一个目录锁比较好,防止多个进程打开同一个目录?
func Open(opt *Options) *DB {
c := utils.NewCloser()
db := &DB{opt: opt}
// 初始化vlog结构
db.initVLog()
// 初始化LSM结构
db.lsm = lsm.NewLSM(&lsm.Options{
WorkDir: opt.WorkDir,
MemTableSize: opt.MemTableSize,
SSTableMaxSz: opt.SSTableMaxSz,
BlockSize: 8 * 1024,
BloomFalsePositive: 0, //0.01,
BaseLevelSize: 10 << 20,
LevelSizeMultiplier: 10,
BaseTableSize: 5 << 20,
TableSizeMultiplier: 2,
NumLevelZeroTables: 15,
MaxLevelNum: 7,
NumCompactors: 1,
DiscardStatsCh: &(db.vlog.lfDiscardStats.flushChan),
})
// 初始化统计信息
db.stats = newStats(opt)
// 启动 sstable 的合并压缩过程
go db.lsm.StartCompacter()
// 准备vlog gc
c.Add(1)
db.writeCh = make(chan *request)
db.flushChan = make(chan flushTask, 16)
go db.doWrites(c)
// 启动 info 统计过程
go db.stats.StartStats()
return db
}
func (db *DB) Close() error {
db.vlog.lfDiscardStats.closer.Close()
if err := db.lsm.Close(); err != nil {
return err
}
if err := db.vlog.close(); err != nil {
return err
}
if err := db.stats.close(); err != nil {
return err
}
return nil
}
func (db *DB) Del(key []byte) error {
// 写入一个值为nil的entry 作为墓碑消息实现删除
return db.Set(&utils.Entry{
Key: key,
Value: nil,
ExpiresAt: 0,
})
}
func (db *DB) Set(data *utils.Entry) error {
if data == nil || len(data.Key) == 0 {
return utils.ErrEmptyKey
}
// 做一些必要性的检查
// 如果value 大于一个阈值 则创建值指针,并将其写入vlog中
var (
vp *utils.ValuePtr
err error
)
data.Key = utils.KeyWithTs(data.Key, math.MaxUint32)
// 如果value不应该直接写入LSM 则先写入 vlog文件,这时必须保证vlog具有重放功能
// 以便于崩溃后恢复数据
if !db.shouldWriteValueToLSM(data) {
if vp, err = db.vlog.newValuePtr(data); err != nil {
return err
}
data.Meta |= utils.BitValuePointer
data.Value = vp.Encode()
}
return db.lsm.Set(data)
}
func (db *DB) Get(key []byte) (*utils.Entry, error) {
if len(key) == 0 {
return nil, utils.ErrEmptyKey
}
originKey := key
var (
entry *utils.Entry
err error
)
key = utils.KeyWithTs(key, math.MaxUint32)
// 从LSM中查询entry,这时不确定entry是不是值指针
if entry, err = db.lsm.Get(key); err != nil {
return entry, err
}
// 检查从lsm拿到的value是否是value ptr,是则从vlog中拿值
if entry != nil && utils.IsValuePtr(entry) {
var vp utils.ValuePtr
vp.Decode(entry.Value)
result, cb, err := db.vlog.read(&vp)
defer utils.RunCallback(cb)
if err != nil {
return nil, err
}
entry.Value = utils.SafeCopy(nil, result)
}
if lsm.IsDeletedOrExpired(entry) {
return nil, utils.ErrKeyNotFound
}
entry.Key = originKey
return entry, nil
}
func (db *DB) Info() *Stats {
// 读取stats结构,打包数据并返回
return db.stats
}
// RunValueLogGC triggers a value log garbage collection.
func (db *DB) RunValueLogGC(discardRatio float64) error {
if discardRatio >= 1.0 || discardRatio <= 0.0 {
return utils.ErrInvalidRequest
}
// Find head on disk
headKey := utils.KeyWithTs(head, math.MaxUint64)
val, err := db.lsm.Get(headKey)
if err != nil {
if err == utils.ErrKeyNotFound {
val = &utils.Entry{
Key: headKey,
Value: []byte{},
}
} else {
return errors.Wrap(err, "Retrieving head from on-disk LSM")
}
}
// 内部key head 一定是value ptr 不需要检查内容
var head utils.ValuePtr
if len(val.Value) > 0 {
head.Decode(val.Value)
}
// Pick a log file and run GC
return db.vlog.runGC(discardRatio, &head)
}
func (db *DB) shouldWriteValueToLSM(e *utils.Entry) bool {
return int64(len(e.Value)) < db.opt.ValueThreshold
}
func (db *DB) sendToWriteCh(entries []*utils.Entry) (*request, error) {
if atomic.LoadInt32(&db.blockWrites) == 1 {
return nil, utils.ErrBlockedWrites
}
var count, size int64
for _, e := range entries {
size += int64(e.EstimateSize(int(db.opt.ValueThreshold)))
count++
}
if count >= db.opt.MaxBatchCount || size >= db.opt.MaxBatchSize {
return nil, utils.ErrTxnTooBig
}
// TODO 尝试使用对象复用,后面entry对象也应该使用
req := requestPool.Get().(*request)
req.reset()
req.Entries = entries
req.Wg.Add(1)
req.IncrRef() // for db write
db.writeCh <- req // Handled in doWrites.
return req, nil
}
// Check(kv.BatchSet(entries))
func (db *DB) batchSet(entries []*utils.Entry) error {
req, err := db.sendToWriteCh(entries)
if err != nil {
return err
}
return req.Wait()
}
func (db *DB) doWrites(lc *utils.Closer) {
defer lc.Done()
pendingCh := make(chan struct{}, 1)
writeRequests := func(reqs []*request) {
if err := db.writeRequests(reqs); err != nil {
utils.Err(fmt.Errorf("writeRequests: %v", err))
}
<-pendingCh
}
// This variable tracks the number of pending writes.
reqLen := new(expvar.Int)
reqs := make([]*request, 0, 10)
for {
var r *request
select {
case r = <-db.writeCh:
case <-lc.CloseSignal:
goto closedCase
}
for {
reqs = append(reqs, r)
reqLen.Set(int64(len(reqs)))
if len(reqs) >= 3*utils.KVWriteChCapacity {
pendingCh <- struct{}{} // blocking.
goto writeCase
}
select {
// Either push to pending, or continue to pick from writeCh.
case r = <-db.writeCh:
case pendingCh <- struct{}{}:
goto writeCase
case <-lc.CloseSignal:
goto closedCase
}
}
closedCase:
// All the pending request are drained.
// Don't close the writeCh, because it has be used in several places.
for {
select {
case r = <-db.writeCh:
reqs = append(reqs, r)
default:
pendingCh <- struct{}{} // Push to pending before doing a write.
writeRequests(reqs)
return
}
}
writeCase:
go writeRequests(reqs)
reqs = make([]*request, 0, 10)
reqLen.Set(0)
}
}
// writeRequests is called serially by only one goroutine.
func (db *DB) writeRequests(reqs []*request) error {
if len(reqs) == 0 {
return nil
}
done := func(err error) {
for _, r := range reqs {
r.Err = err
r.Wg.Done()
}
}
err := db.vlog.write(reqs)
if err != nil {
done(err)
return err
}
var count int
for _, b := range reqs {
if len(b.Entries) == 0 {
continue
}
count += len(b.Entries)
if err != nil {
done(err)
return errors.Wrap(err, "writeRequests")
}
if err := db.writeToLSM(b); err != nil {
done(err)
return errors.Wrap(err, "writeRequests")
}
db.Lock()
db.updateHead(b.Ptrs)
db.Unlock()
}
done(nil)
return nil
}
func (db *DB) writeToLSM(b *request) error {
if len(b.Ptrs) != len(b.Entries) {
return errors.Errorf("Ptrs and Entries don't match: %+v", b)
}
for i, entry := range b.Entries {
if db.shouldWriteValueToLSM(entry) { // Will include deletion / tombstone case.
entry.Meta = entry.Meta &^ utils.BitValuePointer
} else {
entry.Meta = entry.Meta | utils.BitValuePointer
entry.Value = b.Ptrs[i].Encode()
}
db.lsm.Set(entry)
}
return nil
}
func (req *request) IncrRef() {
atomic.AddInt32(&req.ref, 1)
}
func (req *request) DecrRef() {
nRef := atomic.AddInt32(&req.ref, -1)
if nRef > 0 {
return
}
req.Entries = nil
requestPool.Put(req)
}
func (req *request) Wait() error {
req.Wg.Wait()
err := req.Err
req.DecrRef() // DecrRef after writing to DB.
return err
}
// 结构体
type flushTask struct {
mt *utils.Skiplist
vptr *utils.ValuePtr
dropPrefixes [][]byte
}
func (db *DB) pushHead(ft flushTask) error {
// Ensure we never push a zero valued head pointer.
if ft.vptr.IsZero() {
return errors.New("Head should not be zero")
}
fmt.Printf("Storing value log head: %+v\n", ft.vptr)
val := ft.vptr.Encode()
// Pick the max commit ts, so in case of crash, our read ts would be higher than all the
// commits.
headTs := utils.KeyWithTs(head, uint64(time.Now().Unix()/1e9))
ft.mt.Add(&utils.Entry{
Key: headTs,
Value: val,
})
return nil
}