mirror of
https://github.com/status-im/go-waku.git
synced 2025-01-27 05:56:07 +00:00
500 lines
13 KiB
Go
500 lines
13 KiB
Go
package noise
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"crypto/ed25519"
|
|
"crypto/rand"
|
|
"crypto/sha256"
|
|
"errors"
|
|
"sync"
|
|
|
|
n "github.com/waku-org/go-noise"
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/pb"
|
|
"github.com/waku-org/go-waku/waku/v2/utils"
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
var ErrPairingTimeout = errors.New("pairing has timed out")
|
|
|
|
type Sender interface {
|
|
Publish(ctx context.Context, contentTopic string, payload *n.PayloadV2) error
|
|
}
|
|
|
|
type Receiver interface {
|
|
// Subscribe will return a channel to obtain next message received in a content topic
|
|
Subscribe(ctx context.Context, contentTopic string) <-chan *pb.WakuMessage
|
|
}
|
|
|
|
type Pairing struct {
|
|
sync.RWMutex
|
|
ContentTopic string
|
|
msgCh <-chan *pb.WakuMessage
|
|
randomFixLenVal []byte
|
|
myCommittedStaticKey []byte
|
|
params PairingParameters
|
|
handshake *n.Handshake
|
|
authCode string
|
|
authCodeEmitted chan string
|
|
authCodeConfirmed chan bool
|
|
messenger NoiseMessenger
|
|
logger *zap.Logger
|
|
|
|
started bool
|
|
completed bool
|
|
hsResult *n.HandshakeResult
|
|
}
|
|
|
|
type PairingParameterOption func(*PairingParameters) error
|
|
|
|
func WithInitiatorParameters(qrString string, qrMessageNametag n.MessageNametag) PairingParameterOption {
|
|
return func(params *PairingParameters) error {
|
|
params.initiator = true
|
|
qr, err := StringToQR(qrString)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
params.qr = qr
|
|
params.qrMessageNametag = qrMessageNametag
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func WithResponderParameters(applicationName, applicationVersion, shardID string, qrMessageNameTag *n.MessageNametag) PairingParameterOption {
|
|
return func(params *PairingParameters) error {
|
|
params.initiator = false
|
|
if qrMessageNameTag == nil {
|
|
b := make([]byte, n.MessageNametagLength)
|
|
_, err := rand.Read(b)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
params.qrMessageNametag = n.BytesToMessageNametag(b)
|
|
} else {
|
|
params.qrMessageNametag = *qrMessageNameTag
|
|
}
|
|
params.qr = NewQR(applicationName, applicationVersion, shardID, params.ephemeralPublicKey, params.myCommitedStaticKey)
|
|
return nil
|
|
}
|
|
}
|
|
|
|
const DefaultApplicationName = "waku-noise-sessions"
|
|
const DefaultApplicationVersion = "0.1"
|
|
const DefaultShardID = "10"
|
|
|
|
func WithDefaultResponderParameters() PairingParameterOption {
|
|
return WithResponderParameters(DefaultApplicationName, DefaultApplicationVersion, DefaultShardID, nil)
|
|
}
|
|
|
|
type PairingParameters struct {
|
|
myCommitedStaticKey []byte
|
|
ephemeralPublicKey ed25519.PublicKey
|
|
initiator bool
|
|
qr QR
|
|
qrMessageNametag n.MessageNametag
|
|
}
|
|
|
|
func NewPairing(myStaticKey n.Keypair, myEphemeralKey n.Keypair, opts PairingParameterOption, messenger NoiseMessenger, logger *zap.Logger) (*Pairing, error) {
|
|
b := make([]byte, 32)
|
|
_, err := rand.Read(b)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
myCommitedStaticKey := n.CommitPublicKey(sha256.New, myStaticKey.Public, b)
|
|
|
|
var params PairingParameters
|
|
params.myCommitedStaticKey = myCommitedStaticKey
|
|
params.ephemeralPublicKey = myEphemeralKey.Public
|
|
err = opts(¶ms)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
hs, err := n.NewHandshake_WakuPairing_25519_ChaChaPoly_SHA256(myStaticKey, myEphemeralKey, params.initiator, params.qr.Bytes(), params.qr.ephemeralPublicKey)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
contentTopic := "/" + params.qr.applicationName + "/" + params.qr.applicationVersion + "/wakunoise/1/sessions_shard-" + params.qr.shardID + "/proto"
|
|
|
|
// TODO: check if subscription is removed on stop
|
|
msgCh := messenger.Subscribe(context.Background(), contentTopic)
|
|
|
|
return &Pairing{
|
|
ContentTopic: contentTopic,
|
|
msgCh: msgCh,
|
|
randomFixLenVal: b, // this represents r or s depending if you're responder or initiator
|
|
myCommittedStaticKey: myCommitedStaticKey,
|
|
authCodeEmitted: make(chan string, 1),
|
|
authCodeConfirmed: make(chan bool, 1),
|
|
params: params,
|
|
handshake: hs,
|
|
messenger: messenger,
|
|
logger: logger.Named("waku-pairing1"),
|
|
}, nil
|
|
}
|
|
|
|
func (p *Pairing) PairingInfo() (qrString string, qrMessageNametag n.MessageNametag) {
|
|
p.RLock()
|
|
defer p.RUnlock()
|
|
return p.params.qr.String(), p.params.qrMessageNametag
|
|
}
|
|
|
|
func (p *Pairing) Execute(ctx context.Context) error {
|
|
p.RLock()
|
|
if p.started {
|
|
p.RUnlock()
|
|
return errors.New("pairing already executed. Create new pairing object")
|
|
}
|
|
|
|
defer p.messenger.Stop()
|
|
|
|
p.RUnlock()
|
|
p.Lock()
|
|
p.started = true
|
|
p.Unlock()
|
|
|
|
var doneCh <-chan error
|
|
if p.params.initiator {
|
|
doneCh = p.initiatorHandshake(ctx, p.msgCh)
|
|
} else {
|
|
doneCh = p.responderHandshake(ctx, p.msgCh)
|
|
}
|
|
|
|
select {
|
|
case <-ctx.Done():
|
|
p.Lock()
|
|
defer p.Unlock()
|
|
return ErrPairingTimeout
|
|
case err := <-doneCh:
|
|
return err
|
|
}
|
|
}
|
|
|
|
func (p *Pairing) isAuthCodeConfirmed(ctx context.Context) (bool, error) {
|
|
// wait for user to confirm or not, or for the whole pairing process to time out
|
|
select {
|
|
case <-ctx.Done():
|
|
return false, ErrPairingTimeout
|
|
case confirmed := <-p.authCodeConfirmed:
|
|
return confirmed, nil
|
|
}
|
|
}
|
|
|
|
func (p *Pairing) executeReadStepWithNextMessage(ctx context.Context, nextMsgChan <-chan *pb.WakuMessage, messageNametag n.MessageNametag) (*n.HandshakeStepResult, error) {
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return nil, ErrPairingTimeout
|
|
case msg := <-nextMsgChan:
|
|
payload, err := DecodePayloadV2(msg)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
step, err := p.handshake.Step(payload, nil, messageNametag)
|
|
if err != nil {
|
|
if errors.Is(err, n.ErrNametagNotExpected) || errors.Is(err, n.ErrUnexpectedMessageNametag) {
|
|
p.logger.Debug(err.Error())
|
|
continue
|
|
}
|
|
|
|
return nil, err
|
|
}
|
|
return step, nil
|
|
}
|
|
}
|
|
}
|
|
|
|
func (p *Pairing) initiatorHandshake(ctx context.Context, msgCh <-chan *pb.WakuMessage) (doneCh chan error) {
|
|
doneCh = make(chan error, 1)
|
|
|
|
go func() {
|
|
defer utils.LogOnPanic()
|
|
defer close(doneCh)
|
|
// The handshake initiator writes a Waku2 payload v2 containing the handshake message
|
|
// and the (encrypted) transport message
|
|
// The message is sent with a messageNametag equal to the one received through the QR code
|
|
hsStep, err := p.handshake.Step(nil, p.myCommittedStaticKey, p.params.qrMessageNametag)
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
// We prepare a message from initiator's payload2
|
|
// At this point wakuMsg is sent over the Waku network to receiver content topic
|
|
err = p.messenger.Publish(ctx, p.ContentTopic, hsStep.PayloadV2)
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
// We generate an authorization code using the handshake state
|
|
// this check has to be confirmed with a user interaction, comparing auth codes in both ends
|
|
authCode, err := p.handshake.Authcode()
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
p.Lock()
|
|
p.authCode = authCode
|
|
p.Unlock()
|
|
|
|
p.authCodeEmitted <- authCode
|
|
|
|
p.logger.Info("waiting for authcode confirmation....")
|
|
|
|
confirmed, err := p.isAuthCodeConfirmed(ctx)
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
if !confirmed {
|
|
p.logger.Info("authcode not confirmed")
|
|
doneCh <- errors.New("authcode not confirmed")
|
|
return
|
|
}
|
|
|
|
// 2nd step
|
|
// <- sB, eAsB {r}
|
|
hsMessageNametag, err := p.handshake.ToMessageNametag()
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
hsStep, err = p.executeReadStepWithNextMessage(ctx, msgCh, hsMessageNametag)
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
// Initiator further checks if receiver's commitment opens to receiver's static key received
|
|
expectedReceiverCommittedStaticKey := n.CommitPublicKey(sha256.New, p.handshake.RemoteStaticPublicKey(), hsStep.TransportMessage)
|
|
if !bytes.Equal(expectedReceiverCommittedStaticKey, p.params.qr.committedStaticKey) {
|
|
doneCh <- errors.New("expected committed static key does not match the receiver actual committed static key")
|
|
return
|
|
}
|
|
|
|
// 3rd step
|
|
// -> sA, sAeB, sAsB {s}
|
|
// Similarly as in first step, the initiator writes a Waku2 payload containing the handshake message and the (encrypted) transport message
|
|
hsMessageNametag, err = p.handshake.ToMessageNametag()
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
hsStep, err = p.handshake.Step(nil, p.randomFixLenVal, hsMessageNametag)
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
err = p.messenger.Publish(ctx, p.ContentTopic, hsStep.PayloadV2)
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
hsResult, err := p.handshake.FinalizeHandshake()
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
// Secure Transfer Phase
|
|
if !p.handshake.IsComplete() {
|
|
doneCh <- errors.New("handshake is in undefined state")
|
|
return
|
|
}
|
|
|
|
p.Lock()
|
|
p.hsResult = hsResult
|
|
p.completed = true
|
|
p.Unlock()
|
|
}()
|
|
|
|
return doneCh
|
|
}
|
|
|
|
func (p *Pairing) responderHandshake(ctx context.Context, msgCh <-chan *pb.WakuMessage) (doneCh chan error) {
|
|
doneCh = make(chan error, 1)
|
|
|
|
func() {
|
|
defer close(doneCh)
|
|
|
|
// the received reads the initiator's payloads, and returns the (decrypted) transport message the initiator sent
|
|
// Note that the received verifies if the received payloadV2 has the expected messageNametag set
|
|
hsStep, err := p.executeReadStepWithNextMessage(ctx, msgCh, p.params.qrMessageNametag)
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
initiatorCommittedStaticKey := hsStep.TransportMessage
|
|
|
|
// We generate an authorization code using the handshake state
|
|
// this check has to be confirmed with a user interaction, comparing auth codes in both ends
|
|
authCode, err := p.handshake.Authcode()
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
p.Lock()
|
|
p.authCode = authCode
|
|
p.Unlock()
|
|
|
|
p.authCodeEmitted <- authCode
|
|
|
|
p.logger.Info("waiting for authcode confirmation....")
|
|
|
|
confirmed, err := p.isAuthCodeConfirmed(ctx)
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
if !confirmed {
|
|
p.logger.Info("authcode not confirmed")
|
|
doneCh <- errors.New("authcode not confirmed")
|
|
return
|
|
}
|
|
|
|
// 2nd step
|
|
// <- sB, eAsB {r}
|
|
// Receiver writes and returns a payload
|
|
hsMessageNametag, err := p.handshake.ToMessageNametag()
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
hsStep, err = p.handshake.Step(nil, p.randomFixLenVal, hsMessageNametag)
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
// We prepare a Waku message from receiver's payload2
|
|
err = p.messenger.Publish(ctx, p.ContentTopic, hsStep.PayloadV2)
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
// 3rd step
|
|
// -> sA, sAeB, sAsB {s}
|
|
hsMessageNametag, err = p.handshake.ToMessageNametag()
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
// The receiver reads the initiator's payload sent by the initiator
|
|
hsStep, err = p.executeReadStepWithNextMessage(ctx, msgCh, hsMessageNametag)
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
// The receiver further checks if the initiator's commitment opens to the initiator's static key received
|
|
expectedInitiatorCommittedStaticKey := n.CommitPublicKey(sha256.New, p.handshake.RemoteStaticPublicKey(), hsStep.TransportMessage)
|
|
if !bytes.Equal(expectedInitiatorCommittedStaticKey, initiatorCommittedStaticKey) {
|
|
doneCh <- errors.New("expected committed static key does not match the initiator actual committed static key")
|
|
return
|
|
}
|
|
|
|
hsResult, err := p.handshake.FinalizeHandshake()
|
|
if err != nil {
|
|
doneCh <- err
|
|
return
|
|
}
|
|
|
|
// Secure Transfer Phase
|
|
if !p.handshake.IsComplete() {
|
|
doneCh <- errors.New("handshake is in undefined state")
|
|
return
|
|
}
|
|
|
|
p.Lock()
|
|
p.completed = true
|
|
p.hsResult = hsResult
|
|
p.Unlock()
|
|
}()
|
|
return doneCh
|
|
}
|
|
|
|
func (p *Pairing) HandshakeComplete() bool {
|
|
p.RLock()
|
|
defer p.RUnlock()
|
|
return p.completed
|
|
}
|
|
|
|
// Returns a WakuMessage with version 2 and encrypted payload
|
|
func (p *Pairing) Encrypt(plaintext []byte) (*pb.WakuMessage, error) {
|
|
p.RLock()
|
|
defer p.RUnlock()
|
|
if !p.completed {
|
|
return nil, errors.New("pairing is not complete")
|
|
}
|
|
|
|
payload, err := p.hsResult.WriteMessage(plaintext, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
encodedPayload, err := EncodePayloadV2(payload)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
encodedPayload.ContentTopic = p.ContentTopic
|
|
|
|
return encodedPayload, nil
|
|
}
|
|
|
|
func (p *Pairing) Decrypt(msg *pb.WakuMessage) ([]byte, error) {
|
|
p.RLock()
|
|
defer p.RUnlock()
|
|
if !p.completed {
|
|
return nil, errors.New("pairing is not complete")
|
|
}
|
|
|
|
payload, err := DecodePayloadV2(msg)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return p.hsResult.ReadMessage(payload, nil)
|
|
}
|
|
|
|
func (p *Pairing) ConfirmAuthCode(confirmed bool) error {
|
|
p.RLock()
|
|
authcode := p.authCode
|
|
p.RUnlock()
|
|
|
|
if authcode != "" {
|
|
p.authCodeConfirmed <- confirmed
|
|
return nil
|
|
}
|
|
|
|
return errors.New("authcode has not been generated yet")
|
|
}
|
|
|
|
func (p *Pairing) AuthCode() <-chan string {
|
|
ch := make(chan string, 1)
|
|
|
|
p.Lock()
|
|
authcode := p.authCode
|
|
p.Unlock()
|
|
|
|
if authcode != "" {
|
|
ch <- authcode
|
|
} else {
|
|
ch <- <-p.authCodeEmitted
|
|
}
|
|
close(ch)
|
|
return ch
|
|
}
|