Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Release v943060 #980

Closed
wants to merge 113 commits into from
Closed
Changes from 1 commit
Commits
Show all changes
113 commits
Select commit Hold shift + click to select a range
9f5355d
ci: forbid eslint warnings
CertainLach Feb 17, 2023
4bcabac
ci: run clippy
CertainLach Feb 17, 2023
e1f20e1
fix: set prop for not existed token (#933)
bugrazoid Jun 14, 2023
71175e3
feat: update MarketV2 contract
Jun 16, 2023
c575769
migration(app-promo): added number(s) of unhandled blocks
PraetorP Jun 15, 2023
6f79f09
ci collator-selection for dockerhub
BuddyGlas Jun 23, 2023
65414d6
feat(app-promo): types for `Currency` trait support has been removed …
PraetorP Jun 23, 2023
e041df2
Test for market sponsoring
uandysmith Jun 26, 2023
8196877
fix: add `from` to `confirmSponsorship`
Jun 26, 2023
f661a22
test: tests for sending `value` with `evm.call`
Jun 27, 2023
a7c577e
Update market + substrate sponsoring example
uandysmith Jun 27, 2023
54a7967
Remove console.log
uandysmith Jun 27, 2023
9c3f000
fix: return TokenData with owner for old runtimes (#957)
fairax Jun 28, 2023
ef738f7
Merge pull request #958 from UniqueNetwork/ci/collator-selection
CertainLach Jun 28, 2023
1913bf6
Merge pull request #954 from UniqueNetwork/feature/update_market_v2_c…
CertainLach Jun 29, 2023
07a896e
fix: update unique-frontier dependency
Jun 30, 2023
1ecd8fc
Merge pull request #961 from UniqueNetwork/test/send_value_to_contrac…
CertainLach Jun 30, 2023
d8a735e
refactor!: decouple pallet-collator-selection from pallet-configurati…
PraetorP Jun 30, 2023
83d67e5
fix: Rust Analyzer for common mod (#966)
Jul 10, 2023
83605d4
fix: sponsoring tests
Jul 11, 2023
28d97ca
test: add bench for nesting (#963)
PraetorP Jul 11, 2023
724706b
ci: wait for try-runtime logs (#960)
BuddyGlas Jul 11, 2023
8d1c3c9
build: update polkadot to v0.9.43
CertainLach Jul 16, 2023
4de16ed
build: bump spec_version
CertainLach Jul 17, 2023
4d3b620
ci: fix launch-config
CertainLach Jul 17, 2023
55ce0a6
refactor: drop TransactionConverter support
CertainLach Jun 16, 2023
f28d992
feat: update BaseFee storage from pallet-configuration
CertainLach Jun 16, 2023
ec57cb9
refactor: ethereum RPC/tasks initialization
CertainLach Jun 16, 2023
a73db73
style: fix formatting
CertainLach Jul 17, 2023
1b4b057
ci: remove deprecated arguments
CertainLach Jul 20, 2023
ebfc6b1
ci: remove ws-port leftovers
CertainLach Jul 21, 2023
8d3362a
ci: regenerate .env
CertainLach Jul 21, 2023
471fdb4
build: update dependencies
CertainLach Jul 21, 2023
fc1ccc4
Merge pull request #968 from UniqueNetwork/feature/update-polkadot-v0…
CertainLach Aug 14, 2023
e7df43a
ci: node only update changes (#956)
BuddyGlas Aug 14, 2023
950d276
Merge pull request #888 from UniqueNetwork/ci/forbid-warnings
CertainLach Aug 15, 2023
20ac01d
feat: eth all-in-one create_collection (#971)
fairax Aug 16, 2023
1a4aa2d
feat: introduce democracy (#965)
PraetorP Aug 23, 2023
905f9b3
refactor: fix complex value type support in evm-coder
CertainLach Aug 29, 2023
0ce92f0
fix: tests cleanup
uandysmith Aug 29, 2023
e12df81
test(evm): using value with Option::None is not allowed
CertainLach Aug 30, 2023
dff02f7
fix: restore unrecognized selector error message
CertainLach Aug 30, 2023
e24c985
style: fix formatting
CertainLach Aug 30, 2023
66f7f7d
fix: createCollection flags test
uandysmith Aug 30, 2023
c466f2a
test: fix build and style
CertainLach Aug 30, 2023
369a51d
Merge pull request #974 from UniqueNetwork/fix/evm-coder-leftovers
CertainLach Aug 30, 2023
fc2ce00
ci: specify endpoints for xcm containers
CertainLach Aug 23, 2023
7fac866
ci: update .env
CertainLach Aug 23, 2023
6586b5a
ci: remove old xcm workflow
CertainLach Aug 25, 2023
37e846d
test: upgrade to yarn berry
CertainLach Aug 25, 2023
b8ffe22
ci: update env generator
CertainLach Aug 25, 2023
86762e7
ci: use baedeker for xcm workflow
CertainLach Aug 25, 2023
c1adeef
test: regenerate types
CertainLach Aug 27, 2023
12f3485
ci: use compose discovery
CertainLach Aug 30, 2023
d83685b
test: fix unique imports and eslint
uandysmith Aug 31, 2023
ca74825
ci: statemints wants para-ed keys
CertainLach Aug 31, 2023
0722057
test: make testing more resilent to random failures
CertainLach Aug 31, 2023
9acbee5
ci: do not regenerate types on every run
CertainLach Sep 1, 2023
21a0683
test: rewrite metadata fetch to js
CertainLach Sep 1, 2023
24a0f65
ci: add polkadex to xcm workflow
BuddyGlas Aug 31, 2023
3cd8b28
ci(xcm): run for all networks
CertainLach Sep 1, 2023
693c003
test: restore gov tests
CertainLach Sep 1, 2023
a17eff7
Merge pull request #972 from UniqueNetwork/ci/baedeker
CertainLach Sep 1, 2023
42448e1
fix: governance setup inconsistencies
mrshiposha Sep 1, 2023
3e9a8d1
Merge pull request #976 from UniqueNetwork/fix/governance-setup-incon…
CertainLach Sep 2, 2023
edf428e
test(gov): added neg votes
PraetorP Sep 4, 2023
c6d53cc
Merge pull request #977 from UniqueNetwork/fix/gov-council-test
CertainLach Sep 4, 2023
a797a1d
feat: add opal gov timings
mrshiposha Sep 4, 2023
0e75cd9
Merge pull request #978 from UniqueNetwork/fix/opal-governance-timings
CertainLach Sep 4, 2023
5b9d45c
enable governance workflow for develop and master branch
BuddyGlas Sep 5, 2023
98753ec
Merge pull request #981 from UniqueNetwork/ci/governance-workflow
CertainLach Sep 5, 2023
56c4087
ci: relay_chain argument for baedeker
CertainLach Sep 3, 2023
1a650ec
ci: move nodeonly to baedeker
CertainLach Sep 3, 2023
ec8104e
Merge pull request #983 from UniqueNetwork/ci/baedeker-node-only
CertainLach Sep 5, 2023
3050b51
test: add event asserts (#982)
Maksandre Sep 6, 2023
bf4df5f
fix: xcm tests
mrshiposha Sep 6, 2023
a4cc0ef
feat: baedeker helper scripts
CertainLach Sep 6, 2023
3567b2a
ci: set repoDir tla-str by default
CertainLach Sep 6, 2023
432eea3
ci: move collator selection to baedeker (#979)
BuddyGlas Sep 6, 2023
33d4989
Merge pull request #985 from UniqueNetwork/fix/xcm-release-60
CertainLach Sep 6, 2023
d3757b6
Merge pull request #984 from UniqueNetwork/dev/baedeker
CertainLach Sep 6, 2023
320ac9b
ci: update nodejs version to 18 (#987)
BuddyGlas Sep 7, 2023
911b03c
fix: wasm runtime version section
CertainLach Sep 8, 2023
8483c29
test: runtime upgrade helpers
CertainLach Sep 8, 2023
129199c
ci: ability to disable disable default actions
CertainLach Sep 8, 2023
5a3f942
ci: move forkless-nodata to baedeker
CertainLach Sep 8, 2023
f266535
ci: move forkless-data to baedeker
CertainLach Sep 8, 2023
dfae293
ci: fix bdk dev paths
CertainLach Sep 8, 2023
d892f52
ci: fix codestyle failures
CertainLach Sep 8, 2023
f4daa42
fix: collator-selection tests
mrshiposha Sep 8, 2023
13175ce
ci: forkless misc fixes
CertainLach Sep 8, 2023
a1bfaca
style: fix clippy warning
CertainLach Sep 8, 2023
cb2cc11
fix: workflow cargo build
mrshiposha Sep 8, 2023
1d7b145
ci: missing wasm_name for opal
CertainLach Sep 8, 2023
0e9500c
fix: simplify test timings features
mrshiposha Sep 8, 2023
0caf783
fix: use plain node names instead of test account names
mrshiposha Sep 8, 2023
d6fc80a
fix: workflow baedeker args
mrshiposha Sep 8, 2023
70415c3
fix: session-test-timingS
mrshiposha Sep 8, 2023
d27c764
fix: stash accounts discovery move to baedeker-library
mrshiposha Sep 8, 2023
62d7507
fix: maintenance tests - disable MM after the suite
mrshiposha Sep 8, 2023
e6eb884
Merge pull request #989 from UniqueNetwork/fix/maintenance-tests-rele…
CertainLach Sep 8, 2023
095843d
fix: increase session test timings - 5 mins
mrshiposha Sep 8, 2023
f3f56cc
Merge pull request #988 from UniqueNetwork/fix/collector-selection-te…
CertainLach Sep 8, 2023
4badfb7
ci: forkless-nodata wants new nodes in runtime
CertainLach Sep 10, 2023
950f6ba
ci: respect RELAY_CHAIN_TYPE
CertainLach Sep 11, 2023
31ef540
style: fix clippy warning
CertainLach Sep 11, 2023
af0cd94
build: baedeker reset helper script
CertainLach Sep 11, 2023
1454e80
style: fix rest of clippy warnings
CertainLach Sep 11, 2023
716e4d5
ci: fix relay spec name
CertainLach Sep 11, 2023
afa2cb8
ci: fix rewrite ordering
CertainLach Sep 11, 2023
8bfa768
feat: add .envrc-dev
mrshiposha Sep 11, 2023
348c8c0
Merge pull request #991 from UniqueNetwork/feature/envrc-dev
CertainLach Sep 11, 2023
2fcc0ad
test: forbid creating ApiPromise without set endpoint
CertainLach Sep 11, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Test for market sponsoring
uandysmith committed Jun 26, 2023

Verified

This commit was created on GitHub.com and signed with GitHub’s verified signature. The key has expired.
commit e041df2412cb340014bd899cd33382746f6d84fd
1 change: 1 addition & 0 deletions tests/package.json
Original file line number Diff line number Diff line change
@@ -47,6 +47,7 @@
"testEthNesting": "yarn _test './**/eth/nesting/**/*.*test.ts'",
"testEthFractionalizer": "yarn _test './**/eth/fractionalizer/**/*.*test.ts'",
"testEthMarketplace": "yarn _test './**/eth/marketplace/**/*.*test.ts'",
"testEthMarket": "yarn _test './**/eth/marketplace-v2/**/*.*test.ts'",
"testSub": "yarn _test './**/sub/**/*.*test.ts'",
"testSubNesting": "yarn _test './**/sub/nesting/**/*.*test.ts'",
"testEvent": "yarn _test ./src/check-event/*.*test.ts",
130 changes: 72 additions & 58 deletions tests/src/eth/marketplace-v2/Market.sol
Original file line number Diff line number Diff line change
@@ -1,14 +1,16 @@
// SPDX-License-Identifier: UNLICENSED
pragma solidity 0.8.17;

import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import "@openzeppelin/contracts/utils/introspection/ERC165Checker.sol";
import "@openzeppelin/contracts/token/ERC721/IERC721.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
import { UniqueNFT, CrossAddress } from "@unique-nft/solidity-interfaces/contracts/UniqueNFT.sol";
import { UniqueFungible, CrossAddress as CrossAddressF } from "@unique-nft/solidity-interfaces/contracts/UniqueFungible.sol";
import "@unique-nft/solidity-interfaces/contracts/CollectionHelpers.sol";
import "./royalty/UniqueRoyaltyHelper.sol";

contract Market {
contract Market is Ownable, ReentrancyGuard {
using ERC165Checker for address;

struct Order {
@@ -21,7 +23,7 @@ contract Market {
}

uint32 public constant version = 0;
uint32 public constant buildVersion = 1;
uint32 public constant buildVersion = 3;
bytes4 private constant InterfaceId_ERC721 = 0x80ac58cd;
bytes4 private constant InterfaceId_ERC165 = 0x5755c3f2;
CollectionHelpers private constant collectionHelpers =
@@ -31,7 +33,6 @@ contract Market {
uint32 private idCount = 1;
uint32 public marketFee;
uint64 public ctime;
address selfAddress;
address public ownerAddress;
mapping(address => bool) public admins;

@@ -57,15 +58,11 @@ contract Market {
error OrderNotFound();
error TooManyAmountRequested();
error NotEnoughMoneyError();
error InvalidRoyaltiesError(uint256 totalRoyalty);
error FailTransferToken(string reason);

modifier onlyOwner() {
require(msg.sender == ownerAddress, "Only owner can");
_;
}

modifier onlyAdmin() {
require(msg.sender == ownerAddress || admins[msg.sender], "Only admin can");
require(msg.sender == this.owner() || admins[msg.sender], "Only admin can");
_;
}

@@ -85,12 +82,9 @@ contract Market {
marketFee = fee;
ctime = timestamp;

if (marketFee == 0 || marketFee >= 100) {
if (marketFee >= 100) {
revert InvalidMarketFee();
}

ownerAddress = msg.sender;
selfAddress = address(this);
}

function getErc721(uint32 collectionId) private view returns (IERC721) {
@@ -114,34 +108,33 @@ contract Market {
return IERC721(collectionAddress);
}

// ################################################################
// Set new contract owner #
// ################################################################

function setOwner() public onlyOwner {
ownerAddress = msg.sender;
}

// ################################################################
// Add new admin #
// ################################################################

/**
* Add new admin. Only owner or an existing admin can add admins.
*
* @param admin: Address of a new admin to add
*/
function addAdmin(address admin) public onlyAdmin {
admins[admin] = true;
}

// ################################################################
// Remove admin #
// ################################################################

/**
* Remove an admin. Only owner or an existing admin can remove admins.
*
* @param admin: Address of a new admin to add
*/
function removeAdmin(address admin) public onlyAdmin {
delete admins[admin];
}

// ################################################################
// Place a token for sale #
// ################################################################

/**
* Place an NFT or RFT token for sale. It must be pre-approved for transfers by this contract address.
*
* @param collectionId: ID of the token collection
* @param tokenId: ID of the token
* @param price: Price (with proper network currency decimals)
* @param amount: Number of token fractions to list (must always be 1 for NFT)
* @param seller: The seller cross-address (the beneficiary account to receive payment, may be different from transaction sender)
*/
function put(
uint32 collectionId,
uint32 tokenId,
@@ -166,7 +159,7 @@ contract Market {
revert SellerIsNotOwner();
}

if (erc721.getApproved(tokenId) != selfAddress) {
if (erc721.getApproved(tokenId) != address(this)) {
revert TokenIsNotApproved();
}

@@ -185,21 +178,27 @@ contract Market {
emit TokenIsUpForSale(version, order);
}

// ################################################################
// Get order #
// ################################################################

/**
* Get information about the listed token order
*
* @param collectionId: ID of the token collection
* @param tokenId: ID of the token
* @return The order information
*/
function getOrder(
uint32 collectionId,
uint32 tokenId
) external view returns (Order memory) {
return orders[collectionId][tokenId];
}

// ################################################################
// Revoke the token from the sale #
// ################################################################

/**
* Revoke the token from the sale. Only the original lister can use this method.
*
* @param collectionId: ID of the token collection
* @param tokenId: ID of the token
* @param amount: Number of token fractions to de-list (must always be 1 for NFT)
*/
function revoke(
uint32 collectionId,
uint32 tokenId,
@@ -241,10 +240,12 @@ contract Market {
emit TokenRevoke(version, order, amount);
}

// ################################################################
// Check approved #
// ################################################################

/**
* Test if the token is still approved to be transferred by this contract and delete the order if not.
*
* @param collectionId: ID of the token collection
* @param tokenId: ID of the token
*/
function checkApproved(uint32 collectionId, uint32 tokenId) public onlyAdmin {
Order memory order = orders[collectionId][tokenId];
if (order.price == 0) {
@@ -253,7 +254,7 @@ contract Market {

IERC721 erc721 = getErc721(collectionId);

if (erc721.getApproved(tokenId) != selfAddress || erc721.ownerOf(tokenId) != getAddressFromCrossAccount(order.seller)) {
if (erc721.getApproved(tokenId) != address(this) || erc721.ownerOf(tokenId) != getAddressFromCrossAccount(order.seller)) {
uint32 amount = order.amount;
order.amount = 0;
emit TokenRevoke(version, order, amount);
@@ -272,6 +273,12 @@ contract Market {
}
}

/**
* Revoke the token from the sale. Only the contract admin can use this method.
*
* @param collectionId: ID of the token collection
* @param tokenId: ID of the token
*/
function revokeAdmin(uint32 collectionId, uint32 tokenId) public onlyAdmin {
Order memory order = orders[collectionId][tokenId];
if (order.price == 0) {
@@ -285,16 +292,20 @@ contract Market {
delete orders[collectionId][tokenId];
}

// ################################################################
// Buy a token #
// ################################################################

/**
* Buy a token (partially for an RFT).
*
* @param collectionId: ID of the token collection
* @param tokenId: ID of the token
* @param amount: Number of token fractions to buy (must always be 1 for NFT)
* @param buyer: Cross-address of the buyer, eth part must be equal to the transaction signer address
*/
function buy(
uint32 collectionId,
uint32 tokenId,
uint32 amount,
CrossAddress memory buyer
) public payable validCrossAddress(buyer.eth, buyer.sub) {
) public payable validCrossAddress(buyer.eth, buyer.sub) nonReentrant {
if (msg.value == 0) {
revert InvalidArgument("msg.value must not be zero");
}
@@ -319,7 +330,7 @@ contract Market {
}

IERC721 erc721 = getErc721(order.collectionId);
if (erc721.getApproved(tokenId) != selfAddress) {
if (erc721.getApproved(tokenId) != address(this)) {
revert TokenIsNotApproved();
}

@@ -339,13 +350,16 @@ contract Market {
order.tokenId
);

(uint256 totalRoyalty, RoyaltyAmount[] memory royalties) = sendRoyalties(collectionAddress, tokenId, totalValue);
(uint256 totalRoyalty, RoyaltyAmount[] memory royalties) = sendRoyalties(collectionAddress, tokenId, totalValue - feeValue);

if (totalRoyalty >= totalValue - feeValue) {
revert InvalidRoyaltiesError(totalRoyalty);
}

sendMoney(order.seller, totalValue - feeValue - totalRoyalty);

if (msg.value > totalValue) {
// todo, send money to signer or buyer ?
payable(msg.sender).transfer(msg.value - totalValue);
sendMoney(buyer, msg.value - totalValue);
}

emit TokenIsPurchased(version, order, amount, buyer, royalties);
@@ -356,7 +370,7 @@ contract Market {

UniqueFungible fungible = UniqueFungible(collectionAddress);

CrossAddressF memory fromF = CrossAddressF(selfAddress, 0);
CrossAddressF memory fromF = CrossAddressF(address(this), 0);
CrossAddressF memory toF = CrossAddressF(to.eth, to.sub);

fungible.transferFromCross(fromF, toF, money);
@@ -379,7 +393,7 @@ contract Market {
}

function withdraw(address transferTo) public onlyOwner {
uint256 balance = selfAddress.balance;
uint256 balance = address(this).balance;

if (balance > 0) {
payable(transferTo).transfer(balance);
62 changes: 55 additions & 7 deletions tests/src/eth/marketplace-v2/marketplace.test.ts
Original file line number Diff line number Diff line change
@@ -16,7 +16,7 @@

import {IKeyringPair} from '@polkadot/types/types';
import {readFile} from 'fs/promises';
import {EthUniqueHelper, itEth, usingEthPlaygrounds} from '../util';
import {EthUniqueHelper, SponsoringMode, itEth, usingEthPlaygrounds} from '../util';
import {makeNames} from '../../util';
import {expect} from 'chai';
import Web3 from 'web3';
@@ -50,6 +50,18 @@ describe('Market V2 Contract', () => {
solPath: '@openzeppelin/contracts/utils/introspection/IERC165.sol',
fsPath: `${dirname}/../../../node_modules/@openzeppelin/contracts/utils/introspection/IERC165.sol`,
},
{
solPath: '@openzeppelin/contracts/access/Ownable.sol',
fsPath: `${dirname}/../../../node_modules/@openzeppelin/contracts/access/Ownable.sol`,
},
{
solPath: '@openzeppelin/contracts/utils/Context.sol',
fsPath: `${dirname}/../../../node_modules/@openzeppelin/contracts/utils/Context.sol`,
},
{
solPath: '@openzeppelin/contracts/security/ReentrancyGuard.sol',
fsPath: `${dirname}/../../../node_modules/@openzeppelin/contracts/security/ReentrancyGuard.sol`,
},
{
solPath: '@openzeppelin/contracts/utils/introspection/ERC165Checker.sol',
fsPath: `${dirname}/../../../node_modules/@openzeppelin/contracts/utils/introspection/ERC165Checker.sol`,
@@ -94,26 +106,62 @@ describe('Market V2 Contract', () => {
});

itEth('Put + Buy [eth]', async ({helper}) => {
const marketOwner = await helper.eth.createAccountWithBalance(donor, 600n);
const ONE_TOKEN = helper.balance.getOneTokenNominal();
const PRICE = 2n * ONE_TOKEN; // 2 UNQ
const marketOwner = await helper.eth.createAccountWithBalance(donor, 60000n);
const market = await deployMarket(helper, marketOwner);
const contractHelpers = helper.ethNativeContract.contractHelpers(marketOwner);
await contractHelpers.methods.selfSponsoredEnable(market.options.address).send({from: marketOwner});
await helper.eth.transferBalanceFromSubstrate(donor, market.options.address, 10n);

// TODO: this should work too, instead of selfSponsoring!
// await contractHelpers.methods.setSponsor(market.options.address, marketOwner).send({from: marketOwner})
// await contractHelpers.methods.confirmSponsorship(market.options.address);

await contractHelpers.methods.setSponsoringMode(market.options.address, SponsoringMode.Generous).send({from: marketOwner});
await contractHelpers.methods.setSponsoringRateLimit(market.options.address, 0).send({from: marketOwner});

const {collectionId, collectionAddress} = await helper.eth.createNFTCollection(marketOwner, 'Sponsor', 'absolutely anything', 'ROC');
const collection = await helper.ethNativeContract.collection(collectionAddress, 'nft', marketOwner);
const collection = helper.ethNativeContract.collection(collectionAddress, 'nft', marketOwner, true);
await collection.methods.setCollectionSponsor(marketOwner).send({from: marketOwner});
await collection.methods.confirmCollectionSponsorship().send({from: marketOwner});

const sellerCross = await helper.ethCrossAccount.createAccountWithBalance(donor, 600n);
const sellerCross = helper.ethCrossAccount.createAccount();
const result = await collection.methods.mintCross(sellerCross, []).send();
const tokenId = result.events.Transfer.returnValues.tokenId;
await collection.methods.approve(market.options.address, tokenId).send({from: sellerCross.eth});

// Seller has no funds at all, his transactions are sponsored
const sellerBalance = await helper.balance.getEthereum(sellerCross.eth);
expect(sellerBalance).to.be.eq(0n);

const putResult = await market.methods.put(collectionId, tokenId, 1, 1, sellerCross).send({from: sellerCross.eth});
const putResult = await market.methods.put(collectionId, tokenId, PRICE.toString(), 1, sellerCross).send({
from: sellerCross.eth, gasLimit: 1_000_000
});
expect(putResult.events.TokenIsUpForSale).is.not.undefined;

// Seller balance are still 0
const sellerBalanceAfter = await helper.balance.getEthereum(sellerCross.eth);
expect(sellerBalanceAfter).to.be.eq(0n);

let ownerCross = await collection.methods.ownerOfCross(tokenId).call();
expect(ownerCross.eth).to.be.eq(sellerCross.eth);
expect(ownerCross.sub).to.be.eq(sellerCross.sub);

const buyerCross = await helper.ethCrossAccount.createAccountWithBalance(donor, 600n);
const buyResult = await market.methods.buy(collectionId, tokenId, 1, buyerCross).send({from: buyerCross.eth, value: 1});
const buyerCross = await helper.ethCrossAccount.createAccountWithBalance(donor, 10n);
console.log('before buy');

// Buyer has only 10 UNQ
const buyerBalance = await helper.balance.getEthereum(buyerCross.eth);
expect(buyerBalance).to.be.eq(10n * ONE_TOKEN)

const buyResult = await market.methods.buy(collectionId, tokenId, 1, buyerCross).send({from: buyerCross.eth, value: PRICE.toString(), gasLimit: 1_000_000});
expect(buyResult.events.TokenIsPurchased).is.not.undefined;

// Buyer pays only value, transaction use sponsoring
const buyerBalanceAfter = await helper.balance.getEthereum(buyerCross.eth);
expect(buyerBalanceAfter).to.be.eq(10n * ONE_TOKEN - PRICE);

ownerCross = await collection.methods.ownerOfCross(tokenId).call();
expect(ownerCross.eth).to.be.eq(buyerCross.eth);
expect(ownerCross.sub).to.be.eq(buyerCross.sub);