package core import ( "sync" "github.com/CityOfZion/neo-go/pkg/core/transaction" "github.com/CityOfZion/neo-go/pkg/util" ) // Cache is data structure with fixed type key of Uint256, but has a // generic value. Used for block, tx and header cache types. type Cache struct { lock sync.RWMutex m map[util.Uint256]interface{} } // txWithHeight is an ugly wrapper to fit the needs of Blockchain's GetTransaction. type txWithHeight struct { tx *transaction.Transaction height uint32 } // NewCache returns a ready to use Cache object. func NewCache() *Cache { return &Cache{ m: make(map[util.Uint256]interface{}), } } // GetTransaction will return a Transaction type from the cache. func (c *Cache) GetTransaction(h util.Uint256) (*transaction.Transaction, uint32, bool) { c.lock.RLock() defer c.lock.RUnlock() if v, ok := c.m[h]; ok { txh, ok := v.(txWithHeight) if ok { return txh.tx, txh.height, ok } } return nil, 0, false } // GetBlock will return a Block type from the cache. func (c *Cache) GetBlock(h util.Uint256) (block *Block, ok bool) { c.lock.RLock() defer c.lock.RUnlock() return c.getBlock(h) } func (c *Cache) getBlock(h util.Uint256) (block *Block, ok bool) { if v, b := c.m[h]; b { block, ok = v.(*Block) return } return } // Add adds the given hash along with its value to the cache. func (c *Cache) Add(h util.Uint256, v interface{}) { c.lock.Lock() defer c.lock.Unlock() c.add(h, v) } func (c *Cache) add(h util.Uint256, v interface{}) { c.m[h] = v block, ok := v.(*Block) if ok { for _, tx := range block.Transactions { c.m[tx.Hash()] = txWithHeight{tx, block.Index} } } } func (c *Cache) has(h util.Uint256) bool { _, ok := c.m[h] return ok } // Has returns whether the cache contains the given hash. func (c *Cache) Has(h util.Uint256) bool { c.lock.Lock() defer c.lock.Unlock() return c.has(h) } // Len return the number of items present in the cache. func (c *Cache) Len() int { c.lock.RLock() defer c.lock.RUnlock() return len(c.m) } // Delete removes the item out of the cache. func (c *Cache) Delete(h util.Uint256) { c.lock.Lock() defer c.lock.Unlock() block, ok := c.m[h].(*Block) if ok { for _, tx := range block.Transactions { delete(c.m, tx.Hash()) } } delete(c.m, h) } // ReapStrangeBlocks drops blocks from cache that don't fit into the // blkHeight-headHeight interval. Cache should only contain blocks that we // expect to get and store. func (c *Cache) ReapStrangeBlocks(blkHeight, headHeight uint32) { c.lock.Lock() defer c.lock.Unlock() for i, b := range c.m { block, ok := b.(*Block) if ok && (block.Index < blkHeight || block.Index > headHeight) { delete(c.m, i) } } }