2022-07-05 17:28:34 -04:00
|
|
|
package rln
|
|
|
|
|
|
|
|
import (
|
2022-07-28 10:04:33 -04:00
|
|
|
"context"
|
|
|
|
"crypto/ecdsa"
|
2022-07-05 17:28:34 -04:00
|
|
|
"errors"
|
|
|
|
|
2022-07-28 10:04:33 -04:00
|
|
|
"github.com/ethereum/go-ethereum/common"
|
2022-11-09 15:53:01 -04:00
|
|
|
"github.com/waku-org/go-waku/waku/v2/protocol/relay"
|
2022-10-27 11:23:20 -04:00
|
|
|
r "github.com/waku-org/go-zerokit-rln/rln"
|
2022-07-05 17:28:34 -04:00
|
|
|
"go.uber.org/zap"
|
|
|
|
)
|
|
|
|
|
|
|
|
func RlnRelayStatic(
|
2022-07-28 10:04:33 -04:00
|
|
|
ctx context.Context,
|
2022-07-05 17:28:34 -04:00
|
|
|
relay *relay.WakuRelay,
|
|
|
|
group []r.IDCommitment,
|
|
|
|
memKeyPair r.MembershipKeyPair,
|
|
|
|
memIndex r.MembershipIndex,
|
|
|
|
pubsubTopic string,
|
|
|
|
contentTopic string,
|
|
|
|
spamHandler SpamHandler,
|
|
|
|
log *zap.Logger,
|
|
|
|
) (*WakuRLNRelay, error) {
|
|
|
|
log = log.Named("rln-static")
|
|
|
|
|
|
|
|
log.Info("mounting rln-relay in off-chain/static mode")
|
|
|
|
|
|
|
|
// check the peer's index and the inclusion of user's identity commitment in the group
|
|
|
|
if memKeyPair.IDCommitment != group[int(memIndex)] {
|
|
|
|
return nil, errors.New("peer's IDCommitment does not match commitment in group")
|
|
|
|
}
|
|
|
|
|
2022-10-04 19:15:39 -04:00
|
|
|
rlnInstance, err := r.NewRLN()
|
2022-07-05 17:28:34 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// create the WakuRLNRelay
|
|
|
|
rlnPeer := &WakuRLNRelay{
|
2022-07-28 10:04:33 -04:00
|
|
|
ctx: ctx,
|
2022-10-10 18:08:35 -04:00
|
|
|
membershipKeyPair: &memKeyPair,
|
2022-07-05 17:28:34 -04:00
|
|
|
membershipIndex: memIndex,
|
|
|
|
RLN: rlnInstance,
|
|
|
|
pubsubTopic: pubsubTopic,
|
|
|
|
contentTopic: contentTopic,
|
|
|
|
log: log,
|
2022-07-06 16:29:20 -04:00
|
|
|
nullifierLog: make(map[r.Epoch][]r.ProofMetadata),
|
2022-07-05 17:28:34 -04:00
|
|
|
}
|
|
|
|
|
2022-10-21 15:49:55 -04:00
|
|
|
root, err := rlnPeer.RLN.GetMerkleRoot()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
rlnPeer.validMerkleRoots = append(rlnPeer.validMerkleRoots, root)
|
|
|
|
|
|
|
|
// add members to the Merkle tree
|
|
|
|
for _, member := range group {
|
|
|
|
if err := rlnPeer.insertMember(member); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-05 17:28:34 -04:00
|
|
|
// adds a topic validator for the supplied pubsub topic at the relay protocol
|
|
|
|
// messages published on this pubsub topic will be relayed upon a successful validation, otherwise they will be dropped
|
|
|
|
// the topic validator checks for the correct non-spamming proof of the message
|
2022-07-06 16:29:20 -04:00
|
|
|
err = rlnPeer.addValidator(relay, pubsubTopic, contentTopic, spamHandler)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-07-05 17:28:34 -04:00
|
|
|
|
|
|
|
log.Info("rln relay topic validator mounted", zap.String("pubsubTopic", pubsubTopic), zap.String("contentTopic", contentTopic))
|
|
|
|
|
|
|
|
return rlnPeer, nil
|
|
|
|
}
|
2022-07-28 10:04:33 -04:00
|
|
|
|
|
|
|
func RlnRelayDynamic(
|
|
|
|
ctx context.Context,
|
|
|
|
relay *relay.WakuRelay,
|
|
|
|
ethClientAddr string,
|
|
|
|
ethAccountPrivateKey *ecdsa.PrivateKey,
|
|
|
|
memContractAddr common.Address,
|
|
|
|
memKeyPair *r.MembershipKeyPair,
|
|
|
|
memIndex r.MembershipIndex,
|
|
|
|
pubsubTopic string,
|
|
|
|
contentTopic string,
|
|
|
|
spamHandler SpamHandler,
|
2022-09-11 17:08:58 -04:00
|
|
|
registrationHandler RegistrationHandler,
|
2022-07-28 10:04:33 -04:00
|
|
|
log *zap.Logger,
|
|
|
|
) (*WakuRLNRelay, error) {
|
|
|
|
log = log.Named("rln-dynamic")
|
|
|
|
|
|
|
|
log.Info("mounting rln-relay in onchain/dynamic mode")
|
|
|
|
|
2022-10-04 19:15:39 -04:00
|
|
|
rlnInstance, err := r.NewRLN()
|
2022-07-28 10:04:33 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// create the WakuRLNRelay
|
|
|
|
rlnPeer := &WakuRLNRelay{
|
|
|
|
ctx: ctx,
|
|
|
|
membershipIndex: memIndex,
|
|
|
|
membershipContractAddress: memContractAddr,
|
|
|
|
ethClientAddress: ethClientAddr,
|
|
|
|
ethAccountPrivateKey: ethAccountPrivateKey,
|
|
|
|
RLN: rlnInstance,
|
|
|
|
pubsubTopic: pubsubTopic,
|
|
|
|
contentTopic: contentTopic,
|
|
|
|
log: log,
|
|
|
|
nullifierLog: make(map[r.Epoch][]r.ProofMetadata),
|
2022-09-11 17:08:58 -04:00
|
|
|
registrationHandler: registrationHandler,
|
2022-07-28 10:04:33 -04:00
|
|
|
}
|
|
|
|
|
2022-10-21 15:49:55 -04:00
|
|
|
root, err := rlnPeer.RLN.GetMerkleRoot()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
rlnPeer.validMerkleRoots = append(rlnPeer.validMerkleRoots, root)
|
|
|
|
|
2022-07-28 10:04:33 -04:00
|
|
|
// prepare rln membership key pair
|
2022-10-10 18:08:35 -04:00
|
|
|
if memKeyPair == nil && ethAccountPrivateKey != nil {
|
2022-07-28 10:04:33 -04:00
|
|
|
log.Debug("no rln-relay key is provided, generating one")
|
|
|
|
memKeyPair, err = rlnInstance.MembershipKeyGen()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-10-10 18:08:35 -04:00
|
|
|
rlnPeer.membershipKeyPair = memKeyPair
|
2022-08-04 17:39:12 -04:00
|
|
|
|
2022-07-28 10:04:33 -04:00
|
|
|
// register the rln-relay peer to the membership contract
|
|
|
|
membershipIndex, err := rlnPeer.Register(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
rlnPeer.membershipIndex = *membershipIndex
|
|
|
|
|
|
|
|
log.Info("registered peer into the membership contract")
|
2022-10-10 18:08:35 -04:00
|
|
|
} else if memKeyPair != nil {
|
|
|
|
rlnPeer.membershipKeyPair = memKeyPair
|
2022-07-28 10:04:33 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
handler := func(pubkey r.IDCommitment, index r.MembershipIndex) error {
|
2022-10-21 15:49:55 -04:00
|
|
|
return rlnPeer.insertMember(pubkey)
|
2022-07-28 10:04:33 -04:00
|
|
|
}
|
|
|
|
|
2022-08-15 19:29:19 -04:00
|
|
|
errChan := make(chan error)
|
|
|
|
go rlnPeer.HandleGroupUpdates(handler, errChan)
|
|
|
|
err = <-errChan
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-07-28 10:04:33 -04:00
|
|
|
|
|
|
|
// adds a topic validator for the supplied pubsub topic at the relay protocol
|
|
|
|
// messages published on this pubsub topic will be relayed upon a successful validation, otherwise they will be dropped
|
|
|
|
// the topic validator checks for the correct non-spamming proof of the message
|
|
|
|
err = rlnPeer.addValidator(relay, pubsubTopic, contentTopic, spamHandler)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Info("rln relay topic validator mounted", zap.String("pubsubTopic", pubsubTopic), zap.String("contentTopic", contentTopic))
|
|
|
|
|
|
|
|
return rlnPeer, nil
|
|
|
|
|
|
|
|
}
|