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:
parent
991c9fb7dd
commit
086b44c1f4
@ -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 bucket.Put(key[:], b.Bytes())
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
return err
|
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{}{}
|
s.cache[proof.Key()] = struct{}{}
|
||||||
|
|
||||||
return nil
|
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[:])
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
return err
|
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)
|
delete(s.cache, key)
|
||||||
|
|
||||||
return nil
|
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
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user