UNPKG

@ignored/statemanager

Version:
152 lines (136 loc) 4.99 kB
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() } }