channeldb/waitingproof: improve external consistency of store

This commit synchronizes the in-memory cache with the
on-disk state to ensure the waiting proof store is
externally consistent. Currently, there are scenarios
where the in-memory state is updated, and not reverted
if the write fails. The general fix is to wait to apply
modifications until the write succeeds, and use a
read/write lock to synchronize access with db operations.
This commit is contained in:
Conner Fromknecht 2018-06-04 19:20:45 -07:00
parent 991c9fb7dd
commit 086b44c1f4
No known key found for this signature in database
GPG Key ID: 39DE78FBE6ACB0EF

@ -2,6 +2,7 @@ package channeldb
import ( import (
"encoding/binary" "encoding/binary"
"sync"
"io" "io"
@ -35,6 +36,7 @@ type WaitingProofStore struct {
// calls, when object isn't stored in it. // calls, when object isn't stored in it.
cache map[WaitingProofKey]struct{} cache map[WaitingProofKey]struct{}
db *DB db *DB
mu sync.RWMutex
} }
// NewWaitingProofStore creates new instance of proofs storage. // NewWaitingProofStore creates new instance of proofs storage.
@ -56,7 +58,10 @@ func NewWaitingProofStore(db *DB) (*WaitingProofStore, error) {
// Add adds new waiting proof in the storage. // Add adds new waiting proof in the storage.
func (s *WaitingProofStore) Add(proof *WaitingProof) error { func (s *WaitingProofStore) Add(proof *WaitingProof) error {
return s.db.Batch(func(tx *bolt.Tx) error { s.mu.Lock()
defer s.mu.Unlock()
err := s.db.Update(func(tx *bolt.Tx) error {
var err error var err error
var b bytes.Buffer var b bytes.Buffer
@ -72,36 +77,47 @@ func (s *WaitingProofStore) Add(proof *WaitingProof) error {
} }
key := proof.Key() key := proof.Key()
if err := bucket.Put(key[:], b.Bytes()); err != nil {
return err
}
s.cache[proof.Key()] = struct{}{} return bucket.Put(key[:], b.Bytes())
return nil
}) })
if err != nil {
return err
}
// Knowing that the write succeeded, we can now update the in-memory
// cache with the proof's key.
s.cache[proof.Key()] = struct{}{}
return nil
} }
// Remove removes the proof from storage by its key. // Remove removes the proof from storage by its key.
func (s *WaitingProofStore) Remove(key WaitingProofKey) error { func (s *WaitingProofStore) Remove(key WaitingProofKey) error {
s.mu.Lock()
defer s.mu.Unlock()
if _, ok := s.cache[key]; !ok { if _, ok := s.cache[key]; !ok {
return ErrWaitingProofNotFound return ErrWaitingProofNotFound
} }
return s.db.Batch(func(tx *bolt.Tx) error { err := s.db.Update(func(tx *bolt.Tx) error {
// Get or create the top bucket. // Get or create the top bucket.
bucket := tx.Bucket(waitingProofsBucketKey) bucket := tx.Bucket(waitingProofsBucketKey)
if bucket == nil { if bucket == nil {
return ErrWaitingProofNotFound return ErrWaitingProofNotFound
} }
if err := bucket.Delete(key[:]); err != nil { return bucket.Delete(key[:])
return err
}
delete(s.cache, key)
return nil
}) })
if err != nil {
return err
}
// Since the proof was successfully deleted from the store, we can now
// remove it from the in-memory cache.
delete(s.cache, key)
return nil
} }
// ForAll iterates thought all waiting proofs and passing the waiting proof // ForAll iterates thought all waiting proofs and passing the waiting proof
@ -135,6 +151,9 @@ func (s *WaitingProofStore) ForAll(cb func(*WaitingProof) error) error {
func (s *WaitingProofStore) Get(key WaitingProofKey) (*WaitingProof, error) { func (s *WaitingProofStore) Get(key WaitingProofKey) (*WaitingProof, error) {
proof := &WaitingProof{} proof := &WaitingProof{}
s.mu.RLock()
defer s.mu.RUnlock()
if _, ok := s.cache[key]; !ok { if _, ok := s.cache[key]; !ok {
return nil, ErrWaitingProofNotFound return nil, ErrWaitingProofNotFound
} }