Skip to content

Commit

Permalink
Added tests for FunctionDecoder. Closes #1
Browse files Browse the repository at this point in the history
  • Loading branch information
GFJHogue committed Oct 30, 2017
1 parent 7cde592 commit 84929bb
Show file tree
Hide file tree
Showing 5 changed files with 226 additions and 13 deletions.
4 changes: 1 addition & 3 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,10 +1,8 @@
##### CAUTION: This package is not fully tested (for now)!

# ethereum-tx-decoder

[![npm version](https://badge.fury.io/js/ethereum-tx-decoder.svg)](https://badge.fury.io/js/ethereum-tx-decoder)[![Build Status](https://travis-ci.org/GFJHogue/ethereum-tx-decoder.svg?branch=master)](https://travis-ci.org/GFJHogue/ethereum-tx-decoder)

Fully decode function parameters from raw transactions!
Lightweight utility for decoding function parameters from Ethereum transactions.

- A lightweight utility with minimal dependencies.

Expand Down
12 changes: 8 additions & 4 deletions package.json
Original file line number Diff line number Diff line change
@@ -1,13 +1,19 @@
{
"name": "ethereum-tx-decoder",
"version": "0.0.1",
"description": "Takes raw transactions and fully decodes them!",
"version": "1.0.0",
"description": "Lightweight utility for decoding function parameters from Ethereum transactions.",
"keywords": [
"ethereum",
"transaction",
"smart",
"contract",
"abi",
"function",
"functions",
"parameters",
"params",
"event",
"events",
"decoder",
"decode",
"RLP",
Expand Down Expand Up @@ -39,8 +45,6 @@
"ethers-utils": "2.1.3"
},
"devDependencies": {
"ethereumjs-tx": "1.3.3",
"ethereumjs-util": "5.1.2",
"ethers-wallet": "2.1.3",
"mocha": "4.0.1"
}
Expand Down
9 changes: 3 additions & 6 deletions test/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,8 @@

Test generation taken from `ethers.js`.

## Status

- So far only `decodeTx()` is tested (`decodeTransaction.js`).
- `decodeTx()` is tested (`decodeTransaction.js`).
* Using tests replicated from `ethers.Wallet` for `Wallet.parseTransaction()`

## Backlog

- `FunctionDecoder.js`
- `FunctionDecoder` is tested (`FunctionDecoder.js`).
* Using tests derived from `ethers.Interface` for `Interface.decodeParams()`
214 changes: 214 additions & 0 deletions test/test-contract-interface.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,214 @@
'use strict';

var assert = require('assert');

var utils = require('./utils');

// Set this to true to compare results against Web3 coder
var testWeb3 = false;
var testEthereumLib = false;


var FORMAT_WEB3 = 'web3';
var FORMAT_ETHEREUM_LIB = 'ethereum-lib';

var web3Coder = null;
if (testWeb3) {
console.log('Web3 Version:', require('web3/package.json').version);
web3Coder = require('web3/lib/solidity/coder');
}

var ethereumLibCoder = null;
if (testEthereumLib) {
console.log('EthereumLib (abi) Version:', require('ethereumjs-abi/package.json').version);
ethereumLibCoder = require('ethereumjs-abi');
}

function equals(a, b) {
// Array (treat recursively)
if (Array.isArray(a)) {
if (!Array.isArray(b) || a.length !== b.length) { return false; }
for (var i = 0; i < a.length; i++) {
if (!equals(a[i], b[i])) { return false; }
}
return true;
}

if (testWeb3) {
if (a.toJSON && a.minus) { a = utils.bigNumberify(a.toString(10)); }
if (b.toJSON && b.minus) { b = utils.bigNumberify(b.toString(10)); }
}

// EthereumLib returns addresses as 160-bit big numbers
if (testEthereumLib) {
if (a.match && a.match(/^0x[0-9A-Fa-f]{40}$/) && b.imul) {
a = utils.bigNumberify(a);
}
if (b.match && b.match(/^0x[0-9A-Fa-f]{40}$/) && a.imul) {
b = utils.bigNumberify(a);
}
}

// BigNumber
if (a.eq) {
if (!a.eq(b)) { return false; }
return true;
}

if (testEthereumLib) {
if (a.buffer) { a = '0x' + (new Buffer(a)).toString('hex'); }
if (b.buffer) { b = '0x' + (new Buffer(b)).toString('hex'); }
if (Buffer.isBuffer(a)) { a = '0x' + a.toString('hex'); }
if (Buffer.isBuffer(b)) { b = '0x' + b.toString('hex'); }
}

// Uint8Array
if (a.buffer) {
if (!utils.isHexString(b)) { return false; }
b = utils.arrayify(b);

if (!b.buffer || a.length !== b.length) { return false; }
for (var i = 0; i < a.length; i++) {
if (a[i] !== b[i]) { return false; }
}

return true;
}

if (testWeb3) {
if (a.match && a.match(/^0x[0-9A-Fa-f]{40}$/)) { a = a.toLowerCase(); }
if (b.match && b.match(/^0x[0-9A-Fa-f]{40}$/)) { b = b.toLowerCase(); }
}

// Something else
return a === b;
}


function getValues(object, format) {
if (Array.isArray(object)) {
var result = [];
object.forEach(function(object) {
result.push(getValues(object, format));
});
return result;
}

switch (object.type) {
case 'number':
if (format === FORMAT_WEB3) {
var value = utils.bigNumberify(object.value);
value.round = function() {
return this;
}
value.lessThan = function(other) {
return this.lt(other);
}
var toString = value.toString.bind(value);
var toHexString = value.toHexString.bind(value);
Object.defineProperty(value, 'toString', {
value: function(format) {
if (format === 16) { return toHexString().substring(2); }
return toString();
}
});
return value;
}
return utils.bigNumberify(object.value);

case 'boolean':
case 'string':
return object.value;

case 'buffer':
if (format === FORMAT_WEB3) {
return object.value.toLowerCase();
}
return utils.arrayify(object.value);

default:
throw new Error('invalid type - ' + object.type);
}
}

describe('Contract Interface ABI Decoding', function() {
var Interface = require('ethers-contracts/index.js').Interface;
var FunctionDecoder = require('../src/FunctionDecoder.js');

var tests = utils.loadTests('contract-interface');
tests.forEach(function(test) {
var values = getValues(JSON.parse(test.normalizedValues));
var types = JSON.parse(test.types);
var result = test.result;

var title = test.name + ' => (' + test.types + ') = (' + test.normalizedValues + ')';

// Tweak the abi a bit to assume the test function takes the outputs as inputs.
var abi = JSON.parse(test.interface);
abi[0].inputs = abi[0].outputs;

var iface = new Interface(abi);
var sighash = iface.functions.test.sighash;
var encoded = Interface.encodeParams(types, values);

var fnData = sighash + encoded.substring(2);

it(('decodes parameters - ' + test.name + ' - ' + test.types), function() {
var fnDecoder = new FunctionDecoder(iface);

var decoded = fnDecoder.decodeFn(fnData);
var decodedArray = Array.prototype.slice.call(decoded);;

assert.ok(equals(values, decodedArray), 'decoded parameters - ' + title);

if (testWeb3) {
var badWeb3 = [ 'random-1116' ];
if (badWeb3.indexOf(test.name) >= 0) {
assert.ok(false, 'This testcase hangs in Web3');
} else {
var valuesWeb3 = getValues(JSON.parse(test.normalizedValues), FORMAT_WEB3);
var web3Decoded = web3Coder.decodeParams(types, result.substring(2));
//console.dir(valuesWeb3, { depth: null });
//console.dir(web3Decoded, { depth: null });
assert.ok(equals(valuesWeb3, web3Decoded), 'web3 decoded data - ' + title);
}
}

if (testEthereumLib) {
// Bugs (I think)
// - Addresses are returned as Big Numbers (not a bug; but odd?)
// - random-1969: offset bug when decoding
// - bytes[3][]; the regex is unhappy

var badEthereumLib = [
// Bug in elementaryName(name); should read 'return 'int256' + name.slice(3)'
'sol-23', 'sol-24', 'random-22', 'random-220', 'random-254', 'random-431', 'random-468', 'random-537', 'random-544', 'random-566', 'random-604', 'random-708', 'random-713', 'random-761', 'random-767', 'random-884', 'random-934', 'random-937', 'random-963', 'random-1048', 'random-1103', 'random-1189', 'random-1326', 'random-1441', 'random-1456', 'random-1592', 'random-1741', 'random-1834', 'random-1862', 'random-1876', 'random-1895',

// I think the problem is the regex; fails on things like 'bytes[3][]';
// throws 'Cannot read property '1' of null' errors
'random-9', 'random-19', 'random-22', 'random-33', 'random-35', 'random-52', 'random-57', 'random-58', 'random-61', 'random-67', 'random-86', 'random-103', 'random-116', 'random-136', 'random-147', 'random-171', 'random-173', 'random-175', 'random-220', 'random-251', 'random-254', 'random-298', 'random-301', 'random-341', 'random-343', 'random-357', 'random-392', 'random-405', 'random-419', 'random-429', 'random-431', 'random-435', 'random-457', 'random-461', 'random-466', 'random-468', 'random-471', 'random-472', 'random-477', 'random-483', 'random-484', 'random-488', 'random-493', 'random-500', 'random-505', 'random-515', 'random-519', 'random-537', 'random-541', 'random-544', 'random-562', 'random-566', 'random-573', 'random-582', 'random-587', 'random-588', 'random-604', 'random-607', 'random-621', 'random-629', 'random-633', 'random-642', 'random-643', 'random-650', 'random-660', 'random-671', 'random-674', 'random-691', 'random-708', 'random-712', 'random-713', 'random-740', 'random-741', 'random-747', 'random-761', 'random-763', 'random-767', 'random-772', 'random-781', 'random-810', 'random-820', 'random-823', 'random-869', 'random-884', 'random-891', 'random-903', 'random-925', 'random-934', 'random-937', 'random-938', 'random-962', 'random-963', 'random-964', 'random-988', 'random-999', 'random-1002', 'random-1017', 'random-1046', 'random-1048', 'random-1050', 'random-1068', 'random-1075', 'random-1079', 'random-1100', 'random-1103', 'random-1104', 'random-1116', 'random-1117', 'random-1152', 'random-1168', 'random-1189', 'random-1197', 'random-1202', 'random-1217', 'random-1240', 'random-1246', 'random-1247', 'random-1257', 'random-1263', 'random-1267', 'random-1275', 'random-1293', 'random-1303', 'random-1308', 'random-1310', 'random-1326', 'random-1338', 'random-1348', 'random-1371', 'random-1393', 'random-1401', 'random-1406', 'random-1410', 'random-1441', 'random-1449', 'random-1456', 'random-1476', 'random-1478', 'random-1498', 'random-1508', 'random-1529', 'random-1553', 'random-1555', 'random-1576', 'random-1592', 'random-1597', 'random-1598', 'random-1601', 'random-1620', 'random-1627', 'random-1662', 'random-1672', 'random-1713', 'random-1741', 'random-1747', 'random-1756', 'random-1759', 'random-1761', 'random-1762', 'random-1783', 'random-1790', 'random-1795', 'random-1797', 'random-1800', 'random-1807', 'random-1814', 'random-1820', 'random-1823', 'random-1829', 'random-1834', 'random-1841', 'random-1853', 'random-1854', 'random-1862', 'random-1865', 'random-1866', 'random-1869', 'random-1876', 'random-1883', 'random-1895', 'random-1896', 'random-1897', 'random-1912', 'random-1929', 'random-1935', 'random-1965', 'random-1988', 'random-1995',

// I think these are offset bugs; throws 'Decoded uint exceeds width: ...' errors
'random-640', 'random-1358', 'random-1381', 'random-1969',
];

if (badEthereumLib.indexOf(test.name) >= 0) {
assert.ok(false, 'This testcase seems to fail');

} else {
//console.log(result);
var resultBuffer = new Buffer(result.substring(2), 'hex');
var valuesEthereumLib = getValues(JSON.parse(test.normalizedValues), FORMAT_ETHEREUM_LIB);
//console.log('V', valuesEthereumLib);
var ethereumLibDecoded = ethereumLibCoder.rawDecode(types, resultBuffer);
//console.log('R', types, ethereumLibDecoded);
//console.log('E', valuesEthereumLib, ethereumLibDecoded, equals(valuesEthereumLib, ethereumLibDecoded));
//console.log(ethereumLibDecoded);
assert.ok(equals(valuesEthereumLib, ethereumLibDecoded),
'ethereum-lib decoded data - ' + title);
}
}
});

});
});
Binary file added test/tests/contract-interface.json.gz
Binary file not shown.

0 comments on commit 84929bb

Please sign in to comment.