|
|
package txselector
import ( "sort"
"github.com/hermeznetwork/hermez-node/common" "github.com/hermeznetwork/hermez-node/db/l2db" "github.com/hermeznetwork/hermez-node/db/statedb" )
// txs implements the interface Sort for an array of Tx
type txs []*common.PoolL2Tx
func (t txs) Len() int { return len(t) } func (t txs) Swap(i, j int) { t[i], t[j] = t[j], t[i] } func (t txs) Less(i, j int) bool { return t[i].AbsoluteFee > t[j].AbsoluteFee }
// TxSelector implements all the functionalities to select the txs for the next batch
type TxSelector struct { // MaxL1UserTxs is the maximum L1-user-tx for a batch
MaxL1UserTxs uint64 // MaxL1OperatorTxs is the maximum L1-operator-tx for a batch
MaxL1OperatorTxs uint64 // MaxTxs is the maximum txs for a batch
MaxTxs uint64
l2db *l2db.L2DB localAccountsDB *statedb.LocalStateDB }
// NewTxSelector returns a *TxSelector
func NewTxSelector(dbpath string, synchronizerStateDB *statedb.StateDB, l2 *l2db.L2DB, maxL1UserTxs, maxL1OperatorTxs, maxTxs uint64) (*TxSelector, error) { localAccountsDB, err := statedb.NewLocalStateDB(dbpath, synchronizerStateDB, false, 0) // without merkletree
if err != nil { return nil, err }
return &TxSelector{ MaxL1UserTxs: maxL1UserTxs, MaxL1OperatorTxs: maxL1OperatorTxs, MaxTxs: maxTxs, l2db: l2, localAccountsDB: localAccountsDB, }, nil }
// Reset tells the TxSelector to get it's internal AccountsDB
// from the required `batchNum`
func (txsel *TxSelector) Reset(batchNum uint64) error { err := txsel.localAccountsDB.Reset(batchNum, true) if err != nil { return err } return nil }
// GetL2TxSelection returns a selection of the L2Txs for the next batch, from the L2DB pool
func (txsel *TxSelector) GetL2TxSelection(batchNum uint64) ([]*common.PoolL2Tx, error) { // get pending l2-tx from tx-pool
l2TxsRaw, err := txsel.l2db.GetPendingTxs() // once l2db ready, maybe use parameter 'batchNum'
if err != nil { return nil, err }
// discard the txs that don't have an Account in the AccountDB
var validTxs txs for _, tx := range l2TxsRaw { _, err = txsel.localAccountsDB.GetAccount(tx.FromIdx) if err == nil { // if FromIdx has an account into the AccountsDB
validTxs = append(validTxs, tx) } }
// get most profitable L2-tx
txs := txsel.getL2Profitable(validTxs, txsel.MaxTxs)
// apply L2-tx to local AccountDB, make checkpoint tagged with BatchID
// update balances
// update nonces
// return selected txs
return txs, nil }
// GetL1L2TxSelection returns the selection of L1 + L2 txs
func (txsel *TxSelector) GetL1L2TxSelection(batchNum uint64, l1txs []*common.L1Tx) ([]*common.L1Tx, []*common.L1Tx, []*common.PoolL2Tx, error) { // apply l1-user-tx to localAccountDB
// create new leaves
// update balances
// update nonces
// get pending l2-tx from tx-pool
l2TxsRaw, err := txsel.l2db.GetPendingTxs() // (batchID)
if err != nil { return nil, nil, nil, err }
// discard the txs that don't have an Account in the AccountDB
// neither appear in the AccountCreationAuthsDB
var validTxs txs for _, tx := range l2TxsRaw { if txsel.checkIfAccountExistOrPending(tx.FromIdx) { // if FromIdx has an account into the AccountsDB
validTxs = append(validTxs, tx) } }
// prepare (from the selected l2txs) pending to create from the AccountCreationAuthsDB
var accountCreationAuths []*common.Account // TODO once DB ready:
// if tx.ToIdx is in AccountCreationAuthsDB, take it and add it to
// the array 'accountCreationAuths'
// for _, tx := range l2txs {
// account, err := txsel.localAccountsDB.GetAccount(tx.ToIdx)
// if err != nil {
// return nil, nil, nil, err
// }
// if accountToCreate, ok := txsel.DB.AccountCreationAuthsDB[accountID]; ok {
// accountCreationAuths = append(accountCreationAuths, accountToCreate)
// }
// }
// create L1-operator-tx for each L2-tx selected in which the recipient does not have account
l1OperatorTxs := txsel.createL1OperatorTxForL2Tx(accountCreationAuths) // only with the L2-tx selected ones
// get most profitable L2-tx
maxL2Txs := txsel.MaxTxs - uint64(len(l1OperatorTxs)) // - len(l1UserTxs)
l2txs := txsel.getL2Profitable(validTxs, maxL2Txs)
return l1txs, l1OperatorTxs, l2txs, nil }
func (txsel *TxSelector) checkIfAccountExistOrPending(idx common.Idx) bool { // check if account exist in AccountDB
_, err := txsel.localAccountsDB.GetAccount(idx) if err != nil { return false } // check if account is pending to create
// TODO need a method in the DB to get the PendingRegisters
// if _, ok := txsel.DB.AccountCreationAuthsDB[accountID]; ok {
// return true
// }
return false }
func (txsel *TxSelector) getL2Profitable(txs txs, max uint64) txs { sort.Sort(txs) return txs[:max] } func (txsel *TxSelector) createL1OperatorTxForL2Tx(accounts []*common.Account) []*common.L1Tx { //
return nil }
|