2017-10-19 04:28:45 +03:00
'use strict' ;
var assert = require ( 'assert' ) ;
var utils = require ( './utils' ) ;
2018-07-23 10:02:39 +03:00
var ethers = utils . getEthers ( _ _filename ) ;
2017-10-19 04:28:45 +03:00
function equals ( a , b ) {
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 ;
}
return a === b ;
}
describe ( 'Test Contract Address Generation' , function ( ) {
// @TODO: Mine a large collection of these from the blockchain
2018-03-05 03:31:09 +03:00
var getContractAddress = ethers . utils . getContractAddress ;
2017-10-19 04:28:45 +03:00
// Transaction: 0x939aa17985bc2a52a0c1cba9497ef09e092355a805a8150e30e24b753bac6864
var Tests = [
{
address : '0x3474627D4F63A678266BC17171D87f8570936622' ,
name : 'tx-0x939aa17985bc2a52a0c1cba9497ef09e092355a805a8150e30e24b753bac6864' ,
tx : {
from : '0xb2682160c482eb985ec9f3e364eec0a904c44c23' ,
nonce : 10 ,
}
} ,
// Ropsten: 0x5bdfd14fcc917abc2f02a30721d152a6f147f09e8cbaad4e0d5405d646c5c3e1
{
address : '0x0CcCC7507aEDf9FEaF8C8D731421746e16b4d39D' ,
name : 'zero-nonce' ,
tx : {
from : '0xc6af6e1a78a6752c7f8cd63877eb789a2adb776c' ,
nonce : 0
}
} ,
]
Tests . forEach ( function ( test ) {
it ( ( 'Computes the transaction address - ' + test . name ) , function ( ) {
2018-07-16 07:20:36 +03:00
this . timeout ( 120000 ) ;
2017-10-19 04:28:45 +03:00
assert . equal ( getContractAddress ( test . tx ) , test . address , 'computes the transaction address' ) ;
} ) ;
} ) ;
} ) ;
describe ( 'Test RLP Coder' , function ( ) {
2018-03-05 03:31:09 +03:00
var rlp = ethers . utils . RLP ;
2017-10-19 04:28:45 +03:00
var tests = utils . loadTests ( 'rlp-coder' ) ;
tests . forEach ( function ( test ) {
it ( ( 'RLP coder encoded - ' + test . name ) , function ( ) {
2018-07-16 07:20:36 +03:00
this . timeout ( 120000 ) ;
2017-10-19 04:28:45 +03:00
assert . equal ( rlp . encode ( test . decoded ) , test . encoded , 'RLP encoded - ' + test . name ) ;
} ) ;
} ) ;
tests . forEach ( function ( test ) {
it ( ( 'RLP coder decoded - ' + test . name ) , function ( ) {
2018-07-16 07:20:36 +03:00
this . timeout ( 120000 ) ;
2017-10-19 04:28:45 +03:00
assert . ok ( equals ( rlp . decode ( test . encoded ) , test . decoded ) ,
'RLP decoded - ' + test . name ) ;
} ) ;
} ) ;
} ) ;
describe ( 'Test Unit Conversion' , function ( ) {
var tests = utils . loadTests ( 'units' ) ;
2018-01-10 02:25:44 +03:00
2017-10-19 04:28:45 +03:00
tests . forEach ( function ( test ) {
2018-03-05 03:31:09 +03:00
var wei = ethers . utils . bigNumberify ( test . wei ) ;
2017-10-19 04:28:45 +03:00
var formatting = test . format || { } ;
2018-09-26 23:11:07 +03:00
// We no longer support padding
if ( formatting . pad ) { return ; }
2017-10-19 04:28:45 +03:00
it ( ( 'parses ' + test . ether + ' ether' ) , function ( ) {
2018-09-26 23:11:07 +03:00
assert . ok ( ethers . utils . parseEther ( test . ether . replace ( /,/g , '' ) ) . eq ( wei ) ,
2017-10-19 04:28:45 +03:00
'parsing ether failed - ' + test . name ) ;
} ) ;
2018-01-10 02:25:44 +03:00
it ( ( 'formats ' + wei . toString ( ) + ' wei (options: ' + JSON . stringify ( formatting ) + ')' ) , function ( ) {
2018-09-26 23:11:07 +03:00
var v = ethers . utils . formatEther ( wei ) ;
if ( formatting . commify ) { v = ethers . utils . commify ( v ) ; }
assert . equal ( v , test . etherFormat ,
2017-10-19 04:28:45 +03:00
'formatting wei failed - ' + test . name ) ;
} ) ;
} ) ;
2018-01-10 02:25:44 +03:00
tests . forEach ( function ( test ) {
2018-03-05 03:31:09 +03:00
var wei = ethers . utils . bigNumberify ( test . wei ) ;
2018-01-10 02:25:44 +03:00
var formatting = test . format || { } ;
2018-08-21 14:23:11 +03:00
[ 'kwei' , 'mwei' , 'gwei' , 'szabo' , 'finney' , 'satoshi' ] . forEach ( function ( name ) {
2018-04-05 22:46:44 +03:00
var unitName = name ;
if ( name === 'satoshi' ) { unitName = 8 ; }
2018-01-10 02:25:44 +03:00
if ( test [ name ] ) {
it ( ( 'parses ' + test [ name ] + ' ' + name ) , function ( ) {
2018-07-16 07:20:36 +03:00
this . timeout ( 120000 ) ;
2018-09-26 23:11:07 +03:00
assert . ok ( ethers . utils . parseUnits ( test [ name ] . replace ( /,/g , '' ) , unitName ) . eq ( wei ) ,
2018-01-10 02:25:44 +03:00
( 'parsing ' + name + ' failed - ' + test . name ) ) ;
} ) ;
}
if ( test [ name + '_format' ] ) {
it ( ( 'formats ' + wei . toString ( ) + ' ' + name + ' (options: ' + JSON . stringify ( formatting ) + ')' ) , function ( ) {
2018-09-26 23:29:16 +03:00
var v = ethers . utils . formatUnits ( wei , unitName ) ;
2018-09-26 23:11:07 +03:00
if ( formatting . commify ) { v = ethers . utils . commify ( v ) ; }
assert . equal ( v , test [ name + '_format' ] ,
2018-01-10 02:25:44 +03:00
( 'formats ' + name + ' - ' + test . name ) ) ;
} ) ;
}
} ) ;
} ) ;
2017-10-19 04:28:45 +03:00
} ) ;
2018-01-10 02:25:44 +03:00
2017-10-19 04:28:45 +03:00
describe ( 'Test Namehash' , function ( ) {
var tests = utils . loadTests ( 'namehash' ) ;
tests . forEach ( function ( test ) {
it ( ( 'computes namehash - "' + test . name + '"' ) , function ( ) {
2018-07-16 07:20:36 +03:00
this . timeout ( 120000 ) ;
2018-03-05 03:31:09 +03:00
assert . equal ( ethers . utils . namehash ( test . name ) , test . expected ,
2017-10-19 04:28:45 +03:00
'computes namehash(' + test . name + ')' ) ;
} ) ;
} ) ;
} ) ;
2017-11-23 10:11:08 +03:00
describe ( 'Test ID Hash Function' , function ( ) {
2017-10-19 04:28:45 +03:00
var tests = [
{
name : 'setAddr signature hash' ,
text : 'setAddr(bytes32,address)' ,
expected : '0xd5fa2b00b0756613052388dd576d941ba16904757996b8bb03a737ef4fd1f9ce'
}
]
tests . forEach ( function ( test ) {
it ( ( 'computes id - ' + test . name ) , function ( ) {
2018-07-16 07:20:36 +03:00
this . timeout ( 120000 ) ;
2018-03-05 03:31:09 +03:00
var value = ethers . utils . id ( test . text ) ;
2017-10-19 04:28:45 +03:00
assert . equal ( value , test . expected ,
'computes id(' + test . text + ')' ) ;
} ) ;
} ) ;
} ) ;
2017-11-23 10:11:08 +03:00
describe ( 'Test Solidity Hash Functions' , function ( ) {
var tests = utils . loadTests ( 'solidity-hashes' ) ;
2018-03-05 03:31:09 +03:00
[ 'Keccak256' , 'Sha256' ] . forEach ( function ( funcName ) {
2017-11-23 10:11:08 +03:00
it ( ( 'computes ' + funcName + ' correctly' ) , function ( ) {
2018-07-16 07:20:36 +03:00
this . timeout ( 120000 ) ;
2017-11-23 10:11:08 +03:00
tests . forEach ( function ( test , index ) {
2018-03-05 03:31:09 +03:00
var result = ethers . utils [ 'solidity' + funcName ] ( test . types , test . values ) ;
assert . equal ( result , test [ funcName . toLowerCase ( ) ] ,
2017-11-23 10:11:08 +03:00
( 'computes solidity-' + funcName + '(' + JSON . stringify ( test . values ) + ') - ' + test . types ) ) ;
} ) ;
} ) ;
} ) ;
} ) ;
describe ( 'Test Hash Functions' , function ( ) {
var tests = utils . loadTests ( 'hashes' ) ;
it ( 'computes keccak256 correctly' , function ( ) {
2018-07-16 07:20:36 +03:00
this . timeout ( 120000 ) ;
2017-11-23 10:11:08 +03:00
tests . forEach ( function ( test ) {
2018-03-05 03:31:09 +03:00
assert . equal ( ethers . utils . keccak256 ( test . data ) , test . keccak256 , ( 'Keccak256 - ' + test . data ) ) ;
2017-11-23 10:11:08 +03:00
} ) ;
} ) ;
it ( 'computes sha2566 correctly' , function ( ) {
2018-07-16 07:20:36 +03:00
this . timeout ( 120000 ) ;
2017-11-23 10:11:08 +03:00
tests . forEach ( function ( test ) {
2018-03-05 03:31:09 +03:00
assert . equal ( ethers . utils . sha256 ( test . data ) , test . sha256 , ( 'SHA256 - ' + test . data ) ) ;
2017-11-23 10:11:08 +03:00
} ) ;
} ) ;
} ) ;
2018-04-12 22:18:11 +03:00
describe ( 'Test Solidity splitSignature' , function ( ) {
it ( 'splits a canonical signature' , function ( ) {
2018-07-16 07:20:36 +03:00
this . timeout ( 120000 ) ;
2018-04-12 22:18:11 +03:00
var r = '0xdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef' ;
var s = '0xcafe1a7ecafe1a7ecafe1a7ecafe1a7ecafe1a7ecafe1a7ecafe1a7ecafe1a7e' ;
for ( var v = 27 ; v <= 28 ; v ++ ) {
2018-06-13 22:39:39 +03:00
var signature = ethers . utils . concat ( [ r , s , [ v ] ] ) ;
var sig = ethers . utils . splitSignature ( signature ) ;
2018-04-12 22:18:11 +03:00
assert . equal ( sig . r , r , 'split r correctly' ) ;
assert . equal ( sig . s , s , 'split s correctly' ) ;
assert . equal ( sig . v , v , 'split v correctly' ) ;
}
} ) ;
it ( 'splits a legacy signature' , function ( ) {
2018-07-16 07:20:36 +03:00
this . timeout ( 120000 ) ;
2018-04-12 22:18:11 +03:00
var r = '0xdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeefdeadbeef' ;
var s = '0xcafe1a7ecafe1a7ecafe1a7ecafe1a7ecafe1a7ecafe1a7ecafe1a7ecafe1a7e' ;
for ( var v = 27 ; v <= 28 ; v ++ ) {
2018-06-13 22:39:39 +03:00
var signature = ethers . utils . concat ( [ r , s , [ v - 27 ] ] ) ;
var sig = ethers . utils . splitSignature ( signature ) ;
2018-04-12 22:18:11 +03:00
assert . equal ( sig . r , r , 'split r correctly' ) ;
assert . equal ( sig . s , s , 'split s correctly' ) ;
assert . equal ( sig . v , v , 'split v correctly' ) ;
}
} ) ;
} ) ;
2018-04-14 23:10:26 +03:00
describe ( 'Test Base64 coder' , function ( ) {
// https://en.wikipedia.org/wiki/Base64#Examples
it ( 'encodes and decodes the example from wikipedia' , function ( ) {
2018-07-16 07:20:36 +03:00
this . timeout ( 120000 ) ;
2018-04-14 23:10:26 +03:00
var decodedText = 'Man is distinguished, not only by his reason, but by this singular passion from other animals, which is a lust of the mind, that by a perseverance of delight in the continued and indefatigable generation of knowledge, exceeds the short vehemence of any carnal pleasure.' ;
2018-06-13 22:39:39 +03:00
var decoded = ethers . utils . toUtf8Bytes ( decodedText ) ;
2018-04-14 23:10:26 +03:00
var encoded = 'TWFuIGlzIGRpc3Rpbmd1aXNoZWQsIG5vdCBvbmx5IGJ5IGhpcyByZWFzb24sIGJ1dCBieSB0aGlzIHNpbmd1bGFyIHBhc3Npb24gZnJvbSBvdGhlciBhbmltYWxzLCB3aGljaCBpcyBhIGx1c3Qgb2YgdGhlIG1pbmQsIHRoYXQgYnkgYSBwZXJzZXZlcmFuY2Ugb2YgZGVsaWdodCBpbiB0aGUgY29udGludWVkIGFuZCBpbmRlZmF0aWdhYmxlIGdlbmVyYXRpb24gb2Yga25vd2xlZGdlLCBleGNlZWRzIHRoZSBzaG9ydCB2ZWhlbWVuY2Ugb2YgYW55IGNhcm5hbCBwbGVhc3VyZS4=' ;
2018-06-13 22:39:39 +03:00
assert . equal ( ethers . utils . base64 . encode ( decoded ) , encoded , 'encodes to base64 string' ) ;
assert . equal ( ethers . utils . toUtf8String ( ethers . utils . base64 . decode ( encoded ) ) , decodedText , 'decodes from base64 sstring' ) ;
2018-04-14 23:10:26 +03:00
} ) ;
} ) ;
2018-08-21 14:14:37 +03:00
describe ( 'Test UTF-8 coder' , function ( ) {
var BadUTF = [
// See: https://en.wikipedia.org/wiki/UTF-8#Overlong_encodings
{ bytes : [ 0xF0 , 0x82 , 0x82 , 0xAC ] , reason : 'overlong' , name : 'wikipedia overlong encoded Euro sign' } ,
{ bytes : [ 0xc0 , 0x80 ] , reason : 'overlong' , name : '2-byte overlong - 0xc080' } ,
{ bytes : [ 0xc0 , 0xbf ] , reason : 'overlong' , name : '2-byte overlong - 0xc0bf' } ,
{ bytes : [ 0xc1 , 0x80 ] , reason : 'overlong' , name : '2-byte overlong - 0xc180' } ,
{ bytes : [ 0xc1 , 0xbf ] , reason : 'overlong' , name : '2-byte overlong - 0xc1bf' } ,
// Reserved UTF-16 Surrogate halves
{ bytes : [ 0xed , 0xa0 , 0x80 ] , reason : 'utf-16 surrogate' , name : 'utf-16 surrogate - U+d800' } ,
{ bytes : [ 0xed , 0xbf , 0xbf ] , reason : 'utf-16 surrogate' , name : 'utf-16 surrogate - U+dfff' } ,
// a leading byte not followed by enough continuation bytes
{ bytes : [ 0xdf ] , reason : 'too short' , name : 'too short - 2-bytes - 0x00' } ,
{ bytes : [ 0xe0 ] , reason : 'too short' , name : 'too short - 3-bytes' } ,
{ bytes : [ 0xe0 , 0x80 ] , reason : 'too short' , name : 'too short - 3-bytes with 1' } ,
{ bytes : [ 0x80 ] , reason : 'unexpected continuation byte' , name : 'unexpected continuation byte' } ,
{ bytes : [ 0xc2 , 0x00 ] , reason : 'invalid continuation byte' , name : 'invalid continuation byte - 0xc200' } ,
{ bytes : [ 0xc2 , 0x40 ] , reason : 'invalid continuation byte' , name : 'invalid continuation byte - 0xc240' } ,
{ bytes : [ 0xc2 , 0xc0 ] , reason : 'invalid continuation byte' , name : 'invalid continuation byte - 0xc2c0' } ,
// Out of range
{ bytes : [ 0xf4 , 0x90 , 0x80 , 0x80 ] , reason : 'out-of-range' , name : 'out of range' } ,
] ;
BadUTF . forEach ( function ( test ) {
it ( 'toUtf8String - ' + test . name , function ( ) {
assert . throws ( function ( ) {
var result = ethers . utils . toUtf8String ( test . bytes ) ;
console . log ( 'Result' , result ) ;
} , function ( error ) {
return ( error . message . split ( ';' ) . pop ( ) . trim ( ) === test . reason )
} , test . name ) ;
} ) ;
} ) ;
it ( 'toUtf8String - random conversions' , function ( ) {
this . timeout ( 200000 ) ;
function randomChar ( seed ) {
switch ( utils . randomNumber ( seed + '-range' , 0 , 4 ) ) {
case 0 :
return String . fromCharCode ( utils . randomNumber ( seed + '-value' , 0 , 0x100 ) ) ;
case 1 :
return String . fromCharCode ( utils . randomNumber ( seed + '-value' , 0 , 0xd800 ) ) ;
case 2 :
return String . fromCharCode ( utils . randomNumber ( seed + '-value' , 0xdfff + 1 , 0xffff ) ) ;
case 3 :
var left = utils . randomNumber ( seed + '-value' , 0xd800 , 0xdbff + 1 ) ;
var right = utils . randomNumber ( seed + '-value' , 0xdc00 , 0xdfff + 1 ) ;
return String . fromCharCode ( left , right ) ;
}
throw new Error ( 'this should not happen' ) ;
}
function randomString ( seed ) {
var length = utils . randomNumber ( seed + '-length' , 1 , 5 ) ;
var str = '' ;
for ( var i = 0 ; i < length ; i ++ ) {
str += randomChar ( seed + '-char-' + i ) ;
}
return str ;
}
for ( var i = 0 ; i < 100000 ; i ++ ) {
var seed = 'test-' + String ( i ) ;
var str = randomString ( seed ) ;
var bytes = ethers . utils . toUtf8Bytes ( str )
var str2 = ethers . utils . toUtf8String ( bytes ) ;
assert . ok ( Buffer . from ( str ) . equals ( Buffer . from ( bytes ) ) , 'bytes not generated correctly - ' + bytes )
assert . equal ( str2 , str , 'conversion not reflexive - ' + bytes ) ;
}
} ) ;
} ) ;
describe ( 'Test Bytes32String coder' , function ( ) {
// @TODO: a LOT more test cases; generated from Solidity
it ( "encodes an ens name" , function ( ) {
var str = "ricmoo.firefly.eth" ;
var bytes32 = ethers . utils . formatBytes32String ( str ) ;
var str2 = ethers . utils . parseBytes32String ( bytes32 ) ;
assert . equal ( bytes32 , '0x7269636d6f6f2e66697265666c792e6574680000000000000000000000000000' , 'formatted correctly' ) ;
assert . equal ( str2 , str , "parsed correctly" ) ;
} ) ;
} ) ;
2018-12-13 00:10:28 +03:00
describe ( 'Test BigNumber' , function ( ) {
it ( "computes absoltue values" , function ( ) {
function testAbs ( test ) {
var value = ethers . utils . bigNumberify ( test . value ) ;
var expected = ethers . utils . bigNumberify ( test . expected ) ;
assert . ok ( value . abs ( ) . eq ( expected ) , 'BigNumber.abs - ' + test . value ) ;
}
[
{ value : "0x0" , expected : "0x0" } ,
{ value : "-0x0" , expected : "0x0" } ,
{ value : "0x5" , expected : "0x5" } ,
{ value : "-0x5" , expected : "0x5" } ,
{ value : "0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" , expected : "0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" } ,
{ value : "-0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" , expected : "0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" } ,
{ value : "0xfffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" , expected : "0xfffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" } ,
{ value : "-0xfffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" , expected : "0xfffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" } ,
] . forEach ( testAbs ) ;
} ) ;
} ) ;