@0x:contracts-exchange-libs
Added unit tests to exchange-libs
This commit is contained in:
parent
e3aa76cd09
commit
c318b849fe
@ -173,4 +173,28 @@ contract TestLibs is
|
|||||||
_addFillResults(totalFillResults, singleFillResults);
|
_addFillResults(totalFillResults, singleFillResults);
|
||||||
return totalFillResults;
|
return totalFillResults;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function hashOrder(Order memory order)
|
||||||
|
public
|
||||||
|
pure
|
||||||
|
returns (bytes32)
|
||||||
|
{
|
||||||
|
return _hashOrder(order);
|
||||||
|
}
|
||||||
|
|
||||||
|
function hashZeroExTransaction(ZeroExTransaction memory transaction)
|
||||||
|
public
|
||||||
|
pure
|
||||||
|
returns (bytes32)
|
||||||
|
{
|
||||||
|
return _hashZeroExTransaction(transaction);
|
||||||
|
}
|
||||||
|
|
||||||
|
function hashEIP712ExchangeMessage(bytes32 hashStruct)
|
||||||
|
public
|
||||||
|
view
|
||||||
|
returns (bytes32)
|
||||||
|
{
|
||||||
|
return _hashEIP712ExchangeMessage(hashStruct);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
68
contracts/exchange-libs/test/lib_eip712_exchange_domain.ts
Normal file
68
contracts/exchange-libs/test/lib_eip712_exchange_domain.ts
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
import { blockchainTests, constants, describe, expect, hexRandom } from '@0x/contracts-test-utils';
|
||||||
|
import { eip712Utils, orderHashUtils } from '@0x/order-utils';
|
||||||
|
import { Order } from '@0x/types';
|
||||||
|
import { BigNumber, signTypedDataUtils } from '@0x/utils';
|
||||||
|
import * as ethUtil from 'ethereumjs-util';
|
||||||
|
import * as _ from 'lodash';
|
||||||
|
|
||||||
|
import { artifacts, TestLibsContract } from '../src';
|
||||||
|
|
||||||
|
blockchainTests('LibEIP712ExchangeDomain', env => {
|
||||||
|
let libsContract: TestLibsContract;
|
||||||
|
let exchangeDomainHash: string;
|
||||||
|
const CHAIN_ID = 1337;
|
||||||
|
|
||||||
|
// Random generator functions
|
||||||
|
const randomAddress = () => hexRandom(constants.ADDRESS_LENGTH);
|
||||||
|
const randomHash = () => hexRandom(constants.WORD_LENGTH);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests a specific instance of EIP712 message hashing.
|
||||||
|
* @param lib The LibEIP712 contract to call.
|
||||||
|
* @param domainHash The hash of the EIP712 domain of this instance.
|
||||||
|
* @param hashStruct The hash of the struct of this instance.
|
||||||
|
*/
|
||||||
|
async function testHashEIP712MessageAsync(hashStruct: string): Promise<void> {
|
||||||
|
// Remove the hex-prefix from the exchangeDomainHash and the hashStruct
|
||||||
|
const unprefixedHashStruct = hashStruct.slice(2, hashStruct.length);
|
||||||
|
|
||||||
|
// Hash the provided input to get the expected hash
|
||||||
|
const input = '0x1901'.concat(exchangeDomainHash, unprefixedHashStruct);
|
||||||
|
const expectedHash = '0x'.concat(ethUtil.sha3(input).toString('hex'));
|
||||||
|
|
||||||
|
// Get the actual hash by calling the smart contract
|
||||||
|
const actualHash = await libsContract.hashEIP712ExchangeMessage.callAsync(hashStruct);
|
||||||
|
|
||||||
|
// Verify that the actual hash matches the expected hash
|
||||||
|
expect(actualHash).to.be.eq(expectedHash);
|
||||||
|
}
|
||||||
|
|
||||||
|
before(async () => {
|
||||||
|
libsContract = await TestLibsContract.deployFrom0xArtifactAsync(
|
||||||
|
artifacts.TestLibs,
|
||||||
|
env.provider,
|
||||||
|
env.txDefaults,
|
||||||
|
new BigNumber(CHAIN_ID),
|
||||||
|
);
|
||||||
|
|
||||||
|
// Generate the domain hash of 0x Exchange V3
|
||||||
|
exchangeDomainHash = signTypedDataUtils
|
||||||
|
.generateDomainHash({
|
||||||
|
name: '0x Protocol',
|
||||||
|
version: '3.0.0',
|
||||||
|
chainId: CHAIN_ID,
|
||||||
|
verifyingContractAddress: libsContract.address,
|
||||||
|
})
|
||||||
|
.toString('hex');
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('hashEIP712ExchangeMessage', () => {
|
||||||
|
it('should correctly match an empty hash', async () => {
|
||||||
|
await testHashEIP712MessageAsync(constants.NULL_BYTES32);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should correctly match a non-empty hash', async () => {
|
||||||
|
await testHashEIP712MessageAsync(randomHash());
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
133
contracts/exchange-libs/test/lib_order.ts
Normal file
133
contracts/exchange-libs/test/lib_order.ts
Normal file
@ -0,0 +1,133 @@
|
|||||||
|
import { blockchainTests, constants, describe, expect, hexRandom } from '@0x/contracts-test-utils';
|
||||||
|
import { eip712Utils, orderHashUtils } from '@0x/order-utils';
|
||||||
|
import { Order } from '@0x/types';
|
||||||
|
import { BigNumber, signTypedDataUtils } from '@0x/utils';
|
||||||
|
import * as _ from 'lodash';
|
||||||
|
|
||||||
|
import { artifacts, TestLibsContract } from '../src';
|
||||||
|
|
||||||
|
blockchainTests('LibOrder', env => {
|
||||||
|
const CHAIN_ID = 1337;
|
||||||
|
let libsContract: TestLibsContract;
|
||||||
|
|
||||||
|
const randomAddress = () => hexRandom(constants.ADDRESS_LENGTH);
|
||||||
|
const randomHash = () => hexRandom(constants.WORD_LENGTH);
|
||||||
|
const randomUint256 = () => new BigNumber(randomHash());
|
||||||
|
const randomAssetData = () => hexRandom(36);
|
||||||
|
|
||||||
|
const EMPTY_ORDER: Order = {
|
||||||
|
domain: {
|
||||||
|
verifyingContractAddress: constants.NULL_ADDRESS,
|
||||||
|
chainId: 0,
|
||||||
|
},
|
||||||
|
senderAddress: constants.NULL_ADDRESS,
|
||||||
|
makerAddress: constants.NULL_ADDRESS,
|
||||||
|
takerAddress: constants.NULL_ADDRESS,
|
||||||
|
makerFee: constants.ZERO_AMOUNT,
|
||||||
|
takerFee: constants.ZERO_AMOUNT,
|
||||||
|
makerAssetAmount: constants.ZERO_AMOUNT,
|
||||||
|
takerAssetAmount: constants.ZERO_AMOUNT,
|
||||||
|
makerAssetData: constants.NULL_BYTES,
|
||||||
|
takerAssetData: constants.NULL_BYTES,
|
||||||
|
makerFeeAssetData: constants.NULL_BYTES,
|
||||||
|
takerFeeAssetData: constants.NULL_BYTES,
|
||||||
|
salt: constants.ZERO_AMOUNT,
|
||||||
|
feeRecipientAddress: constants.NULL_ADDRESS,
|
||||||
|
expirationTimeSeconds: constants.ZERO_AMOUNT,
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests the `_hashOrder()` function against a reference hash.
|
||||||
|
*/
|
||||||
|
async function testHashOrderAsync(order: Order): Promise<void> {
|
||||||
|
const typedData = eip712Utils.createOrderTypedData(order);
|
||||||
|
const expectedHash = '0x'.concat(
|
||||||
|
signTypedDataUtils.generateTypedDataHashWithoutDomain(typedData).toString('hex'),
|
||||||
|
);
|
||||||
|
const actualHash = await libsContract.hashOrder.callAsync(order);
|
||||||
|
expect(actualHash).to.be.eq(expectedHash);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests the `getOrderHash()` function against a reference hash.
|
||||||
|
*/
|
||||||
|
async function testGetOrderHashAsync(order: Order): Promise<void> {
|
||||||
|
const expectedHash = orderHashUtils.getOrderHashHex(order);
|
||||||
|
const actualHash = await libsContract.getOrderHash.callAsync(order);
|
||||||
|
expect(actualHash).to.be.eq(expectedHash);
|
||||||
|
}
|
||||||
|
|
||||||
|
before(async () => {
|
||||||
|
libsContract = await TestLibsContract.deployFrom0xArtifactAsync(
|
||||||
|
artifacts.TestLibs,
|
||||||
|
env.provider,
|
||||||
|
env.txDefaults,
|
||||||
|
new BigNumber(CHAIN_ID),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('getOrderHash', () => {
|
||||||
|
it('should correctly hash an empty order', async () => {
|
||||||
|
await testGetOrderHashAsync({
|
||||||
|
...EMPTY_ORDER,
|
||||||
|
domain: {
|
||||||
|
verifyingContractAddress: libsContract.address,
|
||||||
|
chainId: 1337,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should correctly hash a non-empty order', async () => {
|
||||||
|
await testGetOrderHashAsync({
|
||||||
|
domain: {
|
||||||
|
verifyingContractAddress: libsContract.address,
|
||||||
|
chainId: 1337,
|
||||||
|
},
|
||||||
|
senderAddress: randomAddress(),
|
||||||
|
makerAddress: randomAddress(),
|
||||||
|
takerAddress: randomAddress(),
|
||||||
|
makerFee: randomUint256(),
|
||||||
|
takerFee: randomUint256(),
|
||||||
|
makerAssetAmount: randomUint256(),
|
||||||
|
takerAssetAmount: randomUint256(),
|
||||||
|
makerAssetData: randomAssetData(),
|
||||||
|
takerAssetData: randomAssetData(),
|
||||||
|
makerFeeAssetData: randomAssetData(),
|
||||||
|
takerFeeAssetData: randomAssetData(),
|
||||||
|
salt: randomUint256(),
|
||||||
|
feeRecipientAddress: randomAddress(),
|
||||||
|
expirationTimeSeconds: randomUint256(),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('hashOrder', () => {
|
||||||
|
it('should correctly hash an empty order', async () => {
|
||||||
|
await testHashOrderAsync(EMPTY_ORDER);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should correctly hash a non-empty order', async () => {
|
||||||
|
await testHashOrderAsync({
|
||||||
|
// The domain is not used in this test, so it's okay if it is left empty.
|
||||||
|
domain: {
|
||||||
|
verifyingContractAddress: constants.NULL_ADDRESS,
|
||||||
|
chainId: 0,
|
||||||
|
},
|
||||||
|
senderAddress: randomAddress(),
|
||||||
|
makerAddress: randomAddress(),
|
||||||
|
takerAddress: randomAddress(),
|
||||||
|
makerFee: randomUint256(),
|
||||||
|
takerFee: randomUint256(),
|
||||||
|
makerAssetAmount: randomUint256(),
|
||||||
|
takerAssetAmount: randomUint256(),
|
||||||
|
makerAssetData: randomAssetData(),
|
||||||
|
takerAssetData: randomAssetData(),
|
||||||
|
makerFeeAssetData: randomAssetData(),
|
||||||
|
takerFeeAssetData: randomAssetData(),
|
||||||
|
salt: randomUint256(),
|
||||||
|
feeRecipientAddress: randomAddress(),
|
||||||
|
expirationTimeSeconds: randomUint256(),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
104
contracts/exchange-libs/test/lib_zero_ex_transaction.ts
Normal file
104
contracts/exchange-libs/test/lib_zero_ex_transaction.ts
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
import { blockchainTests, constants, describe, expect, hexRandom } from '@0x/contracts-test-utils';
|
||||||
|
import { eip712Utils } from '@0x/order-utils';
|
||||||
|
import { ZeroExTransaction } from '@0x/types';
|
||||||
|
import { BigNumber, signTypedDataUtils } from '@0x/utils';
|
||||||
|
import * as _ from 'lodash';
|
||||||
|
|
||||||
|
import { artifacts, TestLibsContract } from '../src';
|
||||||
|
|
||||||
|
blockchainTests('LibZeroExTransaction', env => {
|
||||||
|
const CHAIN_ID = 1337;
|
||||||
|
let libsContract: TestLibsContract;
|
||||||
|
|
||||||
|
const randomAddress = () => hexRandom(constants.ADDRESS_LENGTH);
|
||||||
|
const randomHash = () => hexRandom(constants.WORD_LENGTH);
|
||||||
|
const randomUint256 = () => new BigNumber(randomHash());
|
||||||
|
const randomAssetData = () => hexRandom(36);
|
||||||
|
|
||||||
|
const EMPTY_TRANSACTION: ZeroExTransaction = {
|
||||||
|
salt: constants.ZERO_AMOUNT,
|
||||||
|
expirationTimeSeconds: constants.ZERO_AMOUNT,
|
||||||
|
signerAddress: constants.NULL_ADDRESS,
|
||||||
|
data: constants.NULL_BYTES,
|
||||||
|
domain: {
|
||||||
|
verifyingContractAddress: constants.NULL_ADDRESS,
|
||||||
|
chainId: 0,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests the `_hashZeroExTransaction()` function against a reference hash.
|
||||||
|
*/
|
||||||
|
async function testHashZeroExTransactionAsync(transaction: ZeroExTransaction): Promise<void> {
|
||||||
|
const typedData = eip712Utils.createZeroExTransactionTypedData(transaction);
|
||||||
|
const expectedHash = '0x'.concat(
|
||||||
|
signTypedDataUtils.generateTypedDataHashWithoutDomain(typedData).toString('hex'),
|
||||||
|
);
|
||||||
|
const actualHash = await libsContract.hashZeroExTransaction.callAsync(transaction);
|
||||||
|
expect(actualHash).to.be.eq(expectedHash);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests the `getTransactionHash()` function against a reference hash.
|
||||||
|
*/
|
||||||
|
async function testGetTransactionHashAsync(transaction: ZeroExTransaction): Promise<void> {
|
||||||
|
const typedData = eip712Utils.createZeroExTransactionTypedData(transaction);
|
||||||
|
const expectedHash = '0x'.concat(signTypedDataUtils.generateTypedDataHash(typedData).toString('hex'));
|
||||||
|
const actualHash = await libsContract.getTransactionHash.callAsync(transaction);
|
||||||
|
expect(actualHash).to.be.eq(expectedHash);
|
||||||
|
}
|
||||||
|
|
||||||
|
before(async () => {
|
||||||
|
libsContract = await TestLibsContract.deployFrom0xArtifactAsync(
|
||||||
|
artifacts.TestLibs,
|
||||||
|
env.provider,
|
||||||
|
env.txDefaults,
|
||||||
|
new BigNumber(CHAIN_ID),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('getTransactionHash', () => {
|
||||||
|
it('should correctly hash an empty transaction', async () => {
|
||||||
|
await testGetTransactionHashAsync({
|
||||||
|
...EMPTY_TRANSACTION,
|
||||||
|
domain: {
|
||||||
|
verifyingContractAddress: libsContract.address,
|
||||||
|
chainId: 1337,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should correctly hash a non-empty order', async () => {
|
||||||
|
await testGetTransactionHashAsync({
|
||||||
|
salt: randomUint256(),
|
||||||
|
expirationTimeSeconds: randomUint256(),
|
||||||
|
signerAddress: randomAddress(),
|
||||||
|
data: randomAssetData(),
|
||||||
|
domain: {
|
||||||
|
verifyingContractAddress: libsContract.address,
|
||||||
|
chainId: 1337,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('hashOrder', () => {
|
||||||
|
it('should correctly hash an empty order', async () => {
|
||||||
|
await testHashZeroExTransactionAsync(EMPTY_TRANSACTION);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should correctly hash a non-empty order', async () => {
|
||||||
|
await testHashZeroExTransactionAsync({
|
||||||
|
salt: randomUint256(),
|
||||||
|
expirationTimeSeconds: randomUint256(),
|
||||||
|
signerAddress: randomAddress(),
|
||||||
|
data: randomAssetData(),
|
||||||
|
// The domain is not used in this test, so it's okay if it is left empty.
|
||||||
|
domain: {
|
||||||
|
verifyingContractAddress: constants.NULL_ADDRESS,
|
||||||
|
chainId: 0,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
@ -41,25 +41,6 @@ async function testHashEIP712DomainAsync(
|
|||||||
expect(actualHash).to.be.eq(hexConcat(expectedHash));
|
expect(actualHash).to.be.eq(hexConcat(expectedHash));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Tests a specific instance of EIP712 message hashing.
|
|
||||||
* @param lib The LibEIP712 contract to call.
|
|
||||||
* @param domainHash The hash of the EIP712 domain of this instance.
|
|
||||||
* @param hashStruct The hash of the struct of this instance.
|
|
||||||
*/
|
|
||||||
async function testHashEIP712MessageAsync(
|
|
||||||
lib: TestLibEIP712Contract,
|
|
||||||
domainHash: string,
|
|
||||||
hashStruct: string,
|
|
||||||
): Promise<void> {
|
|
||||||
const input = '0x1901'.concat(
|
|
||||||
domainHash.slice(2, domainHash.length).concat(hashStruct.slice(2, hashStruct.length)),
|
|
||||||
);
|
|
||||||
const expectedHash = '0x'.concat(ethUtil.sha3(input).toString('hex'));
|
|
||||||
const actualHash = await lib.externalHashEIP712Message.callAsync(domainHash, hashStruct);
|
|
||||||
expect(actualHash).to.be.eq(expectedHash);
|
|
||||||
}
|
|
||||||
|
|
||||||
describe('LibEIP712', () => {
|
describe('LibEIP712', () => {
|
||||||
let lib: TestLibEIP712Contract;
|
let lib: TestLibEIP712Contract;
|
||||||
|
|
||||||
@ -73,6 +54,32 @@ describe('LibEIP712', () => {
|
|||||||
await blockchainLifecycle.revertAsync();
|
await blockchainLifecycle.revertAsync();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests a specific instance of EIP712 message hashing.
|
||||||
|
* @param lib The LibEIP712 contract to call.
|
||||||
|
* @param domainHash The hash of the EIP712 domain of this instance.
|
||||||
|
* @param hashStruct The hash of the struct of this instance.
|
||||||
|
*/
|
||||||
|
async function testHashEIP712MessageAsync(
|
||||||
|
lib: TestLibEIP712Contract,
|
||||||
|
domainHash: string,
|
||||||
|
hashStruct: string,
|
||||||
|
): Promise<void> {
|
||||||
|
// Remove the hex prefix from the domain hash and the hash struct
|
||||||
|
const unprefixedDomainHash = domainHash.slice(2, domainHash.length);
|
||||||
|
const unprefixedHashStruct = hashStruct.slice(2, hashStruct.length);
|
||||||
|
|
||||||
|
// Hash the provided input to get the expected hash
|
||||||
|
const input = '0x1901'.concat(unprefixedDomainHash.concat(unprefixedHashStruct));
|
||||||
|
const expectedHash = '0x'.concat(ethUtil.sha3(input).toString('hex'));
|
||||||
|
|
||||||
|
// Get the actual hash by calling the smart contract
|
||||||
|
const actualHash = await lib.externalHashEIP712Message.callAsync(domainHash, hashStruct);
|
||||||
|
|
||||||
|
// Verify that the actual hash matches the expected hash
|
||||||
|
expect(actualHash).to.be.eq(expectedHash);
|
||||||
|
}
|
||||||
|
|
||||||
describe('_hashEIP712Domain', async () => {
|
describe('_hashEIP712Domain', async () => {
|
||||||
it('should correctly hash empty input', async () => {
|
it('should correctly hash empty input', async () => {
|
||||||
await testHashEIP712DomainAsync(lib, '', '', 0, constants.NULL_ADDRESS);
|
await testHashEIP712DomainAsync(lib, '', '', 0, constants.NULL_ADDRESS);
|
||||||
|
@ -20,6 +20,15 @@ export const signTypedDataUtils = {
|
|||||||
]),
|
]),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Generates the EIP712 Typed Data hash for a typed data object without using the domain field. This
|
||||||
|
* makes hashing easier for non-EIP712 data.
|
||||||
|
* @param typedData An object that conforms to the EIP712TypedData interface
|
||||||
|
* @return A Buffer containing the hash of the typed data.
|
||||||
|
*/
|
||||||
|
generateTypedDataHashWithoutDomain(typedData: EIP712TypedData): Buffer {
|
||||||
|
return signTypedDataUtils._structHash(typedData.primaryType, typedData.message, typedData.types);
|
||||||
|
},
|
||||||
/**
|
/**
|
||||||
* Generates the hash of a EIP712 Domain with the default schema
|
* Generates the hash of a EIP712 Domain with the default schema
|
||||||
* @param domain An EIP712 domain with the default schema containing a name, version, chain id,
|
* @param domain An EIP712 domain with the default schema containing a name, version, chain id,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user