import type { IDecodedMessage, IDecoder, IEncoder, IMessage, IProtoMessage, IRateLimitProof, } from "@waku/interfaces"; import debug from "debug"; import { RlnMessage, toRLNSignal } from "./message.js"; import { MembershipKey, RLNInstance } from "./rln.js"; const log = debug("waku:rln:encoder"); export class RLNEncoder implements IEncoder { private readonly idKey: Uint8Array; constructor( private encoder: IEncoder, private rlnInstance: RLNInstance, private index: number, membershipKey: MembershipKey ) { if (index < 0) throw "invalid membership index"; this.idKey = membershipKey.IDKey; } async toWire(message: IMessage): Promise { message.rateLimitProof = await this.generateProof(message); log("Proof generated", message.rateLimitProof); return this.encoder.toWire(message); } async toProtoObj(message: IMessage): Promise { const protoMessage = await this.encoder.toProtoObj(message); if (!protoMessage) return; protoMessage.contentTopic = this.contentTopic; protoMessage.rateLimitProof = await this.generateProof(message); log("Proof generated", protoMessage.rateLimitProof); return protoMessage; } private async generateProof(message: IMessage): Promise { const signal = toRLNSignal(this.contentTopic, message); console.time("proof_gen_timer"); const proof = await this.rlnInstance.generateRLNProof( signal, this.index, message.timestamp, this.idKey ); console.timeEnd("proof_gen_timer"); return proof; } get contentTopic(): string { return this.encoder.contentTopic; } get ephemeral(): boolean { return this.encoder.ephemeral; } } type RLNEncoderOptions = { encoder: IEncoder; rlnInstance: RLNInstance; index: number; membershipKey: MembershipKey; }; export const createRLNEncoder = (options: RLNEncoderOptions): RLNEncoder => { return new RLNEncoder( options.encoder, options.rlnInstance, options.index, options.membershipKey ); }; export class RLNDecoder implements IDecoder> { constructor(private rlnInstance: RLNInstance, private decoder: IDecoder) {} get contentTopic(): string { return this.decoder.contentTopic; } fromWireToProtoObj(bytes: Uint8Array): Promise { const protoMessage = this.decoder.fromWireToProtoObj(bytes); log("Message decoded", protoMessage); return Promise.resolve(protoMessage); } async fromProtoObj( pubSubTopic: string, proto: IProtoMessage ): Promise | undefined> { const msg: T | undefined = await this.decoder.fromProtoObj( pubSubTopic, proto ); if (!msg) return; return new RlnMessage(this.rlnInstance, msg, proto.rateLimitProof); } } type RLNDecoderOptions = { decoder: IDecoder; rlnInstance: RLNInstance; }; export const createRLNDecoder = ( options: RLNDecoderOptions ): RLNDecoder => { return new RLNDecoder(options.rlnInstance, options.decoder); };