arc_cache.go 4.16 KB
Newer Older
1 2 3
package blockstore

import (
4
	"context"
5

6 7 8 9
	lru "github.com/hashicorp/golang-lru"
	blocks "github.com/ipfs/go-block-format"
	cid "github.com/ipfs/go-cid"
	metrics "github.com/ipfs/go-metrics-interface"
10 11
)

12 13 14
type cacheHave bool
type cacheSize int

15 16 17
// arccache wraps a BlockStore with an Adaptive Replacement Cache (ARC) for
// block Cids. This provides block access-time improvements, allowing
// to short-cut many searches without query-ing the underlying datastore.
18
type arccache struct {
Steven Allen's avatar
Steven Allen committed
19
	arc        *lru.TwoQueueCache
20
	blockstore Blockstore
21 22 23

	hits  metrics.Counter
	total metrics.Counter
24 25
}

26
func newARCCachedBS(ctx context.Context, bs Blockstore, lruSize int) (*arccache, error) {
Steven Allen's avatar
Steven Allen committed
27
	arc, err := lru.New2Q(lruSize)
28 29 30
	if err != nil {
		return nil, err
	}
31 32 33
	c := &arccache{arc: arc, blockstore: bs}
	c.hits = metrics.NewCtx(ctx, "arc.hits_total", "Number of ARC cache hits").Counter()
	c.total = metrics.NewCtx(ctx, "arc_total", "Total number of ARC cache requests").Counter()
34

35
	return c, nil
36 37
}

38
func (b *arccache) DeleteBlock(k cid.Cid) error {
39
	if has, _, ok := b.hasCached(k); ok && !has {
40 41 42 43 44 45
		return ErrNotFound
	}

	b.arc.Remove(k) // Invalidate cache before deleting.
	err := b.blockstore.DeleteBlock(k)
	switch err {
46
	case nil, ErrNotFound:
47
		b.cacheHave(k, false)
48
		return err
49 50 51 52 53 54 55
	default:
		return err
	}
}

// if ok == false has is inconclusive
// if ok == true then has respons to question: is it contained
56
func (b *arccache) hasCached(k cid.Cid) (has bool, size int, ok bool) {
57
	b.total.Inc()
58 59
	if !k.Defined() {
		log.Error("undefined cid in arccache")
60 61
		// Return cache invalid so the call to blockstore happens
		// in case of invalid key and correct error is created.
62
		return false, -1, false
63
	}
64

65
	h, ok := b.arc.Get(k.KeyString())
66
	if ok {
67
		b.hits.Inc()
68 69 70 71 72
		switch h := h.(type) {
		case cacheHave:
			return bool(h), -1, true
		case cacheSize:
			return true, int(h), true
73
		}
74
	}
75
	return false, -1, false
76 77
}

78
func (b *arccache) Has(k cid.Cid) (bool, error) {
79 80 81 82 83 84
	if has, _, ok := b.hasCached(k); ok {
		return has, nil
	}
	has, err := b.blockstore.Has(k)
	if err != nil {
		return false, err
85
	}
86 87
	b.cacheHave(k, has)
	return has, nil
88
}
89

90
func (b *arccache) GetSize(k cid.Cid) (int, error) {
91 92 93 94 95
	if has, blockSize, ok := b.hasCached(k); ok {
		if has {
			return blockSize, nil
		}
		return -1, ErrNotFound
96 97
	}
	blockSize, err := b.blockstore.GetSize(k)
98
	if err == ErrNotFound {
99
		b.cacheHave(k, false)
100
	} else if err == nil {
101
		b.cacheSize(k, blockSize)
102
	}
103
	return blockSize, err
104 105
}

106 107 108
func (b *arccache) Get(k cid.Cid) (blocks.Block, error) {
	if !k.Defined() {
		log.Error("undefined cid in arc cache")
109 110 111
		return nil, ErrNotFound
	}

112
	if has, _, ok := b.hasCached(k); ok && !has {
113 114 115 116 117
		return nil, ErrNotFound
	}

	bl, err := b.blockstore.Get(k)
	if bl == nil && err == ErrNotFound {
118
		b.cacheHave(k, false)
119
	} else if bl != nil {
120
		b.cacheSize(k, len(bl.RawData()))
121 122 123 124 125
	}
	return bl, err
}

func (b *arccache) Put(bl blocks.Block) error {
126
	if has, _, ok := b.hasCached(bl.Cid()); ok && has {
127 128 129 130 131
		return nil
	}

	err := b.blockstore.Put(bl)
	if err == nil {
132
		b.cacheSize(bl.Cid(), len(bl.RawData()))
133 134 135 136 137 138 139
	}
	return err
}

func (b *arccache) PutMany(bs []blocks.Block) error {
	var good []blocks.Block
	for _, block := range bs {
140 141
		// call put on block if result is inconclusive or we are sure that
		// the block isn't in storage
142
		if has, _, ok := b.hasCached(block.Cid()); !ok || (ok && !has) {
143 144 145
			good = append(good, block)
		}
	}
146
	err := b.blockstore.PutMany(good)
147 148
	if err != nil {
		return err
149
	}
150
	for _, block := range good {
151
		b.cacheSize(block.Cid(), len(block.RawData()))
152 153
	}
	return nil
154 155
}

156 157 158 159
func (b *arccache) HashOnRead(enabled bool) {
	b.blockstore.HashOnRead(enabled)
}

160
func (b *arccache) cacheHave(c cid.Cid, have bool) {
161 162 163
	b.arc.Add(c.KeyString(), cacheHave(have))
}

164
func (b *arccache) cacheSize(c cid.Cid, blockSize int) {
165
	b.arc.Add(c.KeyString(), cacheSize(blockSize))
166 167
}

168
func (b *arccache) AllKeysChan(ctx context.Context) (<-chan cid.Cid, error) {
169 170 171 172 173 174 175 176 177 178 179 180 181 182
	return b.blockstore.AllKeysChan(ctx)
}

func (b *arccache) GCLock() Unlocker {
	return b.blockstore.(GCBlockstore).GCLock()
}

func (b *arccache) PinLock() Unlocker {
	return b.blockstore.(GCBlockstore).PinLock()
}

func (b *arccache) GCRequested() bool {
	return b.blockstore.(GCBlockstore).GCRequested()
}