go-nwaku/examples/chat2/main.go

260 lines
7.0 KiB
Go

package main
import (
"context"
"crypto/rand"
"encoding/hex"
"encoding/json"
"flag"
"fmt"
"io/ioutil"
"log"
mrand "math/rand"
//"net"
"net/http"
"os"
"time"
//"github.com/ethereum/go-ethereum/crypto"
logging "github.com/ipfs/go-log"
"github.com/libp2p/go-libp2p-core/peer"
// TODO Replace with wrapper
// "github.com/status-im/go-waku/waku/v2/node"
// "github.com/status-im/go-waku/waku/v2/protocol/store"
"github.com/ethereum/go-ethereum/rpc"
"github.com/status-im/go-nwaku/nwaku"
)
var DefaultContentTopic string = "/toy-chat/2/huilong/proto"
func main() {
mrand.Seed(time.Now().UTC().UnixNano())
nickFlag := flag.String("nick", "", "nickname to use in chat. will be generated if empty")
fleetFlag := flag.String("fleet", "wakuv2.prod", "Select the fleet to connect to. (wakuv2.prod, wakuv2.test)")
contentTopicFlag := flag.String("contenttopic", DefaultContentTopic, "content topic to use for the chat")
nodeKeyFlag := flag.String("nodekey", "", "private key for this node. will be generated if empty")
staticNodeFlag := flag.String("staticnode", "", "connects to a node. will get a random node from fleets.status.im if empty")
storeNodeFlag := flag.String("storenode", "", "connects to a store node to retrieve messages. will get a random node from fleets.status.im if empty")
//port := flag.Int("port", 0, "port. Will be random if 0")
payloadV1Flag := flag.Bool("payloadV1", false, "use Waku v1 payload encoding/encryption. default false")
flag.Parse()
// TODO Replace with JSON RPC, unclear if needed?
//hostAddr, _ := net.ResolveTCPAddr("tcp", fmt.Sprintf("0.0.0.0:%d", *port))
if *fleetFlag != "wakuv2.prod" && *fleetFlag != "wakuv2.test" {
fmt.Println("Invalid fleet. Valid values are wakuv2.prod and wakuv2.test")
return
}
// use the nickname from the cli flag, or a default if blank
nodekey := *nodeKeyFlag
if len(nodekey) == 0 {
var err error
nodekey, err = randomHex(32)
if err != nil {
fmt.Println("Could not generate random key")
return
}
}
//prvKey, err := crypto.HexToECDSA(nodekey)
//log.Printf("NYI prvKey use", prvKey)
log.Printf("Starting node...")
nodeStopped := make(chan bool, 1)
// TODO Doesn't seem like SIGINT hook is hit here
go nwaku.StartNode(nodeStopped)
// NOTE: RPC server needs time to start, this can be improved
time.Sleep(2 * time.Second)
// TODO Start wrapper node
// Start wrapper here
// Assumes node started
client, _ := rpc.Dial("http://127.0.0.1:8545")
ctx := context.Background()
// wakuNode, err := node.New(ctx,
// node.WithPrivateKey(prvKey),
// node.WithHostAddress([]net.Addr{hostAddr}),
// node.WithWakuRelay(),
// node.WithWakuStore(false),
// )
// if err != nil {
// fmt.Print(err)
// return
// }
// use the nickname from the cli flag, or a default if blank
nick := *nickFlag
if len(nick) == 0 {
// TODO Replace with wrapper, need additional support in WakuInfo JSON-RPC response
// For now random string
// nick = defaultNick(wakuNode.Host().ID())
nick = defaultNick("go-nwaku-test")
}
// join the chat
// Same as above
// TODO Replace wakuNode
chat, err := NewChat(client, "go-nwaku-test", *contentTopicFlag, *payloadV1Flag, nick)
if err != nil {
panic(err)
}
// Display panic level to reduce log noise
lvl, err := logging.LevelFromString("panic")
if err != nil {
panic(err)
}
logging.SetAllLoggers(lvl)
ui := NewChatUI(ctx, chat)
// Connect to a static node or use random node from fleets.status.im
go func() {
time.Sleep(200 * time.Millisecond)
staticnode := *staticNodeFlag
storenode := *storeNodeFlag
//var fleetData []byte
if len(staticnode) == 0 || len(storenode) == 0 {
// XXX Context timeouts
//fleetData = getFleetData()
}
if len(staticnode) == 0 {
ui.displayMessage(fmt.Sprintf("No static peers configured. Choosing one at random from %s fleet...", *fleetFlag))
// XXX Disable for now
//staticnode = getRandomFleetNode(fleetData, *fleetFlag)
}
// TODO Call with JSON RPC
// err = wakuNode.DialPeer(staticnode)
// if err != nil {
// ui.displayMessage("Could not connect to peer: " + err.Error())
// return
// } else {
// ui.displayMessage("Connected to peer: " + staticnode)
// }
// if len(storenode) == 0 {
// ui.displayMessage(fmt.Sprintf("No store node configured. Choosing one at random from %s fleet...", *fleetFlag))
// storenode = getRandomFleetNode(fleetData, *fleetFlag)
// }
// // TODO Call with JSON RPC
// storeNodeId, err := wakuNode.AddStorePeer(storenode)
// if err != nil {
// ui.displayMessage("Could not connect to storenode: " + err.Error())
// return
// } else {
// ui.displayMessage("Connected to storenode: " + storenode)
// }
time.Sleep(300 * time.Millisecond)
ui.displayMessage("Querying historic messages")
// TODO Call with JSON RPC
//tCtx, _ := context.WithTimeout(ctx, 5*time.Second)
// response, err := wakuNode.Query(tCtx, []string{*contentTopicFlag}, 0, 0,
// store.WithAutomaticRequestId(),
// store.WithPeer(*storeNodeId),
// store.WithPaging(true, 0))
// Query messages
var contentTopic = "/toy-chat/2/huilong/proto"
var response, err = nwaku.GetWakuStoreMessages(client, contentTopic)
if err != nil {
ui.displayMessage("Could not query storenode: " + err.Error())
} else {
chat.displayMessages(response.Messages)
}
}()
//draw the UI
if err = ui.Run(); err != nil {
printErr("error running text UI: %s", err)
}
<-nodeStopped
log.Printf("exiting main")
}
// Generates a random hex string with a length of n
func randomHex(n int) (string, error) {
bytes := make([]byte, n)
if _, err := rand.Read(bytes); err != nil {
return "", err
}
return hex.EncodeToString(bytes), nil
}
// printErr is like fmt.Printf, but writes to stderr.
func printErr(m string, args ...interface{}) {
fmt.Fprintf(os.Stderr, m, args...)
}
// defaultNick generates a nickname based on the $USER environment variable and
// the last 8 chars of a peer ID.
func defaultNick(p peer.ID) string {
return fmt.Sprintf("%s-%s", os.Getenv("USER"), shortID(p))
}
// shortID returns the last 8 chars of a base58-encoded peer id.
func shortID(p peer.ID) string {
pretty := p.Pretty()
return pretty[len(pretty)-8:]
}
func getFleetData() []byte {
url := "https://fleets.status.im"
httpClient := http.Client{
Timeout: time.Second * 2,
}
req, err := http.NewRequest(http.MethodGet, url, nil)
if err != nil {
log.Fatal(err)
}
res, getErr := httpClient.Do(req)
if getErr != nil {
log.Fatal(getErr)
}
if res.Body != nil {
defer res.Body.Close()
}
body, readErr := ioutil.ReadAll(res.Body)
if readErr != nil {
log.Fatal(readErr)
}
return body
}
func getRandomFleetNode(data []byte, fleetId string) string {
var result map[string]interface{}
json.Unmarshal(data, &result)
fleets := result["fleets"].(map[string]interface{})
fleet := fleets[fleetId].(map[string]interface{})
waku := fleet["waku"].(map[string]interface{})
var wakunodes []string
for v := range waku {
wakunodes = append(wakunodes, v)
break
}
randKey := wakunodes[mrand.Intn(len(wakunodes))]
return waku[randKey].(string)
}