Skip to content

Commit

Permalink
added CIP-23
Browse files Browse the repository at this point in the history
  • Loading branch information
Crisgarner committed Oct 26, 2023
1 parent 2497956 commit 8aa699e
Show file tree
Hide file tree
Showing 2 changed files with 170 additions and 5 deletions.
11 changes: 6 additions & 5 deletions scripts/CIP-22.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,13 +13,14 @@ import { BigNumber } from "ethers";
async function main() {
const ethers = hre.ethers;
let multisig = "0xa70b638B70154EdfCbb8DbbBd04900F328F32c35";
let ethReceiver = "0x714E5202297d4981839E7F5EE6E6D4641dFF8769"
let ctxAmount = ethers.utils.parseEther("195625");
let ethAmount = ethers.utils.parseEther("7.3")
let ctx = await deployments.get("Ctx");
let ctxContract = await ethers.getContractAt("Ctx", ctx.address);

const abi = new ethers.utils.AbiCoder();
const targets = [multisig, ctx.address];
const targets = [ethReceiver, ctx.address];
const values = [ethAmount, BigNumber.from(0)];
const signatures = ["", "transfer(address,uint256)"];
const calldatas = [
Expand All @@ -36,8 +37,8 @@ async function main() {

let ctxbalance = await ctxContract.balanceOf(multisig);
console.log("multisig old CTX balance", ethers.utils.formatEther(ctxbalance));
let ethBalance = await ethers.provider.getBalance(multisig);
console.log("multisig old ETH balance", ethers.utils.formatEther(ethBalance));
let ethBalance = await ethers.provider.getBalance(ethReceiver);
console.log("ethReceiver old ETH balance", ethers.utils.formatEther(ethBalance));
if (hardhatArguments.network === "hardhat") {
//Fund Multisign with ETH
await fundMultisign("10000000000000000000");
Expand All @@ -62,9 +63,9 @@ async function main() {
"multisig new CTX balance",
ethers.utils.formatEther(ctxbalance)
);
ethBalance = await ethers.provider.getBalance(multisig);
ethBalance = await ethers.provider.getBalance(ethReceiver);
console.log(
"multisig new ETH balance",
"ethReceiver new ETH balance",
ethers.utils.formatEther(ethBalance)
);
}
Expand Down
164 changes: 164 additions & 0 deletions scripts/CIP-23.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,164 @@
// run with
// npx hardhat run ./scripts/CIP-23.ts --network hardhat
import hre, { deployments, network, hardhatArguments } from "hardhat";
import { hexDataLength } from "@ethersproject/bytes";
import { L1ToL2MessageGasEstimator } from "@arbitrum/sdk/dist/lib/message/L1ToL2MessageGasEstimator";
import { L1TransactionReceipt, L1ToL2MessageStatus, EthBridger, getL2Network } from "@arbitrum/sdk";
import "ethers";

async function calculateArbitrumRelayerGasParams(
payLoad: string,
l1Provider: any,
l2Provider: any,
l1MessageRelayerAddress: string,
l2MessageExecutorProxyAddress: string,
MainnetTimelockAddress: string
) {
const newMessageBytes = ethers.utils.defaultAbiCoder.encode(["bytes"], [payLoad])
const newMessageBytesLength = hexDataLength(newMessageBytes) + 4 // 4 bytes func identifier

/**
* Now we can query the submission price using a helper method; the first value returned tells us the best cost of our transaction; that's what we'll be using.
* The second value (nextUpdateTimestamp) tells us when the base cost will next update (base cost changes over time with chain congestion; the value updates every 24 hours). We won't actually use it here, but generally it's useful info to have.
*/
const l1ToL2MessageGasEstimate = new L1ToL2MessageGasEstimator(l2Provider)

const _submissionPriceWei = await l1ToL2MessageGasEstimate.estimateSubmissionFee(
l1Provider,
await l1Provider.getGasPrice(),
newMessageBytesLength
)

console.log(`Current retryable base submission price: ${_submissionPriceWei.toString()}`)

/**
* ...Okay, but on the off chance we end up underpaying, our retryable ticket simply fails.
* This is highly unlikely, but just to be safe, let's increase the amount we'll be paying (the difference between the actual cost and the amount we pay gets refunded to our address on L2 anyway)
* In nitro, submission fee will be charged in L1 based on L1 basefee, revert on L1 side upon insufficient fee.
*/
const submissionPriceWei = _submissionPriceWei.mul(10)
/**
* Now we'll figure out the gas we need to send for L2 execution; this requires the L2 gas price and gas limit for our L2 transaction
*/

/**
* For the L2 gas price, we simply query it from the L2 provider, as we would when using L1
*/
const gasPriceBid = await l2Provider.getGasPrice()
console.log(`L2 gas price: ${gasPriceBid.toString()}`)

/**
* For the gas limit, we'll use the estimateRetryableTicketGasLimit method in Arbitrum SDK
*/

/**
* First, we need to calculate the calldata for the function being called (setGreeting())
*/
const ABI = ["function executeMessage(bytes calldata payLoad)"]
const iface = new ethers.utils.Interface(ABI)
const calldata = iface.encodeFunctionData("executeMessage", [payLoad])

const maxGas = await l1ToL2MessageGasEstimate.estimateRetryableTicketGasLimit(
{
from: await l1MessageRelayerAddress,
to: l2MessageExecutorProxyAddress,
l2CallValue: ethers.BigNumber.from("0"),
excessFeeRefundAddress: MainnetTimelockAddress,
callValueRefundAddress: MainnetTimelockAddress,
data: calldata,
},
ethers.utils.parseEther("1")
)
/**
* With these three values, we can calculate the total callvalue we'll need our L1 transaction to send to L2
*/
const callValue = submissionPriceWei.add(gasPriceBid.mul(maxGas))

console.log("submissionPriceWei ", submissionPriceWei)
console.log("maxGas ", maxGas)
console.log("gasPriceBid ", gasPriceBid)
console.log("callValue ", callValue)
return [submissionPriceWei, maxGas, gasPriceBid, callValue, calldata]
}

async function main() {
const l1Provider = new ethers.providers.JsonRpcProvider(process.env.MAINNET_API_URL)
const l2Provider = new ethers.providers.JsonRpcProvider(process.env.ARBITRUM_API_URL)
ethers.Wallet.createRandom()
const _wallet = ethers.Wallet.createRandom()
const l1Wallet = new ethers.Wallet(_wallet, l1Provider)
const l2Wallet = new ethers.Wallet(_wallet, l1Provider)

const l2Network = await getL2Network(l2Provider)
const ethBridger = new EthBridger(l2Network)
const inboxAddress = ethBridger.l2Network.ethBridge.inbox
const abiCoder = new ethers.utils.AbiCoder()

// addresses
const MainnetTimelockAddress = "0xa54074b2cc0e96a43048d4a68472F7F046aC0DA8"
const l1MessageRelayerAddress = "0x209c23DB16298504354112fa4210d368e1d564dA"
const l2MessageExecutorProxyAddress = "0x3769b6aA269995297a539BEd7a463105466733A5"
const ArbitrumTreasuryAddress = "0x9474B771Fb46E538cfED114Ca816A3e25Bb346CF"
const ArbitrumMultisigAddress = "0x8705b41F9193f05ba166a1D5C0771E9cB2Ca0aa3"
const ARBTokenAddress = "0x912CE59144191C1204E64559FE8253a0e49E6548"
const MainnetMultisig = "0xa70b638B70154EdfCbb8DbbBd04900F328F32c35"
const CTXTokenAddress = "0x321C2fE4446C7c963dc41Dd58879AF648838f98D"

const _L1MessageRelayer = await (
await hre.ethers.getContractFactory("L1MessageRelayer")
).connect(l1Wallet);
const l1MessageRelayer = _L1MessageRelayer.attach(l1MessageRelayerAddress)
const _L2MessageExecutorProxy = await (
await hre.ethers.getContractFactory("L2MessageExecutorProxy")
).connect(l2Wallet);
const l2MessageExecutorProxy = _L2MessageExecutorProxy.attach(l2MessageExecutorProxyAddress)

const ARBTokensToTransfer = ethers.utils.parseEther("25000")
let CTXTokensToTransfer = ethers.utils.parseEther("229000")

const TreasuryABI = ["function executeTransaction(address,uint256,string,bytes)"]
const TreasuryIface = new ethers.utils.Interface(TreasuryABI);
const _calldata = TreasuryIface.encodeFunctionData("executeTransaction", [
ARBTokenAddress,
0,
"transfer(address,uint256)",
abiCoder.encode(["address", "uint256"], [ArbitrumMultisigAddress, ARBTokensToTransfer]),
]);
const payLoad = abiCoder.encode(["address", "bytes"], [ArbitrumTreasuryAddress, _calldata])

const [submissionPriceWei, maxGas, gasPriceBid, callValue, calldata] =
await calculateArbitrumRelayerGasParams(
payLoad,
l1Provider,
l2Provider,
l1MessageRelayerAddress,
l2MessageExecutorProxyAddress,
MainnetTimelockAddress
)

const targets = [l1MessageRelayerAddress, CTXTokenAddress];
const values = [callValue, 0];
const signatures = ["relayMessage(address,bytes,uint256,uint256,uint256)","transfer(address,uint256)"]
const calldatas = [
abiCoder.encode(
["address", "bytes", "uint256", "uint256", "uint256"],
[l2MessageExecutorProxyAddress, calldata, submissionPriceWei, maxGas, gasPriceBid]
),
abiCoder.encode(["address", "uint256"], [MainnetMultisig, CTXTokensToTransfer])
];

const description = "CIP-23: 2023 Operating Expenditures Treasury Transfers for Q4"
console.log("_".repeat(100))
console.log("targets:", targets)
console.log("values:", values.toString())
console.log("signatures:", signatures)
console.log("calldatas:", calldatas)
console.log("description:", description)
}

main()
.then(() => process.exit(0))
.catch((error) => {
console.error(error);
process.exit(1);
});

0 comments on commit 8aa699e

Please sign in to comment.