2018-09-20 13:25:32 +02:00
// node-canary tests whether a P2P peer is responding correctly.
2018-07-12 16:50:49 +02:00
package main
import (
"context"
"errors"
"flag"
"fmt"
stdlog "log"
"os"
"path"
"path/filepath"
"time"
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/crypto/sha3"
"github.com/ethereum/go-ethereum/log"
"github.com/ethereum/go-ethereum/p2p"
"github.com/ethereum/go-ethereum/p2p/discv5"
"github.com/status-im/status-go/api"
"github.com/status-im/status-go/logutils"
"github.com/status-im/status-go/params"
"github.com/status-im/status-go/rpc"
"github.com/status-im/status-go/services/shhext"
"github.com/status-im/status-go/t/helpers"
2018-09-25 10:05:38 +03:00
whisper "github.com/status-im/whisper/whisperv6"
2018-07-12 16:50:49 +02:00
"golang.org/x/crypto/ssh/terminal"
)
const (
mailboxPassword = "status-offline-inbox"
)
// All general log messages in this package should be routed through this logger.
2018-09-20 19:15:35 +02:00
var logger = log . New ( "package" , "status-go/cmd/node-canary" )
2018-07-12 16:50:49 +02:00
var (
2018-09-20 13:25:32 +02:00
staticEnodeAddr = flag . String ( "staticnode" , "" , "static node enode address to test (e.g. enode://3f04db09bedc8d85a198de94c84da73aa7782fafc61b28c525ec5cca5a6cc16be7ebbb5cd001780f71d8408d35a2f6326faa1e524d9d8875294172ebec988743@172.16.238.10:30303)" )
mailserverEnodeAddr = flag . String ( "mailserver" , "" , "mailserver enode address to test (e.g. enode://1da276e34126e93babf24ec88aac1a7602b4cbb2e11b0961d0ab5e989ca9c261aa7f7c1c85f15550a5f1e5a5ca2305b53b9280cf5894d5ecf7d257b173136d40@167.99.209.61:30504)" )
publicChannel = flag . String ( "channel" , "status" , "The public channel name to retrieve historic messages from (used with 'mailserver' flag)" )
2018-10-27 17:39:06 +02:00
timeout = flag . Int ( "timeout" , 10 , "Timeout when connecting to node or fetching messages from mailserver, in seconds" )
2018-09-20 13:25:32 +02:00
period = flag . Int ( "period" , 24 * 60 * 60 , "How far in the past to request messages from mailserver, in seconds" )
minPow = flag . Float64 ( "shh.pow" , params . WhisperMinimumPoW , "PoW for messages to be added to queue, in float format" )
ttl = flag . Int ( "shh.ttl" , params . WhisperTTL , "Time to live for messages, in seconds" )
homePath = flag . String ( "home-dir" , "." , "Home directory where state is stored" )
logLevel = flag . String ( "log" , "INFO" , ` Log level, one of: "ERROR", "WARN", "INFO", "DEBUG", and "TRACE" ` )
logFile = flag . String ( "logfile" , "" , "Path to the log file" )
logWithoutColors = flag . Bool ( "log-without-color" , false , "Disables log colors" )
2018-07-12 16:50:49 +02:00
)
func main ( ) {
2018-09-20 13:25:32 +02:00
var err error
var staticParsedNode , mailserverParsedNode * discv5 . Node
if * staticEnodeAddr != "" {
staticParsedNode , err = discv5 . ParseNode ( * staticEnodeAddr )
if err != nil {
logger . Crit ( "Invalid static address specified" , "staticEnodeAddr" , * staticEnodeAddr , "error" , err )
os . Exit ( 1 )
}
2018-07-12 16:50:49 +02:00
}
2018-09-20 13:25:32 +02:00
if * mailserverEnodeAddr != "" {
mailserverParsedNode , err = discv5 . ParseNode ( * mailserverEnodeAddr )
if err != nil {
logger . Crit ( "Invalid mailserver address specified" , "mailserverEnodeAddr" , * mailserverEnodeAddr , "error" , err )
os . Exit ( 1 )
}
}
if staticParsedNode != nil {
verifyStaticNodeBehavior ( staticParsedNode )
logger . Info ( "Connected to static node correctly" , "address" , * staticEnodeAddr )
os . Exit ( 0 )
2018-07-12 16:50:49 +02:00
}
2018-09-20 13:25:32 +02:00
if mailserverParsedNode != nil {
verifyMailserverBehavior ( mailserverParsedNode )
logger . Info ( "Mailserver responded correctly" , "address" , * mailserverEnodeAddr )
os . Exit ( 0 )
}
logger . Crit ( "No address specified" )
os . Exit ( 1 )
2018-07-12 16:50:49 +02:00
}
func init ( ) {
flag . Parse ( )
colors := ! ( * logWithoutColors )
if colors {
colors = terminal . IsTerminal ( int ( os . Stdin . Fd ( ) ) )
}
if err := logutils . OverrideRootLog ( * logLevel != "" , * logLevel , * logFile , colors ) ; err != nil {
stdlog . Fatalf ( "Error initializing logger: %s" , err )
}
}
func verifyMailserverBehavior ( mailserverNode * discv5 . Node ) {
clientBackend , err := startClientNode ( )
if err != nil {
logger . Error ( "Node start failed" , "error" , err )
os . Exit ( 1 )
}
defer func ( ) { _ = clientBackend . StopNode ( ) } ( )
clientNode := clientBackend . StatusNode ( )
clientWhisperService , err := clientNode . WhisperService ( )
if err != nil {
logger . Error ( "Could not retrieve Whisper service" , "error" , err )
os . Exit ( 1 )
}
clientShhExtService , err := clientNode . ShhExtService ( )
if err != nil {
logger . Error ( "Could not retrieve shhext service" , "error" , err )
os . Exit ( 1 )
}
// add mailserver peer to client
2018-10-27 17:39:06 +02:00
clientErrCh := helpers . WaitForPeerAsync ( clientNode . Server ( ) , * mailserverEnodeAddr , p2p . PeerEventTypeAdd , time . Duration ( * timeout ) * time . Second )
2018-09-20 13:25:32 +02:00
err = clientNode . AddPeer ( * mailserverEnodeAddr )
2018-07-12 16:50:49 +02:00
if err != nil {
logger . Error ( "Failed to add mailserver peer to client" , "error" , err )
os . Exit ( 1 )
}
err = <- clientErrCh
if err != nil {
logger . Error ( "Error detected while waiting for mailserver peer to be added" , "error" , err )
os . Exit ( 1 )
}
// add mailserver sym key
mailServerKeyID , err := clientWhisperService . AddSymKeyFromPassword ( mailboxPassword )
if err != nil {
logger . Error ( "Error adding mailserver sym key to client peer" , "error" , err )
os . Exit ( 1 )
}
mailboxPeer := mailserverNode . ID [ : ]
mailboxPeerStr := mailserverNode . ID . String ( )
err = clientWhisperService . AllowP2PMessagesFromPeer ( mailboxPeer )
if err != nil {
logger . Error ( "Failed to allow P2P messages from peer" , "error" , err )
os . Exit ( 1 )
}
clientRPCClient := clientNode . RPCClient ( )
// TODO: Replace chat implementation with github.com/status-im/status-go-sdk
_ , topic , _ , err := joinPublicChat ( clientWhisperService , clientRPCClient , * publicChannel )
if err != nil {
logger . Error ( "Failed to join public chat" , "error" , err )
os . Exit ( 1 )
}
// watch for envelopes to be available in filters in the client
envelopeAvailableWatcher := make ( chan whisper . EnvelopeEvent , 1024 )
sub := clientWhisperService . SubscribeEnvelopeEvents ( envelopeAvailableWatcher )
defer sub . Unsubscribe ( )
// watch for mailserver responses in the client
mailServerResponseWatcher := make ( chan whisper . EnvelopeEvent , 1024 )
sub = clientWhisperService . SubscribeEnvelopeEvents ( mailServerResponseWatcher )
defer sub . Unsubscribe ( )
// request messages from mailbox
shhextAPI := shhext . NewPublicAPI ( clientShhExtService )
requestIDBytes , err := shhextAPI . RequestMessages ( context . TODO ( ) ,
shhext . MessagesRequest {
MailServerPeer : mailboxPeerStr ,
From : uint32 ( clientWhisperService . GetCurrentTime ( ) . Add ( - time . Duration ( * period ) * time . Second ) . Unix ( ) ) ,
Limit : 1 ,
Topic : topic ,
SymKeyID : mailServerKeyID ,
2018-10-27 17:39:06 +02:00
Timeout : time . Duration ( * timeout ) * time . Second ,
2018-07-12 16:50:49 +02:00
} )
if err != nil {
logger . Error ( "Error requesting historic messages from mailserver" , "error" , err )
os . Exit ( 2 )
}
requestID := common . BytesToHash ( requestIDBytes )
// wait for mailserver response
2018-10-27 17:39:06 +02:00
resp , err := waitForMailServerResponse ( mailServerResponseWatcher , requestID , time . Duration ( * timeout ) * time . Second )
2018-07-12 16:50:49 +02:00
if err != nil {
logger . Error ( "Error waiting for mailserver response" , "error" , err )
os . Exit ( 3 )
}
// wait for last envelope sent by the mailserver to be available for filters
err = waitForEnvelopeEvents ( envelopeAvailableWatcher , [ ] string { resp . LastEnvelopeHash . String ( ) } , whisper . EventEnvelopeAvailable )
if err != nil {
logger . Error ( "Error waiting for envelopes to be available to client filter" , "error" , err )
os . Exit ( 4 )
}
}
2018-09-20 13:25:32 +02:00
func verifyStaticNodeBehavior ( staticNode * discv5 . Node ) {
clientBackend , err := startClientNode ( )
if err != nil {
logger . Error ( "Node start failed" , "error" , err )
os . Exit ( 1 )
}
defer func ( ) { _ = clientBackend . StopNode ( ) } ( )
clientNode := clientBackend . StatusNode ( )
// wait for peer to be added to client
clientErrCh := helpers . WaitForPeerAsync ( clientNode . Server ( ) , * staticEnodeAddr , p2p . PeerEventTypeAdd , 5 * time . Second )
err = <- clientErrCh
if err != nil {
logger . Error ( "Error detected while waiting for static peer to be added" , "error" , err )
os . Exit ( 1 )
}
// wait to check if peer remains connected to client
clientErrCh = helpers . WaitForPeerAsync ( clientNode . Server ( ) , * staticEnodeAddr , p2p . PeerEventTypeDrop , 5 * time . Second )
err = <- clientErrCh
peers := clientNode . GethNode ( ) . Server ( ) . Peers ( )
if len ( peers ) != 1 {
logger . Error ( "Failed to add static peer" , "error" , err )
os . Exit ( 1 )
}
}
2018-07-12 16:50:49 +02:00
// makeNodeConfig parses incoming CLI options and returns node configuration object
func makeNodeConfig ( ) ( * params . NodeConfig , error ) {
err := error ( nil )
workDir := ""
if path . IsAbs ( * homePath ) {
workDir = * homePath
} else {
workDir , err = filepath . Abs ( filepath . Dir ( os . Args [ 0 ] ) )
if err == nil {
workDir = path . Join ( workDir , * homePath )
}
}
if err != nil {
return nil , err
}
2018-09-21 16:09:31 +02:00
nodeConfig , err := params . NewNodeConfigWithDefaults ( path . Join ( workDir , ".ethereum" ) , uint64 ( params . RopstenNetworkID ) )
2018-07-12 16:50:49 +02:00
if err != nil {
return nil , err
}
if * logLevel != "" {
nodeConfig . LogLevel = * logLevel
nodeConfig . LogEnabled = true
}
if * logFile != "" {
nodeConfig . LogFile = * logFile
}
2018-10-27 15:20:17 +02:00
nodeConfig . NoDiscovery = true
2018-09-19 14:45:43 +02:00
nodeConfig . ListenAddr = ""
2018-09-20 13:25:32 +02:00
if * staticEnodeAddr != "" {
nodeConfig . ClusterConfig . Enabled = true
2018-10-04 12:40:26 +02:00
nodeConfig . ClusterConfig . Fleet = params . FleetUndefined
2018-09-20 13:25:32 +02:00
nodeConfig . ClusterConfig . StaticNodes = [ ] string {
* staticEnodeAddr ,
}
}
2018-07-12 16:50:49 +02:00
return whisperConfig ( nodeConfig )
}
// whisperConfig creates node configuration object from flags
func whisperConfig ( nodeConfig * params . NodeConfig ) ( * params . NodeConfig , error ) {
2018-09-20 13:25:32 +02:00
whisperConfig := & nodeConfig . WhisperConfig
2018-07-12 16:50:49 +02:00
whisperConfig . Enabled = true
whisperConfig . LightClient = true
whisperConfig . MinimumPoW = * minPow
whisperConfig . TTL = * ttl
whisperConfig . EnableNTPSync = true
return nodeConfig , nil
}
func startClientNode ( ) ( * api . StatusBackend , error ) {
config , err := makeNodeConfig ( )
if err != nil {
return nil , err
}
clientBackend := api . NewStatusBackend ( )
err = clientBackend . StartNode ( config )
if err != nil {
return nil , err
}
return clientBackend , err
}
func joinPublicChat ( w * whisper . Whisper , rpcClient * rpc . Client , name string ) ( string , whisper . TopicType , string , error ) {
keyID , err := w . AddSymKeyFromPassword ( name )
if err != nil {
return "" , whisper . TopicType { } , "" , err
}
h := sha3 . NewKeccak256 ( )
_ , err = h . Write ( [ ] byte ( name ) )
if err != nil {
return "" , whisper . TopicType { } , "" , err
}
fullTopic := h . Sum ( nil )
topic := whisper . BytesToTopic ( fullTopic )
whisperAPI := whisper . NewPublicWhisperAPI ( w )
filterID , err := whisperAPI . NewMessageFilter ( whisper . Criteria { SymKeyID : keyID , Topics : [ ] whisper . TopicType { topic } } )
return keyID , topic , filterID , err
}
func waitForMailServerResponse ( events chan whisper . EnvelopeEvent , requestID common . Hash , timeout time . Duration ) ( * whisper . MailServerResponse , error ) {
timeoutTimer := time . NewTimer ( timeout )
for {
select {
case event := <- events :
if event . Hash == requestID {
resp , err := decodeMailServerResponse ( event )
if resp != nil || err != nil {
timeoutTimer . Stop ( )
return resp , err
}
}
case <- timeoutTimer . C :
return nil , errors . New ( "timed out waiting for mailserver response" )
}
}
}
func decodeMailServerResponse ( event whisper . EnvelopeEvent ) ( * whisper . MailServerResponse , error ) {
switch event . Event {
case whisper . EventMailServerRequestCompleted :
resp , ok := event . Data . ( * whisper . MailServerResponse )
if ! ok {
return nil , errors . New ( "failed to convert event to a *MailServerResponse" )
}
return resp , nil
case whisper . EventMailServerRequestExpired :
return nil , errors . New ( "no messages available from mailserver" )
default :
return nil , errors . New ( "unexpected event type" )
}
}
func waitForEnvelopeEvents ( events chan whisper . EnvelopeEvent , hashes [ ] string , event whisper . EventType ) error {
check := make ( map [ string ] struct { } )
for _ , hash := range hashes {
check [ hash ] = struct { } { }
}
timeout := time . NewTimer ( time . Second * 5 )
for {
select {
case e := <- events :
if e . Event == event {
delete ( check , e . Hash . String ( ) )
if len ( check ) == 0 {
timeout . Stop ( )
return nil
}
}
case <- timeout . C :
return fmt . Errorf ( "timed out while waiting for event on envelopes. event: %s" , event )
}
}
}