package statedb import ( "errors" "github.com/hermeznetwork/hermez-node/common" "github.com/iden3/go-merkletree" "github.com/iden3/go-merkletree/db" "github.com/iden3/go-merkletree/db/leveldb" "github.com/iden3/go-merkletree/db/memory" ) // ErrStateDBWithoutMT is used when a method that requires a MerkleTree is called in a StateDB that does not have a MerkleTree defined var ErrStateDBWithoutMT = errors.New("Can not call method to use MerkleTree in a StateDB without MerkleTree") // ErrAccountAlreadyExists is used when CreateAccount is called and the Account already exists var ErrAccountAlreadyExists = errors.New("Can not CreateAccount because Account already exists") // StateDB represents the StateDB object type StateDB struct { db db.Storage mt *merkletree.MerkleTree } // NewStateDB creates a new StateDB, allowing to use an in-memory or in-disk // storage func NewStateDB(path string, inDisk bool, withMT bool, nLevels int) (*StateDB, error) { var sto db.Storage var err error if inDisk { sto, err = leveldb.NewLevelDbStorage(path, false) if err != nil { return nil, err } } else { sto = memory.NewMemoryStorage() } var mt *merkletree.MerkleTree = nil if withMT { mt, err = merkletree.NewMerkleTree(sto, nLevels) if err != nil { return nil, err } } return &StateDB{ db: sto, mt: mt, }, nil } // CheckPointAt does a checkpoint at the given batchNum in the defined path func (s *StateDB) CheckPointAt(batchNum int, path string) error { // TODO return nil } // Reset resets the StateDB to the checkpoint at the given batchNum func (s *StateDB) Reset(batchNum int) error { // TODO return nil } // Checkpoints returns a list of the checkpoints (batchNums) func (s *StateDB) Checkpoints() ([]int, error) { // TODO //batchnums, err return nil, nil } // GetAccount returns the account for the given Idx func (s *StateDB) GetAccount(idx common.Idx) (*common.Account, error) { vBytes, err := s.db.Get(idx.Bytes()) if err != nil { return nil, err } accBytes, err := s.db.Get(vBytes) if err != nil { return nil, err } var b [32 * common.NLEAFELEMS]byte copy(b[:], accBytes) return common.AccountFromBytes(b) } // CreateAccount creates a new Account in the StateDB for the given Idx. // MerkleTree is not affected. func (s *StateDB) CreateAccount(idx common.Idx, account *common.Account) error { // store at the DB the key: v, and value: leaf.Bytes() v, err := account.HashValue() if err != nil { return err } accountBytes, err := account.Bytes() if err != nil { return err } // store the Leaf value tx, err := s.db.NewTx() if err != nil { return err } _, err = tx.Get(idx.Bytes()) if err != db.ErrNotFound { return ErrAccountAlreadyExists } tx.Put(v.Bytes(), accountBytes[:]) tx.Put(idx.Bytes(), v.Bytes()) return tx.Commit() } // UpdateAccount updates the Account in the StateDB for the given Idx. // MerkleTree is not affected. func (s *StateDB) UpdateAccount(idx common.Idx, account *common.Account) error { // store at the DB the key: v, and value: leaf.Bytes() v, err := account.HashValue() if err != nil { return err } accountBytes, err := account.Bytes() if err != nil { return err } tx, err := s.db.NewTx() if err != nil { return err } tx.Put(v.Bytes(), accountBytes[:]) tx.Put(idx.Bytes(), v.Bytes()) return tx.Commit() } // MTCreateAccount creates a new Account in the StateDB for the given Idx, // and updates the MerkleTree, returning a CircomProcessorProof func (s *StateDB) MTCreateAccount(idx common.Idx, account *common.Account) (*merkletree.CircomProcessorProof, error) { if s.mt == nil { return nil, ErrStateDBWithoutMT } err := s.CreateAccount(idx, account) if err != nil { return nil, err } v, err := account.HashValue() // already computed in s.CreateAccount, next iteration reuse first computation if err != nil { return nil, err } // Add k & v into the MT return s.mt.AddAndGetCircomProof(idx.BigInt(), v) } // MTUpdateAccount updates the Account in the StateDB for the given Idx, and // updates the MerkleTree, returning a CircomProcessorProof func (s *StateDB) MTUpdateAccount(idx common.Idx, account *common.Account) (*merkletree.CircomProcessorProof, error) { if s.mt == nil { return nil, ErrStateDBWithoutMT } err := s.UpdateAccount(idx, account) if err != nil { return nil, err } v, err := account.HashValue() // already computed in s.CreateAccount, next iteration reuse first computation if err != nil { return nil, err } // Add k & v into the MT return s.mt.Update(idx.BigInt(), v) } // MTGetProof returns the CircomVerifierProof for a given Idx func (s *StateDB) MTGetProof(idx common.Idx) (*merkletree.CircomVerifierProof, error) { if s.mt == nil { return nil, ErrStateDBWithoutMT } return s.mt.GenerateCircomVerifierProof(idx.BigInt(), s.mt.Root()) } // LocalStateDB represents the local StateDB which allows to make copies from // the synchronizer StateDB, and is used by the tx-selector and the // batch-builder. LocalStateDB is an in-memory storage. type LocalStateDB struct { *StateDB synchronizerStateDB *StateDB } // NewLocalStateDB returns a new LocalStateDB connected to the given // synchronizerDB func NewLocalStateDB(synchronizerDB *StateDB, withMT bool, nLevels int) (*LocalStateDB, error) { s, err := NewStateDB("", false, withMT, nLevels) if err != nil { return nil, err } return &LocalStateDB{ s, synchronizerDB, }, nil } // Reset performs a reset, getting the state from // LocalStateDB.synchronizerStateDB for the given batchNum func (l *LocalStateDB) Reset(batchNum int, fromSynchronizer bool) error { // TODO // if fromSynchronizer==true: // make copy from l.synchronizerStateDB at the batchNum to the localStateDB // if synchronizerStateDB does not have batchNum, return err // the localStateDB checkpoint is set to batchNum // else fromSynchronizer==false: // the localStateDB checkpoint is set to batchNum // if localStateDB does not have batchNum, return err return nil }