diff --git a/packages/base-contract/src/index.ts b/packages/base-contract/src/index.ts index b12796c9bd..09a1dacb14 100644 --- a/packages/base-contract/src/index.ts +++ b/packages/base-contract/src/index.ts @@ -16,7 +16,7 @@ import { } from 'ethereum-types'; import Account from 'ethereumjs-account'; import * as util from 'ethereumjs-util'; -import * as ethereumJsVm from 'ethereumjs-vm'; +import { default as VM } from 'ethereumjs-vm'; import PStateManager from 'ethereumjs-vm/dist/state/promisified'; import * as ethers from 'ethers'; import * as _ from 'lodash'; @@ -31,7 +31,6 @@ export interface AbiEncoderByFunctionSignature { [key: string]: AbiEncoder.Method; } -const VM = ethereumJsVm.default; const ARBITRARY_PRIVATE_KEY = 'e331b6d69882b4cb4ea581d88e0b604039a3de5967688d3dcffdd2270c0fd109'; // tslint:disable: max-classes-per-file diff --git a/packages/base-contract/tsconfig.json b/packages/base-contract/tsconfig.json index 718e623c74..bf70fcb4cd 100644 --- a/packages/base-contract/tsconfig.json +++ b/packages/base-contract/tsconfig.json @@ -4,5 +4,6 @@ "outDir": "lib", "rootDir": "." }, + "typeRoots": ["node_modules/@0x/typescript-typings/types", "node_modules/@types"], "include": ["src/**/*", "test/**/*"] } diff --git a/packages/typescript-typings/types/ethereumjs-vm/index.d.ts b/packages/typescript-typings/types/ethereumjs-vm/index.d.ts new file mode 100644 index 0000000000..594fe1a91a --- /dev/null +++ b/packages/typescript-typings/types/ethereumjs-vm/index.d.ts @@ -0,0 +1,334 @@ +declare module 'ethereumjs-vm' { + import BN = require('bn.js'); + type Common = any; // from ethereumjs-common + type Account = any; // from ethereumjs-account + type Blockchain = any; // from ethereumjs-blockchain + + export default class VM { + opts: VmOpts; + stateManager: StateManager; + constructor(opts: VmOpts); + runCall(opts: RunCallOpts): Promise; + } + + export interface VMOpts { + chain: string; + hardfork: string; + stateManager: StateManager; + state: any; + blockchain: Blockchain; + activatePrecompiles: boolean; + allowUnlimitedContractSize: boolean; + common: Common; + } + + export interface RunCallOpts { + block?: any; + gasPrice?: Buffer; + origin?: Buffer; + caller?: Buffer; + gasLimit?: Buffer; + to?: Buffer; + value?: Buffer; + data?: Buffer; + code?: Buffer; + depth?: number; + compiled?: boolean; + static?: boolean; + salt?: Buffer; + selfdestruct?: { [k: string]: boolean }; + delegatecall?: boolean; + } + + export interface EVMResult { + gasUsed: BN; + createdAddress?: Buffer; + execResult: ExecResult; + } + + interface ExecResult { + runState?: RunState; + exceptionError?: VmError; + gas?: BN; + gasUsed: BN; + returnValue: Buffer; + logs?: any[]; + gasRefund?: BN; + selfdestruct?: { [k: string]: Buffer }; + } + interface RunState { + programCounter: number; + opCode: number; + memory: Memory; + memoryWordCount: BN; + highestMemCost: BN; + stack: Stack; + code: Buffer; + validJumps: number[]; + _common: Common; + stateManager: StateManager; + eei: EEI; + } + + class Memory { + _store: number[]; + constructor(); + extend(offset: number, size: number): void; + write(offset: number, size: number, value: Buffer): void; + read(offset: number, size: number): Buffer; + } + + class Stack { + _store: BN[]; + constructor(); + length(): number; + push(value: BN): void; + pop(): BN; + popN(num: number): BN[]; + swap(position: number): void; + dup(position: number): void; + } + + export class StateManager { + _common: Common; + _trie: any; + _storageTries: any; + _cache: Cache; + _touched: Set; + _touchedStack: Set[]; + _checkpointCount: number; + _originalStorageCache: Map>; + + constructor(opts: StateManagerOpts); + copy(): StateManager; + getAccount(address: Buffer, cb: any): void; + putAccount(address: Buffer, account: Account, cb: any): void; + putContractCode(address: Buffer, value: Buffer, cb: any): void; + getContractCode(address: Buffer, cb: any): void; + _lookupStorageTrie(address: Buffer, cb: any): void; + _getStorageTrie(address: Buffer, cb: any): void; + getContractStorage(address: Buffer, key: Buffer, cb: any): void; + getOriginalContractStorage(address: Buffer, key: Buffer, cb: any): void; + _modifyContractStorage(address: Buffer, modifyTrie: any, cb: any): void; + putContractStorage(address: Buffer, key: Buffer, value: Buffer, cb: any): void; + clearContractStorage(address: Buffer, cb: any): void; + checkpoint(cb: any): void; + commit(cb: any): void; + revert(cb: any): void; + getStateRoot(cb: any): void; + setStateRoot(stateRoot: Buffer, cb: any): void; + dumpStorage(address: Buffer, cb: any): void; + hasGenesisState(cb: any): void; + generateCanonicalGenesis(cb: any): void; + generateGenesis(initState: any, cb: any): void; + accountIsEmpty(address: Buffer, cb: any): void; + cleanupTouchedAccounts(cb: any): void; + _clearOriginalStorageCache(): void; + } + + class Cache { + _cache: any; + _checkpoints: any[]; + _trie: any; + constructor(trie: any); + put(key: Buffer, val: Account, fromTrie: boolean): void; + get(key: Buffer): Account; + lookup(key: Buffer): Account | undefined; + _lookupAccount(address: Buffer, cb: any): void; + getOrLoad(key: Buffer, cb: any): void; + warm(addresses: string[], cb: any): void; + flush(cb: any): void; + checkpoint(): void; + revert(): void; + commit(): void; + clear(): void; + del(key: Buffer): void; + _update(key: Buffer, val: Account, modified: boolean, deleted: boolean): void; + } + + interface StateManagerOpts { + common?: Common; + trie?: any; + } + + class EEI { + _env: Env; + _result: RunResult; + _state: PStateManager; + _evm: EVM; + _lastReturned: Buffer; + _common: Common; + _gasLeft: BN; + constructor(env: Env, state: PStateManager, evm: EVM, common: Common, gasLeft: BN); + useGas(amount: BN): void; + refundGas(amount: BN): void; + getAddress(): Buffer; + getExternalBalance(address: Buffer): Promise; + getSelfBalance(): BN; + getCaller(): BN; + getCallValue(): BN; + getCallData(): Buffer; + getCallDataSize(): BN; + getCodeSize(): BN; + getCode(): Buffer; + isStatic(): boolean; + getExternalCodeSize(address: BN): Promise; + getExternalCode(address: BN | Buffer): Promise; + getReturnDataSize(): BN; + getReturnData(): Buffer; + getTxGasPrice(): BN; + getTxOrigin(): BN; + getBlockNumber(): BN; + getBlockCoinbase(): BN; + getBlockTimestamp(): BN; + getBlockDifficulty(): BN; + getBlockGasLimit(): BN; + getChainId(): BN; + getBlockHash(num: BN): Promise; + storageStore(key: Buffer, value: Buffer): Promise; + storageLoad(key: Buffer): Promise; + getGasLeft(): BN; + finish(returnData: Buffer): void; + revert(returnData: Buffer): void; + selfDestruct(toAddress: Buffer): Promise; + _selfDestruct(toAddress: Buffer): Promise; + log(data: Buffer, numberOfTopics: number, topics: Buffer[]): void; + call(gasLimit: BN, address: Buffer, value: BN, data: Buffer): Promise; + callCode(gasLimit: BN, address: Buffer, value: BN, data: Buffer): Promise; + callStatic(gasLimit: BN, address: Buffer, value: BN, data: Buffer): Promise; + callDelegate(gasLimit: BN, address: Buffer, value: BN, data: Buffer): Promise; + _baseCall(msg: Message): Promise; + create(gasLimit: BN, value: BN, data: Buffer, salt: Buffer | null): Promise; + create2(gasLimit: BN, value: BN, data: Buffer, salt: Buffer): Promise; + isAccountEmpty(address: Buffer): Promise; + private _getReturnCode(results: EVMResult): any; + } + + interface Env { + blockchain: Blockchain; + address: Buffer; + caller: Buffer; + callData: Buffer; + callValue: BN; + code: Buffer; + isStatic: boolean; + depth: number; + gasPrice: Buffer; + origin: Buffer; + block: any; + contract: Account; + } + + interface RunResult { + logs: any; + returnValue?: Buffer; + gasRefund: BN; + selfdestruct: { [k: string]: Buffer }; + } + + export class PStateManager { + _wrapped: StateManager; + constructor(wrapped: StateManager); + copy(): PStateManager; + getAccount(addr: Buffer): Promise; + putAccount(addr: Buffer, account: Account): Promise; + putContractCode(addr: Buffer, code: Buffer): Promise; + getContractCode(addr: Buffer): Promise; + getContractStorage(addr: Buffer, key: Buffer): Promise; + getOriginalContractStorage(addr: Buffer, key: Buffer): Promise; + putContractStorage(addr: Buffer, key: Buffer, value: Buffer): Promise; + clearContractStorage(addr: Buffer): Promise; + checkpoint(): Promise; + commit(): Promise; + revert(): Promise; + getStateRoot(): Promise; + setStateRoot(root: Buffer): Promise; + dumpStorage(address: Buffer): Promise; + hasGenesisState(): Promise; + generateCanonicalGenesis(): Promise; + generateGenesis(initState: any): Promise; + accountIsEmpty(address: Buffer): Promise; + cleanupTouchedAccounts(): Promise; + } + + interface StorageDump { + [key: string]: string; + } + + class EVM { + _vm: any; + _state: PStateManager; + _tx: TxContext; + _block: any; + constructor(vm: any, txContext: TxContext, block: any); + executeMessage(message: Message): Promise; + _executeCall(message: Message): Promise; + _executeCreate(message: Message): Promise; + runInterpreter(message: Message, opts: InterpreterOpts): Promise; + getPrecompile(address: Buffer): PrecompileFunc; + runPrecompile(code: PrecompileFunc, data: Buffer, gasLimit: BN): ExecResult; + _loadCode(message: Message): Promise; + _generateAddress(message: Message): Promise; + _reduceSenderBalance(account: Account, message: Message): Promise; + _addToBalance(toAccount: Account, message: Message): Promise; + _touchAccount(address: Buffer): Promise; + } + + class TxContext { + gasPrice: Buffer; + origin: Buffer; + constructor(gasPrice: Buffer, origin: Buffer); + } + + class Message { + to: Buffer; + value: BN; + caller: Buffer; + gasLimit: BN; + data: Buffer; + depth: number; + code: Buffer | PrecompileFunc; + _codeAddress: Buffer; + isStatic: boolean; + isCompiled: boolean; + salt: Buffer; + selfdestruct: any; + delegatecall: boolean; + constructor(opts: any); + codeAddress(): Buffer; + } + + interface InterpreterOpts { + pc?: number; + } + + interface PrecompileFunc { + (opts: PrecompileInput): ExecResult; + } + + interface PrecompileInput { + data: Buffer; + gasLimit: BN; + _common: Common; + } + + class VmError { + error: ERROR; + errorType: string; + constructor(error: ERROR); + } + + enum ERROR { + OUT_OF_GAS = 'out of gas', + STACK_UNDERFLOW = 'stack underflow', + STACK_OVERFLOW = 'stack overflow', + INVALID_JUMP = 'invalid JUMP', + INVALID_OPCODE = 'invalid opcode', + OUT_OF_RANGE = 'value out of range', + REVERT = 'revert', + STATIC_STATE_CHANGE = 'static state change', + INTERNAL_ERROR = 'internal error', + CREATE_COLLISION = 'create collision', + STOP = 'stop', + } +}