@ignored/statemanager
Version:
An Ethereum statemanager implementation
152 lines (136 loc) • 4.99 kB
text/typescript
import { Chain, Common, Hardfork } from '@ignored/common'
import { debug as createDebugLogger } from 'debug'
import type { Cache } from './cache'
import type { AccountFields } from './interface'
import type { DefaultStateManagerOpts } from './stateManager'
import type { Account, Address } from '@ignored/util'
import type { Debugger } from 'debug'
/**
* Abstract BaseStateManager class for the non-storage-backend
* related functionality parts of a StateManager like keeping
* track of accessed storage (`EIP-2929`) or touched accounts
* (`EIP-158`).
*
* This is not a full StateManager implementation in itself but
* can be used to ease implementing an own StateManager.
*
* Note that the implementation is pretty new (October 2021)
* and we cannot guarantee a stable interface yet.
*/
export abstract class BaseStateManager {
_common: Common
_debug: Debugger
_cache!: Cache
/**
* StateManager is run in DEBUG mode (default: false)
* Taken from DEBUG environment variable
*
* Safeguards on debug() calls are added for
* performance reasons to avoid string literal evaluation
* @hidden
*/
protected readonly DEBUG: boolean = false
/**
* Needs to be called from the subclass constructor
*/
constructor(opts: DefaultStateManagerOpts) {
let common = opts.common
if (!common) {
common = new Common({ chain: Chain.Mainnet, hardfork: Hardfork.Petersburg })
}
this._common = common
// Safeguard if "process" is not available (browser)
if (typeof process?.env.DEBUG !== 'undefined') {
this.DEBUG = true
}
this._debug = createDebugLogger('statemanager:statemanager')
}
/**
* Gets the account associated with `address`. Returns an empty account if the account does not exist.
* @param address - Address of the `account` to get
*/
async getAccount(address: Address): Promise<Account> {
const account = await this._cache.getOrLoad(address)
return account
}
/**
* Saves an account into state under the provided `address`.
* @param address - Address under which to store `account`
* @param account - The account to store
*/
async putAccount(address: Address, account: Account): Promise<void> {
if (this.DEBUG) {
this._debug(
`Save account address=${address} nonce=${account.nonce} balance=${
account.balance
} contract=${account.isContract() ? 'yes' : 'no'} empty=${account.isEmpty() ? 'yes' : 'no'}`
)
}
this._cache.put(address, account)
}
/**
* Gets the account associated with `address`, modifies the given account
* fields, then saves the account into state. Account fields can include
* `nonce`, `balance`, `storageRoot`, and `codeHash`.
* @param address - Address of the account to modify
* @param accountFields - Object containing account fields and values to modify
*/
async modifyAccountFields(address: Address, accountFields: AccountFields): Promise<void> {
const account = await this.getAccount(address)
account.nonce = accountFields.nonce ?? account.nonce
account.balance = accountFields.balance ?? account.balance
account.storageRoot = accountFields.storageRoot ?? account.storageRoot
account.codeHash = accountFields.codeHash ?? account.codeHash
await this.putAccount(address, account)
}
/**
* Deletes an account from state under the provided `address`. The account will also be removed from the state trie.
* @param address - Address of the account which should be deleted
*/
async deleteAccount(address: Address) {
if (this.DEBUG) {
this._debug(`Delete account ${address}`)
}
this._cache.del(address)
}
async accountIsEmpty(address: Address): Promise<boolean> {
const account = await this.getAccount(address)
return account.isEmpty()
}
abstract putContractCode(address: Address, value: Buffer): Promise<void>
abstract getContractStorage(address: Address, key: Buffer): Promise<Buffer>
abstract putContractStorage(address: Address, key: Buffer, value: Buffer): Promise<void>
/**
* Checkpoints the current state of the StateManager instance.
* State changes that follow can then be committed by calling
* `commit` or `reverted` by calling rollback.
*
* Partial implementation, called from the subclass.
*/
async checkpoint(): Promise<void> {
this._cache.checkpoint()
}
/**
* Commits the current change-set to the instance since the
* last call to checkpoint.
*
* Partial implementation, called from the subclass.
*/
async commit(): Promise<void> {
// setup cache checkpointing
this._cache.commit()
}
/**
* Reverts the current change-set to the instance since the
* last call to checkpoint.
*
* Partial implementation , called from the subclass.
*/
async revert(): Promise<void> {
// setup cache checkpointing
this._cache.revert()
}
async flush(): Promise<void> {
await this._cache.flush()
}
}