Skip to content
This repository has been archived by the owner on Mar 25, 2024. It is now read-only.

Tests for L2EthToken.sol Contract #88

Open
wants to merge 16 commits into
base: dev
Choose a base branch
from
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -16,3 +16,5 @@ yarn-error.log*

.vscode
target/

era_test_node.log
227 changes: 227 additions & 0 deletions test/L2EthToken.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,227 @@
import { expect } from "chai";
import { ethers } from "hardhat";
import type { Wallet } from "zksync-web3";
import { Provider } from "zksync-web3";
import { IMailbox__factory, L2EthToken__factory } from "../typechain-types";
import type { L2EthToken } from "../typechain-types";
import { deployContractOnAddress, getWallets } from "./shared/utils";
import * as hre from "hardhat";
import type { BigNumber } from "ethers";
import { TEST_BOOTLOADER_FORMAL_ADDRESS, TEST_ETH_TOKEN_SYSTEM_CONTRACT_ADDRESS } from "./shared/constants";
import { prepareEnvironment, setResult } from "./shared/mocks";

describe("L2EthToken tests", () => {
let walletFrom: Wallet;
let walletTo: Wallet;
let l2EthToken: L2EthToken;
let bootloaderAccount: ethers.Signer;
let l1Receiver: Wallet;

before(async () => {
await prepareEnvironment();
walletFrom = getWallets()[0];
walletTo = getWallets()[1];
l1Receiver = getWallets()[2];
await deployContractOnAddress(TEST_ETH_TOKEN_SYSTEM_CONTRACT_ADDRESS, "L2EthToken");
l2EthToken = L2EthToken__factory.connect(TEST_ETH_TOKEN_SYSTEM_CONTRACT_ADDRESS, walletFrom);
bootloaderAccount = await ethers.getImpersonatedSigner(TEST_BOOTLOADER_FORMAL_ADDRESS);
});

describe("mint", () => {
it("successful", async () => {
const initialSupply: BigNumber = await l2EthToken.totalSupply();
const initialBalanceOfWallet: BigNumber = await l2EthToken.balanceOf(walletFrom.address);
const amountToMint: BigNumber = ethers.utils.parseEther("10.0");

await expect(l2EthToken.connect(bootloaderAccount).mint(walletFrom.address, amountToMint))
.to.emit(l2EthToken, "Mint")
.withArgs(walletFrom.address, amountToMint);

const finalSupply: BigNumber = await l2EthToken.totalSupply();
const balanceOfWallet: BigNumber = await l2EthToken.balanceOf(walletFrom.address);

expect(finalSupply).to.equal(initialSupply.add(amountToMint));
expect(balanceOfWallet).to.equal(initialBalanceOfWallet.add(amountToMint));
});

it("not called by bootloader", async () => {
const amountToMint: BigNumber = ethers.utils.parseEther("10.0");
await expect(l2EthToken.connect(walletTo.address).mint(walletFrom.address, amountToMint)).to.be.rejectedWith(
"Callable only by the bootloader"
);
});
});

describe("transfer", () => {
it("transfer successfully", async () => {
await l2EthToken.connect(bootloaderAccount).mint(walletFrom.address, ethers.utils.parseEther("100.0"));

const senderBalandeBeforeTransfer: BigNumber = await l2EthToken.balanceOf(walletFrom.address);
const recipientBalanceBeforeTransfer: BigNumber = await l2EthToken.balanceOf(walletTo.address);

const amountToTransfer = ethers.utils.parseEther("10.0");

await expect(
l2EthToken.connect(bootloaderAccount).transferFromTo(walletFrom.address, walletTo.address, amountToTransfer)
)
.to.emit(l2EthToken, "Transfer")
.withArgs(walletFrom.address, walletTo.address, amountToTransfer);

const senderBalanceAfterTransfer: BigNumber = await l2EthToken.balanceOf(walletFrom.address);
const recipientBalanceAfterTransfer: BigNumber = await l2EthToken.balanceOf(walletTo.address);
expect(senderBalanceAfterTransfer).to.be.eq(senderBalandeBeforeTransfer.sub(amountToTransfer));
expect(recipientBalanceAfterTransfer).to.be.eq(recipientBalanceBeforeTransfer.add(amountToTransfer));
});

it("no tranfser due to insufficient balance", async () => {
await l2EthToken.connect(bootloaderAccount).mint(walletFrom.address, ethers.utils.parseEther("5.0"));
const amountToTransfer: BigNumber = ethers.utils.parseEther("100000000000000000.0");

await expect(
l2EthToken.connect(bootloaderAccount).transferFromTo(walletFrom.address, walletTo.address, amountToTransfer)
).to.be.rejectedWith("Transfer amount exceeds balance");
});

it("no transfer - require special access", async () => {
const provider: Provider = new Provider((hre.network.config as any).url);
neotheprogramist marked this conversation as resolved.
Show resolved Hide resolved
const maliciousWallet: Wallet = ethers.Wallet.createRandom().connect(provider);
await l2EthToken.connect(bootloaderAccount).mint(maliciousWallet.address, ethers.utils.parseEther("20.0"));

const amountToTransfer: BigNumber = ethers.utils.parseEther("20.0");

await expect(
l2EthToken.connect(maliciousWallet).transferFromTo(maliciousWallet.address, walletTo.address, amountToTransfer)
).to.be.rejectedWith("Only system contracts with special access can call this method");
});
});

describe("balanceOf", () => {
it("correct balance", async () => {
const amountToMint: BigNumber = ethers.utils.parseEther("10.0");
await l2EthToken.connect(bootloaderAccount).mint(walletFrom.address, amountToMint);
const balance = await l2EthToken.balanceOf(walletFrom.address);
expect(balance).to.equal(ethers.utils.parseEther("115.0"));
neotheprogramist marked this conversation as resolved.
Show resolved Hide resolved
});

it("wrong address", async () => {
neotheprogramist marked this conversation as resolved.
Show resolved Hide resolved
const balance = await l2EthToken.balanceOf("0xdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef");
expect(balance).to.equal(0);
});
});

describe("name", () => {
it("correct name", async () => {
const name = await l2EthToken.name();
expect(name).to.equal("Ether");
});
});

describe("symbol", () => {
it("correct symbol", async () => {
const symbol = await l2EthToken.symbol();
expect(symbol).to.equal("ETH");
});
});

describe("decimals", () => {
it("correct decimals", async () => {
const decimals = await l2EthToken.decimals();
expect(decimals).to.equal(18);
});
});

describe("withdraw", () => {
it("successful, correct contract balance and total supply", async () => {
const iface = IMailbox__factory.createInterface();
const selector = iface.getSighash("finalizeEthWithdrawal");
const amountToWithdraw: BigNumber = ethers.utils.parseEther("1.0");

const message: string = ethers.utils.solidityPack(
["bytes4", "address", "uint256"],
[selector, l1Receiver.address, amountToWithdraw]
);

await setResult("L1Messenger", "sendToL1", [message], {
failure: false,
returnData: ethers.utils.defaultAbiCoder.encode(["bytes32"], [ethers.utils.keccak256(message)]),
});

const amountToMint: BigNumber = ethers.utils.parseEther("100.0");
await l2EthToken.connect(bootloaderAccount).mint(l2EthToken.address, amountToMint);

const balanceBeforeWithdrawal: BigNumber = await l2EthToken.balanceOf(l2EthToken.address);
const totalSupplyBefore = await l2EthToken.totalSupply();

await expect(l2EthToken.connect(walletFrom).withdraw(l1Receiver.address, { value: amountToWithdraw }))
.to.emit(l2EthToken, "Withdrawal")
.withArgs(walletFrom.address, l1Receiver.address, amountToWithdraw);

const balanceAfterWithdrawal: BigNumber = await l2EthToken.balanceOf(l2EthToken.address);
const totalSupplyAfter = await l2EthToken.totalSupply();

expect(balanceAfterWithdrawal).to.equal(balanceBeforeWithdrawal.sub(amountToWithdraw));
expect(totalSupplyAfter).to.equal(totalSupplyBefore.sub(amountToWithdraw));
});

it("big amount to withdraw, underflow contract balance", async () => {
const iface = IMailbox__factory.createInterface();
const selector = iface.getSighash("finalizeEthWithdrawal");
const amountToWithdraw: BigNumber = ethers.utils.parseEther("300.0");

const message: string = ethers.utils.solidityPack(
["bytes4", "address", "uint256"],
[selector, l1Receiver.address, amountToWithdraw]
);

await setResult("L1Messenger", "sendToL1", [message], {
failure: false,
returnData: ethers.utils.defaultAbiCoder.encode(["bytes32"], [ethers.utils.keccak256(message)]),
});

const amountToMint: BigNumber = ethers.utils.parseEther("100.0");
await l2EthToken.connect(bootloaderAccount).mint(l2EthToken.address, amountToMint);

const balanceBeforeWithdrawal: BigNumber = await l2EthToken.balanceOf(l2EthToken.address);

await expect(l2EthToken.connect(walletFrom).withdraw(l1Receiver.address, { value: amountToWithdraw }))
.to.emit(l2EthToken, "Withdrawal")
.withArgs(walletFrom.address, l1Receiver.address, amountToWithdraw);

const balanceAfterWithdrawal: BigNumber = await l2EthToken.balanceOf(l2EthToken.address);
expect(balanceAfterWithdrawal).not.equal(balanceBeforeWithdrawal.sub(amountToWithdraw));
neotheprogramist marked this conversation as resolved.
Show resolved Hide resolved
});
});

describe("withdrawWithMessage", () => {
it("successful", async () => {
const iface = IMailbox__factory.createInterface();
const selector = iface.getSighash("finalizeEthWithdrawal");
const amountToWidthdraw: BigNumber = ethers.utils.parseEther("1.0");
const additionalData: string = ethers.utils.defaultAbiCoder.encode(["string"], ["additional data"]);
const message: string = ethers.utils.solidityPack(
["bytes4", "address", "uint256", "address", "bytes"],
[selector, l1Receiver.address, amountToWidthdraw, walletFrom.address, additionalData]
);

await setResult("L1Messenger", "sendToL1", [message], {
failure: false,
returnData: ethers.utils.defaultAbiCoder.encode(["bytes32"], [ethers.utils.keccak256(message)]),
});

const amountToWithdraw: BigNumber = ethers.utils.parseEther("1.0");
const totalSupplyBefore = await l2EthToken.totalSupply();
const balanceBeforeWithdrawal: BigNumber = await l2EthToken.balanceOf(l2EthToken.address);
await expect(
l2EthToken.connect(walletFrom).withdrawWithMessage(l1Receiver.address, additionalData, {
value: amountToWithdraw,
})
)
.to.emit(l2EthToken, "WithdrawalWithMessage")
.withArgs(walletFrom.address, l1Receiver.address, amountToWithdraw, additionalData);
const totalSupplyAfter = await l2EthToken.totalSupply();
const balanceAfterWithdrawal: BigNumber = await l2EthToken.balanceOf(l2EthToken.address);
expect(balanceAfterWithdrawal).to.equal(balanceBeforeWithdrawal.sub(amountToWithdraw));
expect(totalSupplyAfter).to.equal(totalSupplyBefore.sub(amountToWithdraw));
});
});
});