2016-05-06 12:40:23 +03:00
|
|
|
// Copyright 2016 The go-ethereum Authors
|
|
|
|
// This file is part of go-ethereum.
|
|
|
|
//
|
|
|
|
// go-ethereum is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// go-ethereum is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with go-ethereum. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2016-06-08 13:53:07 +03:00
|
|
|
"crypto/rand"
|
|
|
|
"math/big"
|
2016-05-06 12:40:23 +03:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"runtime"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2016-11-25 18:55:06 +03:00
|
|
|
"github.com/ethereum/go-ethereum/params"
|
2017-04-12 17:27:23 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
ipcAPIs = "admin:1.0 debug:1.0 eth:1.0 miner:1.0 net:1.0 personal:1.0 rpc:1.0 shh:1.0 txpool:1.0 web3:1.0"
|
|
|
|
httpAPIs = "eth:1.0 net:1.0 rpc:1.0 web3:1.0"
|
2016-05-06 12:40:23 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
// Tests that a node embedded within a console can be started up properly and
|
|
|
|
// then terminated by closing the input stream.
|
|
|
|
func TestConsoleWelcome(t *testing.T) {
|
|
|
|
coinbase := "0x8605cdbbdb6d264aa742e77020dcbc58fcdce182"
|
|
|
|
|
|
|
|
// Start a geth console, make sure it's cleaned up and terminate the console
|
2016-06-02 23:33:57 +03:00
|
|
|
geth := runGeth(t,
|
|
|
|
"--port", "0", "--maxpeers", "0", "--nodiscover", "--nat", "none",
|
|
|
|
"--etherbase", coinbase, "--shh",
|
|
|
|
"console")
|
2016-05-06 12:40:23 +03:00
|
|
|
|
|
|
|
// Gather all the infos the welcome message needs to contain
|
|
|
|
geth.setTemplateFunc("goos", func() string { return runtime.GOOS })
|
2017-04-05 22:51:01 +03:00
|
|
|
geth.setTemplateFunc("goarch", func() string { return runtime.GOARCH })
|
2016-05-06 12:40:23 +03:00
|
|
|
geth.setTemplateFunc("gover", runtime.Version)
|
2016-11-25 18:55:06 +03:00
|
|
|
geth.setTemplateFunc("gethver", func() string { return params.Version })
|
2016-05-06 12:40:23 +03:00
|
|
|
geth.setTemplateFunc("niltime", func() string { return time.Unix(0, 0).Format(time.RFC1123) })
|
2017-04-12 17:27:23 +03:00
|
|
|
geth.setTemplateFunc("apis", func() string { return ipcAPIs })
|
2016-05-06 12:40:23 +03:00
|
|
|
|
|
|
|
// Verify the actual welcome message to the required template
|
|
|
|
geth.expect(`
|
|
|
|
Welcome to the Geth JavaScript console!
|
|
|
|
|
2017-04-05 22:51:01 +03:00
|
|
|
instance: Geth/v{{gethver}}/{{goos}}-{{goarch}}/{{gover}}
|
2016-05-06 12:40:23 +03:00
|
|
|
coinbase: {{.Etherbase}}
|
|
|
|
at block: 0 ({{niltime}})
|
|
|
|
datadir: {{.Datadir}}
|
2017-04-12 17:27:23 +03:00
|
|
|
modules: {{apis}}
|
2016-05-06 12:40:23 +03:00
|
|
|
|
2016-06-02 23:33:57 +03:00
|
|
|
> {{.InputLine "exit"}}
|
2016-05-06 12:40:23 +03:00
|
|
|
`)
|
2016-06-02 23:33:57 +03:00
|
|
|
geth.expectExit()
|
2016-05-06 12:40:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Tests that a console can be attached to a running node via various means.
|
|
|
|
func TestIPCAttachWelcome(t *testing.T) {
|
|
|
|
// Configure the instance for IPC attachement
|
|
|
|
coinbase := "0x8605cdbbdb6d264aa742e77020dcbc58fcdce182"
|
|
|
|
var ipc string
|
|
|
|
if runtime.GOOS == "windows" {
|
2016-06-08 13:53:07 +03:00
|
|
|
ipc = `\\.\pipe\geth` + strconv.Itoa(trulyRandInt(100000, 999999))
|
2016-05-06 12:40:23 +03:00
|
|
|
} else {
|
|
|
|
ws := tmpdir(t)
|
|
|
|
defer os.RemoveAll(ws)
|
|
|
|
ipc = filepath.Join(ws, "geth.ipc")
|
|
|
|
}
|
2016-06-02 23:33:57 +03:00
|
|
|
// Note: we need --shh because testAttachWelcome checks for default
|
|
|
|
// list of ipc modules and shh is included there.
|
|
|
|
geth := runGeth(t,
|
|
|
|
"--port", "0", "--maxpeers", "0", "--nodiscover", "--nat", "none",
|
|
|
|
"--etherbase", coinbase, "--shh", "--ipcpath", ipc)
|
2016-05-06 12:40:23 +03:00
|
|
|
|
|
|
|
time.Sleep(2 * time.Second) // Simple way to wait for the RPC endpoint to open
|
2017-04-12 17:27:23 +03:00
|
|
|
testAttachWelcome(t, geth, "ipc:"+ipc, ipcAPIs)
|
2016-06-02 23:33:57 +03:00
|
|
|
|
|
|
|
geth.interrupt()
|
|
|
|
geth.expectExit()
|
2016-05-06 12:40:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestHTTPAttachWelcome(t *testing.T) {
|
|
|
|
coinbase := "0x8605cdbbdb6d264aa742e77020dcbc58fcdce182"
|
2016-06-08 13:53:07 +03:00
|
|
|
port := strconv.Itoa(trulyRandInt(1024, 65536)) // Yeah, sometimes this will fail, sorry :P
|
2016-06-02 23:33:57 +03:00
|
|
|
geth := runGeth(t,
|
|
|
|
"--port", "0", "--maxpeers", "0", "--nodiscover", "--nat", "none",
|
|
|
|
"--etherbase", coinbase, "--rpc", "--rpcport", port)
|
2016-05-06 12:40:23 +03:00
|
|
|
|
|
|
|
time.Sleep(2 * time.Second) // Simple way to wait for the RPC endpoint to open
|
2017-04-12 17:27:23 +03:00
|
|
|
testAttachWelcome(t, geth, "http://localhost:"+port, httpAPIs)
|
2016-06-02 23:33:57 +03:00
|
|
|
|
|
|
|
geth.interrupt()
|
|
|
|
geth.expectExit()
|
2016-05-06 12:40:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestWSAttachWelcome(t *testing.T) {
|
|
|
|
coinbase := "0x8605cdbbdb6d264aa742e77020dcbc58fcdce182"
|
2016-06-08 13:53:07 +03:00
|
|
|
port := strconv.Itoa(trulyRandInt(1024, 65536)) // Yeah, sometimes this will fail, sorry :P
|
2016-05-06 12:40:23 +03:00
|
|
|
|
2016-06-02 23:33:57 +03:00
|
|
|
geth := runGeth(t,
|
|
|
|
"--port", "0", "--maxpeers", "0", "--nodiscover", "--nat", "none",
|
|
|
|
"--etherbase", coinbase, "--ws", "--wsport", port)
|
2016-05-06 12:40:23 +03:00
|
|
|
|
|
|
|
time.Sleep(2 * time.Second) // Simple way to wait for the RPC endpoint to open
|
2017-04-12 17:27:23 +03:00
|
|
|
testAttachWelcome(t, geth, "ws://localhost:"+port, httpAPIs)
|
2016-06-02 23:33:57 +03:00
|
|
|
|
|
|
|
geth.interrupt()
|
|
|
|
geth.expectExit()
|
2016-05-06 12:40:23 +03:00
|
|
|
}
|
|
|
|
|
2017-04-12 17:27:23 +03:00
|
|
|
func testAttachWelcome(t *testing.T, geth *testgeth, endpoint, apis string) {
|
2016-05-06 12:40:23 +03:00
|
|
|
// Attach to a running geth note and terminate immediately
|
|
|
|
attach := runGeth(t, "attach", endpoint)
|
|
|
|
defer attach.expectExit()
|
|
|
|
attach.stdin.Close()
|
|
|
|
|
|
|
|
// Gather all the infos the welcome message needs to contain
|
|
|
|
attach.setTemplateFunc("goos", func() string { return runtime.GOOS })
|
2017-04-05 22:51:01 +03:00
|
|
|
attach.setTemplateFunc("goarch", func() string { return runtime.GOARCH })
|
2016-05-06 12:40:23 +03:00
|
|
|
attach.setTemplateFunc("gover", runtime.Version)
|
2016-11-25 18:55:06 +03:00
|
|
|
attach.setTemplateFunc("gethver", func() string { return params.Version })
|
2016-05-06 12:40:23 +03:00
|
|
|
attach.setTemplateFunc("etherbase", func() string { return geth.Etherbase })
|
|
|
|
attach.setTemplateFunc("niltime", func() string { return time.Unix(0, 0).Format(time.RFC1123) })
|
|
|
|
attach.setTemplateFunc("ipc", func() bool { return strings.HasPrefix(endpoint, "ipc") })
|
|
|
|
attach.setTemplateFunc("datadir", func() string { return geth.Datadir })
|
2017-04-12 17:27:23 +03:00
|
|
|
attach.setTemplateFunc("apis", func() string { return apis })
|
2016-05-06 12:40:23 +03:00
|
|
|
|
|
|
|
// Verify the actual welcome message to the required template
|
|
|
|
attach.expect(`
|
|
|
|
Welcome to the Geth JavaScript console!
|
|
|
|
|
2017-04-05 22:51:01 +03:00
|
|
|
instance: Geth/v{{gethver}}/{{goos}}-{{goarch}}/{{gover}}
|
2016-05-06 12:40:23 +03:00
|
|
|
coinbase: {{etherbase}}
|
|
|
|
at block: 0 ({{niltime}}){{if ipc}}
|
|
|
|
datadir: {{datadir}}{{end}}
|
2017-04-12 17:27:23 +03:00
|
|
|
modules: {{apis}}
|
2016-05-06 12:40:23 +03:00
|
|
|
|
2016-06-02 23:33:57 +03:00
|
|
|
> {{.InputLine "exit" }}
|
2016-05-06 12:40:23 +03:00
|
|
|
`)
|
2016-06-02 23:33:57 +03:00
|
|
|
attach.expectExit()
|
2016-05-06 12:40:23 +03:00
|
|
|
}
|
2016-06-08 13:53:07 +03:00
|
|
|
|
|
|
|
// trulyRandInt generates a crypto random integer used by the console tests to
|
|
|
|
// not clash network ports with other tests running cocurrently.
|
|
|
|
func trulyRandInt(lo, hi int) int {
|
|
|
|
num, _ := rand.Int(rand.Reader, big.NewInt(int64(hi-lo)))
|
|
|
|
return int(num.Int64()) + lo
|
|
|
|
}
|