consolecmd_test.go 5.74 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
// 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 (
20 21
	"crypto/rand"
	"math/big"
22 23 24 25 26 27 28 29
	"os"
	"path/filepath"
	"runtime"
	"strconv"
	"strings"
	"testing"
	"time"

30
	"github.com/ethereum/go-ethereum/params"
31 32 33
)

const (
34
	ipcAPIs  = "admin:1.0 debug:1.0 eth:1.0 ethash:1.0 miner:1.0 net:1.0 personal:1.0 rpc:1.0 txpool:1.0 web3:1.0"
35
	httpAPIs = "eth:1.0 net:1.0 rpc:1.0 web3:1.0"
36 37
)

38 39 40 41 42 43 44
// spawns geth with the given command line args, using a set of flags to minimise
// memory and disk IO. If the args don't set --datadir, the
// child g gets a temporary data directory.
func runMinimalGeth(t *testing.T, args ...string) *testgeth {
	// --ropsten to make the 'writing genesis to disk' faster (no accounts)
	// --networkid=1337 to avoid cache bump
	// --syncmode=full to avoid allocating fast sync bloom
45
	allArgs := []string{"--ropsten", "--networkid", "1337", "--syncmode=full", "--port", "0",
46 47 48 49
		"--nat", "none", "--nodiscover", "--maxpeers", "0", "--cache", "64"}
	return runGeth(t, append(allArgs, args...)...)
}

50 51 52 53 54 55
// 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
56
	geth := runMinimalGeth(t, "--miner.etherbase", coinbase, "console")
57 58

	// Gather all the infos the welcome message needs to contain
59 60 61
	geth.SetTemplateFunc("goos", func() string { return runtime.GOOS })
	geth.SetTemplateFunc("goarch", func() string { return runtime.GOARCH })
	geth.SetTemplateFunc("gover", runtime.Version)
62
	geth.SetTemplateFunc("gethver", func() string { return params.VersionWithCommit("", "") })
63 64 65
	geth.SetTemplateFunc("niltime", func() string {
		return time.Unix(0, 0).Format("Mon Jan 02 2006 15:04:05 GMT-0700 (MST)")
	})
66
	geth.SetTemplateFunc("apis", func() string { return ipcAPIs })
67 68

	// Verify the actual welcome message to the required template
69
	geth.Expect(`
70 71
Welcome to the Geth JavaScript console!

72
instance: Geth/v{{gethver}}/{{goos}}-{{goarch}}/{{gover}}
73 74 75
coinbase: {{.Etherbase}}
at block: 0 ({{niltime}})
 datadir: {{.Datadir}}
76
 modules: {{apis}}
77

78
To exit, press ctrl-d or type exit
79
> {{.InputLine "exit"}}
80
`)
81
	geth.ExpectExit()
82 83 84
}

// Tests that a console can be attached to a running node via various means.
85 86 87 88 89 90
func TestAttachWelcome(t *testing.T) {
	var (
		ipc      string
		httpPort string
		wsPort   string
	)
91
	// Configure the instance for IPC attachment
92
	if runtime.GOOS == "windows" {
93
		ipc = `\\.\pipe\geth` + strconv.Itoa(trulyRandInt(100000, 999999))
94 95 96 97 98
	} else {
		ws := tmpdir(t)
		defer os.RemoveAll(ws)
		ipc = filepath.Join(ws, "geth.ipc")
	}
99 100 101 102
	// And HTTP + WS attachment
	p := trulyRandInt(1024, 65533) // Yeah, sometimes this will fail, sorry :P
	httpPort = strconv.Itoa(p)
	wsPort = strconv.Itoa(p + 1)
103
	geth := runMinimalGeth(t, "--miner.etherbase", "0x8605cdbbdb6d264aa742e77020dcbc58fcdce182",
104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120
		"--ipcpath", ipc,
		"--http", "--http.port", httpPort,
		"--ws", "--ws.port", wsPort)
	t.Run("ipc", func(t *testing.T) {
		waitForEndpoint(t, ipc, 3*time.Second)
		testAttachWelcome(t, geth, "ipc:"+ipc, ipcAPIs)
	})
	t.Run("http", func(t *testing.T) {
		endpoint := "http://127.0.0.1:" + httpPort
		waitForEndpoint(t, endpoint, 3*time.Second)
		testAttachWelcome(t, geth, endpoint, httpAPIs)
	})
	t.Run("ws", func(t *testing.T) {
		endpoint := "ws://127.0.0.1:" + wsPort
		waitForEndpoint(t, endpoint, 3*time.Second)
		testAttachWelcome(t, geth, endpoint, httpAPIs)
	})
121 122
}

123
func testAttachWelcome(t *testing.T, geth *testgeth, endpoint, apis string) {
124 125
	// Attach to a running geth note and terminate immediately
	attach := runGeth(t, "attach", endpoint)
126 127
	defer attach.ExpectExit()
	attach.CloseStdin()
128 129

	// Gather all the infos the welcome message needs to contain
130 131 132
	attach.SetTemplateFunc("goos", func() string { return runtime.GOOS })
	attach.SetTemplateFunc("goarch", func() string { return runtime.GOARCH })
	attach.SetTemplateFunc("gover", runtime.Version)
133
	attach.SetTemplateFunc("gethver", func() string { return params.VersionWithCommit("", "") })
134
	attach.SetTemplateFunc("etherbase", func() string { return geth.Etherbase })
135 136 137
	attach.SetTemplateFunc("niltime", func() string {
		return time.Unix(0, 0).Format("Mon Jan 02 2006 15:04:05 GMT-0700 (MST)")
	})
138 139 140
	attach.SetTemplateFunc("ipc", func() bool { return strings.HasPrefix(endpoint, "ipc") })
	attach.SetTemplateFunc("datadir", func() string { return geth.Datadir })
	attach.SetTemplateFunc("apis", func() string { return apis })
141 142

	// Verify the actual welcome message to the required template
143
	attach.Expect(`
144 145
Welcome to the Geth JavaScript console!

146
instance: Geth/v{{gethver}}/{{goos}}-{{goarch}}/{{gover}}
147 148 149
coinbase: {{etherbase}}
at block: 0 ({{niltime}}){{if ipc}}
 datadir: {{datadir}}{{end}}
150
 modules: {{apis}}
151

152
To exit, press ctrl-d or type exit
153
> {{.InputLine "exit" }}
154
`)
155
	attach.ExpectExit()
156
}
157 158 159 160 161 162 163

// 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
}