utils.ts 5.68 KB
Newer Older
1 2
import { expect } from 'chai'

3
/* Imports: External */
4 5 6 7 8 9 10
import {
  Contract,
  Wallet,
  constants,
  providers,
  BigNumberish,
  BigNumber,
11
  utils,
12
} from 'ethers'
13 14 15 16 17 18 19 20 21 22 23
import {
  getContractFactory,
  getContractInterface,
  predeploys,
} from '@eth-optimism/contracts'
import { injectL2Context, remove0x, Watcher } from '@eth-optimism/core-utils'
import { cleanEnv, str, num, bool } from 'envalid'
import dotenv from 'dotenv'

/* Imports: Internal */
import { Direction, waitForXDomainTransaction } from './watcher-utils'
24

25 26
export const GWEI = BigNumber.from(1e9)

27 28 29 30
if (process.env.IS_LIVE_NETWORK === 'true') {
  dotenv.config()
}

31
const env = cleanEnv(process.env, {
32 33
  L1_URL: str({ default: 'http://localhost:9545' }),
  L2_URL: str({ default: 'http://localhost:8545' }),
34
  VERIFIER_URL: str({ default: 'http://localhost:8547' }),
35
  REPLICA_URL: str({ default: 'http://localhost:8549' }),
36 37
  L1_POLLING_INTERVAL: num({ default: 10 }),
  L2_POLLING_INTERVAL: num({ default: 10 }),
38
  VERIFIER_POLLING_INTERVAL: num({ default: 10 }),
39
  REPLICA_POLLING_INTERVAL: num({ default: 10 }),
40 41 42 43 44 45 46
  PRIVATE_KEY: str({
    default:
      '0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80',
  }),
  ADDRESS_MANAGER: str({
    default: '0x5FbDB2315678afecb367f032d93F642f64180aa3',
  }),
47 48
  L2_CHAINID: num({ default: 420 }),
  IS_LIVE_NETWORK: bool({ default: false }),
49 50
})

51
// The hardhat instance
52 53 54 55 56
export const l1Provider = new providers.JsonRpcProvider(env.L1_URL)
l1Provider.pollingInterval = env.L1_POLLING_INTERVAL

export const l2Provider = new providers.JsonRpcProvider(env.L2_URL)
l2Provider.pollingInterval = env.L2_POLLING_INTERVAL
57

58 59 60
export const verifierProvider = new providers.JsonRpcProvider(env.VERIFIER_URL)
verifierProvider.pollingInterval = env.VERIFIER_POLLING_INTERVAL

61 62 63
export const replicaProvider = new providers.JsonRpcProvider(env.REPLICA_URL)
replicaProvider.pollingInterval = env.REPLICA_POLLING_INTERVAL

64
// The sequencer private key which is funded on L1
65
export const l1Wallet = new Wallet(env.PRIVATE_KEY, l1Provider)
66 67 68 69 70 71 72 73

// A random private key which should always be funded with deposits from L1 -> L2
// if it's using non-0 gas price
export const l2Wallet = l1Wallet.connect(l2Provider)

// Predeploys
export const PROXY_SEQUENCER_ENTRYPOINT_ADDRESS =
  '0x4200000000000000000000000000000000000004'
74
export const OVM_ETH_ADDRESS = predeploys.OVM_ETH
75

76 77 78
export const L2_CHAINID = env.L2_CHAINID
export const IS_LIVE_NETWORK = env.IS_LIVE_NETWORK

79 80 81
export const getAddressManager = (provider: any) => {
  return getContractFactory('Lib_AddressManager')
    .connect(provider)
82
    .attach(env.ADDRESS_MANAGER)
83
}
84

85 86 87 88 89
// Gets the bridge contract
export const getL1Bridge = async (wallet: Wallet, AddressManager: Contract) => {
  const l1BridgeInterface = getContractInterface('OVM_L1StandardBridge')
  const ProxyBridgeAddress = await AddressManager.getAddress(
    'Proxy__OVM_L1StandardBridge'
90
  )
91 92 93 94 95 96 97 98 99 100 101

  if (
    !utils.isAddress(ProxyBridgeAddress) ||
    ProxyBridgeAddress === constants.AddressZero
  ) {
    throw new Error('Proxy__OVM_L1StandardBridge not found')
  }

  const OVM_L1StandardBridge = new Contract(
    ProxyBridgeAddress,
    l1BridgeInterface,
102 103
    wallet
  )
104 105 106 107 108
  return OVM_L1StandardBridge
}

export const getL2Bridge = async (wallet: Wallet) => {
  const L2BridgeInterface = getContractInterface('OVM_L2StandardBridge')
109

110 111 112 113 114 115
  const OVM_L2StandardBridge = new Contract(
    predeploys.OVM_L2StandardBridge,
    L2BridgeInterface,
    wallet
  )
  return OVM_L2StandardBridge
116 117 118 119 120 121 122 123 124 125 126 127 128 129
}

export const getOvmEth = (wallet: Wallet) => {
  const OVM_ETH = new Contract(
    OVM_ETH_ADDRESS,
    getContractInterface('OVM_ETH'),
    wallet
  )

  return OVM_ETH
}

export const fundUser = async (
  watcher: Watcher,
130
  bridge: Contract,
131 132 133 134 135
  amount: BigNumberish,
  recipient?: string
) => {
  const value = BigNumber.from(amount)
  const tx = recipient
136 137
    ? bridge.depositETHTo(recipient, 1_300_000, '0x', { value })
    : bridge.depositETH(1_300_000, '0x', { value })
138

139 140 141 142
  await waitForXDomainTransaction(watcher, tx, Direction.L1ToL2)
}

export const sleep = (ms: number) => new Promise((r) => setTimeout(r, ms))
143 144 145 146 147

const abiCoder = new utils.AbiCoder()
export const encodeSolidityRevertMessage = (_reason: string): string => {
  return '0x08c379a0' + remove0x(abiCoder.encode(['string'], [_reason]))
}
148 149 150 151 152 153 154 155

export const DEFAULT_TRANSACTION = {
  to: '0x' + '1234'.repeat(10),
  gasLimit: 33600000000001,
  gasPrice: 0,
  data: '0x',
  value: 0,
}
156

157 158 159 160 161
interface percentDeviationRange {
  upperPercentDeviation: number
  lowerPercentDeviation?: number
}

162 163 164
export const expectApprox = (
  actual: BigNumber | number,
  target: BigNumber | number,
165
  { upperPercentDeviation, lowerPercentDeviation = 100 }: percentDeviationRange
166 167 168 169 170
) => {
  actual = BigNumber.from(actual)
  target = BigNumber.from(target)

  const validDeviations =
171 172 173 174
    upperPercentDeviation >= 0 &&
    upperPercentDeviation <= 100 &&
    lowerPercentDeviation >= 0 &&
    lowerPercentDeviation <= 100
175 176 177 178 179
  if (!validDeviations) {
    throw new Error(
      'Upper and lower deviation percentage arguments should be between 0 and 100'
    )
  }
180 181
  const upper = target.mul(100 + upperPercentDeviation).div(100)
  const lower = target.mul(100 - lowerPercentDeviation).div(100)
182 183 184

  expect(
    actual.lte(upper),
185
    `Actual value is more than ${upperPercentDeviation}% greater than target`
186 187 188
  ).to.be.true
  expect(
    actual.gte(lower),
189
    `Actual value is more than ${lowerPercentDeviation}% less than target`
190 191
  ).to.be.true
}
192

193 194 195 196 197 198 199 200 201 202 203 204 205 206
export const waitForL2Geth = async (
  provider: providers.JsonRpcProvider
): Promise<providers.JsonRpcProvider> => {
  let ready: boolean = false
  while (!ready) {
    try {
      await provider.getNetwork()
      ready = true
    } catch (error) {
      await sleep(1000)
    }
  }
  return injectL2Context(provider)
}