266 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			266 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
import { blockchainTests, constants, expect, hexConcat, hexRandom, hexSlice } from '@0x/contracts-test-utils';
 | 
						|
import { StakingRevertErrors } from '@0x/order-utils';
 | 
						|
import { cartesianProduct } from 'js-combinatorics';
 | 
						|
 | 
						|
import { artifacts, TestLibProxyContract, TestLibProxyReceiverContract } from '../../src';
 | 
						|
 | 
						|
blockchainTests.resets('LibProxy', env => {
 | 
						|
    let proxy: TestLibProxyContract;
 | 
						|
    let receiver: TestLibProxyReceiverContract;
 | 
						|
 | 
						|
    before(async () => {
 | 
						|
        proxy = await TestLibProxyContract.deployFrom0xArtifactAsync(
 | 
						|
            artifacts.TestLibProxy,
 | 
						|
            env.provider,
 | 
						|
            env.txDefaults,
 | 
						|
            artifacts,
 | 
						|
        );
 | 
						|
 | 
						|
        receiver = await TestLibProxyReceiverContract.deployFrom0xArtifactAsync(
 | 
						|
            artifacts.TestLibProxyReceiver,
 | 
						|
            env.provider,
 | 
						|
            env.txDefaults,
 | 
						|
            artifacts,
 | 
						|
        );
 | 
						|
    });
 | 
						|
 | 
						|
    enum RevertRule {
 | 
						|
        RevertOnError,
 | 
						|
        AlwaysRevert,
 | 
						|
        NeverRevert,
 | 
						|
    }
 | 
						|
 | 
						|
    interface PublicProxyCallArgs {
 | 
						|
        destination: string;
 | 
						|
        revertRule: RevertRule;
 | 
						|
        customEgressSelector: string;
 | 
						|
        ignoreIngressSelector: boolean;
 | 
						|
        calldata: string;
 | 
						|
    }
 | 
						|
 | 
						|
    // Choose a random 4 byte string of calldata to send and prepend with `0x00` to ensure
 | 
						|
    // that it does not call `externalProxyCall` by accident. This calldata will make the fallback
 | 
						|
    // in `TestLibProxyReceiver` fail because it is 4 bytes long.
 | 
						|
    function constructRandomFailureCalldata(): string {
 | 
						|
        return hexConcat('0x00', hexRandom(3));
 | 
						|
    }
 | 
						|
 | 
						|
    // Choose a random 24 byte string of calldata to send and prepend with `0x00` to ensure
 | 
						|
    // that it does not call `externalProxyCall` by accident. This calldata will make the fallback
 | 
						|
    // in `TestLibProxyReceiver` succeed because it isn't 4 bytes long.
 | 
						|
    function constructRandomSuccessCalldata(): string {
 | 
						|
        return hexConcat('0x00', hexRandom(35));
 | 
						|
    }
 | 
						|
 | 
						|
    // Exposes `publicProxyCall()` with useful default arguments.
 | 
						|
    async function publicProxyCallAsync(args: Partial<PublicProxyCallArgs>): Promise<[boolean, string]> {
 | 
						|
        return proxy.publicProxyCall.callAsync(
 | 
						|
            {
 | 
						|
                destination: args.destination || receiver.address,
 | 
						|
                revertRule: args.revertRule || RevertRule.RevertOnError,
 | 
						|
                customEgressSelector: args.customEgressSelector || constants.NULL_BYTES4,
 | 
						|
                ignoreIngressSelector: args.ignoreIngressSelector || false,
 | 
						|
            },
 | 
						|
            args.calldata || constructRandomSuccessCalldata(),
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    // Verifies that the result of a given call to `proxyCall()` results in specified outcome
 | 
						|
    function verifyPostConditions(result: [boolean, string], success: boolean, calldata: string): void {
 | 
						|
        expect(result[0]).to.be.eq(success);
 | 
						|
        expect(result[1]).to.be.eq(calldata);
 | 
						|
    }
 | 
						|
 | 
						|
    // Verifies that the result of a given call to `proxyCall()` results in `ProxyDestinationCannotBeNilError`
 | 
						|
    function verifyDestinationZeroError(result: [boolean, string]): void {
 | 
						|
        const expectedError = new StakingRevertErrors.ProxyDestinationCannotBeNilError();
 | 
						|
        expect(result[0]).to.be.false();
 | 
						|
        expect(result[1]).to.be.eq(expectedError.encode());
 | 
						|
    }
 | 
						|
 | 
						|
    describe('proxyCall', () => {
 | 
						|
        describe('Failure Conditions', () => {
 | 
						|
            it('should revert when the destination is address zero', async () => {
 | 
						|
                verifyDestinationZeroError(await publicProxyCallAsync({ destination: constants.NULL_ADDRESS }));
 | 
						|
            });
 | 
						|
 | 
						|
            it('should revert when the destination is address zero and revertRule == AlwaysRevert', async () => {
 | 
						|
                verifyDestinationZeroError(
 | 
						|
                    await publicProxyCallAsync({
 | 
						|
                        destination: constants.NULL_ADDRESS,
 | 
						|
                        revertRule: RevertRule.AlwaysRevert,
 | 
						|
                    }),
 | 
						|
                );
 | 
						|
            });
 | 
						|
 | 
						|
            it('should revert when the destination is address zero and revertRule == NeverRevert', async () => {
 | 
						|
                verifyDestinationZeroError(
 | 
						|
                    await publicProxyCallAsync({
 | 
						|
                        destination: constants.NULL_ADDRESS,
 | 
						|
                        revertRule: RevertRule.NeverRevert,
 | 
						|
                    }),
 | 
						|
                );
 | 
						|
            });
 | 
						|
        });
 | 
						|
 | 
						|
        describe('RevertRule Checks', () => {
 | 
						|
            it('should revert with the correct data when the call succeeds and revertRule = AlwaysRevert', async () => {
 | 
						|
                const calldata = constructRandomSuccessCalldata();
 | 
						|
 | 
						|
                // Ensure that the returndata (the provided calldata) is correct.
 | 
						|
                verifyPostConditions(
 | 
						|
                    await publicProxyCallAsync({
 | 
						|
                        calldata,
 | 
						|
                        revertRule: RevertRule.AlwaysRevert,
 | 
						|
                    }),
 | 
						|
                    false,
 | 
						|
                    calldata,
 | 
						|
                );
 | 
						|
            });
 | 
						|
 | 
						|
            it('should revert with the correct data when the call falls and revertRule = AlwaysRevert', async () => {
 | 
						|
                const calldata = constructRandomFailureCalldata();
 | 
						|
 | 
						|
                // Ensure that the returndata (the provided calldata) is correct.
 | 
						|
                verifyPostConditions(
 | 
						|
                    await publicProxyCallAsync({
 | 
						|
                        calldata,
 | 
						|
                        revertRule: RevertRule.AlwaysRevert,
 | 
						|
                    }),
 | 
						|
                    false,
 | 
						|
                    calldata,
 | 
						|
                );
 | 
						|
            });
 | 
						|
 | 
						|
            it('should succeed with the correct data when the call succeeds and revertRule = NeverRevert', async () => {
 | 
						|
                const calldata = constructRandomSuccessCalldata();
 | 
						|
 | 
						|
                // Ensure that the returndata (the provided calldata) is correct.
 | 
						|
                verifyPostConditions(
 | 
						|
                    await publicProxyCallAsync({
 | 
						|
                        calldata,
 | 
						|
                        revertRule: RevertRule.NeverRevert,
 | 
						|
                    }),
 | 
						|
                    true,
 | 
						|
                    calldata,
 | 
						|
                );
 | 
						|
            });
 | 
						|
 | 
						|
            it('should succeed with the correct data when the call falls and revertRule = NeverRevert', async () => {
 | 
						|
                const calldata = constructRandomFailureCalldata();
 | 
						|
 | 
						|
                // Ensure that the returndata (the provided calldata) is correct.
 | 
						|
                verifyPostConditions(
 | 
						|
                    await publicProxyCallAsync({
 | 
						|
                        calldata,
 | 
						|
                        revertRule: RevertRule.NeverRevert,
 | 
						|
                    }),
 | 
						|
                    true,
 | 
						|
                    calldata,
 | 
						|
                );
 | 
						|
            });
 | 
						|
 | 
						|
            it('should succeed with the correct data when the call succeeds and revertRule = RevertOnError', async () => {
 | 
						|
                const calldata = constructRandomSuccessCalldata();
 | 
						|
 | 
						|
                // Ensure that the returndata (the provided calldata) is correct.
 | 
						|
                verifyPostConditions(
 | 
						|
                    await publicProxyCallAsync({
 | 
						|
                        calldata,
 | 
						|
                    }),
 | 
						|
                    true,
 | 
						|
                    calldata,
 | 
						|
                );
 | 
						|
            });
 | 
						|
 | 
						|
            it('should revert with the correct data when the call falls and revertRule = RevertOnError', async () => {
 | 
						|
                const calldata = constructRandomFailureCalldata();
 | 
						|
 | 
						|
                // Ensure that the returndata (the provided calldata) is correct.
 | 
						|
                verifyPostConditions(
 | 
						|
                    await publicProxyCallAsync({
 | 
						|
                        calldata,
 | 
						|
                    }),
 | 
						|
                    false,
 | 
						|
                    calldata,
 | 
						|
                );
 | 
						|
            });
 | 
						|
        });
 | 
						|
 | 
						|
        describe('Combinatorial Tests', () => {
 | 
						|
            // Combinatorial Scenarios for `proxyCall()`.
 | 
						|
            function getCombinatorialTestDescription(params: [RevertRule, boolean, string, string]): string {
 | 
						|
                const REVERT_RULE_NAMES = ['RevertOnError', 'AlwaysRevert', 'NeverRevert'];
 | 
						|
                return [
 | 
						|
                    `revertRule: ${REVERT_RULE_NAMES[params[0]]}`,
 | 
						|
                    `ignoreIngressSelector: ${params[1]}`,
 | 
						|
                    `customEgressSelector: ${params[2]}`,
 | 
						|
                    `calldata: ${
 | 
						|
                        params[3].length / 2 - 2 > 4
 | 
						|
                            ? // tslint:disable-next-line
 | 
						|
                              hexSlice(params[3], 0, 4) + '...'
 | 
						|
                            : params[3]
 | 
						|
                    }`,
 | 
						|
                ].join(', ');
 | 
						|
            }
 | 
						|
 | 
						|
            const scenarios = [
 | 
						|
                // revertRule
 | 
						|
                [RevertRule.RevertOnError, RevertRule.AlwaysRevert, RevertRule.NeverRevert],
 | 
						|
                // ignoreIngressSelector
 | 
						|
                [false, true],
 | 
						|
                // customEgressSelector
 | 
						|
                [
 | 
						|
                    constants.NULL_BYTES4,
 | 
						|
                    // Random failure calldata is used because it is nonzero and
 | 
						|
                    // won't collide.
 | 
						|
                    constructRandomFailureCalldata(),
 | 
						|
                ],
 | 
						|
                // calldata
 | 
						|
                [constructRandomFailureCalldata(), constructRandomSuccessCalldata()],
 | 
						|
            ] as [RevertRule[], boolean[], string[], string[]];
 | 
						|
 | 
						|
            for (const params of cartesianProduct(...scenarios).toArray()) {
 | 
						|
                const [revertRule, shouldIgnoreIngressSelector, customEgressSelector, calldata] = params;
 | 
						|
                it(getCombinatorialTestDescription(params), async () => {
 | 
						|
                    // Determine whether or not the call should succeed.
 | 
						|
                    let shouldSucceed = true;
 | 
						|
                    if (
 | 
						|
                        ((shouldIgnoreIngressSelector && customEgressSelector !== constants.NULL_BYTES4) ||
 | 
						|
                            (!shouldIgnoreIngressSelector && customEgressSelector === constants.NULL_BYTES4)) &&
 | 
						|
                        calldata.length === 10 // This corresponds to a hex length of 4
 | 
						|
                    ) {
 | 
						|
                        shouldSucceed = false;
 | 
						|
                    }
 | 
						|
 | 
						|
                    // Override the above success value if the RevertRule defines the success.
 | 
						|
                    if (revertRule === RevertRule.AlwaysRevert) {
 | 
						|
                        shouldSucceed = false;
 | 
						|
                    }
 | 
						|
                    if (revertRule === RevertRule.NeverRevert) {
 | 
						|
                        shouldSucceed = true;
 | 
						|
                    }
 | 
						|
 | 
						|
                    // Construct the data that should be returned.
 | 
						|
                    let returnData = calldata;
 | 
						|
                    if (shouldIgnoreIngressSelector) {
 | 
						|
                        returnData = hexSlice(returnData, 4);
 | 
						|
                    }
 | 
						|
                    if (customEgressSelector !== constants.NULL_BYTES4) {
 | 
						|
                        returnData = hexConcat(customEgressSelector, returnData);
 | 
						|
                    }
 | 
						|
 | 
						|
                    const [didSucceed, actualReturnData] = await publicProxyCallAsync({
 | 
						|
                        calldata,
 | 
						|
                        customEgressSelector,
 | 
						|
                        ignoreIngressSelector: shouldIgnoreIngressSelector,
 | 
						|
                        revertRule,
 | 
						|
                    });
 | 
						|
                    expect(didSucceed).to.be.eq(shouldSucceed);
 | 
						|
                    expect(actualReturnData).to.be.eq(returnData);
 | 
						|
                });
 | 
						|
            }
 | 
						|
        });
 | 
						|
    });
 | 
						|
});
 |