package debug_test import ( "bufio" "fmt" "io/ioutil" "net" "os" "strconv" "strings" "sync" "testing" "github.com/status-im/status-go/api" "github.com/status-im/status-go/cmd/statusd/debug" "github.com/status-im/status-go/params" testifyAssert "github.com/stretchr/testify/assert" ) const stopNodeCommandLine = "StopNode()" // TestInvalidExpressions tests invalid expressions. func TestInvalidExpressions(t *testing.T) { assert := testifyAssert.New(t) startDebugging(assert) conn := connectDebug(assert) tests := []struct { commandLine string replies []string }{ { commandLine: "", replies: []string{"[0] cannot read command: 1:2: expected operand, found 'EOF'"}, }, { commandLine: "1 + 1", replies: []string{"[0] cannot read command: invalid command line: \"1 + 1\\n\""}, }, { commandLine: "func() { panic(42) }", replies: []string{"[0] cannot read command: invalid command line: \"func() { panic(42) }\\n\""}, }, { commandLine: "DoesNotExist()", replies: []string{"[0] cannot execute command: command \"DoesNotExist\" not found"}, }, { commandLine: "node.Start()", replies: []string{"[0] cannot read command: invalid expression: \"node.Start()\\n\""}, }, } for _, test := range tests { replies := sendCommandLine(assert, conn, test.commandLine) assert.Equal(test.replies, replies) } } // TestStartStopNode tests starting and stopping a node remotely. func TestStartStopNode(t *testing.T) { assert := testifyAssert.New(t) configJSON, cleanup, err := mkConfigJSON("start-stop-node") assert.NoError(err) defer cleanup() startDebugging(assert) conn := connectDebug(assert) commandLine := fmt.Sprintf("StartNode(%q)", configJSON) replies := sendCommandLine(assert, conn, commandLine) assert.Len(replies, 1) assert.Equal("[0] ", replies[0]) replies = sendCommandLine(assert, conn, stopNodeCommandLine) assert.Len(replies, 1) assert.Equal("[0] ", replies[0]) } // TestCreateAccount tests creating an account on the server. func TestCreateAccount(t *testing.T) { assert := testifyAssert.New(t) configJSON, cleanup, err := mkConfigJSON("create-account") assert.NoError(err) defer cleanup() startDebugging(assert) conn := connectDebug(assert) commandLine := fmt.Sprintf("StartNode(%q)", configJSON) replies := sendCommandLine(assert, conn, commandLine) assert.Len(replies, 1) assert.Equal("[0] ", replies[0]) commandLine = fmt.Sprintf("CreateAccount(%q)", "password") replies = sendCommandLine(assert, conn, commandLine) assert.Len(replies, 4) assert.NotEqual("[0] ", replies[0]) assert.NotEqual("[1] ", replies[1]) assert.NotEqual("[2] ", replies[2]) assert.Equal("[3] ", replies[3]) replies = sendCommandLine(assert, conn, stopNodeCommandLine) assert.Len(replies, 1) assert.Equal("[0] ", replies[0]) } // TestSelectAccountLogout tests selecting an account on the server // and logging out afterwards. func TestSelectAccountLogout(t *testing.T) { assert := testifyAssert.New(t) configJSON, cleanup, err := mkConfigJSON("select-account") assert.NoError(err) defer cleanup() startDebugging(assert) conn := connectDebug(assert) commandLine := fmt.Sprintf("StartNode(%q)", configJSON) replies := sendCommandLine(assert, conn, commandLine) assert.Len(replies, 1) assert.Equal("[0] ", replies[0]) commandLine = fmt.Sprintf("CreateAccount(%q)", "password") replies = sendCommandLine(assert, conn, commandLine) assert.Len(replies, 4) assert.NotEqual("[0] ", replies[0]) assert.NotEqual("[1] ", replies[1]) assert.NotEqual("[2] ", replies[2]) assert.Equal("[3] ", replies[3]) address := replies[0][4:] commandLine = fmt.Sprintf("SelectAccount(%q, %q)", address, "password") replies = sendCommandLine(assert, conn, commandLine) assert.Len(replies, 1) assert.Equal("[0] ", replies[0]) commandLine = "Logout()" replies = sendCommandLine(assert, conn, commandLine) assert.Len(replies, 1) assert.Equal("[0] ", replies[0]) replies = sendCommandLine(assert, conn, stopNodeCommandLine) assert.Len(replies, 1) assert.Equal("[0] ", replies[0]) } //----- // HELPERS //----- var ( mu sync.Mutex d *debug.Server ) // startDebugging lazily creates or reuses a debug instance. func startDebugging(assert *testifyAssert.Assertions) { mu.Lock() defer mu.Unlock() if d == nil { var err error backend := api.NewStatusBackend() d, err = debug.New(backend, debug.CLIPort) assert.NoError(err) } } // mkConfigJSON creates a configuration matching to // a temporary directory and a cleanup for that directory. func mkConfigJSON(name string) (string, func(), error) { tmpDir, err := ioutil.TempDir(os.TempDir(), name) if err != nil { return "", nil, err } cleanup := func() { os.RemoveAll(tmpDir) //nolint: errcheck } configJSON := `{ "NetworkId": ` + strconv.Itoa(params.RopstenNetworkID) + `, "DataDir": "` + tmpDir + `", "BackupDisabledDataDir": "` + tmpDir + `", "KeyStoreDir": "` + tmpDir + `/keystore", "LogLevel": "INFO", "NoDiscovery": true, "RPCEnabled": true }` return configJSON, cleanup, nil } // connectDebug connects to the debug instance. func connectDebug(assert *testifyAssert.Assertions) net.Conn { conn, err := net.Dial("tcp", ":51515") assert.NoError(err) return conn } // sendCommandLine sends a command line via the passed connection. // nolint: interfacer func sendCommandLine(assert *testifyAssert.Assertions, conn net.Conn, commandLine string) []string { reader := bufio.NewReader(conn) writer := bufio.NewWriter(conn) _, err := writer.WriteString(commandLine + "\n") assert.NoError(err) err = writer.Flush() assert.NoError(err) countStr, err := reader.ReadString('\n') assert.NoError(err) count, err := strconv.Atoi(strings.TrimSuffix(countStr, "\n")) assert.NoError(err) replies := make([]string, count) for i := 0; i < count; i++ { reply, err := reader.ReadString('\n') assert.NoError(err) replies[i] = strings.TrimSuffix(reply, "\n") } return replies }