-
Notifications
You must be signed in to change notification settings - Fork 169
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Signed-off-by: Ryan Goulding <[email protected]>
- Loading branch information
1 parent
381bc69
commit 9071343
Showing
13 changed files
with
2,304 additions
and
78 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
import pMemoize from 'p-memoize' | ||
import { OmniCounter } from '@/omnicounter/sdk' | ||
import { createEndpointFactory } from '@layerzerolabs/protocol-utils-evm' | ||
import { EndpointFactory } from '@layerzerolabs/protocol-utils' | ||
import { OAppFactory } from '@layerzerolabs/ua-utils' | ||
import { OmniContractFactory } from '@layerzerolabs/utils-evm' | ||
|
||
/** | ||
* Syntactic sugar that creates an instance of EVM `OmniCounter` SDK based on an `OmniPoint` with help of an | ||
* `OmniContractFactory` and an (optional) `EndpointFactory` | ||
* | ||
* @param {OmniContractFactory} contractFactory | ||
* @param {EndpointFactory} [endpointFactory] | ||
* @returns {EndpointFactory<Endpoint>} | ||
*/ | ||
export const createOmniCounterFactory = ( | ||
contractFactory: OmniContractFactory, | ||
endpointFactory: EndpointFactory = createEndpointFactory(contractFactory) | ||
): OAppFactory<OmniCounter> => pMemoize(async (point) => new OmniCounter(await contractFactory(point), endpointFactory)) |
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 |
---|---|---|
@@ -1 +1,2 @@ | ||
export * from './factory' | ||
export * from './sdk' |
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 |
---|---|---|
@@ -1,6 +1,6 @@ | ||
import { OmniTransaction } from '@layerzerolabs/utils' | ||
import { EndpointId } from '@layerzerolabs/lz-definitions' | ||
|
||
export interface IOmniCounterApp { | ||
export interface IOmniCounter { | ||
increment(eid: EndpointId, type: number, options: string): Promise<OmniTransaction> | ||
} |
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,8 @@ | ||
// SPDX-License-Identifier: UNLICENSED | ||
pragma solidity ^0.8.22; | ||
|
||
import { OmniCounter as OmniCounterImpl } from "@layerzerolabs/lz-evm-oapp-v2/contracts/oapp/examples/OmniCounter.sol"; | ||
|
||
contract OmniCounterOApp is OmniCounterImpl { | ||
constructor(address _endpoint, address _owner) OmniCounterImpl(_endpoint, _owner) {} | ||
} |
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
32 changes: 32 additions & 0 deletions
32
packages/ua-utils-evm-hardhat-test/deploy/003_omnicounteroapp.ts
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,32 @@ | ||
import { formatEid } from '@layerzerolabs/utils' | ||
import { type DeployFunction } from 'hardhat-deploy/types' | ||
import assert from 'assert' | ||
|
||
/** | ||
* This deploy function will deploy and configure LayerZero OmniCounter | ||
* | ||
* @param env `HardhatRuntimeEnvironment` | ||
*/ | ||
const deploy: DeployFunction = async ({ getUnnamedAccounts, deployments, network }) => { | ||
assert(network.config.eid != null, `Missing endpoint ID for network ${network.name}`) | ||
|
||
const [deployer] = await getUnnamedAccounts() | ||
assert(deployer, 'Missing deployer') | ||
|
||
await deployments.delete('OmniCounter') | ||
const endpointV2 = await deployments.get('EndpointV2') | ||
const omniCounterDeployment = await deployments.deploy('OmniCounter', { | ||
from: deployer, | ||
args: [endpointV2.address, deployer], | ||
}) | ||
|
||
console.table({ | ||
Network: `${network.name} (endpoint ${formatEid(network.config.eid)})`, | ||
OmniCounter: omniCounterDeployment.address, | ||
}) | ||
} | ||
|
||
deploy.tags = ['OmniCounter'] | ||
deploy.dependencies = ['Bootstrap'] | ||
|
||
export default deploy |
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
21 changes: 21 additions & 0 deletions
21
packages/ua-utils-evm-hardhat-test/test/__utils__/omnicounter.ts
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,21 @@ | ||
import { EndpointId } from '@layerzerolabs/lz-definitions' | ||
import { createNetworkEnvironmentFactory } from '@layerzerolabs/utils-evm-hardhat' | ||
|
||
export const deployOmniCounter = async () => { | ||
const environmentFactory = createNetworkEnvironmentFactory() | ||
const eth = await environmentFactory(EndpointId.ETHEREUM_MAINNET) | ||
const avax = await environmentFactory(EndpointId.AVALANCHE_MAINNET) | ||
|
||
await Promise.all([ | ||
eth.deployments.run('OmniCounter', { writeDeploymentsToFiles: true }), | ||
avax.deployments.run('OmniCounter', { writeDeploymentsToFiles: true }), | ||
]) | ||
} | ||
|
||
export const deployOmniCounterFixture = async () => { | ||
const environmentFactory = createNetworkEnvironmentFactory() | ||
const eth = await environmentFactory(EndpointId.ETHEREUM_MAINNET) | ||
const avax = await environmentFactory(EndpointId.AVALANCHE_MAINNET) | ||
|
||
await Promise.all([eth.deployments.fixture('OmniCounter'), avax.deployments.fixture('OmniCounter')]) | ||
} |
128 changes: 128 additions & 0 deletions
128
packages/ua-utils-evm-hardhat-test/test/omnicounter/options.test.ts
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,128 @@ | ||
import fc from 'fast-check' | ||
import 'hardhat' | ||
import { EventFragment } from '@ethersproject/abi/src.ts/fragments' | ||
import { Log, TransactionReceipt } from '@ethersproject/providers' | ||
import { EndpointId, MainnetEndpointId } from '@layerzerolabs/lz-definitions' | ||
import { Options } from '@layerzerolabs/lz-utility-v2' | ||
import { createOmniCounterFactory, OmniCounter } from '@layerzerolabs/omnicounter-utils-evm' | ||
import { createEndpointFactory } from '@layerzerolabs/protocol-utils-evm' | ||
import { configureOApp } from '@layerzerolabs/ua-utils' | ||
import { OmniTransaction } from '@layerzerolabs/utils' | ||
import { omniContractToPoint } from '@layerzerolabs/utils-evm' | ||
import { | ||
createConnectedContractFactory, | ||
createSignerFactory, | ||
OmniGraphBuilderHardhat, | ||
OmniGraphHardhat, | ||
} from '@layerzerolabs/utils-evm-hardhat' | ||
import { utils } from 'ethers' | ||
import { keccak256, parseEther, toUtf8Bytes } from 'ethers/lib/utils' | ||
import { setupDefaultEndpoint } from '../__utils__/endpoint' | ||
import { deployOmniCounter } from '../__utils__/omnicounter' | ||
|
||
/** | ||
* Find matching emitted events. | ||
* @param {TransactionReceipt} receipt | ||
* @param {EventFragment} frag | ||
* @param {string} contractAddress | ||
* @returns {Log[]} | ||
*/ | ||
const findMatchingEvents = (receipt: TransactionReceipt, frag: utils.EventFragment, contractAddress: string): Log[] => | ||
receipt.logs | ||
.filter((log) => log.topics.includes(keccak256(toUtf8Bytes(frag.format())))) | ||
.filter( | ||
(log) => | ||
log.address && | ||
(contractAddress === undefined || log.address.toLowerCase() === contractAddress.toLowerCase()) | ||
) | ||
|
||
/** | ||
* Parse event logs. | ||
* @param {Log} log | ||
* @param {utils.Interface} context | ||
*/ | ||
const parseArgs = (log: Log, context: utils.Interface): utils.Result => context.parseLog(log).args | ||
|
||
describe('oapp/options', () => { | ||
const ethContract = { eid: EndpointId.ETHEREUM_MAINNET, contractName: 'OmniCounter' } | ||
const avaxContract = { eid: EndpointId.AVALANCHE_MAINNET, contractName: 'OmniCounter' } | ||
|
||
const config: OmniGraphHardhat = { | ||
contracts: [ | ||
{ | ||
contract: ethContract, | ||
}, | ||
{ | ||
contract: avaxContract, | ||
}, | ||
], | ||
connections: [ | ||
{ | ||
from: ethContract, | ||
to: avaxContract, | ||
}, | ||
{ | ||
from: avaxContract, | ||
to: ethContract, | ||
}, | ||
], | ||
} | ||
|
||
beforeEach(async () => { | ||
await deployOmniCounter() | ||
await setupDefaultEndpoint() | ||
}) | ||
|
||
it('lzReceive option', async () => { | ||
const contractFactory = createConnectedContractFactory() | ||
const builder = await OmniGraphBuilderHardhat.fromConfig(config) | ||
const sdkFactory = createOmniCounterFactory(contractFactory) | ||
const signerFactory = createSignerFactory() | ||
|
||
const ethPoint = omniContractToPoint(await contractFactory(ethContract)) | ||
const ethSdk: OmniCounter = await sdkFactory(ethPoint) | ||
const ethSigner = await signerFactory(ethContract.eid) | ||
|
||
const avaxPoint = omniContractToPoint(await contractFactory(avaxContract)) | ||
const avaxSdk = await sdkFactory(avaxPoint) | ||
const avaxSigner = await signerFactory(avaxContract.eid) | ||
|
||
const transactions = await configureOApp(builder.graph, sdkFactory) | ||
for (const transaction of transactions) { | ||
const signer = transaction.point.eid === MainnetEndpointId.ETHEREUM_MAINNET ? ethSigner : avaxSigner | ||
const txResponse = await signer.signAndSend(transaction) | ||
const txReceipt: TransactionReceipt = await txResponse.wait() | ||
expect(txReceipt.status).toBe(1) | ||
} | ||
|
||
expect(await ethSdk.hasPeer(avaxPoint.eid, avaxPoint.address)).toBe(true) | ||
expect(await avaxSdk.hasPeer(ethPoint.eid, ethPoint.address)).toBe(true) | ||
|
||
await fc.assert( | ||
fc.asyncProperty(fc.integer({ min: 200000, max: 100000000000 }), async (p) => { | ||
const options = Options.newOptions().addExecutorLzReceiveOption(p) | ||
const incrementTx: OmniTransaction = { | ||
...(await ethSdk.increment(avaxPoint.eid, 2, options.toHex())), | ||
gasLimit: 500000, | ||
value: parseEther('0').toString(), | ||
} | ||
const incrementTxResponse = await ethSigner.signAndSend(incrementTx) | ||
const incrementTxReceipt: TransactionReceipt = await incrementTxResponse.wait() | ||
expect(incrementTxReceipt.status).toEqual(1) | ||
|
||
const ethEndpointContract = { eid: MainnetEndpointId.ETHEREUM_MAINNET, contractName: 'EndpointV2' } | ||
const ethEndpointPoint = omniContractToPoint(await contractFactory(ethEndpointContract)) | ||
const endpointSdkFactory = createEndpointFactory(contractFactory) | ||
const ethEndpointSdk = await endpointSdkFactory(ethEndpointPoint) | ||
const packetReceived = Object.values( | ||
ethEndpointSdk.contract.contract.interface.events as any as EventFragment[] | ||
).find((frag: EventFragment) => frag.name === 'PacketSent') as EventFragment | ||
const logs = findMatchingEvents(incrementTxReceipt, packetReceived, ethEndpointPoint.address) | ||
if (logs && logs.length > 0) { | ||
const eventArgs = parseArgs(logs[0]!, ethEndpointSdk.contract.contract.interface) | ||
expect(eventArgs.options.toLowerCase() === options.toHex().toLowerCase()) | ||
} | ||
}) | ||
) | ||
}) | ||
}) |
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
Oops, something went wrong.