2023-01-26 18:58:18 +01:00
|
|
|
import { ethers } from "ethers";
|
|
|
|
|
|
|
|
|
|
import { RLN_ABI } from "./constants.js";
|
2023-05-08 18:10:26 -04:00
|
|
|
import { IdentityCredential, RLNInstance } from "./rln.js";
|
2023-01-26 18:58:18 +01:00
|
|
|
|
|
|
|
|
type Member = {
|
|
|
|
|
pubkey: string;
|
|
|
|
|
index: number;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
type ContractOptions = {
|
|
|
|
|
address: string;
|
|
|
|
|
provider: ethers.Signer | ethers.providers.Provider;
|
|
|
|
|
};
|
|
|
|
|
|
2023-04-28 01:20:29 +02:00
|
|
|
type FetchMembersOptions = {
|
|
|
|
|
fromBlock?: number;
|
|
|
|
|
fetchRange?: number;
|
|
|
|
|
fetchChunks?: number;
|
|
|
|
|
};
|
|
|
|
|
|
2023-01-26 18:58:18 +01:00
|
|
|
export class RLNContract {
|
|
|
|
|
private _contract: ethers.Contract;
|
|
|
|
|
private membersFilter: ethers.EventFilter;
|
|
|
|
|
|
|
|
|
|
private _members: Member[] = [];
|
|
|
|
|
|
|
|
|
|
public static async init(
|
|
|
|
|
rlnInstance: RLNInstance,
|
|
|
|
|
options: ContractOptions
|
|
|
|
|
): Promise<RLNContract> {
|
|
|
|
|
const rlnContract = new RLNContract(options);
|
|
|
|
|
|
|
|
|
|
await rlnContract.fetchMembers(rlnInstance);
|
|
|
|
|
rlnContract.subscribeToMembers(rlnInstance);
|
|
|
|
|
|
|
|
|
|
return rlnContract;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
constructor({ address, provider }: ContractOptions) {
|
|
|
|
|
this._contract = new ethers.Contract(address, RLN_ABI, provider);
|
|
|
|
|
this.membersFilter = this.contract.filters.MemberRegistered();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public get contract(): ethers.Contract {
|
|
|
|
|
return this._contract;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public get members(): Member[] {
|
|
|
|
|
return this._members;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async fetchMembers(
|
|
|
|
|
rlnInstance: RLNInstance,
|
2023-04-28 01:20:29 +02:00
|
|
|
options: FetchMembersOptions = {}
|
2023-01-26 18:58:18 +01:00
|
|
|
): Promise<void> {
|
2023-04-28 01:20:29 +02:00
|
|
|
const registeredMemberEvents = await queryFilter(this.contract, {
|
|
|
|
|
...options,
|
|
|
|
|
membersFilter: this.membersFilter,
|
|
|
|
|
});
|
2023-01-26 18:58:18 +01:00
|
|
|
|
|
|
|
|
for (const event of registeredMemberEvents) {
|
|
|
|
|
this.addMemberFromEvent(rlnInstance, event);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public subscribeToMembers(rlnInstance: RLNInstance): void {
|
|
|
|
|
this.contract.on(this.membersFilter, (_pubkey, _index, event) =>
|
|
|
|
|
this.addMemberFromEvent(rlnInstance, event)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private addMemberFromEvent(
|
|
|
|
|
rlnInstance: RLNInstance,
|
|
|
|
|
event: ethers.Event
|
|
|
|
|
): void {
|
|
|
|
|
if (!event.args) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const pubkey: string = event.args.pubkey;
|
|
|
|
|
const index: number = event.args.index;
|
|
|
|
|
|
|
|
|
|
this.members.push({ index, pubkey });
|
|
|
|
|
|
|
|
|
|
const idCommitment = ethers.utils.zeroPad(
|
|
|
|
|
ethers.utils.arrayify(pubkey),
|
|
|
|
|
32
|
|
|
|
|
);
|
|
|
|
|
rlnInstance.insertMember(idCommitment);
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-19 23:28:04 +02:00
|
|
|
public async registerWithSignature(
|
2023-01-26 18:58:18 +01:00
|
|
|
rlnInstance: RLNInstance,
|
|
|
|
|
signature: string
|
|
|
|
|
): Promise<ethers.Event | undefined> {
|
2023-05-08 18:10:26 -04:00
|
|
|
const identityCredential =
|
|
|
|
|
await rlnInstance.generateSeededIdentityCredential(signature);
|
2023-04-19 23:28:04 +02:00
|
|
|
|
2023-05-08 18:10:26 -04:00
|
|
|
return this.registerWithKey(identityCredential);
|
2023-04-19 23:28:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async registerWithKey(
|
2023-05-08 18:10:26 -04:00
|
|
|
credential: IdentityCredential
|
2023-04-19 23:28:04 +02:00
|
|
|
): Promise<ethers.Event | undefined> {
|
2023-01-26 18:58:18 +01:00
|
|
|
const depositValue = await this.contract.MEMBERSHIP_DEPOSIT();
|
|
|
|
|
|
|
|
|
|
const txRegisterResponse: ethers.ContractTransaction =
|
2023-05-08 18:10:26 -04:00
|
|
|
await this.contract.register(credential.IDCommitmentBigInt, {
|
2023-01-26 18:58:18 +01:00
|
|
|
value: depositValue,
|
|
|
|
|
});
|
|
|
|
|
const txRegisterReceipt = await txRegisterResponse.wait();
|
|
|
|
|
|
|
|
|
|
return txRegisterReceipt?.events?.[0];
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-04-28 01:20:29 +02:00
|
|
|
|
|
|
|
|
type CustomQueryOptions = FetchMembersOptions & {
|
|
|
|
|
membersFilter: ethers.EventFilter;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// these value should be tested on other networks
|
|
|
|
|
const FETCH_CHUNK = 5;
|
|
|
|
|
const BLOCK_RANGE = 3000;
|
|
|
|
|
|
|
|
|
|
async function queryFilter(
|
|
|
|
|
contract: ethers.Contract,
|
|
|
|
|
options: CustomQueryOptions
|
|
|
|
|
): Promise<ethers.Event[]> {
|
|
|
|
|
const {
|
|
|
|
|
fromBlock,
|
|
|
|
|
membersFilter,
|
|
|
|
|
fetchRange = BLOCK_RANGE,
|
|
|
|
|
fetchChunks = FETCH_CHUNK,
|
|
|
|
|
} = options;
|
|
|
|
|
|
|
|
|
|
if (!fromBlock) {
|
|
|
|
|
return contract.queryFilter(membersFilter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!contract.signer.provider) {
|
|
|
|
|
throw Error("No provider found on the contract's signer.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const toBlock = await contract.signer.provider.getBlockNumber();
|
|
|
|
|
|
|
|
|
|
if (toBlock - fromBlock < fetchRange) {
|
|
|
|
|
return contract.queryFilter(membersFilter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const events: ethers.Event[][] = [];
|
|
|
|
|
const chunks = splitToChunks(fromBlock, toBlock, fetchRange);
|
|
|
|
|
|
|
|
|
|
for (const portion of takeN<[number, number]>(chunks, fetchChunks)) {
|
|
|
|
|
const promises = portion.map(([left, right]) =>
|
|
|
|
|
ignoreErrors(contract.queryFilter(membersFilter, left, right), [])
|
|
|
|
|
);
|
|
|
|
|
const fetchedEvents = await Promise.all(promises);
|
|
|
|
|
events.push(fetchedEvents.flatMap((v) => v));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return events.flatMap((v) => v);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function splitToChunks(
|
|
|
|
|
from: number,
|
|
|
|
|
to: number,
|
|
|
|
|
step: number
|
|
|
|
|
): Array<[number, number]> {
|
|
|
|
|
const chunks = [];
|
|
|
|
|
|
|
|
|
|
let left = from;
|
|
|
|
|
while (left < to) {
|
|
|
|
|
const right = left + step < to ? left + step : to;
|
|
|
|
|
|
|
|
|
|
chunks.push([left, right] as [number, number]);
|
|
|
|
|
|
|
|
|
|
left = right;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return chunks;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function* takeN<T>(array: T[], size: number): Iterable<T[]> {
|
|
|
|
|
let start = 0;
|
|
|
|
|
let skip = size;
|
|
|
|
|
|
|
|
|
|
while (skip < array.length) {
|
|
|
|
|
const portion = array.slice(start, skip);
|
|
|
|
|
|
|
|
|
|
yield portion;
|
|
|
|
|
|
|
|
|
|
start = skip;
|
|
|
|
|
skip += size;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function ignoreErrors<T>(promise: Promise<T>, defaultValue: T): Promise<T> {
|
|
|
|
|
return promise.catch((err) => {
|
|
|
|
|
console.error(`Ignoring an error during query: ${err?.message}`);
|
|
|
|
|
return defaultValue;
|
|
|
|
|
});
|
|
|
|
|
}
|