2015-04-03 15:37:59 +00:00
package natspec
import (
"fmt"
"io/ioutil"
"math/big"
"os"
"testing"
"github.com/ethereum/go-ethereum/accounts"
"github.com/ethereum/go-ethereum/common"
2015-04-06 06:01:36 +00:00
"github.com/ethereum/go-ethereum/common/docserver"
2015-04-07 09:50:17 +00:00
"github.com/ethereum/go-ethereum/common/natspec"
2015-04-03 15:37:59 +00:00
"github.com/ethereum/go-ethereum/common/resolver"
"github.com/ethereum/go-ethereum/core"
2015-04-07 09:50:17 +00:00
//"github.com/ethereum/go-ethereum/core/state"
2015-04-03 15:37:59 +00:00
//"github.com/ethereum/go-ethereum/core/types"
"github.com/ethereum/go-ethereum/crypto"
"github.com/ethereum/go-ethereum/eth"
2015-04-07 09:50:17 +00:00
"github.com/ethereum/go-ethereum/rpc"
2015-04-03 15:37:59 +00:00
xe "github.com/ethereum/go-ethereum/xeth"
)
type testFrontend struct {
2015-04-06 06:01:36 +00:00
t * testing . T
ethereum * eth . Ethereum
xeth * xe . XEth
2015-04-07 09:50:17 +00:00
api * rpc . EthereumApi
2015-04-06 06:01:36 +00:00
coinbase string
lastConfirm string
2015-04-07 09:50:17 +00:00
makeNatSpec bool
2015-04-03 15:37:59 +00:00
}
2015-04-13 12:35:41 +00:00
const testNotice = "Register key `utils.toHex(_key)` <- content `utils.toHex(_content)`"
const testExpNotice = "Register key 0xadd1a7d961cff0242089674ec2ef6fca671ab15e1fe80e38859fc815b98d88ab <- content 0xc00d5bcc872e17813df6ec5c646bb281a6e2d3b454c2c400c78192adf3344af9"
const testExpNotice2 = ` About to submit transaction (NatSpec notice error "abi key %!x(MISSING) does not match any method %!v(MISSING)"): { "id":6,"jsonrpc":"2.0","method":"eth_transact","params":[ { "from":"0xe273f01c99144c438695e10f24926dc1f9fbf62d","to":"0x0000000000000000000000000000000000000009","value":"100000000000","gas":"100000","gasPrice":"100000","data":"0x31e12c20"}]} `
const testExpNotice3 = ` About to submit transaction (no NatSpec info found for contract): { "id":6,"jsonrpc":"2.0","method":"eth_transact","params":[ { "from":"0xe273f01c99144c438695e10f24926dc1f9fbf62d","to":"0x0000000000000000000000000000000000000008","value":"100000000000","gas":"100000","gasPrice":"100000","data":"0xd66d6c10c00d5bcc872e17813df6ec5c646bb281a6e2d3b454c2c400c78192adf3344af900000000000000000000000066696c653a2f2f2f746573742e636f6e74656e74"}]} `
2015-04-07 09:50:17 +00:00
const testUserDoc = `
{
"source" : "..." ,
"language" : "Solidity" ,
"languageVersion" : 1 ,
"methods" : {
"register(uint256,uint256)" : {
"notice" : "` + testNotice + `"
}
} ,
"invariants" : [
{ "notice" : "" }
] ,
"construction" : [
{ "notice" : "" }
]
}
`
const testABI = `
[ {
"name" : "register" ,
"constant" : false ,
"type" : "function" ,
"inputs" : [ {
"name" : "_key" ,
"type" : "uint256"
} , {
"name" : "_content" ,
"type" : "uint256"
} ] ,
"outputs" : [ ]
} ]
`
const testDocs = `
{
"userdoc" : ` + testUserDoc + ` ,
"abi" : ` + testABI + `
}
`
2015-04-03 15:37:59 +00:00
func ( f * testFrontend ) UnlockAccount ( acc [ ] byte ) bool {
f . t . Logf ( "Unlocking account %v\n" , common . Bytes2Hex ( acc ) )
f . ethereum . AccountManager ( ) . Unlock ( acc , "password" )
return true
}
2015-04-07 09:50:17 +00:00
func ( f * testFrontend ) ConfirmTransaction ( tx string ) bool {
//f.t.Logf("ConfirmTransaction called tx = %v", tx)
if f . makeNatSpec {
ds , err := docserver . New ( "/tmp/" )
if err != nil {
f . t . Errorf ( "Error creating DocServer: %v" , err )
}
2015-04-08 10:35:02 +00:00
f . lastConfirm = natspec . GetNotice ( f . xeth , tx , ds )
2015-04-07 09:50:17 +00:00
}
2015-04-06 06:01:36 +00:00
return true
}
2015-04-03 15:37:59 +00:00
var port = 30300
2015-04-06 06:01:36 +00:00
func testEth ( t * testing . T ) ( ethereum * eth . Ethereum , err error ) {
2015-04-03 15:37:59 +00:00
os . RemoveAll ( "/tmp/eth/" )
err = os . MkdirAll ( "/tmp/eth/keys/e273f01c99144c438695e10f24926dc1f9fbf62d/" , os . ModePerm )
if err != nil {
t . Errorf ( "%v" , err )
return
}
err = os . MkdirAll ( "/tmp/eth/data" , os . ModePerm )
if err != nil {
t . Errorf ( "%v" , err )
return
}
ks := crypto . NewKeyStorePlain ( "/tmp/eth/keys" )
ioutil . WriteFile ( "/tmp/eth/keys/e273f01c99144c438695e10f24926dc1f9fbf62d/e273f01c99144c438695e10f24926dc1f9fbf62d" ,
[ ] byte ( ` { "Id":"RhRXD+fNRKS4jx+7ZfEsNA==","Address":"4nPwHJkUTEOGleEPJJJtwfn79i0=","PrivateKey":"h4ACVpe74uIvi5Cg/2tX/Yrm2xdr3J7QoMbMtNX2CNc="} ` ) , os . ModePerm )
port ++
ethereum , err = eth . New ( & eth . Config {
DataDir : "/tmp/eth" ,
AccountManager : accounts . NewManager ( ks ) ,
Port : fmt . Sprintf ( "%d" , port ) ,
MaxPeers : 10 ,
Name : "test" ,
} )
if err != nil {
t . Errorf ( "%v" , err )
return
}
return
}
2015-04-06 06:01:36 +00:00
func testInit ( t * testing . T ) ( self * testFrontend ) {
ethereum , err := testEth ( t )
if err != nil {
t . Errorf ( "error creating jsre, got %v" , err )
return
}
err = ethereum . Start ( )
if err != nil {
t . Errorf ( "error starting ethereum: %v" , err )
return
}
self = & testFrontend { t : t , ethereum : ethereum }
self . xeth = xe . New ( ethereum , self )
2015-04-07 09:50:17 +00:00
self . api = rpc . NewEthereumApi ( self . xeth )
2015-04-06 06:01:36 +00:00
addr := self . xeth . Coinbase ( )
self . coinbase = addr
if addr != "0x" + core . TestAccount {
t . Errorf ( "CoinBase %v does not match TestAccount 0x%v" , addr , core . TestAccount )
}
t . Logf ( "CoinBase is %v" , addr )
balance := self . xeth . BalanceAt ( core . TestAccount )
2015-04-13 12:35:41 +00:00
/ * if balance != core . TestBalance {
2015-04-06 06:01:36 +00:00
t . Errorf ( "Balance %v does not match TestBalance %v" , balance , core . TestBalance )
2015-04-13 12:35:41 +00:00
} * /
2015-04-06 06:01:36 +00:00
t . Logf ( "Balance is %v" , balance )
return
}
2015-04-03 15:37:59 +00:00
func ( self * testFrontend ) insertTx ( addr , contract , fnsig string , args [ ] string ) {
2015-04-07 12:46:14 +00:00
//cb := common.HexToAddress(self.coinbase)
//coinbase := self.ethereum.ChainManager().State().GetStateObject(cb)
2015-04-03 15:37:59 +00:00
hash := common . Bytes2Hex ( crypto . Sha3 ( [ ] byte ( fnsig ) ) )
data := "0x" + hash [ 0 : 8 ]
for _ , arg := range args {
data = data + common . Bytes2Hex ( common . Hex2BytesFixed ( arg , 32 ) )
}
self . t . Logf ( "Tx data: %v" , data )
2015-04-07 09:50:17 +00:00
jsontx := `
[ {
"from" : "` + addr + `" ,
"to" : "0x` + contract + `" ,
"value" : "100000000000" ,
"gas" : "100000" ,
"gasPrice" : "100000" ,
"data" : "` + data + `"
} ]
`
req := & rpc . RpcRequest {
Jsonrpc : "2.0" ,
Method : "eth_transact" ,
Params : [ ] byte ( jsontx ) ,
2015-04-07 12:46:14 +00:00
Id : 6 ,
2015-04-07 09:50:17 +00:00
}
var reply interface { }
err0 := self . api . GetRequestReply ( req , & reply )
if err0 != nil {
self . t . Errorf ( "GetRequestReply error: %v" , err0 )
}
//self.xeth.Transact(addr, contract, "100000000000", "100000", "100000", data)
}
func ( self * testFrontend ) applyTxs ( ) {
2015-04-03 15:37:59 +00:00
2015-04-07 09:50:17 +00:00
cb := common . HexToAddress ( self . coinbase )
stateDb := self . ethereum . ChainManager ( ) . State ( ) . Copy ( )
2015-04-03 15:37:59 +00:00
block := self . ethereum . ChainManager ( ) . NewBlock ( cb )
2015-04-07 09:50:17 +00:00
coinbase := stateDb . GetStateObject ( cb )
coinbase . SetGasPool ( big . NewInt ( 1000000 ) )
2015-04-03 15:37:59 +00:00
txs := self . ethereum . TxPool ( ) . GetTransactions ( )
2015-04-07 12:46:14 +00:00
for i := 0 ; i < len ( txs ) ; i ++ {
for _ , tx := range txs {
if tx . Nonce ( ) == uint64 ( i ) {
_ , gas , err := core . ApplyMessage ( core . NewEnv ( stateDb , self . ethereum . ChainManager ( ) , tx , block ) , tx , coinbase )
//self.ethereum.TxPool().RemoveSet([]*types.Transaction{tx})
self . t . Logf ( "ApplyMessage: gas %v err %v" , gas , err )
}
}
2015-04-07 09:50:17 +00:00
}
2015-04-03 15:37:59 +00:00
self . ethereum . TxPool ( ) . RemoveSet ( txs )
2015-04-07 09:50:17 +00:00
self . xeth = self . xeth . WithState ( stateDb )
2015-04-03 15:37:59 +00:00
}
2015-04-06 06:01:36 +00:00
func ( self * testFrontend ) registerURL ( hash common . Hash , url string ) {
hashHex := common . Bytes2Hex ( hash [ : ] )
urlHex := common . Bytes2Hex ( [ ] byte ( url ) )
2015-04-07 09:50:17 +00:00
self . insertTx ( self . coinbase , core . ContractAddrURLhint , "register(uint256,uint256)" , [ ] string { hashHex , urlHex } )
}
func ( self * testFrontend ) setOwner ( ) {
self . insertTx ( self . coinbase , core . ContractAddrHashReg , "setowner()" , [ ] string { } )
/ * owner := self . xeth . StorageAt ( "0x" + core . ContractAddrHashReg , "0x0000000000000000000000000000000000000000000000000000000000000000" )
self . t . Logf ( "owner = %v" , owner )
if owner != self . coinbase {
self . t . Errorf ( "setowner() unsuccessful, owner != coinbase" )
} * /
}
func ( self * testFrontend ) registerNatSpec ( codehash , dochash common . Hash ) {
codeHex := common . Bytes2Hex ( codehash [ : ] )
docHex := common . Bytes2Hex ( dochash [ : ] )
self . insertTx ( self . coinbase , core . ContractAddrHashReg , "register(uint256,uint256)" , [ ] string { codeHex , docHex } )
2015-04-06 06:01:36 +00:00
}
2015-04-03 15:37:59 +00:00
2015-04-06 06:01:36 +00:00
func ( self * testFrontend ) testResolver ( ) * resolver . Resolver {
2015-04-07 09:50:17 +00:00
return resolver . New ( self . xeth , resolver . URLHintContractAddress , resolver . HashRegContractAddress )
2015-04-06 06:01:36 +00:00
}
2015-04-03 15:37:59 +00:00
2015-04-06 06:01:36 +00:00
func TestNatspecE2E ( t * testing . T ) {
2015-04-03 15:37:59 +00:00
2015-04-06 06:01:36 +00:00
tf := testInit ( t )
defer tf . ethereum . Stop ( )
2015-04-03 15:37:59 +00:00
2015-04-07 09:50:17 +00:00
ioutil . WriteFile ( "/tmp/test.content" , [ ] byte ( testDocs ) , os . ModePerm )
dochash := common . BytesToHash ( crypto . Sha3 ( [ ] byte ( testDocs ) ) )
2015-04-03 15:37:59 +00:00
2015-04-07 09:50:17 +00:00
codehex := tf . xeth . CodeAt ( core . ContractAddrHashReg )
2015-04-08 10:35:02 +00:00
codehash := common . BytesToHash ( crypto . Sha3 ( common . Hex2Bytes ( codehex [ 2 : ] ) ) )
2015-04-03 15:37:59 +00:00
2015-04-07 09:50:17 +00:00
tf . setOwner ( )
tf . registerNatSpec ( codehash , dochash )
tf . registerURL ( dochash , "file:///test.content" )
tf . applyTxs ( )
2015-04-03 15:37:59 +00:00
2015-04-07 09:50:17 +00:00
chash , err := tf . testResolver ( ) . KeyToContentHash ( codehash )
if err != nil {
t . Errorf ( "Can't find content hash" )
}
t . Logf ( "chash = %x err = %v" , chash , err )
url , err2 := tf . testResolver ( ) . ContentHashToUrl ( dochash )
if err2 != nil {
t . Errorf ( "Can't find URL hint" )
}
t . Logf ( "url = %v err = %v" , url , err2 )
2015-04-08 10:35:02 +00:00
// NatSpec info for register method of HashReg contract installed
// now using the same transactions to check confirm messages
2015-04-07 09:50:17 +00:00
tf . makeNatSpec = true
2015-04-08 10:35:02 +00:00
tf . registerNatSpec ( codehash , dochash )
2015-04-07 09:50:17 +00:00
t . Logf ( "Confirm message: %v\n" , tf . lastConfirm )
if tf . lastConfirm != testExpNotice {
t . Errorf ( "Wrong confirm message, expected '%v', got '%v'" , testExpNotice , tf . lastConfirm )
}
2015-04-03 15:37:59 +00:00
2015-04-08 10:35:02 +00:00
tf . setOwner ( )
t . Logf ( "Confirm message for unknown method: %v\n" , tf . lastConfirm )
if tf . lastConfirm != testExpNotice2 {
t . Errorf ( "Wrong confirm message, expected '%v', got '%v'" , testExpNotice2 , tf . lastConfirm )
}
tf . registerURL ( dochash , "file:///test.content" )
t . Logf ( "Confirm message for unknown contract: %v\n" , tf . lastConfirm )
if tf . lastConfirm != testExpNotice3 {
t . Errorf ( "Wrong confirm message, expected '%v', got '%v'" , testExpNotice3 , tf . lastConfirm )
}
2015-04-03 15:37:59 +00:00
}