-
Notifications
You must be signed in to change notification settings - Fork 61
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(deploy): v1 factory contracts on avaxc testnet
Ticket: COIN-1712
- Loading branch information
1 parent
953f3d2
commit eb95cb1
Showing
6 changed files
with
301 additions
and
8 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,57 @@ | ||
// SPDX-License-Identifier: Apache-2.0 | ||
pragma solidity 0.8.20; | ||
import '../Forwarder.sol'; | ||
import '../ERC20Interface.sol'; | ||
import '../WalletSimple.sol'; | ||
|
||
/** | ||
* | ||
* WalletSimple | ||
* ============ | ||
* | ||
* Basic multi-signer wallet designed for use in a co-signing environment where 2 signatures are required to move funds. | ||
* Typically used in a 2-of-3 signing configuration. Uses ecrecover to allow for 2 signatures in a single transaction. | ||
* | ||
* The first signature is created on the operation hash (see Data Formats) and passed to sendMultiSig/sendMultiSigToken | ||
* The signer is determined by verifyMultiSig(). | ||
* | ||
* The second signature is created by the submitter of the transaction and determined by msg.signer. | ||
* | ||
* Data Formats | ||
* ============ | ||
* | ||
* The signature is created with ethereumjs-util.ecsign(operationHash). | ||
* Like the eth_sign RPC call, it packs the values as a 65-byte array of [r, s, v]. | ||
* Unlike eth_sign, the message is not prefixed. | ||
* | ||
* The operationHash the result of keccak256(prefix, toAddress, value, data, expireTime). | ||
* For ether transactions, `prefix` is "ETHER". | ||
* For token transaction, `prefix` is "ERC20" and `data` is the tokenContractAddress. | ||
* | ||
* | ||
*/ | ||
contract AvaxcWalletSimple is WalletSimple { | ||
/** | ||
* Get the network identifier that signers must sign over | ||
* This provides protection signatures being replayed on other chains | ||
*/ | ||
function getNetworkId() internal override pure returns (string memory) { | ||
return 'AVAX'; | ||
} | ||
|
||
/** | ||
* Get the network identifier that signers must sign over for token transfers | ||
* This provides protection signatures being replayed on other chains | ||
*/ | ||
function getTokenNetworkId() internal override pure returns (string memory) { | ||
return 'AVAX-ERC20'; | ||
} | ||
|
||
/** | ||
* Get the network identifier that signers must sign over for batch transfers | ||
* This provides protection signatures being replayed on other chains | ||
*/ | ||
function getBatchNetworkId() internal override pure returns (string memory) { | ||
return 'AVAX-Batch'; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,188 @@ | ||
import { ethers } from 'hardhat'; | ||
import { BigNumber } from 'ethers'; | ||
const hre = require('hardhat'); | ||
const fs = require('fs'); | ||
|
||
async function main() { | ||
const output = { | ||
walletImplementation: '', | ||
walletFactory: '', | ||
forwarderImplementation: '', | ||
forwarderFactory: '' | ||
}; | ||
|
||
const feeData = await ethers.provider.getFeeData(); | ||
|
||
|
||
const [walletDeployer, forwarderDeployer] = await ethers.getSigners(); | ||
|
||
const gasParams = { | ||
gasPrice: feeData.gasPrice!.mul('2'), | ||
gasLimit: 5000000 | ||
}; | ||
const walletTxCount = await walletDeployer.getTransactionCount(); | ||
|
||
console.log('Deploying wallet contracts....'); | ||
console.log('Wallet Tx Count: ', walletTxCount); | ||
const walletSelfTransactions = 2 - walletTxCount; | ||
for (let i = 0; i < walletSelfTransactions; i++) { | ||
const tx = await walletDeployer.sendTransaction({ | ||
to: walletDeployer.address, | ||
value: ethers.utils.parseEther('0'), | ||
gasPrice: gasParams.gasPrice | ||
}); | ||
await tx.wait(); | ||
console.log(`Self transaction with nonce: ${i} complete`); | ||
} | ||
|
||
const walletImplementationContractName = 'AvaxcWalletSimple'; | ||
const walletFactoryContractName = 'WalletFactory'; | ||
|
||
const WalletImplementation = await ethers.getContractFactory( | ||
walletImplementationContractName, | ||
walletDeployer | ||
); | ||
const walletImplementation = await WalletImplementation.deploy(gasParams); | ||
await walletImplementation.deployed(); | ||
output.walletImplementation = walletImplementation.address; | ||
console.log( | ||
`${walletImplementationContractName} deployed at ` + | ||
walletImplementation.address | ||
); | ||
|
||
const WalletFactory = await ethers.getContractFactory( | ||
walletFactoryContractName, | ||
walletDeployer | ||
); | ||
const walletFactory = await WalletFactory.deploy( | ||
walletImplementation.address, | ||
gasParams | ||
); | ||
await walletFactory.deployed(); | ||
output.walletFactory = walletFactory.address; | ||
console.log( | ||
`${walletFactoryContractName} deployed at ` + walletFactory.address | ||
); | ||
|
||
const forwarderTxCount = await forwarderDeployer.getTransactionCount(); | ||
|
||
console.log('Deploying forwarder contracts....'); | ||
console.log('Forwarder Tx Count: ', forwarderTxCount); | ||
const forwarderSelfTransactions = 234 - forwarderTxCount; | ||
|
||
for (let i = 0; i < forwarderSelfTransactions; i++) { | ||
const tx = await forwarderDeployer.sendTransaction({ | ||
to: forwarderDeployer.address, | ||
value: ethers.utils.parseEther('0'), | ||
gasPrice: gasParams.gasPrice | ||
}); | ||
await tx.wait(); | ||
console.log(`Self transaction with nonce: ${i} complete`); | ||
} | ||
|
||
const forwarderImplementationContractName = 'Forwarder'; | ||
const forwarderFactoryContractName = 'ForwarderFactory'; | ||
|
||
const ForwarderImplementation = await ethers.getContractFactory( | ||
forwarderImplementationContractName, | ||
forwarderDeployer | ||
); | ||
|
||
const forwarderImplementation = await ForwarderImplementation.deploy( | ||
gasParams | ||
); | ||
await forwarderImplementation.deployed(); | ||
output.forwarderImplementation = forwarderImplementation.address; | ||
|
||
console.log( | ||
`${forwarderImplementationContractName} deployed at ` + | ||
forwarderImplementation.address | ||
); | ||
|
||
const ForwarderFactory = await ethers.getContractFactory( | ||
forwarderFactoryContractName, | ||
forwarderDeployer | ||
); | ||
|
||
const forwarderFactory = await ForwarderFactory.deploy( | ||
forwarderImplementation.address, | ||
gasParams | ||
); | ||
|
||
await forwarderFactory.deployed(); | ||
output.forwarderFactory = forwarderFactory.address; | ||
console.log( | ||
`${forwarderFactoryContractName} deployed at ` + forwarderFactory.address | ||
); | ||
|
||
fs.writeFileSync('output.json', JSON.stringify(output)); | ||
|
||
// Wait 5 minutes. It takes some time for the etherscan backend to index the transaction and store the contract. | ||
console.log('Waiting for 5 minutes before verifying....'); | ||
await new Promise((r) => setTimeout(r, 1000 * 300)); | ||
|
||
// We have to wait for a minimum of 10 block confirmations before we can call the etherscan api to verify | ||
|
||
await walletImplementation.deployTransaction.wait(10); | ||
await walletFactory.deployTransaction.wait(10); | ||
await forwarderImplementation.deployTransaction.wait(10); | ||
await forwarderFactory.deployTransaction.wait(10); | ||
|
||
console.log('Done waiting, verifying'); | ||
await verifyContract( | ||
walletImplementationContractName, | ||
walletImplementation.address, | ||
[] | ||
); | ||
await verifyContract('WalletFactory', walletFactory.address, [ | ||
walletImplementation.address | ||
]); | ||
|
||
await verifyContract( | ||
forwarderImplementationContractName, | ||
forwarderImplementation.address, | ||
[] | ||
); | ||
|
||
await verifyContract('ForwarderFactory', forwarderFactory.address, [ | ||
forwarderImplementation.address | ||
]); | ||
|
||
console.log('Contracts verified'); | ||
} | ||
|
||
async function verifyContract( | ||
contractName: string, | ||
contractAddress: string, | ||
constructorArguments: string[], | ||
contract?: string | ||
) { | ||
try { | ||
const verifyContractArgs: { | ||
address: string; | ||
constructorArguments: string[]; | ||
contract?: string; | ||
} = { | ||
address: contractAddress, | ||
constructorArguments: constructorArguments | ||
}; | ||
|
||
if (contract) { | ||
verifyContractArgs.contract = contract; | ||
} | ||
|
||
await hre.run('verify:verify', verifyContractArgs); | ||
} catch (e) { | ||
// @ts-ignore | ||
// We get a failure API response if the source code has already been uploaded, don't throw in this case. | ||
if (!e.message.includes('Reason: Already Verified')) { | ||
throw e; | ||
} | ||
} | ||
console.log(`Verified ${contractName} on Etherscan!`); | ||
} | ||
|
||
main().catch((error) => { | ||
console.error(error); | ||
process.exitCode = 1; | ||
}); |