From 88086b1b096d7dfc787f7ebf14e711b19ea447bb Mon Sep 17 00:00:00 2001 From: CanvasL <746591811@qq.com> Date: Tue, 6 Aug 2024 15:33:44 +0800 Subject: [PATCH 1/4] feat: emit events --- {src => contracts}/AccessToken.sol | 0 {src => contracts}/AccessTokenFactory.sol | 0 {src => contracts}/Marketplace.sol | 128 +++++++++++++++--- contracts/interfaces/IMarketplace.sol | 40 ++++++ contracts/interfaces/IMarketplaceEvents.sol | 61 +++++++++ .../interfaces/IMarketplaceStructs.sol | 2 +- {src => contracts}/interfaces/IProduct.sol | 0 foundry.toml | 2 +- test/AccessToken.t.sol | 4 +- test/AccessTokenFactory.t.sol | 6 +- test/Marketplace.t.sol | 62 ++++----- test/mocks/MockProduct.sol | 2 +- 12 files changed, 249 insertions(+), 58 deletions(-) rename {src => contracts}/AccessToken.sol (100%) rename {src => contracts}/AccessTokenFactory.sol (100%) rename {src => contracts}/Marketplace.sol (78%) create mode 100644 contracts/interfaces/IMarketplace.sol create mode 100644 contracts/interfaces/IMarketplaceEvents.sol rename src/interfaces/IMarketplace.sol => contracts/interfaces/IMarketplaceStructs.sol (98%) rename {src => contracts}/interfaces/IProduct.sol (100%) diff --git a/src/AccessToken.sol b/contracts/AccessToken.sol similarity index 100% rename from src/AccessToken.sol rename to contracts/AccessToken.sol diff --git a/src/AccessTokenFactory.sol b/contracts/AccessTokenFactory.sol similarity index 100% rename from src/AccessTokenFactory.sol rename to contracts/AccessTokenFactory.sol diff --git a/src/Marketplace.sol b/contracts/Marketplace.sol similarity index 78% rename from src/Marketplace.sol rename to contracts/Marketplace.sol index 8b1d6ef..2581263 100644 --- a/src/Marketplace.sol +++ b/contracts/Marketplace.sol @@ -53,22 +53,44 @@ contract Marketplace is IMarketplace, Ownable { _feePoints = feePoints; } - function getListingInfo(address accessToken, uint256 tokenId) external view returns (ListingInfo memory) { + /** + * @inheritdoc IMarketplace + */ + function getListingInfo( + address accessToken, + uint256 tokenId + ) external view returns (ListingInfo memory) { return _listings[accessToken][tokenId]; } - function getRentalInfo(address accessToken, uint256 tokenId, address tenant) external view returns (RentalInfo memory) { + /** + * @inheritdoc IMarketplace + */ + function getRentalInfo( + address accessToken, + uint256 tokenId, + address tenant + ) external view returns (RentalInfo memory) { return _rentals[accessToken][tokenId][tenant]; } + /** + * @inheritdoc IMarketplace + */ function setFeePoints(uint256 feePoints) external onlyOwner { _feePoints = feePoints; } + /** + * @inheritdoc IMarketplace + */ function setTreasury(address payable treasury) external onlyOwner { _treasury = treasury; } + /** + * @inheritdoc IMarketplace + */ function addRentCurrencies( address[] memory rentCurrencies ) external onlyOwner { @@ -77,6 +99,9 @@ contract Marketplace is IMarketplace, Ownable { } } + /** + * @inheritdoc IMarketplace + */ function removeRentCurrencies( address[] memory rentCurrencies ) external onlyOwner { @@ -85,11 +110,12 @@ contract Marketplace is IMarketplace, Ownable { } } + /** + * @inheritdoc IMarketplace + */ function list(ListArgs memory args) public { - address accessToken = ACCESS_TOKEN_FACTORY.getAccessToken( - args.product - ); - if(accessToken == address(0)) { + address accessToken = ACCESS_TOKEN_FACTORY.getAccessToken(args.product); + if (accessToken == address(0)) { accessToken = ACCESS_TOKEN_FACTORY.createAccessToken(args.product); } require( @@ -103,7 +129,8 @@ contract Marketplace is IMarketplace, Ownable { "invalid maximum rental days" ); require( - args.rentCurrency == NATIVE_TOKEN || supportedRentCurrencies[args.rentCurrency], + args.rentCurrency == NATIVE_TOKEN || + supportedRentCurrencies[args.rentCurrency], "unsupported rent currency" ); @@ -122,20 +149,36 @@ contract Marketplace is IMarketplace, Ownable { address(this), args.tokenId ); + + emit List( + msg.sender, + args.product, + accessToken, + args.tokenId, + args.minRentalDays, + args.maxRentalDays, + args.rentCurrency, + args.dailyRent, + args.rentRecipient + ); } + /** + * @inheritdoc IMarketplace + */ function delist(DelistArgs memory args) public { - ListingInfo storage listing = _listings[args.accessToken][ - args.tokenId - ]; + ListingInfo storage listing = _listings[args.accessToken][args.tokenId]; require(listing.owner == msg.sender, "not listing owner"); listing.status = ListingStatus.Delisted; + + emit Delist(msg.sender, args.accessToken, args.tokenId); } + /** + * @inheritdoc IMarketplace + */ function relist(RelistArgs memory args) public { - ListingInfo storage listing = _listings[args.accessToken][ - args.tokenId - ]; + ListingInfo storage listing = _listings[args.accessToken][args.tokenId]; require(listing.owner == msg.sender, "not listing owner"); require(args.minRentalDays > 0, "invalid minimum rental days"); require( @@ -143,7 +186,8 @@ contract Marketplace is IMarketplace, Ownable { "invalid maximum rental days" ); require( - args.rentCurrency == NATIVE_TOKEN || supportedRentCurrencies[args.rentCurrency], + args.rentCurrency == NATIVE_TOKEN || + supportedRentCurrencies[args.rentCurrency], "unsupported rent currency" ); @@ -153,8 +197,22 @@ contract Marketplace is IMarketplace, Ownable { listing.dailyRent = args.dailyRent; listing.rentRecipient = args.rentRecipient; listing.status = ListingStatus.Listing; + + emit Relist( + msg.sender, + args.accessToken, + args.tokenId, + args.minRentalDays, + args.maxRentalDays, + args.rentCurrency, + args.dailyRent, + args.rentRecipient + ); } + /** + * @inheritdoc IMarketplace + */ function rent(RentArgs memory args) public payable { require( _rentals[args.accessToken][args.tokenId][args.tenant].status == @@ -190,12 +248,20 @@ contract Marketplace is IMarketplace, Ownable { args.prepaidRent ); // Mint access token to tenant - AccessToken(args.accessToken).mint( + AccessToken(args.accessToken).mint(args.tenant, args.tokenId); + + emit Rent( args.tenant, - args.tokenId + args.accessToken, + args.tokenId, + args.rentalDays, + args.prepaidRent ); } + /** + * @inheritdoc IMarketplace + */ function payRent(PayRentArgs memory args) public payable { ListingInfo memory listing = _listings[args.accessToken][args.tokenId]; RentalInfo storage rental = _rentals[args.accessToken][args.tokenId][ @@ -209,8 +275,18 @@ contract Marketplace is IMarketplace, Ownable { // Pay rent _payRent(listing, rental, args.rent); + + emit PayRent( + args.tenant, + args.accessToken, + args.tokenId, + args.rent + ); } + /** + * @inheritdoc IMarketplace + */ function endLease(EndLeaseArgs memory args) public { RentalInfo storage rental = _rentals[args.accessToken][args.tokenId][ args.tenant @@ -227,12 +303,20 @@ contract Marketplace is IMarketplace, Ownable { // Burn tenant's access token AccessToken(args.accessToken).burn(args.tokenId); rental.status = RentalStatus.EndedOrNotExist; + + emit EndLease( + args.tenant, + args.accessToken, + args.tokenId, + msg.sender + ); } + /** + * @inheritdoc IMarketplace + */ function withdraw(WithdrawArgs memory args) public { - ListingInfo storage listing = _listings[args.accessToken][ - args.tokenId - ]; + ListingInfo storage listing = _listings[args.accessToken][args.tokenId]; require(listing.owner == msg.sender, "not listing owner"); require( !AccessToken(args.accessToken).isExist(args.tokenId), @@ -245,6 +329,12 @@ contract Marketplace is IMarketplace, Ownable { listing.owner, args.tokenId ); + + emit Withdraw( + msg.sender, + args.accessToken, + args.tokenId + ); } function _payRent( diff --git a/contracts/interfaces/IMarketplace.sol b/contracts/interfaces/IMarketplace.sol new file mode 100644 index 0000000..cf2e99c --- /dev/null +++ b/contracts/interfaces/IMarketplace.sol @@ -0,0 +1,40 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.24; + +import {IMarketplaceStructs} from "./IMarketplaceStructs.sol"; +import {IMarketplaceEvents} from "./IMarketplaceEvents.sol"; + +interface IMarketplace is IMarketplaceStructs, IMarketplaceEvents { + function getListingInfo( + address accessToken, + uint256 tokenId + ) external view returns (ListingInfo memory); + + function getRentalInfo( + address accessToken, + uint256 tokenId, + address tenant + ) external view returns (RentalInfo memory); + + function setFeePoints(uint256 feePoints) external; + + function setTreasury(address payable treasury) external; + + function addRentCurrencies(address[] memory rentCurrencies) external; + + function removeRentCurrencies(address[] memory rentCurrencies) external; + + function list(ListArgs memory args) external; + + function delist(DelistArgs memory args) external; + + function relist(RelistArgs memory args) external; + + function rent(RentArgs memory args) external payable; + + function payRent(PayRentArgs memory args) external payable; + + function endLease(EndLeaseArgs memory args) external; + + function withdraw(WithdrawArgs memory args) external; +} diff --git a/contracts/interfaces/IMarketplaceEvents.sol b/contracts/interfaces/IMarketplaceEvents.sol new file mode 100644 index 0000000..f8da2c8 --- /dev/null +++ b/contracts/interfaces/IMarketplaceEvents.sol @@ -0,0 +1,61 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.24; + +interface IMarketplaceEvents { + event List( + address indexed owner, + address indexed product, + address accessToken, + uint256 indexed tokenId, + uint256 minRentalDays, + uint256 maxRentalDays, + address rentCurrency, + uint256 dailyRent, + address rentRecipient + ); + + event Delist( + address indexed owner, + address indexed accessToken, + uint256 indexed tokenId + ); + + event Relist( + address indexed owner, + address indexed accessToken, + uint256 indexed tokenId, + uint256 minRentalDays, + uint256 maxRentalDays, + address rentCurrency, + uint256 dailyRent, + address rentRecipient + ); + + event Rent( + address indexed tenant, + address indexed accessToken, + uint256 indexed tokenId, + uint256 rentalDays, + uint256 prepaidRent + ); + + event PayRent( + address indexed tenant, + address indexed accessToken, + uint256 indexed tokenId, + uint256 rent + ); + + event EndLease( + address indexed tenant, + address indexed accessToken, + uint256 indexed tokenId, + address operator + ); + + event Withdraw( + address indexed owner, + address indexed accessToken, + uint256 indexed tokenId + ); +} \ No newline at end of file diff --git a/src/interfaces/IMarketplace.sol b/contracts/interfaces/IMarketplaceStructs.sol similarity index 98% rename from src/interfaces/IMarketplace.sol rename to contracts/interfaces/IMarketplaceStructs.sol index 9d8091d..5e13df9 100644 --- a/src/interfaces/IMarketplace.sol +++ b/contracts/interfaces/IMarketplaceStructs.sol @@ -1,7 +1,7 @@ // SPDX-License-Identifier: MIT pragma solidity ^0.8.24; -interface IMarketplace { +interface IMarketplaceStructs { // Statuses of a listing. WithdrawnOrNotExist, which is 0, is effectively the same as never listed before. enum ListingStatus { WithdrawnOrNotExist, diff --git a/src/interfaces/IProduct.sol b/contracts/interfaces/IProduct.sol similarity index 100% rename from src/interfaces/IProduct.sol rename to contracts/interfaces/IProduct.sol diff --git a/foundry.toml b/foundry.toml index d7016e5..c496598 100644 --- a/foundry.toml +++ b/foundry.toml @@ -1,5 +1,5 @@ [profile.default] -src = "src" +src = "contracts" out = "out" libs = ["dependencies"] auto_detect_remappings = true diff --git a/test/AccessToken.t.sol b/test/AccessToken.t.sol index 14d3c92..ba7db42 100644 --- a/test/AccessToken.t.sol +++ b/test/AccessToken.t.sol @@ -2,8 +2,8 @@ pragma solidity ^0.8.24; import {Clones} from "@openzeppelin/contracts/proxy/Clones.sol"; -import {AccessToken} from "../src/AccessToken.sol"; -import {IProduct} from "../src/interfaces/IProduct.sol"; +import {AccessToken} from "../contracts/AccessToken.sol"; +import {IProduct} from "../contracts/interfaces/IProduct.sol"; import {MockProduct} from "./mocks/MockProduct.sol"; import "forge-std/src/Test.sol"; diff --git a/test/AccessTokenFactory.t.sol b/test/AccessTokenFactory.t.sol index c3a4eb3..4054a0e 100644 --- a/test/AccessTokenFactory.t.sol +++ b/test/AccessTokenFactory.t.sol @@ -3,9 +3,9 @@ pragma solidity ^0.8.24; import {Clones} from "@openzeppelin/contracts/proxy/Clones.sol"; import {ERC721} from "@openzeppelin/contracts/token/ERC721/ERC721.sol"; -import {AccessTokenFactory} from "../src/AccessTokenFactory.sol"; -import {AccessToken} from "../src/AccessToken.sol"; -import {IProduct} from "../src/interfaces/IProduct.sol"; +import {AccessTokenFactory} from "../contracts/AccessTokenFactory.sol"; +import {AccessToken} from "../contracts/AccessToken.sol"; +import {IProduct} from "../contracts/interfaces/IProduct.sol"; import {MockProduct} from "./mocks/MockProduct.sol"; import "forge-std/src/Test.sol"; diff --git a/test/Marketplace.t.sol b/test/Marketplace.t.sol index fb45141..1e8e5de 100644 --- a/test/Marketplace.t.sol +++ b/test/Marketplace.t.sol @@ -3,11 +3,11 @@ pragma solidity ^0.8.24; import {Clones} from "@openzeppelin/contracts/proxy/Clones.sol"; import {ERC721} from "@openzeppelin/contracts/token/ERC721/ERC721.sol"; -import {AccessTokenFactory} from "../src/AccessTokenFactory.sol"; -import {IMarketplace} from "../src/interfaces/IMarketplace.sol"; -import {Marketplace} from "../src/Marketplace.sol"; -import {AccessToken} from "../src/AccessToken.sol"; -import {IProduct} from "../src/interfaces/IProduct.sol"; +import {AccessTokenFactory} from "../contracts/AccessTokenFactory.sol"; +import {IMarketplaceStructs} from "../contracts/interfaces/IMarketplaceStructs.sol"; +import {Marketplace} from "../contracts/Marketplace.sol"; +import {AccessToken} from "../contracts/AccessToken.sol"; +import {IProduct} from "../contracts/interfaces/IProduct.sol"; import {MockProduct} from "./mocks/MockProduct.sol"; import "forge-std/src/Test.sol"; @@ -43,7 +43,7 @@ contract AccessTokenFactoryTest is Test { uint256 tokenId = product.mint(address(this)); // List the product token on the marketplace - IMarketplace.ListArgs memory args = IMarketplace.ListArgs({ + IMarketplaceStructs.ListArgs memory args = IMarketplaceStructs.ListArgs({ product: productAddress, tokenId: tokenId, minRentalDays: 1, @@ -57,7 +57,7 @@ contract AccessTokenFactoryTest is Test { marketplace.list(args); // Check that the token is listed - IMarketplace.ListingInfo memory listing = marketplace.getListingInfo( + IMarketplaceStructs.ListingInfo memory listing = marketplace.getListingInfo( address(factory.getAccessToken(productAddress)), tokenId ); @@ -68,7 +68,7 @@ contract AccessTokenFactoryTest is Test { assertEq(listing.dailyRent, 1 ether); assertEq( uint256(listing.status), - uint256(IMarketplace.ListingStatus.Listing) + uint256(IMarketplaceStructs.ListingStatus.Listing) ); } @@ -77,7 +77,7 @@ contract AccessTokenFactoryTest is Test { uint256 tokenId = product.mint(address(this)); // List the product token on the marketplace - IMarketplace.ListArgs memory listArgs = IMarketplace.ListArgs({ + IMarketplaceStructs.ListArgs memory listArgs = IMarketplaceStructs.ListArgs({ product: productAddress, tokenId: tokenId, minRentalDays: 1, @@ -91,7 +91,7 @@ contract AccessTokenFactoryTest is Test { marketplace.list(listArgs); // Delist the token - IMarketplace.DelistArgs memory delistArgs = IMarketplace.DelistArgs({ + IMarketplaceStructs.DelistArgs memory delistArgs = IMarketplaceStructs.DelistArgs({ accessToken: payable(factory.getAccessToken(productAddress)), tokenId: tokenId }); @@ -99,13 +99,13 @@ contract AccessTokenFactoryTest is Test { marketplace.delist(delistArgs); // Check that the token is delisted - IMarketplace.ListingInfo memory listing = marketplace.getListingInfo( + IMarketplaceStructs.ListingInfo memory listing = marketplace.getListingInfo( address(factory.getAccessToken(productAddress)), tokenId ); assertEq( uint256(listing.status), - uint256(IMarketplace.ListingStatus.Delisted) + uint256(IMarketplaceStructs.ListingStatus.Delisted) ); } @@ -117,7 +117,7 @@ contract AccessTokenFactoryTest is Test { vm.deal(tenant, 10 ether); // List the product token on the marketplace - IMarketplace.ListArgs memory listArgs = IMarketplace.ListArgs({ + IMarketplaceStructs.ListArgs memory listArgs = IMarketplaceStructs.ListArgs({ product: productAddress, tokenId: tokenId, minRentalDays: 1, @@ -131,7 +131,7 @@ contract AccessTokenFactoryTest is Test { marketplace.list(listArgs); // Rent the token - IMarketplace.RentArgs memory rentArgs = IMarketplace.RentArgs({ + IMarketplaceStructs.RentArgs memory rentArgs = IMarketplaceStructs.RentArgs({ accessToken: factory.getAccessToken(productAddress), tokenId: tokenId, tenant: tenant, @@ -143,7 +143,7 @@ contract AccessTokenFactoryTest is Test { marketplace.rent{value: 5 ether}(rentArgs); // Check that the token is rented - IMarketplace.RentalInfo memory rental = marketplace.getRentalInfo( + IMarketplaceStructs.RentalInfo memory rental = marketplace.getRentalInfo( address(factory.getAccessToken(productAddress)), tokenId, tenant @@ -154,7 +154,7 @@ contract AccessTokenFactoryTest is Test { assertEq(rental.dailyRent, 1 ether); assertEq( uint256(rental.status), - uint256(IMarketplace.RentalStatus.Renting) + uint256(IMarketplaceStructs.RentalStatus.Renting) ); } @@ -166,7 +166,7 @@ contract AccessTokenFactoryTest is Test { vm.deal(tenant, 20 ether); // List the product token on the marketplace - IMarketplace.ListArgs memory listArgs = IMarketplace.ListArgs({ + IMarketplaceStructs.ListArgs memory listArgs = IMarketplaceStructs.ListArgs({ product: productAddress, tokenId: tokenId, minRentalDays: 1, @@ -180,7 +180,7 @@ contract AccessTokenFactoryTest is Test { marketplace.list(listArgs); // Rent the token - IMarketplace.RentArgs memory rentArgs = IMarketplace.RentArgs({ + IMarketplaceStructs.RentArgs memory rentArgs = IMarketplaceStructs.RentArgs({ accessToken: factory.getAccessToken(productAddress), tokenId: tokenId, tenant: tenant, @@ -191,7 +191,7 @@ contract AccessTokenFactoryTest is Test { vm.prank(tenant); marketplace.rent{value: 5 ether}(rentArgs); - IMarketplace.RentalInfo memory rental = marketplace.getRentalInfo( + IMarketplaceStructs.RentalInfo memory rental = marketplace.getRentalInfo( address(factory.getAccessToken(productAddress)), tokenId, tenant @@ -203,11 +203,11 @@ contract AccessTokenFactoryTest is Test { assertEq(rental.totalPaidRent, 5 ether); assertEq( uint256(rental.status), - uint256(IMarketplace.RentalStatus.Renting) + uint256(IMarketplaceStructs.RentalStatus.Renting) ); // Pay additional rent - IMarketplace.PayRentArgs memory payRentArgs = IMarketplace.PayRentArgs({ + IMarketplaceStructs.PayRentArgs memory payRentArgs = IMarketplaceStructs.PayRentArgs({ accessToken: factory.getAccessToken(productAddress), tokenId: tokenId, tenant: tenant, @@ -235,7 +235,7 @@ contract AccessTokenFactoryTest is Test { vm.deal(tenant, 10 ether); // List the product token on the marketplace - IMarketplace.ListArgs memory listArgs = IMarketplace.ListArgs({ + IMarketplaceStructs.ListArgs memory listArgs = IMarketplaceStructs.ListArgs({ product: productAddress, tokenId: tokenId, minRentalDays: 1, @@ -249,7 +249,7 @@ contract AccessTokenFactoryTest is Test { marketplace.list(listArgs); // Rent the token - IMarketplace.RentArgs memory rentArgs = IMarketplace.RentArgs({ + IMarketplaceStructs.RentArgs memory rentArgs = IMarketplaceStructs.RentArgs({ accessToken: factory.getAccessToken(productAddress), tokenId: tokenId, tenant: tenant, @@ -264,7 +264,7 @@ contract AccessTokenFactoryTest is Test { vm.warp(block.timestamp + 6 days); // End the lease - IMarketplace.EndLeaseArgs memory endLeaseArgs = IMarketplace + IMarketplaceStructs.EndLeaseArgs memory endLeaseArgs = IMarketplaceStructs .EndLeaseArgs({ accessToken: factory.getAccessToken(productAddress), tokenId: tokenId, @@ -275,14 +275,14 @@ contract AccessTokenFactoryTest is Test { marketplace.endLease(endLeaseArgs); // Check that the lease is ended - IMarketplace.RentalInfo memory rental = marketplace.getRentalInfo( + IMarketplaceStructs.RentalInfo memory rental = marketplace.getRentalInfo( address(factory.getAccessToken(productAddress)), tokenId, tenant ); assertEq( uint256(rental.status), - uint256(IMarketplace.RentalStatus.EndedOrNotExist) + uint256(IMarketplaceStructs.RentalStatus.EndedOrNotExist) ); } @@ -291,7 +291,7 @@ contract AccessTokenFactoryTest is Test { uint256 tokenId = product.mint(address(this)); // List the product token on the marketplace - IMarketplace.ListArgs memory listArgs = IMarketplace.ListArgs({ + IMarketplaceStructs.ListArgs memory listArgs = IMarketplaceStructs.ListArgs({ product: productAddress, tokenId: tokenId, minRentalDays: 1, @@ -305,7 +305,7 @@ contract AccessTokenFactoryTest is Test { marketplace.list(listArgs); // Delist the token - IMarketplace.DelistArgs memory delistArgs = IMarketplace.DelistArgs({ + IMarketplaceStructs.DelistArgs memory delistArgs = IMarketplaceStructs.DelistArgs({ accessToken: payable(factory.getAccessToken(productAddress)), tokenId: tokenId }); @@ -313,7 +313,7 @@ contract AccessTokenFactoryTest is Test { marketplace.delist(delistArgs); // Withdraw the token - IMarketplace.WithdrawArgs memory withdrawArgs = IMarketplace + IMarketplaceStructs.WithdrawArgs memory withdrawArgs = IMarketplaceStructs .WithdrawArgs({ accessToken: factory.getAccessToken(productAddress), tokenId: tokenId @@ -322,13 +322,13 @@ contract AccessTokenFactoryTest is Test { marketplace.withdraw(withdrawArgs); // Check that the token is withdrawn - IMarketplace.ListingInfo memory listing = marketplace.getListingInfo( + IMarketplaceStructs.ListingInfo memory listing = marketplace.getListingInfo( address(factory.getAccessToken(productAddress)), tokenId ); assertEq( uint256(listing.status), - uint256(IMarketplace.ListingStatus.WithdrawnOrNotExist) + uint256(IMarketplaceStructs.ListingStatus.WithdrawnOrNotExist) ); } diff --git a/test/mocks/MockProduct.sol b/test/mocks/MockProduct.sol index d03de89..29e9ad6 100644 --- a/test/mocks/MockProduct.sol +++ b/test/mocks/MockProduct.sol @@ -5,7 +5,7 @@ import {Initializable} from "@openzeppelin/contracts-upgradeable/proxy/utils/Ini import {OwnableUpgradeable} from "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol"; import {ERC721Upgradeable} from "@openzeppelin/contracts-upgradeable/token/ERC721/ERC721Upgradeable.sol"; import {IERC165} from "@openzeppelin/contracts/utils/introspection/IERC165.sol"; -import {IProduct} from "../../src/interfaces/IProduct.sol"; +import {IProduct} from "../../contracts/interfaces/IProduct.sol"; contract MockProduct is Initializable, From f1f4e9b1562440410dc22bb848f5ccb62697d1bb Mon Sep 17 00:00:00 2001 From: CanvasL <746591811@qq.com> Date: Tue, 6 Aug 2024 15:37:13 +0800 Subject: [PATCH 2/4] chore: visibility --- contracts/Marketplace.sol | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/contracts/Marketplace.sol b/contracts/Marketplace.sol index 2581263..83a3c79 100644 --- a/contracts/Marketplace.sol +++ b/contracts/Marketplace.sol @@ -12,15 +12,15 @@ import {IMarketplace} from "./interfaces/IMarketplace.sol"; contract Marketplace is IMarketplace, Ownable { using SafeERC20 for IERC20; - address private constant NATIVE_TOKEN = address(0); + address public constant NATIVE_TOKEN = address(0); uint256 public constant MAX_POINTS = 10000; AccessTokenFactory public immutable ACCESS_TOKEN_FACTORY; - address payable private _treasury; + address payable internal _treasury; - uint256 private _feePoints; + uint256 internal _feePoints; /** * @notice rent currency => is supported From b8378aecbb49cdc127c20a0c61daae41f2c49c9d Mon Sep 17 00:00:00 2001 From: CanvasL <746591811@qq.com> Date: Tue, 6 Aug 2024 17:39:07 +0800 Subject: [PATCH 3/4] feat: tool access-control --- .env.example | 4 +- .gitignore | 6 +- addresses.json | 3 + foundry.toml | 2 + package.json | 15 + pnpm-lock.yaml | 1040 +++++++++++++++++++++++++ pnpm-workspace.yaml | 2 + script/DeployAccessTokenFactory.s.sol | 15 + tools/access-control/README.md | 59 ++ tools/access-control/dist/cli.js | 102 +++ tools/access-control/dist/index.js | 17 + tools/access-control/package.json | 22 + tools/access-control/src/cli.ts | 194 +++++ tools/access-control/src/index.ts | 1 + tools/access-control/tsconfig.json | 10 + 15 files changed, 1490 insertions(+), 2 deletions(-) create mode 100644 addresses.json create mode 100644 package.json create mode 100644 pnpm-lock.yaml create mode 100644 pnpm-workspace.yaml create mode 100644 script/DeployAccessTokenFactory.s.sol create mode 100644 tools/access-control/README.md create mode 100644 tools/access-control/dist/cli.js create mode 100644 tools/access-control/dist/index.js create mode 100644 tools/access-control/package.json create mode 100644 tools/access-control/src/cli.ts create mode 100644 tools/access-control/src/index.ts create mode 100644 tools/access-control/tsconfig.json diff --git a/.env.example b/.env.example index 2cf0664..425c69d 100644 --- a/.env.example +++ b/.env.example @@ -1,4 +1,6 @@ BASE_RPC_URL=https://base.rpc.subquery.network/public -BASE_SEPOLIA_RPC_URL=https://base-sepolia-rpc.publicnode.com +BASE_SEPOLIA_RPC_URL=https://base-sepolia.g.alchemy.com/v2/0ZS0OdXDqBpKt6wkusuFDyi0lLlTFRVf +BNB_RPC_URL=https://binance.llamarpc.com +BNB_TESTNET_RPC_URL=https://public.stackup.sh/api/v1/node/bsc-testnet PRIVATE_KEY= diff --git a/.gitignore b/.gitignore index e9edb29..fb4d303 100644 --- a/.gitignore +++ b/.gitignore @@ -13,4 +13,8 @@ docs/ # Dotenv file .env -dependencies/ \ No newline at end of file +dependencies/ + +node_modules + +bun.lockb \ No newline at end of file diff --git a/addresses.json b/addresses.json new file mode 100644 index 0000000..2020e28 --- /dev/null +++ b/addresses.json @@ -0,0 +1,3 @@ +{ + "BNBTestnet": "0x35a8483444947B2166Aa85837F97FaEf122f5ebb" +} \ No newline at end of file diff --git a/foundry.toml b/foundry.toml index c496598..340c13a 100644 --- a/foundry.toml +++ b/foundry.toml @@ -12,3 +12,5 @@ forge-std = { version = "1.9.1" } [rpc_endpoints] base = "${BASE_RPC_URL}" base_sepolia = "${BASE_SEPOLIA_RPC_URL}" +bnb = "${BNB_RPC_URL}" +bnb_testnet = "${BNB_TESTNET_RPC_URL}" \ No newline at end of file diff --git a/package.json b/package.json new file mode 100644 index 0000000..5783a05 --- /dev/null +++ b/package.json @@ -0,0 +1,15 @@ +{ + "name": "dephy-conduits-contracts", + "type": "module", + "scripts": { + "build": "forge build --sizes", + "deploy:base": "source .env && forge script script/Deploy.s.sol --rpc-url base --broadcast", + "deploy:base_sepolia": "source .env && forge script script/Deploy.s.sol --rpc-url base_sepolia --broadcast", + "deploy:bnb": "source .env && forge script script/Deploy.s.sol --rpc-url bnb --broadcast", + "deploy:bnb_testnet": "source .env && forge script script/Deploy.s.sol --rpc-url bnb_testnet --broadcast" + }, + "devDependencies": { + "@types/node": "^20.14.10", + "tsx": "^4.16.2" + } +} diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml new file mode 100644 index 0000000..c3eac47 --- /dev/null +++ b/pnpm-lock.yaml @@ -0,0 +1,1040 @@ +lockfileVersion: '6.0' + +settings: + autoInstallPeers: true + excludeLinksFromLockfile: false + +importers: + + .: + devDependencies: + '@types/node': + specifier: ^20.14.10 + version: 20.14.10 + tsx: + specifier: ^4.16.2 + version: 4.16.2 + + tools/acl: + dependencies: + ethers: + specifier: ^5.7.2 + version: 5.7.2 + typescript: + specifier: ^5.0.0 + version: 5.0.2 + yargs: + specifier: ^17.7.2 + version: 17.7.2 + devDependencies: + '@types/yargs': + specifier: ^17.0.32 + version: 17.0.32 + rimraf: + specifier: ^6.0.0 + version: 6.0.0 + +packages: + + /@esbuild/aix-ppc64@0.21.5: + resolution: {integrity: sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [aix] + requiresBuild: true + dev: true + optional: true + + /@esbuild/android-arm64@0.21.5: + resolution: {integrity: sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A==} + engines: {node: '>=12'} + cpu: [arm64] + os: [android] + requiresBuild: true + dev: true + optional: true + + /@esbuild/android-arm@0.21.5: + resolution: {integrity: sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg==} + engines: {node: '>=12'} + cpu: [arm] + os: [android] + requiresBuild: true + dev: true + optional: true + + /@esbuild/android-x64@0.21.5: + resolution: {integrity: sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA==} + engines: {node: '>=12'} + cpu: [x64] + os: [android] + requiresBuild: true + dev: true + optional: true + + /@esbuild/darwin-arm64@0.21.5: + resolution: {integrity: sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ==} + engines: {node: '>=12'} + cpu: [arm64] + os: [darwin] + requiresBuild: true + dev: true + optional: true + + /@esbuild/darwin-x64@0.21.5: + resolution: {integrity: sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw==} + engines: {node: '>=12'} + cpu: [x64] + os: [darwin] + requiresBuild: true + dev: true + optional: true + + /@esbuild/freebsd-arm64@0.21.5: + resolution: {integrity: sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g==} + engines: {node: '>=12'} + cpu: [arm64] + os: [freebsd] + requiresBuild: true + dev: true + optional: true + + /@esbuild/freebsd-x64@0.21.5: + resolution: {integrity: sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [freebsd] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-arm64@0.21.5: + resolution: {integrity: sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q==} + engines: {node: '>=12'} + cpu: [arm64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-arm@0.21.5: + resolution: {integrity: sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA==} + engines: {node: '>=12'} + cpu: [arm] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-ia32@0.21.5: + resolution: {integrity: sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg==} + engines: {node: '>=12'} + cpu: [ia32] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-loong64@0.21.5: + resolution: {integrity: sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg==} + engines: {node: '>=12'} + cpu: [loong64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-mips64el@0.21.5: + resolution: {integrity: sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg==} + engines: {node: '>=12'} + cpu: [mips64el] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-ppc64@0.21.5: + resolution: {integrity: sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w==} + engines: {node: '>=12'} + cpu: [ppc64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-riscv64@0.21.5: + resolution: {integrity: sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA==} + engines: {node: '>=12'} + cpu: [riscv64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-s390x@0.21.5: + resolution: {integrity: sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A==} + engines: {node: '>=12'} + cpu: [s390x] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/linux-x64@0.21.5: + resolution: {integrity: sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ==} + engines: {node: '>=12'} + cpu: [x64] + os: [linux] + requiresBuild: true + dev: true + optional: true + + /@esbuild/netbsd-x64@0.21.5: + resolution: {integrity: sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg==} + engines: {node: '>=12'} + cpu: [x64] + os: [netbsd] + requiresBuild: true + dev: true + optional: true + + /@esbuild/openbsd-x64@0.21.5: + resolution: {integrity: sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow==} + engines: {node: '>=12'} + cpu: [x64] + os: [openbsd] + requiresBuild: true + dev: true + optional: true + + /@esbuild/sunos-x64@0.21.5: + resolution: {integrity: sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg==} + engines: {node: '>=12'} + cpu: [x64] + os: [sunos] + requiresBuild: true + dev: true + optional: true + + /@esbuild/win32-arm64@0.21.5: + resolution: {integrity: sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A==} + engines: {node: '>=12'} + cpu: [arm64] + os: [win32] + requiresBuild: true + dev: true + optional: true + + /@esbuild/win32-ia32@0.21.5: + resolution: {integrity: sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA==} + engines: {node: '>=12'} + cpu: [ia32] + os: [win32] + requiresBuild: true + dev: true + optional: true + + /@esbuild/win32-x64@0.21.5: + resolution: {integrity: sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw==} + engines: {node: '>=12'} + cpu: [x64] + os: [win32] + requiresBuild: true + dev: true + optional: true + + /@ethersproject/abi@5.7.0: + resolution: {integrity: sha512-351ktp42TiRcYB3H1OP8yajPeAQstMW/yCFokj/AthP9bLHzQFPlOrxOcwYEDkUAICmOHljvN4K39OMTMUa9RA==} + dependencies: + '@ethersproject/address': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/hash': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/strings': 5.7.0 + dev: false + + /@ethersproject/abstract-provider@5.7.0: + resolution: {integrity: sha512-R41c9UkchKCpAqStMYUpdunjo3pkEvZC3FAwZn5S5MGbXoMQOHIdHItezTETxAO5bevtMApSyEhn9+CHcDsWBw==} + dependencies: + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/networks': 5.7.1 + '@ethersproject/properties': 5.7.0 + '@ethersproject/transactions': 5.7.0 + '@ethersproject/web': 5.7.1 + dev: false + + /@ethersproject/abstract-signer@5.7.0: + resolution: {integrity: sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==} + dependencies: + '@ethersproject/abstract-provider': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + dev: false + + /@ethersproject/address@5.7.0: + resolution: {integrity: sha512-9wYhYt7aghVGo758POM5nqcOMaE168Q6aRLJZwUmiqSrAungkG74gSSeKEIR7ukixesdRZGPgVqme6vmxs1fkA==} + dependencies: + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/rlp': 5.7.0 + dev: false + + /@ethersproject/base64@5.7.0: + resolution: {integrity: sha512-Dr8tcHt2mEbsZr/mwTPIQAf3Ai0Bks/7gTw9dSqk1mQvhW3XvRlmDJr/4n+wg1JmCl16NZue17CDh8xb/vZ0sQ==} + dependencies: + '@ethersproject/bytes': 5.7.0 + dev: false + + /@ethersproject/basex@5.7.0: + resolution: {integrity: sha512-ywlh43GwZLv2Voc2gQVTKBoVQ1mti3d8HK5aMxsfu/nRDnMmNqaSJ3r3n85HBByT8OpoY96SXM1FogC533T4zw==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/properties': 5.7.0 + dev: false + + /@ethersproject/bignumber@5.7.0: + resolution: {integrity: sha512-n1CAdIHRWjSucQO3MC1zPSVgV/6dy/fjL9pMrPP9peL+QxEg9wOsVqwD4+818B6LUEtaXzVHQiuivzRoxPxUGw==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + bn.js: 5.2.1 + dev: false + + /@ethersproject/bytes@5.7.0: + resolution: {integrity: sha512-nsbxwgFXWh9NyYWo+U8atvmMsSdKJprTcICAkvbBffT75qDocbuggBU0SJiVK2MuTrp0q+xvLkTnGMPK1+uA9A==} + dependencies: + '@ethersproject/logger': 5.7.0 + dev: false + + /@ethersproject/constants@5.7.0: + resolution: {integrity: sha512-DHI+y5dBNvkpYUMiRQyxRBYBefZkJfo70VUkUAsRjcPs47muV9evftfZ0PJVCXYbAiCgght0DtcF9srFQmIgWA==} + dependencies: + '@ethersproject/bignumber': 5.7.0 + dev: false + + /@ethersproject/contracts@5.7.0: + resolution: {integrity: sha512-5GJbzEU3X+d33CdfPhcyS+z8MzsTrBGk/sc+G+59+tPa9yFkl6HQ9D6L0QMgNTA9q8dT0XKxxkyp883XsQvbbg==} + dependencies: + '@ethersproject/abi': 5.7.0 + '@ethersproject/abstract-provider': 5.7.0 + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/address': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/transactions': 5.7.0 + dev: false + + /@ethersproject/hash@5.7.0: + resolution: {integrity: sha512-qX5WrQfnah1EFnO5zJv1v46a8HW0+E5xuBBDTwMFZLuVTx0tbU2kkx15NqdjxecrLGatQN9FGQKpb1FKdHCt+g==} + dependencies: + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/address': 5.7.0 + '@ethersproject/base64': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/strings': 5.7.0 + dev: false + + /@ethersproject/hdnode@5.7.0: + resolution: {integrity: sha512-OmyYo9EENBPPf4ERhR7oj6uAtUAhYGqOnIS+jE5pTXvdKBS99ikzq1E7Iv0ZQZ5V36Lqx1qZLeak0Ra16qpeOg==} + dependencies: + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/basex': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/pbkdf2': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/sha2': 5.7.0 + '@ethersproject/signing-key': 5.7.0 + '@ethersproject/strings': 5.7.0 + '@ethersproject/transactions': 5.7.0 + '@ethersproject/wordlists': 5.7.0 + dev: false + + /@ethersproject/json-wallets@5.7.0: + resolution: {integrity: sha512-8oee5Xgu6+RKgJTkvEMl2wDgSPSAQ9MB/3JYjFV9jlKvcYHUXZC+cQp0njgmxdHkYWn8s6/IqIZYm0YWCjO/0g==} + dependencies: + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/address': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/hdnode': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/pbkdf2': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/random': 5.7.0 + '@ethersproject/strings': 5.7.0 + '@ethersproject/transactions': 5.7.0 + aes-js: 3.0.0 + scrypt-js: 3.0.1 + dev: false + + /@ethersproject/keccak256@5.7.0: + resolution: {integrity: sha512-2UcPboeL/iW+pSg6vZ6ydF8tCnv3Iu/8tUmLLzWWGzxWKFFqOBQFLo6uLUv6BDrLgCDfN28RJ/wtByx+jZ4KBg==} + dependencies: + '@ethersproject/bytes': 5.7.0 + js-sha3: 0.8.0 + dev: false + + /@ethersproject/logger@5.7.0: + resolution: {integrity: sha512-0odtFdXu/XHtjQXJYA3u9G0G8btm0ND5Cu8M7i5vhEcE8/HmF4Lbdqanwyv4uQTr2tx6b7fQRmgLrsnpQlmnig==} + dev: false + + /@ethersproject/networks@5.7.1: + resolution: {integrity: sha512-n/MufjFYv3yFcUyfhnXotyDlNdFb7onmkSy8aQERi2PjNcnWQ66xXxa3XlS8nCcA8aJKJjIIMNJTC7tu80GwpQ==} + dependencies: + '@ethersproject/logger': 5.7.0 + dev: false + + /@ethersproject/pbkdf2@5.7.0: + resolution: {integrity: sha512-oR/dBRZR6GTyaofd86DehG72hY6NpAjhabkhxgr3X2FpJtJuodEl2auADWBZfhDHgVCbu3/H/Ocq2uC6dpNjjw==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/sha2': 5.7.0 + dev: false + + /@ethersproject/properties@5.7.0: + resolution: {integrity: sha512-J87jy8suntrAkIZtecpxEPxY//szqr1mlBaYlQ0r4RCaiD2hjheqF9s1LVE8vVuJCXisjIP+JgtK/Do54ej4Sw==} + dependencies: + '@ethersproject/logger': 5.7.0 + dev: false + + /@ethersproject/providers@5.7.2: + resolution: {integrity: sha512-g34EWZ1WWAVgr4aptGlVBF8mhl3VWjv+8hoAnzStu8Ah22VHBsuGzP17eb6xDVRzw895G4W7vvx60lFFur/1Rg==} + dependencies: + '@ethersproject/abstract-provider': 5.7.0 + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/address': 5.7.0 + '@ethersproject/base64': 5.7.0 + '@ethersproject/basex': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/hash': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/networks': 5.7.1 + '@ethersproject/properties': 5.7.0 + '@ethersproject/random': 5.7.0 + '@ethersproject/rlp': 5.7.0 + '@ethersproject/sha2': 5.7.0 + '@ethersproject/strings': 5.7.0 + '@ethersproject/transactions': 5.7.0 + '@ethersproject/web': 5.7.1 + bech32: 1.1.4 + ws: 7.4.6 + transitivePeerDependencies: + - bufferutil + - utf-8-validate + dev: false + + /@ethersproject/random@5.7.0: + resolution: {integrity: sha512-19WjScqRA8IIeWclFme75VMXSBvi4e6InrUNuaR4s5pTF2qNhcGdCUwdxUVGtDDqC00sDLCO93jPQoDUH4HVmQ==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + dev: false + + /@ethersproject/rlp@5.7.0: + resolution: {integrity: sha512-rBxzX2vK8mVF7b0Tol44t5Tb8gomOHkj5guL+HhzQ1yBh/ydjGnpw6at+X6Iw0Kp3OzzzkcKp8N9r0W4kYSs9w==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + dev: false + + /@ethersproject/sha2@5.7.0: + resolution: {integrity: sha512-gKlH42riwb3KYp0reLsFTokByAKoJdgFCwI+CCiX/k+Jm2mbNs6oOaCjYQSlI1+XBVejwH2KrmCbMAT/GnRDQw==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + hash.js: 1.1.7 + dev: false + + /@ethersproject/signing-key@5.7.0: + resolution: {integrity: sha512-MZdy2nL3wO0u7gkB4nA/pEf8lu1TlFswPNmy8AiYkfKTdO6eXBJyUdmHO/ehm/htHw9K/qF8ujnTyUAD+Ry54Q==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + bn.js: 5.2.1 + elliptic: 6.5.4 + hash.js: 1.1.7 + dev: false + + /@ethersproject/solidity@5.7.0: + resolution: {integrity: sha512-HmabMd2Dt/raavyaGukF4XxizWKhKQ24DoLtdNbBmNKUOPqwjsKQSdV9GQtj9CBEea9DlzETlVER1gYeXXBGaA==} + dependencies: + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/sha2': 5.7.0 + '@ethersproject/strings': 5.7.0 + dev: false + + /@ethersproject/strings@5.7.0: + resolution: {integrity: sha512-/9nu+lj0YswRNSH0NXYqrh8775XNyEdUQAuf3f+SmOrnVewcJ5SBNAjF7lpgehKi4abvNNXyf+HX86czCdJ8Mg==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/logger': 5.7.0 + dev: false + + /@ethersproject/transactions@5.7.0: + resolution: {integrity: sha512-kmcNicCp1lp8qanMTC3RIikGgoJ80ztTyvtsFvCYpSCfkjhD0jZ2LOrnbcuxuToLIUYYf+4XwD1rP+B/erDIhQ==} + dependencies: + '@ethersproject/address': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/rlp': 5.7.0 + '@ethersproject/signing-key': 5.7.0 + dev: false + + /@ethersproject/units@5.7.0: + resolution: {integrity: sha512-pD3xLMy3SJu9kG5xDGI7+xhTEmGXlEqXU4OfNapmfnxLVY4EMSSRp7j1k7eezutBPH7RBN/7QPnwR7hzNlEFeg==} + dependencies: + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/logger': 5.7.0 + dev: false + + /@ethersproject/wallet@5.7.0: + resolution: {integrity: sha512-MhmXlJXEJFBFVKrDLB4ZdDzxcBxQ3rLyCkhNqVu3CDYvR97E+8r01UgrI+TI99Le+aYm/in/0vp86guJuM7FCA==} + dependencies: + '@ethersproject/abstract-provider': 5.7.0 + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/address': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/hash': 5.7.0 + '@ethersproject/hdnode': 5.7.0 + '@ethersproject/json-wallets': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/random': 5.7.0 + '@ethersproject/signing-key': 5.7.0 + '@ethersproject/transactions': 5.7.0 + '@ethersproject/wordlists': 5.7.0 + dev: false + + /@ethersproject/web@5.7.1: + resolution: {integrity: sha512-Gueu8lSvyjBWL4cYsWsjh6MtMwM0+H4HvqFPZfB6dV8ctbP9zFAO73VG1cMWae0FLPCtz0peKPpZY8/ugJJX2w==} + dependencies: + '@ethersproject/base64': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/strings': 5.7.0 + dev: false + + /@ethersproject/wordlists@5.7.0: + resolution: {integrity: sha512-S2TFNJNfHWVHNE6cNDjbVlZ6MgE17MIxMbMg2zv3wn+3XSJGosL1m9ZVv3GXCf/2ymSsQ+hRI5IzoMJTG6aoVA==} + dependencies: + '@ethersproject/bytes': 5.7.0 + '@ethersproject/hash': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/strings': 5.7.0 + dev: false + + /@isaacs/cliui@8.0.2: + resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==} + engines: {node: '>=12'} + dependencies: + string-width: 5.1.2 + string-width-cjs: /string-width@4.2.3 + strip-ansi: 7.1.0 + strip-ansi-cjs: /strip-ansi@6.0.1 + wrap-ansi: 8.1.0 + wrap-ansi-cjs: /wrap-ansi@7.0.0 + dev: true + + /@pkgjs/parseargs@0.11.0: + resolution: {integrity: sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg==} + engines: {node: '>=14'} + requiresBuild: true + dev: true + optional: true + + /@types/node@20.14.10: + resolution: {integrity: sha512-MdiXf+nDuMvY0gJKxyfZ7/6UFsETO7mGKF54MVD/ekJS6HdFtpZFBgrh6Pseu64XTb2MLyFPlbW6hj8HYRQNOQ==} + dependencies: + undici-types: 5.26.5 + dev: true + + /@types/yargs-parser@21.0.3: + resolution: {integrity: sha512-I4q9QU9MQv4oEOz4tAHJtNz1cwuLxn2F3xcc2iV5WdqLPpUnj30aUuxt1mAxYTG+oe8CZMV/+6rU4S4gRDzqtQ==} + dev: true + + /@types/yargs@17.0.32: + resolution: {integrity: sha512-xQ67Yc/laOG5uMfX/093MRlGGCIBzZMarVa+gfNKJxWAIgykYpVGkBdbqEzGDDfCrVUj6Hiff4mTZ5BA6TmAog==} + dependencies: + '@types/yargs-parser': 21.0.3 + dev: true + + /aes-js@3.0.0: + resolution: {integrity: sha512-H7wUZRn8WpTq9jocdxQ2c8x2sKo9ZVmzfRE13GiNJXfp7NcKYEdvl3vspKjXox6RIG2VtaRe4JFvxG4rqp2Zuw==} + dev: false + + /ansi-regex@5.0.1: + resolution: {integrity: sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==} + engines: {node: '>=8'} + + /ansi-regex@6.0.1: + resolution: {integrity: sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==} + engines: {node: '>=12'} + dev: true + + /ansi-styles@4.3.0: + resolution: {integrity: sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==} + engines: {node: '>=8'} + dependencies: + color-convert: 2.0.1 + + /ansi-styles@6.2.1: + resolution: {integrity: sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug==} + engines: {node: '>=12'} + dev: true + + /balanced-match@1.0.2: + resolution: {integrity: sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==} + dev: true + + /bech32@1.1.4: + resolution: {integrity: sha512-s0IrSOzLlbvX7yp4WBfPITzpAU8sqQcpsmwXDiKwrG4r491vwCO/XpejasRNl0piBMe/DvP4Tz0mIS/X1DPJBQ==} + dev: false + + /bn.js@4.12.0: + resolution: {integrity: sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==} + dev: false + + /bn.js@5.2.1: + resolution: {integrity: sha512-eXRvHzWyYPBuB4NBy0cmYQjGitUrtqwbvlzP3G6VFnNRbsZQIxQ10PbKKHt8gZ/HW/D/747aDl+QkDqg3KQLMQ==} + dev: false + + /brace-expansion@2.0.1: + resolution: {integrity: sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==} + dependencies: + balanced-match: 1.0.2 + dev: true + + /brorand@1.1.0: + resolution: {integrity: sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==} + dev: false + + /cliui@8.0.1: + resolution: {integrity: sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==} + engines: {node: '>=12'} + dependencies: + string-width: 4.2.3 + strip-ansi: 6.0.1 + wrap-ansi: 7.0.0 + dev: false + + /color-convert@2.0.1: + resolution: {integrity: sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==} + engines: {node: '>=7.0.0'} + dependencies: + color-name: 1.1.4 + + /color-name@1.1.4: + resolution: {integrity: sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==} + + /cross-spawn@7.0.3: + resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==} + engines: {node: '>= 8'} + dependencies: + path-key: 3.1.1 + shebang-command: 2.0.0 + which: 2.0.2 + dev: true + + /eastasianwidth@0.2.0: + resolution: {integrity: sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==} + dev: true + + /elliptic@6.5.4: + resolution: {integrity: sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ==} + dependencies: + bn.js: 4.12.0 + brorand: 1.1.0 + hash.js: 1.1.7 + hmac-drbg: 1.0.1 + inherits: 2.0.4 + minimalistic-assert: 1.0.1 + minimalistic-crypto-utils: 1.0.1 + dev: false + + /emoji-regex@8.0.0: + resolution: {integrity: sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==} + + /emoji-regex@9.2.2: + resolution: {integrity: sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==} + dev: true + + /esbuild@0.21.5: + resolution: {integrity: sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw==} + engines: {node: '>=12'} + hasBin: true + requiresBuild: true + optionalDependencies: + '@esbuild/aix-ppc64': 0.21.5 + '@esbuild/android-arm': 0.21.5 + '@esbuild/android-arm64': 0.21.5 + '@esbuild/android-x64': 0.21.5 + '@esbuild/darwin-arm64': 0.21.5 + '@esbuild/darwin-x64': 0.21.5 + '@esbuild/freebsd-arm64': 0.21.5 + '@esbuild/freebsd-x64': 0.21.5 + '@esbuild/linux-arm': 0.21.5 + '@esbuild/linux-arm64': 0.21.5 + '@esbuild/linux-ia32': 0.21.5 + '@esbuild/linux-loong64': 0.21.5 + '@esbuild/linux-mips64el': 0.21.5 + '@esbuild/linux-ppc64': 0.21.5 + '@esbuild/linux-riscv64': 0.21.5 + '@esbuild/linux-s390x': 0.21.5 + '@esbuild/linux-x64': 0.21.5 + '@esbuild/netbsd-x64': 0.21.5 + '@esbuild/openbsd-x64': 0.21.5 + '@esbuild/sunos-x64': 0.21.5 + '@esbuild/win32-arm64': 0.21.5 + '@esbuild/win32-ia32': 0.21.5 + '@esbuild/win32-x64': 0.21.5 + dev: true + + /escalade@3.1.2: + resolution: {integrity: sha512-ErCHMCae19vR8vQGe50xIsVomy19rg6gFu3+r3jkEO46suLMWBksvVyoGgQV+jOfl84ZSOSlmv6Gxa89PmTGmA==} + engines: {node: '>=6'} + dev: false + + /ethers@5.7.2: + resolution: {integrity: sha512-wswUsmWo1aOK8rR7DIKiWSw9DbLWe6x98Jrn8wcTflTVvaXhAMaB5zGAXy0GYQEQp9iO1iSHWVyARQm11zUtyg==} + dependencies: + '@ethersproject/abi': 5.7.0 + '@ethersproject/abstract-provider': 5.7.0 + '@ethersproject/abstract-signer': 5.7.0 + '@ethersproject/address': 5.7.0 + '@ethersproject/base64': 5.7.0 + '@ethersproject/basex': 5.7.0 + '@ethersproject/bignumber': 5.7.0 + '@ethersproject/bytes': 5.7.0 + '@ethersproject/constants': 5.7.0 + '@ethersproject/contracts': 5.7.0 + '@ethersproject/hash': 5.7.0 + '@ethersproject/hdnode': 5.7.0 + '@ethersproject/json-wallets': 5.7.0 + '@ethersproject/keccak256': 5.7.0 + '@ethersproject/logger': 5.7.0 + '@ethersproject/networks': 5.7.1 + '@ethersproject/pbkdf2': 5.7.0 + '@ethersproject/properties': 5.7.0 + '@ethersproject/providers': 5.7.2 + '@ethersproject/random': 5.7.0 + '@ethersproject/rlp': 5.7.0 + '@ethersproject/sha2': 5.7.0 + '@ethersproject/signing-key': 5.7.0 + '@ethersproject/solidity': 5.7.0 + '@ethersproject/strings': 5.7.0 + '@ethersproject/transactions': 5.7.0 + '@ethersproject/units': 5.7.0 + '@ethersproject/wallet': 5.7.0 + '@ethersproject/web': 5.7.1 + '@ethersproject/wordlists': 5.7.0 + transitivePeerDependencies: + - bufferutil + - utf-8-validate + dev: false + + /foreground-child@3.2.1: + resolution: {integrity: sha512-PXUUyLqrR2XCWICfv6ukppP96sdFwWbNEnfEMt7jNsISjMsvaLNinAHNDYyvkyU+SZG2BTSbT5NjG+vZslfGTA==} + engines: {node: '>=14'} + dependencies: + cross-spawn: 7.0.3 + signal-exit: 4.1.0 + dev: true + + /fsevents@2.3.3: + resolution: {integrity: sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==} + engines: {node: ^8.16.0 || ^10.6.0 || >=11.0.0} + os: [darwin] + requiresBuild: true + dev: true + optional: true + + /get-caller-file@2.0.5: + resolution: {integrity: sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==} + engines: {node: 6.* || 8.* || >= 10.*} + dev: false + + /get-tsconfig@4.7.6: + resolution: {integrity: sha512-ZAqrLlu18NbDdRaHq+AKXzAmqIUPswPWKUchfytdAjiRFnCe5ojG2bstg6mRiZabkKfCoL/e98pbBELIV/YCeA==} + dependencies: + resolve-pkg-maps: 1.0.0 + dev: true + + /glob@11.0.0: + resolution: {integrity: sha512-9UiX/Bl6J2yaBbxKoEBRm4Cipxgok8kQYcOPEhScPwebu2I0HoQOuYdIO6S3hLuWoZgpDpwQZMzTFxgpkyT76g==} + engines: {node: 20 || >=22} + hasBin: true + dependencies: + foreground-child: 3.2.1 + jackspeak: 4.0.1 + minimatch: 10.0.1 + minipass: 7.1.2 + package-json-from-dist: 1.0.0 + path-scurry: 2.0.0 + dev: true + + /hash.js@1.1.7: + resolution: {integrity: sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==} + dependencies: + inherits: 2.0.4 + minimalistic-assert: 1.0.1 + dev: false + + /hmac-drbg@1.0.1: + resolution: {integrity: sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==} + dependencies: + hash.js: 1.1.7 + minimalistic-assert: 1.0.1 + minimalistic-crypto-utils: 1.0.1 + dev: false + + /inherits@2.0.4: + resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==} + dev: false + + /is-fullwidth-code-point@3.0.0: + resolution: {integrity: sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==} + engines: {node: '>=8'} + + /isexe@2.0.0: + resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} + dev: true + + /jackspeak@4.0.1: + resolution: {integrity: sha512-cub8rahkh0Q/bw1+GxP7aeSe29hHHn2V4m29nnDlvCdlgU+3UGxkZp7Z53jLUdpX3jdTO0nJZUDl3xvbWc2Xog==} + engines: {node: 20 || >=22} + dependencies: + '@isaacs/cliui': 8.0.2 + optionalDependencies: + '@pkgjs/parseargs': 0.11.0 + dev: true + + /js-sha3@0.8.0: + resolution: {integrity: sha512-gF1cRrHhIzNfToc802P800N8PpXS+evLLXfsVpowqmAFR9uwbi89WvXg2QspOmXL8QL86J4T1EpFu+yUkwJY3Q==} + dev: false + + /lru-cache@11.0.0: + resolution: {integrity: sha512-Qv32eSV1RSCfhY3fpPE2GNZ8jgM9X7rdAfemLWqTUxwiyIC4jJ6Sy0fZ8H+oLWevO6i4/bizg7c8d8i6bxrzbA==} + engines: {node: 20 || >=22} + dev: true + + /minimalistic-assert@1.0.1: + resolution: {integrity: sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==} + dev: false + + /minimalistic-crypto-utils@1.0.1: + resolution: {integrity: sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==} + dev: false + + /minimatch@10.0.1: + resolution: {integrity: sha512-ethXTt3SGGR+95gudmqJ1eNhRO7eGEGIgYA9vnPatK4/etz2MEVDno5GMCibdMTuBMyElzIlgxMna3K94XDIDQ==} + engines: {node: 20 || >=22} + dependencies: + brace-expansion: 2.0.1 + dev: true + + /minipass@7.1.2: + resolution: {integrity: sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw==} + engines: {node: '>=16 || 14 >=14.17'} + dev: true + + /package-json-from-dist@1.0.0: + resolution: {integrity: sha512-dATvCeZN/8wQsGywez1mzHtTlP22H8OEfPrVMLNr4/eGa+ijtLn/6M5f0dY8UKNrC2O9UCU6SSoG3qRKnt7STw==} + dev: true + + /path-key@3.1.1: + resolution: {integrity: sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==} + engines: {node: '>=8'} + dev: true + + /path-scurry@2.0.0: + resolution: {integrity: sha512-ypGJsmGtdXUOeM5u93TyeIEfEhM6s+ljAhrk5vAvSx8uyY/02OvrZnA0YNGUrPXfpJMgI1ODd3nwz8Npx4O4cg==} + engines: {node: 20 || >=22} + dependencies: + lru-cache: 11.0.0 + minipass: 7.1.2 + dev: true + + /require-directory@2.1.1: + resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==} + engines: {node: '>=0.10.0'} + dev: false + + /resolve-pkg-maps@1.0.0: + resolution: {integrity: sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==} + dev: true + + /rimraf@6.0.0: + resolution: {integrity: sha512-u+yqhM92LW+89cxUQK0SRyvXYQmyuKHx0jkx4W7KfwLGLqJnQM5031Uv1trE4gB9XEXBM/s6MxKlfW95IidqaA==} + engines: {node: 20 || >=22} + hasBin: true + dependencies: + glob: 11.0.0 + dev: true + + /scrypt-js@3.0.1: + resolution: {integrity: sha512-cdwTTnqPu0Hyvf5in5asVdZocVDTNRmR7XEcJuIzMjJeSHybHl7vpB66AzwTaIg6CLSbtjcxc8fqcySfnTkccA==} + dev: false + + /shebang-command@2.0.0: + resolution: {integrity: sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==} + engines: {node: '>=8'} + dependencies: + shebang-regex: 3.0.0 + dev: true + + /shebang-regex@3.0.0: + resolution: {integrity: sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==} + engines: {node: '>=8'} + dev: true + + /signal-exit@4.1.0: + resolution: {integrity: sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==} + engines: {node: '>=14'} + dev: true + + /string-width@4.2.3: + resolution: {integrity: sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==} + engines: {node: '>=8'} + dependencies: + emoji-regex: 8.0.0 + is-fullwidth-code-point: 3.0.0 + strip-ansi: 6.0.1 + + /string-width@5.1.2: + resolution: {integrity: sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==} + engines: {node: '>=12'} + dependencies: + eastasianwidth: 0.2.0 + emoji-regex: 9.2.2 + strip-ansi: 7.1.0 + dev: true + + /strip-ansi@6.0.1: + resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} + engines: {node: '>=8'} + dependencies: + ansi-regex: 5.0.1 + + /strip-ansi@7.1.0: + resolution: {integrity: sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ==} + engines: {node: '>=12'} + dependencies: + ansi-regex: 6.0.1 + dev: true + + /tsx@4.16.2: + resolution: {integrity: sha512-C1uWweJDgdtX2x600HjaFaucXTilT7tgUZHbOE4+ypskZ1OP8CRCSDkCxG6Vya9EwaFIVagWwpaVAn5wzypaqQ==} + engines: {node: '>=18.0.0'} + hasBin: true + dependencies: + esbuild: 0.21.5 + get-tsconfig: 4.7.6 + optionalDependencies: + fsevents: 2.3.3 + dev: true + + /typescript@5.0.2: + resolution: {integrity: sha512-wVORMBGO/FAs/++blGNeAVdbNKtIh1rbBL2EyQ1+J9lClJ93KiiKe8PmFIVdXhHcyv44SL9oglmfeSsndo0jRw==} + engines: {node: '>=12.20'} + hasBin: true + dev: false + + /undici-types@5.26.5: + resolution: {integrity: sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==} + dev: true + + /which@2.0.2: + resolution: {integrity: sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==} + engines: {node: '>= 8'} + hasBin: true + dependencies: + isexe: 2.0.0 + dev: true + + /wrap-ansi@7.0.0: + resolution: {integrity: sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==} + engines: {node: '>=10'} + dependencies: + ansi-styles: 4.3.0 + string-width: 4.2.3 + strip-ansi: 6.0.1 + + /wrap-ansi@8.1.0: + resolution: {integrity: sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ==} + engines: {node: '>=12'} + dependencies: + ansi-styles: 6.2.1 + string-width: 5.1.2 + strip-ansi: 7.1.0 + dev: true + + /ws@7.4.6: + resolution: {integrity: sha512-YmhHDO4MzaDLB+M9ym/mDA5z0naX8j7SIlT8f8z+I0VtzsRbekxEutHSme7NPS2qE8StCYQNUnfWdXta/Yu85A==} + engines: {node: '>=8.3.0'} + peerDependencies: + bufferutil: ^4.0.1 + utf-8-validate: ^5.0.2 + peerDependenciesMeta: + bufferutil: + optional: true + utf-8-validate: + optional: true + dev: false + + /y18n@5.0.8: + resolution: {integrity: sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==} + engines: {node: '>=10'} + dev: false + + /yargs-parser@21.1.1: + resolution: {integrity: sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==} + engines: {node: '>=12'} + dev: false + + /yargs@17.7.2: + resolution: {integrity: sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==} + engines: {node: '>=12'} + dependencies: + cliui: 8.0.1 + escalade: 3.1.2 + get-caller-file: 2.0.5 + require-directory: 2.1.1 + string-width: 4.2.3 + y18n: 5.0.8 + yargs-parser: 21.1.1 + dev: false diff --git a/pnpm-workspace.yaml b/pnpm-workspace.yaml new file mode 100644 index 0000000..df70b1f --- /dev/null +++ b/pnpm-workspace.yaml @@ -0,0 +1,2 @@ +packages: + - 'tools/**' diff --git a/script/DeployAccessTokenFactory.s.sol b/script/DeployAccessTokenFactory.s.sol new file mode 100644 index 0000000..53aa2e5 --- /dev/null +++ b/script/DeployAccessTokenFactory.s.sol @@ -0,0 +1,15 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.24; + +import {AccessTokenFactory} from "../contracts/AccessTokenFactory.sol"; +import "forge-std/src/Script.sol"; + +contract DeployAccessTokenFactory is Script { + function run() public returns (address) { + uint256 deployerPrivateKey = vm.envUint("PRIVATE_KEY"); + vm.startBroadcast(deployerPrivateKey); + AccessTokenFactory factory = new AccessTokenFactory(); + vm.stopBroadcast(); + return address(factory); + } +} diff --git a/tools/access-control/README.md b/tools/access-control/README.md new file mode 100644 index 0000000..91544c4 --- /dev/null +++ b/tools/access-control/README.md @@ -0,0 +1,59 @@ +# Tool Access Control + +## Cli + +Prepare environmental variables: + +```bash +source .env +``` + +### Help + +```bash +pnpm run cli --help +``` + +### Create AccessToken + +```bash +pnpm run cli createAccessToken \ +--rpc $BNB_TESTNET_RPC_URL \ +--privatekey $PRIVATE_KEY \ +--accessTokenFactory {accessTokenFactory address} +--product {product address} +``` + +### Mint + +```bash +pnpm run cli mint \ +--rpc $BNB_TESTNET_RPC_URL \ +--privatekey $PRIVATE_KEY \ +--accessTokenFactory {accessTokenFactory address} +--product {product address} +--tokenId {product token id} +--user {user address} +``` + +### Burn + +```bash +pnpm run cli burn \ +--rpc $BNB_TESTNET_RPC_URL \ +--privatekey $PRIVATE_KEY \ +--accessTokenFactory {accessTokenFactory address} +--product {product address} +--tokenId {product token id} +``` + +### Access Control + +```bash +pnpm run cli accessControl \ +--rpc $BNB_TESTNET_RPC_URL \ +--accessTokenFactory {accessTokenFactory address} +--product {product address} +--tokenId {product token id} +--user {user address} +``` diff --git a/tools/access-control/dist/cli.js b/tools/access-control/dist/cli.js new file mode 100644 index 0000000..8fbd1b4 --- /dev/null +++ b/tools/access-control/dist/cli.js @@ -0,0 +1,102 @@ +"use strict"; +var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { + function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } + return new (P || (P = Promise))(function (resolve, reject) { + function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } + function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } + function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } + step((generator = generator.apply(thisArg, _arguments || [])).next()); + }); +}; +var __importDefault = (this && this.__importDefault) || function (mod) { + return (mod && mod.__esModule) ? mod : { "default": mod }; +}; +Object.defineProperty(exports, "__esModule", { value: true }); +const ethers_1 = require("ethers"); +const yargs_1 = __importDefault(require("yargs")); +const helpers_1 = require("yargs/helpers"); +const fs_1 = __importDefault(require("fs")); +const path_1 = __importDefault(require("path")); +const ACCESS_TOKEN_FACTORY_PATH = "../../../out/AccessTokenFactory.sol/AccessTokenFactory.json"; +const ACCESS_TOKEN_PATH = "../../../out/AccessToken.sol/AccessToken.json"; +(0, yargs_1.default)((0, helpers_1.hideBin)(process.argv)) + .command("createAccessToken", "Create an AccessToken for a given product address.", { + rpc: { type: "string", demandOption: true }, + privatekey: { type: "string", demandOption: true }, + accessTokenFactory: { type: "string", demandOption: true }, + product: { type: "string", demandOption: true }, +}, (args) => __awaiter(void 0, void 0, void 0, function* () { + const wallet = new ethers_1.ethers.Wallet(args.privatekey, new ethers_1.ethers.providers.JsonRpcProvider(args.rpc)); + const accessTokenFactoryArtifactPath = path_1.default.resolve(__dirname, ACCESS_TOKEN_FACTORY_PATH); + const accessTokenFactoryArtifact = JSON.parse(fs_1.default.readFileSync(accessTokenFactoryArtifactPath, "utf-8")); + const accessTokenFactory = new ethers_1.ethers.Contract(args.accessTokenFactory, accessTokenFactoryArtifact.abi, wallet); + let accessTokenAddress = yield accessTokenFactory.getAccessToken(args.product); + if (accessTokenAddress === ethers_1.ethers.constants.AddressZero) { + console.log("Creating AccessToken..."); + const tx = yield accessTokenFactory.createAccessToken(args.product); + yield tx.wait(); + accessTokenAddress = yield accessTokenFactory.getAccessToken(args.product); + } + console.log(`AccessToken created at ${accessTokenAddress}`); +})) + .command("mint", "Mint an AccessToken for a given product and tokenId.", { + rpc: { type: "string", demandOption: true }, + privatekey: { type: "string", demandOption: true }, + accessTokenFactory: { type: "string", demandOption: true }, + product: { type: "string", demandOption: true }, + user: { type: "string", demandOption: true }, + tokenId: { type: "string", demandOption: true }, +}, (args) => __awaiter(void 0, void 0, void 0, function* () { + const wallet = new ethers_1.ethers.Wallet(args.privatekey, new ethers_1.ethers.providers.JsonRpcProvider(args.rpc)); + const accessTokenFactoryArtifactPath = path_1.default.resolve(__dirname, ACCESS_TOKEN_FACTORY_PATH); + const accessTokenFactoryArtifact = JSON.parse(fs_1.default.readFileSync(accessTokenFactoryArtifactPath, "utf-8")); + const accessTokenFactory = new ethers_1.ethers.Contract(args.accessTokenFactory, accessTokenFactoryArtifact.abi, wallet); + const accessTokenAddress = yield accessTokenFactory.getAccessToken(args.product); + const accessTokenArtifactPath = path_1.default.resolve(__dirname, ACCESS_TOKEN_PATH); + const accessTokenArtifact = JSON.parse(fs_1.default.readFileSync(accessTokenArtifactPath, "utf-8")); + const accessToken = new ethers_1.ethers.Contract(accessTokenAddress, accessTokenArtifact.abi, wallet); + console.log("Minting AccessToken..."); + const tx = yield accessToken.mint(args.user, args.tokenId); + yield tx.wait(); + console.log(`AccessToken minted to ${args.user} for tokenId ${args.tokenId}`); +})) + .command("burn", "Burn an AccessToken for a given product and tokenId.", { + rpc: { type: "string", demandOption: true }, + privatekey: { type: "string", demandOption: true }, + accessTokenFactory: { type: "string", demandOption: true }, + product: { type: "string", demandOption: true }, + tokenId: { type: "string", demandOption: true }, +}, (args) => __awaiter(void 0, void 0, void 0, function* () { + const wallet = new ethers_1.ethers.Wallet(args.privatekey, new ethers_1.ethers.providers.JsonRpcProvider(args.rpc)); + const accessTokenFactoryArtifactPath = path_1.default.resolve(__dirname, ACCESS_TOKEN_FACTORY_PATH); + const accessTokenFactoryArtifact = JSON.parse(fs_1.default.readFileSync(accessTokenFactoryArtifactPath, "utf-8")); + const accessTokenFactory = new ethers_1.ethers.Contract(args.accessTokenFactory, accessTokenFactoryArtifact.abi, wallet); + const accessTokenAddress = yield accessTokenFactory.getAccessToken(args.product); + const accessTokenArtifactPath = path_1.default.resolve(__dirname, ACCESS_TOKEN_PATH); + const accessTokenArtifact = JSON.parse(fs_1.default.readFileSync(accessTokenArtifactPath, "utf-8")); + const accessToken = new ethers_1.ethers.Contract(accessTokenAddress, accessTokenArtifact.abi, wallet); + console.log("Burning AccessToken..."); + const tx = yield accessToken.burn(args.tokenId); + yield tx.wait(); + console.log(`AccessToken with tokenId ${args.tokenId} burned`); +})) + .command("accessControl", "Check if a user owns an AccessToken for a given product and tokenId.", { + rpc: { type: "string", demandOption: true }, + accessTokenFactory: { type: "string", demandOption: true }, + product: { type: "string", demandOption: true }, + tokenId: { type: "string", demandOption: true }, + user: { type: "string", demandOption: true }, +}, (args) => __awaiter(void 0, void 0, void 0, function* () { + const provider = new ethers_1.ethers.providers.JsonRpcProvider(args.rpc); + const accessTokenFactoryArtifactPath = path_1.default.resolve(__dirname, ACCESS_TOKEN_FACTORY_PATH); + const accessTokenFactoryArtifact = JSON.parse(fs_1.default.readFileSync(accessTokenFactoryArtifactPath, "utf-8")); + const accessTokenFactory = new ethers_1.ethers.Contract(args.accessTokenFactory, accessTokenFactoryArtifact.abi, provider); + const accessTokenAddress = yield accessTokenFactory.getAccessToken(args.product); + const accessTokenArtifactPath = path_1.default.resolve(__dirname, ACCESS_TOKEN_PATH); + const accessTokenArtifact = JSON.parse(fs_1.default.readFileSync(accessTokenArtifactPath, "utf-8")); + const accessToken = new ethers_1.ethers.Contract(accessTokenAddress, accessTokenArtifact.abi, provider); + console.log("Checking AccessToken ownership..."); + const isOwned = yield accessToken.isUserOwned(args.user, args.tokenId); + console.log(`User ${args.user} owns tokenId ${args.tokenId}: ${isOwned}`); +})) + .help().argv; diff --git a/tools/access-control/dist/index.js b/tools/access-control/dist/index.js new file mode 100644 index 0000000..9b0a458 --- /dev/null +++ b/tools/access-control/dist/index.js @@ -0,0 +1,17 @@ +"use strict"; +var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { + if (k2 === undefined) k2 = k; + var desc = Object.getOwnPropertyDescriptor(m, k); + if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { + desc = { enumerable: true, get: function() { return m[k]; } }; + } + Object.defineProperty(o, k2, desc); +}) : (function(o, m, k, k2) { + if (k2 === undefined) k2 = k; + o[k2] = m[k]; +})); +var __exportStar = (this && this.__exportStar) || function(m, exports) { + for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p); +}; +Object.defineProperty(exports, "__esModule", { value: true }); +__exportStar(require("./cli"), exports); diff --git a/tools/access-control/package.json b/tools/access-control/package.json new file mode 100644 index 0000000..e0abf60 --- /dev/null +++ b/tools/access-control/package.json @@ -0,0 +1,22 @@ +{ + "name": "tool-access-control", + "main": "dist/index.js", + "scripts": { + "build": "rimraf dist/* && tsc -p tsconfig.json", + "cli": "node ./dist/index.js" + }, + "devDependencies": { + "@types/yargs": "^17.0.32", + "rimraf": "^6.0.0" + }, + "bin": { + "acl": "./dist/index.js" + }, + "peerDependencies": { + "typescript": "^5.0.0" + }, + "dependencies": { + "ethers": "^5.7.2", + "yargs": "^17.7.2" + } +} diff --git a/tools/access-control/src/cli.ts b/tools/access-control/src/cli.ts new file mode 100644 index 0000000..2864610 --- /dev/null +++ b/tools/access-control/src/cli.ts @@ -0,0 +1,194 @@ +import { ethers } from "ethers"; +import yargs from "yargs"; +import { hideBin } from "yargs/helpers"; +import fs from "fs"; +import path from "path"; + +const ACCESS_TOKEN_FACTORY_PATH = "../../../out/AccessTokenFactory.sol/AccessTokenFactory.json"; +const ACCESS_TOKEN_PATH = "../../../out/AccessToken.sol/AccessToken.json"; + +yargs(hideBin(process.argv)) + .command( + "createAccessToken", + "Create an AccessToken for a given product address.", + { + rpc: { type: "string", demandOption: true }, + privatekey: { type: "string", demandOption: true }, + accessTokenFactory: { type: "string", demandOption: true }, + product: { type: "string", demandOption: true }, + }, + async (args) => { + const wallet = new ethers.Wallet(args.privatekey, new ethers.providers.JsonRpcProvider(args.rpc)); + + const accessTokenFactoryArtifactPath = path.resolve( + __dirname, + ACCESS_TOKEN_FACTORY_PATH + ); + const accessTokenFactoryArtifact = JSON.parse( + fs.readFileSync(accessTokenFactoryArtifactPath, "utf-8") + ); + + const accessTokenFactory = new ethers.Contract( + args.accessTokenFactory, + accessTokenFactoryArtifact.abi, + wallet + ); + + let accessTokenAddress = await accessTokenFactory.getAccessToken(args.product); + if(accessTokenAddress === ethers.constants.AddressZero) { + console.log("Creating AccessToken..."); + const tx = await accessTokenFactory.createAccessToken(args.product); + await tx.wait(); + accessTokenAddress = await accessTokenFactory.getAccessToken(args.product); + } + console.log(`AccessToken created at ${accessTokenAddress}`); + } + ) + .command( + "mint", + "Mint an AccessToken for a given product and tokenId.", + { + rpc: { type: "string", demandOption: true }, + privatekey: { type: "string", demandOption: true }, + accessTokenFactory: { type: "string", demandOption: true }, + product: { type: "string", demandOption: true }, + user: { type: "string", demandOption: true }, + tokenId: { type: "string", demandOption: true }, + }, + async (args) => { + const wallet = new ethers.Wallet(args.privatekey, new ethers.providers.JsonRpcProvider(args.rpc)); + + const accessTokenFactoryArtifactPath = path.resolve( + __dirname, + ACCESS_TOKEN_FACTORY_PATH + ); + const accessTokenFactoryArtifact = JSON.parse( + fs.readFileSync(accessTokenFactoryArtifactPath, "utf-8") + ); + + const accessTokenFactory = new ethers.Contract( + args.accessTokenFactory, + accessTokenFactoryArtifact.abi, + wallet + ); + + const accessTokenAddress = await accessTokenFactory.getAccessToken(args.product); + + const accessTokenArtifactPath = path.resolve( + __dirname, + ACCESS_TOKEN_PATH + ); + const accessTokenArtifact = JSON.parse( + fs.readFileSync(accessTokenArtifactPath, "utf-8") + ); + + const accessToken = new ethers.Contract( + accessTokenAddress, + accessTokenArtifact.abi, + wallet + ); + + console.log("Minting AccessToken..."); + const tx = await accessToken.mint(args.user, args.tokenId); + await tx.wait(); + console.log(`AccessToken minted to ${args.user} for tokenId ${args.tokenId}`); + } + ) + .command( + "burn", + "Burn an AccessToken for a given product and tokenId.", + { + rpc: { type: "string", demandOption: true }, + privatekey: { type: "string", demandOption: true }, + accessTokenFactory: { type: "string", demandOption: true }, + product: { type: "string", demandOption: true }, + tokenId: { type: "string", demandOption: true }, + }, + async (args) => { + const wallet = new ethers.Wallet(args.privatekey, new ethers.providers.JsonRpcProvider(args.rpc)); + + const accessTokenFactoryArtifactPath = path.resolve( + __dirname, + ACCESS_TOKEN_FACTORY_PATH + ); + const accessTokenFactoryArtifact = JSON.parse( + fs.readFileSync(accessTokenFactoryArtifactPath, "utf-8") + ); + + const accessTokenFactory = new ethers.Contract( + args.accessTokenFactory, + accessTokenFactoryArtifact.abi, + wallet + ); + + const accessTokenAddress = await accessTokenFactory.getAccessToken(args.product); + + const accessTokenArtifactPath = path.resolve( + __dirname, + ACCESS_TOKEN_PATH + ); + const accessTokenArtifact = JSON.parse( + fs.readFileSync(accessTokenArtifactPath, "utf-8") + ); + + const accessToken = new ethers.Contract( + accessTokenAddress, + accessTokenArtifact.abi, + wallet + ); + + console.log("Burning AccessToken..."); + const tx = await accessToken.burn(args.tokenId); + await tx.wait(); + console.log(`AccessToken with tokenId ${args.tokenId} burned`); + } + ) + .command( + "accessControl", + "Check if a user owns an AccessToken for a given product and tokenId.", + { + rpc: { type: "string", demandOption: true }, + accessTokenFactory: { type: "string", demandOption: true }, + product: { type: "string", demandOption: true }, + tokenId: { type: "string", demandOption: true }, + user: { type: "string", demandOption: true }, + }, + async (args) => { + const provider = new ethers.providers.JsonRpcProvider(args.rpc); + + const accessTokenFactoryArtifactPath = path.resolve( + __dirname, + ACCESS_TOKEN_FACTORY_PATH + ); + const accessTokenFactoryArtifact = JSON.parse( + fs.readFileSync(accessTokenFactoryArtifactPath, "utf-8") + ); + + const accessTokenFactory = new ethers.Contract( + args.accessTokenFactory, + accessTokenFactoryArtifact.abi, + provider + ); + + const accessTokenAddress = await accessTokenFactory.getAccessToken(args.product); + + const accessTokenArtifactPath = path.resolve( + __dirname, + ACCESS_TOKEN_PATH + ); + const accessTokenArtifact = JSON.parse( + fs.readFileSync(accessTokenArtifactPath, "utf-8") + ); + + const accessToken = new ethers.Contract( + accessTokenAddress, + accessTokenArtifact.abi, + provider + ); + + console.log("Checking AccessToken ownership..."); + const isOwned = await accessToken.isUserOwned(args.user, args.tokenId); + console.log(`User ${args.user} owns tokenId ${args.tokenId}: ${isOwned}`); + } + ) + .help().argv; diff --git a/tools/access-control/src/index.ts b/tools/access-control/src/index.ts new file mode 100644 index 0000000..84c1dcb --- /dev/null +++ b/tools/access-control/src/index.ts @@ -0,0 +1 @@ +export * from "./cli"; \ No newline at end of file diff --git a/tools/access-control/tsconfig.json b/tools/access-control/tsconfig.json new file mode 100644 index 0000000..33c463f --- /dev/null +++ b/tools/access-control/tsconfig.json @@ -0,0 +1,10 @@ +{ + "compilerOptions": { + "target": "ES2015", + "module": "CommonJS", + "strict": true, + "outDir": "./dist", + "esModuleInterop": true, + "moduleResolution": "node" + }, +} From 6b47a1eb68972aeb73da7426788a8f46d5d5cae8 Mon Sep 17 00:00:00 2001 From: CanvasL <746591811@qq.com> Date: Fri, 9 Aug 2024 13:46:30 +0800 Subject: [PATCH 4/4] feat: list, rent, access control script --- .github/workflows/test.yml | 45 ----------------------- addresses.json | 5 ++- script/AccessControl.s.sol | 51 ++++++++++++++++++++++++++ script/CreateAccessToken.s.sol | 44 +++++++++++++++++++++++ script/DeployMarketplace.s.sol | 33 +++++++++++++++++ script/List.s.sol | 56 +++++++++++++++++++++++++++++ script/Rent.s.sol | 65 ++++++++++++++++++++++++++++++++++ 7 files changed, 253 insertions(+), 46 deletions(-) delete mode 100644 .github/workflows/test.yml create mode 100644 script/AccessControl.s.sol create mode 100644 script/CreateAccessToken.s.sol create mode 100644 script/DeployMarketplace.s.sol create mode 100644 script/List.s.sol create mode 100644 script/Rent.s.sol diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml deleted file mode 100644 index 762a296..0000000 --- a/.github/workflows/test.yml +++ /dev/null @@ -1,45 +0,0 @@ -name: CI - -on: - push: - pull_request: - workflow_dispatch: - -env: - FOUNDRY_PROFILE: ci - -jobs: - check: - strategy: - fail-fast: true - - name: Foundry project - runs-on: ubuntu-latest - steps: - - uses: actions/checkout@v4 - with: - submodules: recursive - - - name: Install Foundry - uses: foundry-rs/foundry-toolchain@v1 - with: - version: nightly - - - name: Show Forge version - run: | - forge --version - - - name: Run Forge fmt - run: | - forge fmt --check - id: fmt - - - name: Run Forge build - run: | - forge build --sizes - id: build - - - name: Run Forge tests - run: | - forge test -vvv - id: test diff --git a/addresses.json b/addresses.json index 2020e28..754f8e1 100644 --- a/addresses.json +++ b/addresses.json @@ -1,3 +1,6 @@ { - "BNBTestnet": "0x35a8483444947B2166Aa85837F97FaEf122f5ebb" + "BNBTestnet": { + "AccessTokenFactory": "0x35a8483444947B2166Aa85837F97FaEf122f5ebb", + "Marketplace": "0x647d77324E241709BaF63D7f96F0C19ecA06E2e0" + } } \ No newline at end of file diff --git a/script/AccessControl.s.sol b/script/AccessControl.s.sol new file mode 100644 index 0000000..588a054 --- /dev/null +++ b/script/AccessControl.s.sol @@ -0,0 +1,51 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.24; + +import {AccessTokenFactory} from "../contracts/AccessTokenFactory.sol"; +import {AccessToken} from "../contracts/AccessToken.sol"; +import {Marketplace} from "../contracts/Marketplace.sol"; +import {IMarketplaceStructs} from "../contracts/interfaces/IMarketplaceStructs.sol"; +import "forge-std/src/Script.sol"; + +contract Rent is Script { + uint256 deployerPrivateKey; + Marketplace marketplace; + IProductFactory productFactory; + AccessTokenFactory accessTokenFactory; + address device; + IProductFactory.DeviceBinding deviceBinding; + AccessToken accessToken; + address user; + + function setUp() public { + deployerPrivateKey = vm.envUint("PRIVATE_KEY"); + marketplace = Marketplace(0x647d77324E241709BaF63D7f96F0C19ecA06E2e0); + productFactory = IProductFactory(0x8E360F40c7875e096e0DbFe0cC01d1db5aFB78B6); + accessTokenFactory = AccessTokenFactory(0x35a8483444947B2166Aa85837F97FaEf122f5ebb); + device = 0xA7fE098F2D4D2cD6bA158E5470d9231AC223bA06; // set device here + user = 0xD0167B1cc6CAb1e4e7C6f38d09EA35171d00b68e; // set user device here + deviceBinding = productFactory.getDeviceBinding(device); + accessToken = AccessToken(accessTokenFactory.getAccessToken(deviceBinding.product)); + } + + function run() public view { + bool accessible = accessToken.isUserOwned(vm.addr(deployerPrivateKey), deviceBinding.tokenId); + console.log("accessible:", accessible); + } +} + +interface IProductFactory { + struct DeviceBinding { + address product; + uint256 tokenId; + } + + /** + * @notice Returns the product address and token ID for a device. + * @param device Address of the device. + * @return DevieInfo Product address and Token ID associated with the device. + */ + function getDeviceBinding( + address device + ) external view returns (DeviceBinding memory); +} diff --git a/script/CreateAccessToken.s.sol b/script/CreateAccessToken.s.sol new file mode 100644 index 0000000..11d52e8 --- /dev/null +++ b/script/CreateAccessToken.s.sol @@ -0,0 +1,44 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.24; + +import {AccessTokenFactory} from "../contracts/AccessTokenFactory.sol"; +import "forge-std/src/Script.sol"; + +contract CreateAccessToken is Script { + uint256 deployerPrivateKey; + AccessTokenFactory accessTokenFactory; + IProductFactory productFactory; + address device; + + function setUp() public { + deployerPrivateKey = vm.envUint("PRIVATE_KEY"); + accessTokenFactory = AccessTokenFactory(0x35a8483444947B2166Aa85837F97FaEf122f5ebb); + productFactory = IProductFactory(0x8E360F40c7875e096e0DbFe0cC01d1db5aFB78B6); + device = 0xA7fE098F2D4D2cD6bA158E5470d9231AC223bA06; // set your device here + } + + function run() public returns (address) { + IProductFactory.DeviceBinding memory deviceBinding = productFactory.getDeviceBinding(device); + vm.startBroadcast(deployerPrivateKey); + address accessToken = accessTokenFactory.createAccessToken(deviceBinding.product); + console.log("AccessToken:", accessToken); + vm.stopBroadcast(); + return address(accessTokenFactory); + } +} + +interface IProductFactory { + struct DeviceBinding { + address product; + uint256 tokenId; + } + + /** + * @notice Returns the product address and token ID for a device. + * @param device Address of the device. + * @return DevieInfo Product address and Token ID associated with the device. + */ + function getDeviceBinding( + address device + ) external view returns (DeviceBinding memory); +} diff --git a/script/DeployMarketplace.s.sol b/script/DeployMarketplace.s.sol new file mode 100644 index 0000000..87b0d92 --- /dev/null +++ b/script/DeployMarketplace.s.sol @@ -0,0 +1,33 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.24; + +import {Marketplace} from "../contracts/Marketplace.sol"; +import "forge-std/src/Script.sol"; + +contract DeployMarketplace is Script { + uint256 deployerPrivateKey; + address accessTokenFactory; + address[] rentCurrencies; + address treasury; + uint256 feePoints; + + function setUp() public { + deployerPrivateKey = vm.envUint("PRIVATE_KEY"); + accessTokenFactory = 0x35a8483444947B2166Aa85837F97FaEf122f5ebb; + treasury = 0x3F3786B67DC1874C3Bd8e8CD61F5eea87604470F; + feePoints = 100; // 100/10000 = 10% + } + + function run() public returns (address) { + vm.startBroadcast(deployerPrivateKey); + Marketplace martketplace = new Marketplace( + vm.addr(deployerPrivateKey), + accessTokenFactory, + rentCurrencies, + payable(treasury), + feePoints + ); + vm.stopBroadcast(); + return address(martketplace); + } +} diff --git a/script/List.s.sol b/script/List.s.sol new file mode 100644 index 0000000..67a516d --- /dev/null +++ b/script/List.s.sol @@ -0,0 +1,56 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.24; + +import {Marketplace} from "../contracts/Marketplace.sol"; +import {IMarketplaceStructs} from "../contracts/interfaces/IMarketplaceStructs.sol"; +import {IProduct} from "../contracts/interfaces/IProduct.sol"; +import "forge-std/src/Script.sol"; + +contract List is Script { + uint256 deployerPrivateKey; + Marketplace marketplace; + IProductFactory productFactory; + address device; + IProductFactory.DeviceBinding deviceBinding; + IMarketplaceStructs.ListArgs listArgs; + + function setUp() public { + deployerPrivateKey = vm.envUint("PRIVATE_KEY"); + marketplace = Marketplace(0x647d77324E241709BaF63D7f96F0C19ecA06E2e0); + productFactory = IProductFactory(0x8E360F40c7875e096e0DbFe0cC01d1db5aFB78B6); + device = 0xA7fE098F2D4D2cD6bA158E5470d9231AC223bA06; // set your device here + deviceBinding = productFactory.getDeviceBinding(device); + listArgs = IMarketplaceStructs.ListArgs({ + product: deviceBinding.product, + tokenId: deviceBinding.tokenId, + minRentalDays: 5, // set min rental days + maxRentalDays: 10, // set max rental days + rentCurrency: address(0), // only whitelisted currency, zero-address means bnb(native token) + dailyRent: 1e14, // set daily rent, here is 0.0001 BNB per day + rentRecipient: vm.addr(deployerPrivateKey) // set rent receiver + }); + } + + function run() public { + vm.startBroadcast(deployerPrivateKey); + IProduct(deviceBinding.product).approve(address(marketplace), deviceBinding.tokenId); + marketplace.list(listArgs); + vm.stopBroadcast(); + } +} + +interface IProductFactory { + struct DeviceBinding { + address product; + uint256 tokenId; + } + + /** + * @notice Returns the product address and token ID for a device. + * @param device Address of the device. + * @return DevieInfo Product address and Token ID associated with the device. + */ + function getDeviceBinding( + address device + ) external view returns (DeviceBinding memory); +} diff --git a/script/Rent.s.sol b/script/Rent.s.sol new file mode 100644 index 0000000..9fa6753 --- /dev/null +++ b/script/Rent.s.sol @@ -0,0 +1,65 @@ +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.24; + +import {AccessTokenFactory} from "../contracts/AccessTokenFactory.sol"; +import {AccessToken} from "../contracts/AccessToken.sol"; +import {Marketplace} from "../contracts/Marketplace.sol"; +import {IMarketplaceStructs} from "../contracts/interfaces/IMarketplaceStructs.sol"; +import "forge-std/src/Script.sol"; + +contract Rent is Script { + uint256 deployerPrivateKey; + Marketplace marketplace; + IProductFactory productFactory; + AccessTokenFactory accessTokenFactory; + address device; + IProductFactory.DeviceBinding deviceBinding; + AccessToken accessToken; + IMarketplaceStructs.ListingInfo listingInfo; + IMarketplaceStructs.RentArgs rentArgs; + + function setUp() public { + deployerPrivateKey = vm.envUint("PRIVATE_KEY"); + marketplace = Marketplace(0x647d77324E241709BaF63D7f96F0C19ecA06E2e0); + productFactory = IProductFactory(0x8E360F40c7875e096e0DbFe0cC01d1db5aFB78B6); + accessTokenFactory = AccessTokenFactory(0x35a8483444947B2166Aa85837F97FaEf122f5ebb); + device = 0xA7fE098F2D4D2cD6bA158E5470d9231AC223bA06; // set your device here + deviceBinding = productFactory.getDeviceBinding(device); + accessToken = AccessToken(accessTokenFactory.getAccessToken(deviceBinding.product)); + listingInfo = marketplace.getListingInfo(address(accessToken), deviceBinding.tokenId); + rentArgs = IMarketplaceStructs.RentArgs({ + accessToken: address(accessToken), + tokenId: deviceBinding.tokenId, + tenant: vm.addr(deployerPrivateKey), // set tenant address, default is caller + rentalDays: 5, // set rental days, min value is min rental days set by device owner + prepaidRent: 5 * listingInfo.dailyRent // set prepaid rent, min value is rentalDays * dailyRent set by device owner + }); + } + + function run() public { + vm.startBroadcast(deployerPrivateKey); + if(listingInfo.rentCurrency == marketplace.NATIVE_TOKEN()) { + marketplace.rent{value: rentArgs.prepaidRent}(rentArgs); + } else { + marketplace.rent(rentArgs); + } + require(accessToken.isUserOwned(vm.addr(deployerPrivateKey), deviceBinding.tokenId)); + vm.stopBroadcast(); + } +} + +interface IProductFactory { + struct DeviceBinding { + address product; + uint256 tokenId; + } + + /** + * @notice Returns the product address and token ID for a device. + * @param device Address of the device. + * @return DevieInfo Product address and Token ID associated with the device. + */ + function getDeviceBinding( + address device + ) external view returns (DeviceBinding memory); +}