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-18 15:31:26 +01:00
solcVersion = "0.9.23"
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 (
2015-05-18 15:31:26 +01:00
versionRE = regexp . MustCompile ( strconv . Quote ( ` "compilerVersion":" ` + solcVersion + ` " ` ) )
2015-05-26 19:35:31 +03:00
testNodeKey = crypto . ToECDSA ( common . Hex2Bytes ( "4b50fa71f5c3eeb8fdc452224b2395af2fcc3d125e06c32c82e048c0559db03f" ) )
2015-04-22 23:11:11 +01:00
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 {
2015-05-12 17:48:21 +02:00
err := self . ethereum . AccountManager ( ) . Unlock ( common . BytesToAddress ( acc ) , "" )
2015-04-22 23:11:11 +01:00
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 )
2015-05-14 17:51:05 +02:00
ks := crypto . NewKeyStorePlain ( filepath . Join ( tmp , "keystore" ) )
2015-04-22 23:11:11 +01:00
am := accounts . NewManager ( ks )
2015-04-22 10:59:27 +02:00
ethereum , err := eth . New ( & eth . Config {
2015-05-26 19:35:31 +03:00
NodeKey : testNodeKey ,
2015-04-22 10:59:27 +02:00
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" ,
2015-05-20 02:04:52 +01:00
SolcPath : testSolcPath ,
2015-03-15 13:43:48 +07:00
} )
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 ( ) }
2015-05-20 02:04:52 +01:00
repl := newJSRE ( ethereum , assetPath , "" , false , tf )
2015-04-22 23:11:11 +01:00
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-05-26 19:35:31 +03:00
want := ` { "DiscPort":0,"IP":"0.0.0.0","ListenAddr":"","Name":"test","NodeID":"4cb2fc32924e94277bf94b5e4c983beedb2eabd5a0bc941db32202735c6625d020ca14a5963d1738af43b6ac0a711d61b1a06de931a499fe2aa0b1a132a902b5","NodeUrl":"enode://4cb2fc32924e94277bf94b5e4c983beedb2eabd5a0bc941db32202735c6625d020ca14a5963d1738af43b6ac0a711d61b1a06de931a499fe2aa0b1a132a902b5@0.0.0.0:0","TCPPort":0,"Td":"131072"} `
2015-04-22 10:59:27 +02:00
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-05-18 16:04:10 +02:00
ethereum . ChainManager ( ) . Reset ( )
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 )
2015-05-20 04:38:20 +01:00
val , err := repl . re . Run ( ` eth.sign(" ` + testAddress + ` ", " ` + testHash + ` ") ` )
2015-05-08 19:37:35 +02:00
// This is a very preliminary test, lacking actual signature verification
if err != nil {
2015-05-20 04:11:48 +01:00
t . Errorf ( "Error running js: %v" , err )
2015-05-08 19:37:35 +02:00
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 )
2015-05-18 15:31:26 +01:00
// time.Sleep(1000 * time.Millisecond)
2015-04-22 23:11:11 +01:00
2015-05-18 15:31:26 +01:00
// checkEvalJSON(t, repl, `eth.getBlock("pending", true).transactions.length`, `2`)
2015-04-22 23:11:11 +01:00
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 {
2015-05-18 15:31:26 +01:00
t . Logf ( "solc not found: mocking contract compilation step" )
2015-05-10 13:46:38 +02:00
} else if sol . Version ( ) != solcVersion {
2015-05-18 15:31:26 +01:00
t . Logf ( "WARNING: solc different version found (%v, test written for %v, may need to update)" , sol . Version ( ) , solcVersion )
2015-05-10 13:46:38 +02:00
}
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 {
2015-05-20 04:11:48 +01:00
checkEvalJSON ( t , repl , ` contract = eth.compile.solidity(source).test ` , string ( contractInfo ) )
2015-04-22 23:11:11 +01:00
}
2015-05-10 13:46:38 +02:00
2015-05-18 15:31:26 +01:00
checkEvalJSON ( t , repl , ` contract.code ` , ` "0x605880600c6000396000f3006000357c010000000000000000000000000000000000000000000000000000000090048063c6888fa114602e57005b603d6004803590602001506047565b8060005260206000f35b60006007820290506053565b91905056" ` )
2015-04-22 23:11:11 +01:00
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 ) ;
2015-05-20 04:11:48 +01:00
multiply7 = Multiply7 . at ( contractaddress ) ;
2015-04-22 23:11:11 +01:00
`
2015-05-18 15:31:26 +01:00
// time.Sleep(1500 * time.Millisecond)
2015-04-22 23:11:11 +01:00
_ , err = repl . re . Run ( callSetup )
if err != nil {
2015-05-18 15:31:26 +01:00
t . Errorf ( "unexpected error setting up contract, got %v" , err )
2015-04-22 23:11:11 +01:00
}
2015-05-18 15:31:26 +01:00
// checkEvalJSON(t, repl, `eth.getBlock("pending", true).transactions.length`, `3`)
2015-04-22 23:11:11 +01:00
// 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 )
}
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 ` )
2015-05-18 15:31:26 +01:00
expNotice = ` About to submit transaction (no NatSpec info found for contract: content hash not found for '0x87e2802265838c7f14bb69eecd2112911af6767907a702eeaa445239fb20711b'): { "params":[ { "to":"0x5dcaace5982778b409c524873b319667eba5d074","data": "0xc6888fa10000000000000000000000000000000000000000000000000000000000000006"}]} `
2015-04-22 23:11:11 +01:00
if repl . lastConfirm != expNotice {
t . Errorf ( "incorrect confirmation message: expected %v, got %v" , expNotice , repl . lastConfirm )
}
2015-05-18 15:31:26 +01:00
var contenthash = ` "0x86d2b7cf1e72e9a7a3f8d96601f0151742a2f780f1526414304fbe413dc7f9bd" `
if sol != nil {
modContractInfo := versionRE . ReplaceAll ( contractInfo , [ ] byte ( ` "compilerVersion":" ` + sol . Version ( ) + ` " ` ) )
_ = modContractInfo
// contenthash = crypto.Sha3(modContractInfo)
}
2015-04-22 23:11:11 +01:00
checkEvalJSON ( t , repl , ` filename = "/tmp/info.json" ` , ` "/tmp/info.json" ` )
2015-05-18 15:31:26 +01:00
checkEvalJSON ( t , repl , ` contenthash = admin.contractInfo.register(primary, contractaddress, contract, filename) ` , contenthash )
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
}