117 lines
2.5 KiB
Go
Raw Normal View History

2021-04-04 13:06:17 -04:00
package main
import (
"chat2/pb"
"context"
2021-04-04 13:06:17 -04:00
"time"
"github.com/golang/protobuf/proto"
"github.com/libp2p/go-libp2p-core/peer"
"github.com/status-im/go-waku/waku/v2/node"
2021-04-21 20:12:51 -04:00
wpb "github.com/status-im/go-waku/waku/v2/protocol/pb"
2021-04-04 13:06:17 -04:00
)
// Chat represents a subscription to a single PubSub topic. Messages
// can be published to the topic with Chat.Publish, and received
// messages are pushed to the Messages channel.
type Chat struct {
// Messages is a channel of messages received from other peers in the chat room
Messages chan *pb.Chat2Message
sub *node.Subscription
node *node.WakuNode
self peer.ID
nick string
}
// NewChat tries to subscribe to the PubSub topic for the room name, returning
// a ChatRoom on success.
2021-04-21 20:12:51 -04:00
func NewChat(n *node.WakuNode, selfID peer.ID, nickname string) (*Chat, error) {
2021-04-04 13:06:17 -04:00
// join the default waku topic and subscribe to it
sub, err := n.Subscribe(nil)
if err != nil {
return nil, err
}
c := &Chat{
node: n,
sub: sub,
self: selfID,
nick: nickname,
Messages: make(chan *pb.Chat2Message, 1024),
}
// start reading messages from the subscription in a loop
go c.readLoop()
return c, nil
}
// Publish sends a message to the pubsub topic.
func (cr *Chat) Publish(ctx context.Context, message string) error {
2021-04-04 13:06:17 -04:00
msg := &pb.Chat2Message{
Timestamp: uint64(time.Now().Unix()),
Nick: cr.nick,
Payload: []byte(message),
}
msgBytes, err := proto.Marshal(msg)
if err != nil {
return err
}
var version uint32 = 0
2021-04-04 15:33:21 -04:00
var timestamp float64 = float64(time.Now().UnixNano())
var keyInfo *node.KeyInfo = &node.KeyInfo{Kind: node.None}
p := new(node.Payload)
p.Data = msgBytes
p.Key = keyInfo
2021-04-04 13:06:17 -04:00
payload, err := p.Encode(0)
2021-04-04 13:06:17 -04:00
if err != nil {
return err
}
2021-04-21 20:12:51 -04:00
wakuMsg := &wpb.WakuMessage{
2021-04-04 13:06:17 -04:00
Payload: payload,
2021-04-07 17:19:39 -04:00
Version: version,
ContentTopic: DefaultContentTopic,
Timestamp: timestamp,
2021-04-04 13:06:17 -04:00
}
_, err = cr.node.Publish(ctx, wakuMsg, nil)
2021-04-11 19:45:42 -04:00
return err
2021-04-04 13:06:17 -04:00
}
2021-04-21 20:12:51 -04:00
func (cr *Chat) decodeMessage(wakumsg *wpb.WakuMessage) {
2021-04-12 14:03:58 -04:00
payload, err := node.DecodePayload(wakumsg, &node.KeyInfo{Kind: node.None})
if err != nil {
return
}
msg := &pb.Chat2Message{}
if err := proto.Unmarshal(payload.Data, msg); err != nil {
return
}
// send valid messages onto the Messages channel
cr.Messages <- msg
}
2021-04-04 13:06:17 -04:00
// readLoop pulls messages from the pubsub topic and pushes them onto the Messages channel.
func (cr *Chat) readLoop() {
for value := range cr.sub.C {
2021-04-12 14:03:58 -04:00
cr.decodeMessage(value.Message())
}
}
2021-04-21 20:12:51 -04:00
func (cr *Chat) displayMessages(messages []*wpb.WakuMessage) {
2021-04-12 14:03:58 -04:00
for _, msg := range messages {
cr.decodeMessage(msg)
2021-04-04 13:06:17 -04:00
}
}