2015-03-15 13:43:48 +07:00
package main
import (
"fmt"
2015-03-25 14:58:52 +00:00
"io/ioutil"
2015-03-15 13:43:48 +07:00
"os"
2015-04-21 18:08:47 +01:00
"path/filepath"
2015-04-22 23:11:11 +01:00
"regexp"
"runtime"
"strconv"
2015-03-15 13:43:48 +07:00
"testing"
"github.com/ethereum/go-ethereum/accounts"
2015-04-22 23:11:11 +01:00
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/common/compiler"
"github.com/ethereum/go-ethereum/common/docserver"
"github.com/ethereum/go-ethereum/common/natspec"
"github.com/ethereum/go-ethereum/common/resolver"
"github.com/ethereum/go-ethereum/core"
"github.com/ethereum/go-ethereum/core/state"
2015-03-15 13:43:48 +07:00
"github.com/ethereum/go-ethereum/crypto"
"github.com/ethereum/go-ethereum/eth"
)
2015-04-22 23:11:11 +01:00
const (
testSolcPath = ""
2015-05-10 13:46:38 +02:00
solcVersion = "0.9.17"
2015-04-22 23:11:11 +01:00
testKey = "e6fab74a43941f82d89cb7faa408e227cdad3153c4720e540e855c19b15e6674"
testAddress = "0x8605cdbbdb6d264aa742e77020dcbc58fcdce182"
testBalance = "10000000000000000000"
2015-05-08 19:37:35 +02:00
// of empty string
testHash = "0xc5d2460186f7233c927e7db2dcc703c0e500b653ca82273b7bfad8045d85a470"
2015-04-22 23:11:11 +01:00
)
var (
testGenesis = ` { " ` + testAddress [ 2 : ] + ` ": { "balance": " ` + testBalance + ` "}} `
)
type testjethre struct {
* jsre
stateDb * state . StateDB
lastConfirm string
ds * docserver . DocServer
}
2015-03-15 13:43:48 +07:00
2015-04-22 23:11:11 +01:00
func ( self * testjethre ) UnlockAccount ( acc [ ] byte ) bool {
err := self . ethereum . AccountManager ( ) . Unlock ( acc , "" )
if err != nil {
panic ( "unable to unlock" )
}
return true
}
func ( self * testjethre ) ConfirmTransaction ( tx string ) bool {
if self . ethereum . NatSpec {
self . lastConfirm = natspec . GetNotice ( self . xeth , tx , self . ds )
}
return true
}
func testJEthRE ( t * testing . T ) ( string , * testjethre , * eth . Ethereum ) {
2015-04-22 10:59:27 +02:00
tmp , err := ioutil . TempDir ( "" , "geth-test" )
2015-03-15 13:43:48 +07:00
if err != nil {
2015-04-22 10:59:27 +02:00
t . Fatal ( err )
2015-03-15 13:43:48 +07:00
}
2015-04-22 23:11:11 +01:00
// set up mock genesis with balance on the testAddress
core . GenesisData = [ ] byte ( testGenesis )
ks := crypto . NewKeyStorePassphrase ( filepath . Join ( tmp , "keys" ) )
am := accounts . NewManager ( ks )
2015-04-22 10:59:27 +02:00
ethereum , err := eth . New ( & eth . Config {
DataDir : tmp ,
2015-04-22 23:11:11 +01:00
AccountManager : am ,
2015-04-22 10:59:27 +02:00
MaxPeers : 0 ,
2015-03-15 13:43:48 +07:00
Name : "test" ,
} )
if err != nil {
2015-04-22 10:59:27 +02:00
t . Fatal ( "%v" , err )
2015-03-15 13:43:48 +07:00
}
2015-04-22 23:11:11 +01:00
keyb , err := crypto . HexToECDSA ( testKey )
if err != nil {
t . Fatal ( err )
}
key := crypto . NewKeyFromECDSA ( keyb )
err = ks . StoreKey ( key , "" )
if err != nil {
t . Fatal ( err )
}
err = am . Unlock ( key . Address , "" )
if err != nil {
t . Fatal ( err )
}
2015-05-12 14:24:11 +02:00
assetPath := filepath . Join ( os . Getenv ( "GOPATH" ) , "src" , "github.com" , "ethereum" , "go-ethereum" , "cmd" , "mist" , "assets" , "ext" )
2015-04-22 23:11:11 +01:00
ds , err := docserver . New ( "/" )
if err != nil {
t . Errorf ( "Error creating DocServer: %v" , err )
}
tf := & testjethre { ds : ds , stateDb : ethereum . ChainManager ( ) . State ( ) . Copy ( ) }
repl := newJSRE ( ethereum , assetPath , testSolcPath , "" , false , tf )
tf . jsre = repl
return tmp , tf , ethereum
2015-03-15 13:43:48 +07:00
}
2015-04-22 23:11:11 +01:00
// this line below is needed for transaction to be applied to the state in testing
// the heavy lifing is done in XEth.ApplyTestTxs
// this is fragile, overwriting xeth will result in
// process leaking since xeth loops cannot quit safely
// should be replaced by proper mining with testDAG for easy full integration tests
// txc, self.xeth = self.xeth.ApplyTestTxs(self.xeth.repl.stateDb, coinbase, txc)
2015-03-15 13:43:48 +07:00
func TestNodeInfo ( t * testing . T ) {
2015-04-22 23:11:11 +01:00
tmp , repl , ethereum := testJEthRE ( t )
2015-04-22 10:59:27 +02:00
if err := ethereum . Start ( ) ; err != nil {
t . Fatalf ( "error starting ethereum: %v" , err )
2015-03-15 13:43:48 +07:00
}
defer ethereum . Stop ( )
2015-04-22 23:11:11 +01:00
defer os . RemoveAll ( tmp )
2015-04-22 10:59:27 +02:00
want := ` { "DiscPort":0,"IP":"0.0.0.0","ListenAddr":"","Name":"test","NodeID":"00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000","NodeUrl":"enode://00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000@0.0.0.0:0","TCPPort":0,"Td":"0"} `
checkEvalJSON ( t , repl , ` admin.nodeInfo() ` , want )
2015-03-15 13:43:48 +07:00
}
func TestAccounts ( t * testing . T ) {
2015-04-22 23:11:11 +01:00
tmp , repl , ethereum := testJEthRE ( t )
2015-04-22 10:59:27 +02:00
if err := ethereum . Start ( ) ; err != nil {
t . Fatalf ( "error starting ethereum: %v" , err )
2015-03-15 13:43:48 +07:00
}
defer ethereum . Stop ( )
2015-04-22 23:11:11 +01:00
defer os . RemoveAll ( tmp )
2015-03-15 13:43:48 +07:00
2015-04-22 23:11:11 +01:00
checkEvalJSON ( t , repl , ` eth.accounts ` , ` [" ` + testAddress + ` "] ` )
checkEvalJSON ( t , repl , ` eth.coinbase ` , ` " ` + testAddress + ` " ` )
2015-03-15 13:43:48 +07:00
2015-04-22 10:59:27 +02:00
val , err := repl . re . Run ( ` admin.newAccount("password") ` )
2015-03-15 13:43:48 +07:00
if err != nil {
t . Errorf ( "expected no error, got %v" , err )
}
2015-04-22 10:59:27 +02:00
addr := val . String ( )
if ! regexp . MustCompile ( ` 0x[0-9a-f] { 40} ` ) . MatchString ( addr ) {
t . Errorf ( "address not hex: %q" , addr )
2015-03-15 13:43:48 +07:00
}
2015-04-22 23:11:11 +01:00
// skip until order fixed #824
// checkEvalJSON(t, repl, `eth.accounts`, `["`+testAddress+`", "`+addr+`"]`)
// checkEvalJSON(t, repl, `eth.coinbase`, `"`+testAddress+`"`)
2015-03-15 13:43:48 +07:00
}
func TestBlockChain ( t * testing . T ) {
2015-04-22 23:11:11 +01:00
tmp , repl , ethereum := testJEthRE ( t )
2015-04-22 10:59:27 +02:00
if err := ethereum . Start ( ) ; err != nil {
t . Fatalf ( "error starting ethereum: %v" , err )
2015-03-15 13:43:48 +07:00
}
defer ethereum . Stop ( )
2015-04-22 23:11:11 +01:00
defer os . RemoveAll ( tmp )
2015-04-22 10:59:27 +02:00
// get current block dump before export/import.
val , err := repl . re . Run ( "JSON.stringify(admin.debug.dumpBlock())" )
2015-03-15 13:43:48 +07:00
if err != nil {
t . Errorf ( "expected no error, got %v" , err )
}
2015-04-22 10:59:27 +02:00
beforeExport := val . String ( )
2015-03-15 13:43:48 +07:00
2015-04-22 10:59:27 +02:00
// do the export
2015-04-22 23:11:11 +01:00
extmp , err := ioutil . TempDir ( "" , "geth-test-export" )
2015-03-15 13:43:48 +07:00
if err != nil {
2015-04-22 10:59:27 +02:00
t . Fatal ( err )
2015-03-15 13:43:48 +07:00
}
2015-04-22 23:11:11 +01:00
defer os . RemoveAll ( extmp )
tmpfile := filepath . Join ( extmp , "export.chain" )
2015-04-22 10:59:27 +02:00
tmpfileq := strconv . Quote ( tmpfile )
2015-03-15 13:43:48 +07:00
2015-04-21 18:08:47 +01:00
checkEvalJSON ( t , repl , ` admin.export( ` + tmpfileq + ` ) ` , ` true ` )
2015-04-22 10:59:27 +02:00
if _ , err := os . Stat ( tmpfile ) ; err != nil {
t . Fatal ( err )
2015-03-15 13:43:48 +07:00
}
2015-04-22 10:59:27 +02:00
// check import, verify that dumpBlock gives the same result.
2015-04-21 18:08:47 +01:00
checkEvalJSON ( t , repl , ` admin.import( ` + tmpfileq + ` ) ` , ` true ` )
2015-04-22 10:59:27 +02:00
checkEvalJSON ( t , repl , ` admin.debug.dumpBlock() ` , beforeExport )
2015-03-15 13:43:48 +07:00
}
func TestMining ( t * testing . T ) {
2015-04-22 23:11:11 +01:00
tmp , repl , ethereum := testJEthRE ( t )
2015-04-22 10:59:27 +02:00
if err := ethereum . Start ( ) ; err != nil {
t . Fatalf ( "error starting ethereum: %v" , err )
2015-03-15 13:43:48 +07:00
}
defer ethereum . Stop ( )
2015-04-22 23:11:11 +01:00
defer os . RemoveAll ( tmp )
2015-04-22 10:59:27 +02:00
checkEvalJSON ( t , repl , ` eth.mining ` , ` false ` )
2015-03-15 13:43:48 +07:00
}
func TestRPC ( t * testing . T ) {
2015-04-22 23:11:11 +01:00
tmp , repl , ethereum := testJEthRE ( t )
2015-04-22 10:59:27 +02:00
if err := ethereum . Start ( ) ; err != nil {
2015-03-15 13:43:48 +07:00
t . Errorf ( "error starting ethereum: %v" , err )
return
}
defer ethereum . Stop ( )
2015-04-22 23:11:11 +01:00
defer os . RemoveAll ( tmp )
2015-03-15 13:43:48 +07:00
2015-04-22 10:59:27 +02:00
checkEvalJSON ( t , repl , ` admin.startRPC("127.0.0.1", 5004) ` , ` true ` )
}
2015-04-22 23:11:11 +01:00
func TestCheckTestAccountBalance ( t * testing . T ) {
tmp , repl , ethereum := testJEthRE ( t )
if err := ethereum . Start ( ) ; err != nil {
t . Errorf ( "error starting ethereum: %v" , err )
return
}
defer ethereum . Stop ( )
defer os . RemoveAll ( tmp )
repl . re . Run ( ` primary = " ` + testAddress + ` " ` )
checkEvalJSON ( t , repl , ` eth.getBalance(primary) ` , ` " ` + testBalance + ` " ` )
}
2015-05-08 19:37:35 +02:00
func TestSignature ( t * testing . T ) {
tmp , repl , ethereum := testJEthRE ( t )
if err := ethereum . Start ( ) ; err != nil {
t . Errorf ( "error starting ethereum: %v" , err )
return
}
defer ethereum . Stop ( )
defer os . RemoveAll ( tmp )
val , err := repl . re . Run ( ` eth.sign( { from: " ` + testAddress + ` ", data: " ` + testHash + ` "}) ` )
// This is a very preliminary test, lacking actual signature verification
if err != nil {
t . Errorf ( "Error runnig js: %v" , err )
return
}
output := val . String ( )
t . Logf ( "Output: %v" , output )
regex := regexp . MustCompile ( ` ^0x[0-9a-f] { 130}$ ` )
if ! regex . MatchString ( output ) {
t . Errorf ( "Signature is not 65 bytes represented in hexadecimal." )
return
}
}
2015-04-22 23:11:11 +01:00
func TestContract ( t * testing . T ) {
tmp , repl , ethereum := testJEthRE ( t )
if err := ethereum . Start ( ) ; err != nil {
t . Errorf ( "error starting ethereum: %v" , err )
return
}
defer ethereum . Stop ( )
defer os . RemoveAll ( tmp )
var txc uint64
coinbase := common . HexToAddress ( testAddress )
resolver . New ( repl . xeth ) . CreateContracts ( coinbase )
source := ` contract test { \n ` +
" /// @notice Will multiply `a` by 7." + ` \n ` +
` function multiply(uint a) returns(uint d) { \n ` +
` return a * 7;\n ` +
` }\n ` +
` }\n `
checkEvalJSON ( t , repl , ` admin.contractInfo.stop() ` , ` true ` )
contractInfo , err := ioutil . ReadFile ( "info_test.json" )
if err != nil {
t . Fatalf ( "%v" , err )
}
checkEvalJSON ( t , repl , ` primary = eth.accounts[0] ` , ` " ` + testAddress + ` " ` )
checkEvalJSON ( t , repl , ` source = " ` + source + ` " ` , ` " ` + source + ` " ` )
2015-05-10 13:46:38 +02:00
// if solc is found with right version, test it, otherwise read from file
sol , err := compiler . New ( "" )
2015-04-22 23:11:11 +01:00
if err != nil {
t . Logf ( "solc not found: skipping compiler test" )
2015-05-10 13:46:38 +02:00
} else if sol . Version ( ) != solcVersion {
err = fmt . Errorf ( "solc wrong version found (%v, expect %v): skipping compiler test" , sol . Version ( ) , solcVersion )
t . Log ( err )
}
if err != nil {
2015-04-22 23:11:11 +01:00
info , err := ioutil . ReadFile ( "info_test.json" )
if err != nil {
t . Fatalf ( "%v" , err )
}
_ , err = repl . re . Run ( ` contract = JSON.parse( ` + strconv . Quote ( string ( info ) ) + ` ) ` )
if err != nil {
t . Errorf ( "%v" , err )
}
} else {
checkEvalJSON ( t , repl , ` contract = eth.compile.solidity(source) ` , string ( contractInfo ) )
}
2015-05-10 13:46:38 +02:00
2015-04-22 23:11:11 +01:00
checkEvalJSON ( t , repl , ` contract.code ` , ` "605280600c6000396000f3006000357c010000000000000000000000000000000000000000000000000000000090048063c6888fa114602e57005b60376004356041565b8060005260206000f35b6000600782029050604d565b91905056" ` )
checkEvalJSON (
t , repl ,
` contractaddress = eth.sendTransaction( { from: primary, data: contract.code }) ` ,
` "0x5dcaace5982778b409c524873b319667eba5d074" ` ,
)
callSetup := ` abiDef = JSON . parse ( ' [ { "constant" : false , "inputs" : [ { "name" : "a" , "type" : "uint256" } ] , "name" : "multiply" , "outputs" : [ { "name" : "d" , "type" : "uint256" } ] , "type" : "function" } ] ' ) ;
Multiply7 = eth . contract ( abiDef ) ;
multiply7 = new Multiply7 ( contractaddress ) ;
`
_ , err = repl . re . Run ( callSetup )
if err != nil {
t . Errorf ( "unexpected error registering, got %v" , err )
}
// updatespec
// why is this sometimes failing?
// checkEvalJSON(t, repl, `multiply7.multiply.call(6)`, `42`)
expNotice := ""
if repl . lastConfirm != expNotice {
t . Errorf ( "incorrect confirmation message: expected %v, got %v" , expNotice , repl . lastConfirm )
}
// why 0?
checkEvalJSON ( t , repl , ` eth.getBlock("pending", true).transactions.length ` , ` 0 ` )
txc , repl . xeth = repl . xeth . ApplyTestTxs ( repl . stateDb , coinbase , txc )
checkEvalJSON ( t , repl , ` admin.contractInfo.start() ` , ` true ` )
checkEvalJSON ( t , repl , ` multiply7.multiply.sendTransaction(6, { from: primary, gas: "1000000", gasPrice: "100000" }) ` , ` undefined ` )
expNotice = ` About to submit transaction (no NatSpec info found for contract: content hash not found for '0x4a6c99e127191d2ee302e42182c338344b39a37a47cdbb17ab0f26b6802eb4d1'): { "params":[ { "to":"0x5dcaace5982778b409c524873b319667eba5d074","data": "0xc6888fa10000000000000000000000000000000000000000000000000000000000000006"}]} `
if repl . lastConfirm != expNotice {
t . Errorf ( "incorrect confirmation message: expected %v, got %v" , expNotice , repl . lastConfirm )
}
checkEvalJSON ( t , repl , ` filename = "/tmp/info.json" ` , ` "/tmp/info.json" ` )
2015-05-10 13:46:38 +02:00
checkEvalJSON ( t , repl , ` contenthash = admin.contractInfo.register(primary, contractaddress, contract, filename) ` , ` "0x0d067e2dd99a4d8f0c0279738b17130dd415a89f24a23f0e7cf68c546ae3089d" ` )
2015-04-22 23:11:11 +01:00
checkEvalJSON ( t , repl , ` admin.contractInfo.registerUrl(primary, contenthash, "file://"+filename) ` , ` true ` )
if err != nil {
t . Errorf ( "unexpected error registering, got %v" , err )
}
checkEvalJSON ( t , repl , ` admin.contractInfo.start() ` , ` true ` )
// update state
txc , repl . xeth = repl . xeth . ApplyTestTxs ( repl . stateDb , coinbase , txc )
checkEvalJSON ( t , repl , ` multiply7.multiply.sendTransaction(6, { from: primary, gas: "1000000", gasPrice: "100000" }) ` , ` undefined ` )
expNotice = "Will multiply 6 by 7."
if repl . lastConfirm != expNotice {
t . Errorf ( "incorrect confirmation message: expected %v, got %v" , expNotice , repl . lastConfirm )
}
}
func checkEvalJSON ( t * testing . T , re * testjethre , expr , want string ) error {
2015-04-21 18:08:47 +01:00
val , err := re . re . Run ( "JSON.stringify(" + expr + ")" )
2015-04-22 10:59:27 +02:00
if err == nil && val . String ( ) != want {
err = fmt . Errorf ( "Output mismatch for `%s`:\ngot: %s\nwant: %s" , expr , val . String ( ) , want )
2015-03-15 13:43:48 +07:00
}
2015-04-22 10:59:27 +02:00
if err != nil {
_ , file , line , _ := runtime . Caller ( 1 )
2015-05-12 14:24:11 +02:00
file = filepath . Base ( file )
2015-04-22 10:59:27 +02:00
fmt . Printf ( "\t%s:%d: %v\n" , file , line , err )
t . Fail ( )
2015-03-15 13:43:48 +07:00
}
2015-04-22 10:59:27 +02:00
return err
2015-03-15 13:43:48 +07:00
}