go-waku/waku/v2/rpc/private.go
Richard Ramos 11d1f8fb0d
feat: result aggregation in resume and enforce max page size (#183)
* feat: result aggregation in resume and enforce max page size
* feat: add WithLogger option to wakunode (#184)
* fix: rebase issues
2022-01-18 14:17:06 -04:00

189 lines
5.1 KiB
Go

package rpc
import (
"crypto/ecdsa"
"crypto/rand"
"encoding/hex"
"fmt"
"net/http"
"sync"
"github.com/ethereum/go-ethereum/crypto"
"github.com/status-im/go-waku/waku/v2/node"
"github.com/status-im/go-waku/waku/v2/protocol/pb"
"go.uber.org/zap"
)
type PrivateService struct {
node *node.WakuNode
log *zap.SugaredLogger
symmetricMessages map[string][]*pb.WakuMessage
symmetricMessagesMutex sync.RWMutex
asymmetricMessages map[string][]*pb.WakuMessage
asymmetricMessagesMutex sync.RWMutex
}
type SymmetricKeyReply struct {
Key string `json:"key"`
}
type KeyPairReply struct {
PrivateKey string `json:"privateKey"`
PulicKey string `json:"publicKey"`
}
type SymmetricMessageArgs struct {
Topic string `json:"topic"`
Message pb.WakuMessage `json:"message"`
SymKey string `json:"symkey"`
}
type AsymmetricMessageArgs struct {
Topic string `json:"topic"`
Message pb.WakuMessage `json:"message"`
PublicKey string `json:"publicKey"`
}
type SymmetricMessagesArgs struct {
Topic string `json:"topic"`
SymKey string `json:"symkey"`
}
type AsymmetricMessagesArgs struct {
Topic string `json:"topic"`
PrivateKey string `json:"privateKey"`
}
func NewPrivateService(node *node.WakuNode, log *zap.SugaredLogger) *PrivateService {
return &PrivateService{
node: node,
symmetricMessages: make(map[string][]*pb.WakuMessage),
asymmetricMessages: make(map[string][]*pb.WakuMessage),
log: log.Named("private"),
}
}
func (p *PrivateService) GetV1SymmetricKey(req *http.Request, args *Empty, reply *SymmetricKeyReply) error {
key := [32]byte{}
_, err := rand.Read(key[:])
if err != nil {
return err
}
reply.Key = hex.EncodeToString(key[:])
return nil
}
func (p *PrivateService) GetV1AsymmetricKeypair(req *http.Request, args *Empty, reply *KeyPairReply) error {
privateKey, err := crypto.GenerateKey()
if err != nil {
return err
}
privateKeyBytes := crypto.FromECDSA(privateKey)
publicKey := privateKey.Public()
publicKeyECDSA, ok := publicKey.(*ecdsa.PublicKey)
if !ok {
return fmt.Errorf("cannot assert type: publicKey is not of type *ecdsa.PublicKey")
}
publicKeyBytes := crypto.FromECDSAPub(publicKeyECDSA)
reply.PrivateKey = hex.EncodeToString(privateKeyBytes[:])
reply.PulicKey = hex.EncodeToString(publicKeyBytes[:])
return nil
}
func (p *PrivateService) PostV1SymmetricMessage(req *http.Request, args *SymmetricMessageArgs, reply *SuccessReply) error {
keyInfo := new(node.KeyInfo)
keyInfo.Kind = node.Symmetric
keyInfo.SymKey = []byte(args.SymKey)
err := node.EncodeWakuMessage(&args.Message, keyInfo)
if err != nil {
reply.Error = err.Error()
reply.Success = false
return nil
}
err = p.node.Publish(req.Context(), &args.Message)
if err != nil {
reply.Error = err.Error()
reply.Success = false
return nil
}
p.symmetricMessagesMutex.Lock()
defer p.symmetricMessagesMutex.Unlock()
if _, ok := p.symmetricMessages[args.Topic]; !ok {
p.symmetricMessages[args.Topic] = make([]*pb.WakuMessage, 0)
}
p.symmetricMessages[args.Topic] = append(p.symmetricMessages[args.Topic], &args.Message)
reply.Success = true
return nil
}
func (p *PrivateService) PostV1AsymmetricMessage(req *http.Request, args *AsymmetricMessageArgs, reply *SuccessReply) error {
keyInfo := new(node.KeyInfo)
keyInfo.Kind = node.Asymmetric
pubKeyBytes, err := hex.DecodeString(args.PublicKey)
if err != nil {
return fmt.Errorf("public key cannot be decoded: %v", err)
}
pubKey, err := crypto.UnmarshalPubkey(pubKeyBytes)
if err != nil {
return fmt.Errorf("public key cannot be unmarshalled: %v", err)
}
keyInfo.PubKey = *pubKey
err = node.EncodeWakuMessage(&args.Message, keyInfo)
if err != nil {
reply.Error = err.Error()
reply.Success = false
return nil
}
err = p.node.Publish(req.Context(), &args.Message)
if err != nil {
reply.Error = err.Error()
reply.Success = false
return nil
}
p.asymmetricMessagesMutex.Lock()
defer p.asymmetricMessagesMutex.Unlock()
if _, ok := p.asymmetricMessages[args.Topic]; !ok {
p.asymmetricMessages[args.Topic] = make([]*pb.WakuMessage, 0)
}
p.asymmetricMessages[args.Topic] = append(p.asymmetricMessages[args.Topic], &args.Message)
reply.Success = true
return nil
}
func (p *PrivateService) GetV1SymmetricMessages(req *http.Request, args *SymmetricMessagesArgs, reply *MessagesReply) error {
p.symmetricMessagesMutex.Lock()
defer p.symmetricMessagesMutex.Unlock()
if _, ok := p.symmetricMessages[args.Topic]; !ok {
return fmt.Errorf("topic %s not subscribed", args.Topic)
}
reply.Messages = p.symmetricMessages[args.Topic]
p.symmetricMessages[args.Topic] = make([]*pb.WakuMessage, 0)
return nil
}
func (p *PrivateService) GetV1AsymmetricMessages(req *http.Request, args *AsymmetricMessagesArgs, reply *MessagesReply) error {
p.asymmetricMessagesMutex.Lock()
defer p.asymmetricMessagesMutex.Unlock()
if _, ok := p.asymmetricMessages[args.Topic]; !ok {
return fmt.Errorf("topic %s not subscribed", args.Topic)
}
reply.Messages = p.asymmetricMessages[args.Topic]
p.asymmetricMessages[args.Topic] = make([]*pb.WakuMessage, 0)
return nil
}