mirror of
https://github.com/waku-org/js-waku.git
synced 2025-01-11 21:15:01 +00:00
feat!: set peer-exchange with default bootstrap (#1469)
* set peer-exchange with default bootstrap * only initialise protocols with bootstrap peers * update package * update package-lock * refactor `getPeers` while setting up a protocol * move codecs to `@waku/interfaces` * lightpush: send messages to multiple peers * only use multiple peers for LP and Filter * fix: ts warnings * lightpush: tests pass * update breaking changes for new API * move codecs back into protocol files * refactor: `getPeers()` * rm: log as an arg * add tsdoc for getPeers * add import * add prettier rule to eslint * add: peer exchange to sdk as a dep * fix eslint error * add try catch * revert unecessary diff * revert unecessary diff * fix imports * convert relaycodecs to array * remove: peerId as an arg for protocol methods * keep peerId as an arg for peer-exchange * remove: peerId from getPeers() * lightpush: extract hardcoded numPeers as a constant * return all peers if numPeers is 0 and increase readability for random peers * refactor considering more than 1 bootstrap peers can exist * use `getPeers` * change arg for `getPeers` to object * address comments * refactor tests for new API * lightpush: make constant the class variable * use `maxBootstrapPeers` instead of `includeBootstrap` * refactor protocols for new API * add tests for `getPeers` * skip getPeers test * rm: only from test * move tests to `base_protocol.spec.ts` * break down `getPeers` into a `filter` method * return all bootstrap peers if arg is 0 * refactor test without stubbing * address comments * update test title * move `filterPeers` to a separate file * address comments & add more test * make test title more verbose * address comments * remove ProtocolOptions * chore: refactor tests for new API * add defaults for getPeers * address comments * rm unneeded comment * address comment: add diversity of node tags to test * address comments * fix: imports
This commit is contained in:
parent
408b79d6a5
commit
81a52a8097
3
package-lock.json
generated
3
package-lock.json
generated
@ -27311,6 +27311,7 @@
|
||||
"@waku/core": "0.0.22",
|
||||
"@waku/dns-discovery": "0.0.16",
|
||||
"@waku/interfaces": "0.0.17",
|
||||
"@waku/peer-exchange": "^0.0.15",
|
||||
"@waku/relay": "0.0.5",
|
||||
"@waku/utils": "0.0.10",
|
||||
"libp2p": "^0.46.8"
|
||||
@ -27629,6 +27630,7 @@
|
||||
"version": "0.0.10",
|
||||
"license": "MIT OR Apache-2.0",
|
||||
"dependencies": {
|
||||
"@waku/interfaces": "^0.0.17",
|
||||
"debug": "^4.3.4",
|
||||
"uint8arrays": "^4.0.4"
|
||||
},
|
||||
@ -31390,6 +31392,7 @@
|
||||
"@waku/core": "0.0.22",
|
||||
"@waku/dns-discovery": "0.0.16",
|
||||
"@waku/interfaces": "0.0.17",
|
||||
"@waku/peer-exchange": "^0.0.15",
|
||||
"@waku/relay": "0.0.5",
|
||||
"@waku/utils": "0.0.10",
|
||||
"cspell": "^7.3.2",
|
||||
|
@ -15,15 +15,11 @@ export * as waku_filter from "./lib/filter/index.js";
|
||||
export { wakuFilter, FilterCodecs } from "./lib/filter/index.js";
|
||||
|
||||
export * as waku_light_push from "./lib/light_push/index.js";
|
||||
export { wakuLightPush, LightPushCodec } from "./lib/light_push/index.js";
|
||||
export { wakuLightPush } from "./lib/light_push/index.js";
|
||||
|
||||
export * as waku_store from "./lib/store/index.js";
|
||||
export {
|
||||
PageDirection,
|
||||
wakuStore,
|
||||
StoreCodec,
|
||||
createCursor
|
||||
} from "./lib/store/index.js";
|
||||
|
||||
export { PageDirection, wakuStore, createCursor } from "./lib/store/index.js";
|
||||
|
||||
export { waitForRemotePeer } from "./lib/wait_for_remote_peer.js";
|
||||
|
||||
|
@ -5,6 +5,7 @@ import { Peer, PeerStore } from "@libp2p/interface/peer-store";
|
||||
import type { IBaseProtocol, Libp2pComponents } from "@waku/interfaces";
|
||||
import { getPeersForProtocol, selectPeerForProtocol } from "@waku/utils/libp2p";
|
||||
|
||||
import { filterPeers } from "./filterPeers.js";
|
||||
import { StreamManager } from "./stream_manager.js";
|
||||
|
||||
/**
|
||||
@ -60,4 +61,32 @@ export class BaseProtocol implements IBaseProtocol {
|
||||
);
|
||||
return peer;
|
||||
}
|
||||
|
||||
/**
|
||||
* Retrieves a list of peers based on the specified criteria.
|
||||
*
|
||||
* @param numPeers - The total number of peers to retrieve. If 0, all peers are returned.
|
||||
* @param maxBootstrapPeers - The maximum number of bootstrap peers to retrieve.
|
||||
* @returns A Promise that resolves to an array of peers based on the specified criteria.
|
||||
*/
|
||||
protected async getPeers(
|
||||
{
|
||||
numPeers,
|
||||
maxBootstrapPeers
|
||||
}: {
|
||||
numPeers: number;
|
||||
maxBootstrapPeers: number;
|
||||
} = {
|
||||
maxBootstrapPeers: 1,
|
||||
numPeers: 0
|
||||
}
|
||||
): Promise<Peer[]> {
|
||||
// Retrieve all peers that support the protocol
|
||||
const allPeersForProtocol = await getPeersForProtocol(this.peerStore, [
|
||||
this.multicodec
|
||||
]);
|
||||
|
||||
// Filter the peers based on the specified criteria
|
||||
return filterPeers(allPeersForProtocol, numPeers, maxBootstrapPeers);
|
||||
}
|
||||
}
|
||||
|
@ -1,5 +1,4 @@
|
||||
import { Stream } from "@libp2p/interface/connection";
|
||||
import type { PeerId } from "@libp2p/interface/peer-id";
|
||||
import type { Peer } from "@libp2p/interface/peer-store";
|
||||
import type { IncomingStreamData } from "@libp2p/interface-internal/registrar";
|
||||
import type {
|
||||
@ -14,7 +13,6 @@ import type {
|
||||
Libp2p,
|
||||
PeerIdStr,
|
||||
ProtocolCreateOptions,
|
||||
ProtocolOptions,
|
||||
PubSubTopic,
|
||||
Unsubscribe
|
||||
} from "@waku/interfaces";
|
||||
@ -228,6 +226,7 @@ class Subscription {
|
||||
class Filter extends BaseProtocol implements IReceiver {
|
||||
private readonly options: ProtocolCreateOptions;
|
||||
private activeSubscriptions = new Map<string, Subscription>();
|
||||
private readonly NUM_PEERS_PROTOCOL = 1;
|
||||
|
||||
private getActiveSubscription(
|
||||
pubSubTopic: PubSubTopic,
|
||||
@ -257,14 +256,16 @@ class Filter extends BaseProtocol implements IReceiver {
|
||||
this.options = options ?? {};
|
||||
}
|
||||
|
||||
async createSubscription(
|
||||
pubSubTopic?: string,
|
||||
peerId?: PeerId
|
||||
): Promise<Subscription> {
|
||||
async createSubscription(pubSubTopic?: string): Promise<Subscription> {
|
||||
const _pubSubTopic =
|
||||
pubSubTopic ?? this.options.pubSubTopic ?? DefaultPubSubTopic;
|
||||
|
||||
const peer = await this.getPeer(peerId);
|
||||
const peer = (
|
||||
await this.getPeers({
|
||||
maxBootstrapPeers: 1,
|
||||
numPeers: this.NUM_PEERS_PROTOCOL
|
||||
})
|
||||
)[0];
|
||||
|
||||
const subscription =
|
||||
this.getActiveSubscription(_pubSubTopic, peer.id.toString()) ??
|
||||
@ -278,10 +279,9 @@ class Filter extends BaseProtocol implements IReceiver {
|
||||
}
|
||||
|
||||
public toSubscriptionIterator<T extends IDecodedMessage>(
|
||||
decoders: IDecoder<T> | IDecoder<T>[],
|
||||
opts?: ProtocolOptions | undefined
|
||||
decoders: IDecoder<T> | IDecoder<T>[]
|
||||
): Promise<IAsyncIterator<T>> {
|
||||
return toAsyncIterator(this, decoders, opts);
|
||||
return toAsyncIterator(this, decoders);
|
||||
}
|
||||
|
||||
/**
|
||||
@ -301,10 +301,9 @@ class Filter extends BaseProtocol implements IReceiver {
|
||||
*/
|
||||
async subscribe<T extends IDecodedMessage>(
|
||||
decoders: IDecoder<T> | IDecoder<T>[],
|
||||
callback: Callback<T>,
|
||||
opts?: ProtocolOptions
|
||||
callback: Callback<T>
|
||||
): Promise<Unsubscribe> {
|
||||
const subscription = await this.createSubscription(undefined, opts?.peerId);
|
||||
const subscription = await this.createSubscription();
|
||||
|
||||
await subscription.subscribe(decoders, callback);
|
||||
|
||||
|
144
packages/core/src/lib/filterPeers.spec.ts
Normal file
144
packages/core/src/lib/filterPeers.spec.ts
Normal file
@ -0,0 +1,144 @@
|
||||
import { Peer } from "@libp2p/interface/peer-store";
|
||||
import type { Tag } from "@libp2p/interface/peer-store";
|
||||
import { createSecp256k1PeerId } from "@libp2p/peer-id-factory";
|
||||
import { Tags } from "@waku/interfaces";
|
||||
import { expect } from "chai";
|
||||
|
||||
import { filterPeers } from "./filterPeers.js";
|
||||
|
||||
describe("filterPeers function", function () {
|
||||
it("should return all peers when numPeers is 0", async function () {
|
||||
const peer1 = await createSecp256k1PeerId();
|
||||
const peer2 = await createSecp256k1PeerId();
|
||||
const peer3 = await createSecp256k1PeerId();
|
||||
|
||||
const mockPeers = [
|
||||
{
|
||||
id: peer1,
|
||||
tags: new Map<string, Tag>([[Tags.BOOTSTRAP, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer2,
|
||||
tags: new Map<string, Tag>([[Tags.BOOTSTRAP, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer3,
|
||||
tags: new Map<string, Tag>([[Tags.PEER_EXCHANGE, { value: 100 }]])
|
||||
}
|
||||
] as unknown as Peer[];
|
||||
|
||||
const result = await filterPeers(mockPeers, 0, 10);
|
||||
expect(result.length).to.deep.equal(mockPeers.length);
|
||||
});
|
||||
|
||||
it("should return all non-bootstrap peers and no bootstrap peer when numPeers is 0 and maxBootstrapPeers is 0", async function () {
|
||||
const peer1 = await createSecp256k1PeerId();
|
||||
const peer2 = await createSecp256k1PeerId();
|
||||
const peer3 = await createSecp256k1PeerId();
|
||||
const peer4 = await createSecp256k1PeerId();
|
||||
|
||||
const mockPeers = [
|
||||
{
|
||||
id: peer1,
|
||||
tags: new Map<string, Tag>([[Tags.BOOTSTRAP, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer2,
|
||||
tags: new Map<string, Tag>([[Tags.BOOTSTRAP, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer3,
|
||||
tags: new Map<string, Tag>([[Tags.PEER_EXCHANGE, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer4,
|
||||
tags: new Map<string, Tag>([[Tags.PEER_EXCHANGE, { value: 100 }]])
|
||||
}
|
||||
] as unknown as Peer[];
|
||||
|
||||
const result = await filterPeers(mockPeers, 0, 0);
|
||||
|
||||
// result should have no bootstrap peers, and a total of 2 peers
|
||||
expect(result.length).to.equal(2);
|
||||
expect(
|
||||
result.filter((peer: Peer) => peer.tags.has(Tags.BOOTSTRAP)).length
|
||||
).to.equal(0);
|
||||
});
|
||||
|
||||
it("should return one bootstrap peer, and all non-boostrap peers, when numPeers is 0 & maxBootstrap is 1", async function () {
|
||||
const peer1 = await createSecp256k1PeerId();
|
||||
const peer2 = await createSecp256k1PeerId();
|
||||
const peer3 = await createSecp256k1PeerId();
|
||||
const peer4 = await createSecp256k1PeerId();
|
||||
const peer5 = await createSecp256k1PeerId();
|
||||
|
||||
const mockPeers = [
|
||||
{
|
||||
id: peer1,
|
||||
tags: new Map<string, Tag>([[Tags.BOOTSTRAP, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer2,
|
||||
tags: new Map<string, Tag>([[Tags.BOOTSTRAP, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer3,
|
||||
tags: new Map<string, Tag>([[Tags.PEER_EXCHANGE, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer4,
|
||||
tags: new Map<string, Tag>([[Tags.PEER_EXCHANGE, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer5,
|
||||
tags: new Map<string, Tag>([[Tags.PEER_EXCHANGE, { value: 100 }]])
|
||||
}
|
||||
] as unknown as Peer[];
|
||||
|
||||
const result = await filterPeers(mockPeers, 0, 1);
|
||||
|
||||
// result should have 1 bootstrap peers, and a total of 4 peers
|
||||
expect(result.length).to.equal(4);
|
||||
expect(
|
||||
result.filter((peer: Peer) => peer.tags.has(Tags.BOOTSTRAP)).length
|
||||
).to.equal(1);
|
||||
});
|
||||
|
||||
it("should return only bootstrap peers up to maxBootstrapPeers", async function () {
|
||||
const peer1 = await createSecp256k1PeerId();
|
||||
const peer2 = await createSecp256k1PeerId();
|
||||
const peer3 = await createSecp256k1PeerId();
|
||||
const peer4 = await createSecp256k1PeerId();
|
||||
const peer5 = await createSecp256k1PeerId();
|
||||
|
||||
const mockPeers = [
|
||||
{
|
||||
id: peer1,
|
||||
tags: new Map<string, Tag>([[Tags.BOOTSTRAP, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer2,
|
||||
tags: new Map<string, Tag>([[Tags.BOOTSTRAP, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer3,
|
||||
tags: new Map<string, Tag>([[Tags.BOOTSTRAP, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer4,
|
||||
tags: new Map<string, Tag>([[Tags.PEER_EXCHANGE, { value: 100 }]])
|
||||
},
|
||||
{
|
||||
id: peer5,
|
||||
tags: new Map<string, Tag>([[Tags.PEER_EXCHANGE, { value: 100 }]])
|
||||
}
|
||||
] as unknown as Peer[];
|
||||
|
||||
const result = await filterPeers(mockPeers, 5, 2);
|
||||
|
||||
// check that result has at least 2 bootstrap peers and no more than 5 peers
|
||||
expect(result.length).to.be.at.least(2);
|
||||
expect(result.length).to.be.at.most(5);
|
||||
expect(result.filter((peer: Peer) => peer.tags.has(Tags.BOOTSTRAP)).length);
|
||||
});
|
||||
});
|
43
packages/core/src/lib/filterPeers.ts
Normal file
43
packages/core/src/lib/filterPeers.ts
Normal file
@ -0,0 +1,43 @@
|
||||
import { Peer } from "@libp2p/interface/peer-store";
|
||||
import { Tags } from "@waku/interfaces";
|
||||
|
||||
/**
|
||||
* Retrieves a list of peers based on the specified criteria.
|
||||
*
|
||||
* @param peers - The list of peers to filter from.
|
||||
* @param numPeers - The total number of peers to retrieve. If 0, all peers are returned.
|
||||
* @param maxBootstrapPeers - The maximum number of bootstrap peers to retrieve.
|
||||
* @returns A Promise that resolves to an array of peers based on the specified criteria.
|
||||
*/
|
||||
export async function filterPeers(
|
||||
peers: Peer[],
|
||||
numPeers: number,
|
||||
maxBootstrapPeers: number
|
||||
): Promise<Peer[]> {
|
||||
// Collect the bootstrap peers up to the specified maximum
|
||||
const bootstrapPeers = peers
|
||||
.filter((peer) => peer.tags.has(Tags.BOOTSTRAP))
|
||||
.slice(0, maxBootstrapPeers);
|
||||
|
||||
// Collect non-bootstrap peers
|
||||
const nonBootstrapPeers = peers.filter(
|
||||
(peer) => !peer.tags.has(Tags.BOOTSTRAP)
|
||||
);
|
||||
|
||||
// If numPeers is 0, return all peers
|
||||
if (numPeers === 0) {
|
||||
return [...bootstrapPeers, ...nonBootstrapPeers];
|
||||
}
|
||||
|
||||
// Initialize the list of selected peers with the bootstrap peers
|
||||
const selectedPeers: Peer[] = [...bootstrapPeers];
|
||||
|
||||
// Fill up to numPeers with remaining random peers if needed
|
||||
while (selectedPeers.length < numPeers && nonBootstrapPeers.length > 0) {
|
||||
const randomIndex = Math.floor(Math.random() * nonBootstrapPeers.length);
|
||||
const randomPeer = nonBootstrapPeers.splice(randomIndex, 1)[0];
|
||||
selectedPeers.push(randomPeer);
|
||||
}
|
||||
|
||||
return selectedPeers;
|
||||
}
|
@ -5,7 +5,6 @@ import {
|
||||
IMessage,
|
||||
Libp2p,
|
||||
ProtocolCreateOptions,
|
||||
ProtocolOptions,
|
||||
SendError,
|
||||
SendResult
|
||||
} from "@waku/interfaces";
|
||||
@ -42,6 +41,7 @@ type PreparePushMessageResult =
|
||||
*/
|
||||
class LightPush extends BaseProtocol implements ILightPush {
|
||||
options: ProtocolCreateOptions;
|
||||
private readonly NUM_PEERS_PROTOCOL = 1;
|
||||
|
||||
constructor(libp2p: Libp2p, options?: ProtocolCreateOptions) {
|
||||
super(LightPushCodec, libp2p.components);
|
||||
@ -80,11 +80,7 @@ class LightPush extends BaseProtocol implements ILightPush {
|
||||
}
|
||||
}
|
||||
|
||||
async send(
|
||||
encoder: IEncoder,
|
||||
message: IMessage,
|
||||
opts?: ProtocolOptions
|
||||
): Promise<SendResult> {
|
||||
async send(encoder: IEncoder, message: IMessage): Promise<SendResult> {
|
||||
const { pubSubTopic = DefaultPubSubTopic } = this.options;
|
||||
const recipients: PeerId[] = [];
|
||||
|
||||
@ -97,49 +93,70 @@ class LightPush extends BaseProtocol implements ILightPush {
|
||||
if (preparationError || !query) {
|
||||
return {
|
||||
recipients,
|
||||
error: preparationError
|
||||
errors: [preparationError]
|
||||
};
|
||||
}
|
||||
|
||||
let error: undefined | SendError = undefined;
|
||||
const peer = await this.getPeer(opts?.peerId);
|
||||
const stream = await this.getStream(peer);
|
||||
const peers = await this.getPeers({
|
||||
maxBootstrapPeers: 1,
|
||||
numPeers: this.NUM_PEERS_PROTOCOL
|
||||
});
|
||||
|
||||
try {
|
||||
const res = await pipe(
|
||||
[query.encode()],
|
||||
lp.encode,
|
||||
stream,
|
||||
lp.decode,
|
||||
async (source) => await all(source)
|
||||
);
|
||||
const promises = peers.map(async (peer) => {
|
||||
let error: SendError | undefined;
|
||||
const stream = await this.getStream(peer);
|
||||
|
||||
try {
|
||||
const bytes = new Uint8ArrayList();
|
||||
res.forEach((chunk) => {
|
||||
bytes.append(chunk);
|
||||
});
|
||||
const res = await pipe(
|
||||
[query.encode()],
|
||||
lp.encode,
|
||||
stream,
|
||||
lp.decode,
|
||||
async (source) => await all(source)
|
||||
);
|
||||
try {
|
||||
const bytes = new Uint8ArrayList();
|
||||
res.forEach((chunk) => {
|
||||
bytes.append(chunk);
|
||||
});
|
||||
|
||||
const response = PushRpc.decode(bytes).response;
|
||||
const response = PushRpc.decode(bytes).response;
|
||||
|
||||
if (response?.isSuccess) {
|
||||
recipients.push(peer.id);
|
||||
} else {
|
||||
log("No response in PushRPC");
|
||||
error = SendError.NO_RPC_RESPONSE;
|
||||
if (response?.isSuccess) {
|
||||
recipients.some((recipient) => recipient.equals(peer.id)) ||
|
||||
recipients.push(peer.id);
|
||||
} else {
|
||||
log("No response in PushRPC");
|
||||
error = SendError.NO_RPC_RESPONSE;
|
||||
}
|
||||
} catch (err) {
|
||||
log("Failed to decode push reply", err);
|
||||
error = SendError.DECODE_FAILED;
|
||||
}
|
||||
} catch (err) {
|
||||
log("Failed to decode push reply", err);
|
||||
error = SendError.DECODE_FAILED;
|
||||
log("Failed to send waku light push request", err);
|
||||
error = SendError.GENERIC_FAIL;
|
||||
}
|
||||
} catch (err) {
|
||||
log("Failed to send waku light push request", err);
|
||||
error = SendError.GENERIC_FAIL;
|
||||
}
|
||||
|
||||
return { recipients, error };
|
||||
});
|
||||
|
||||
const results = await Promise.allSettled(promises);
|
||||
const errors = results
|
||||
.filter(
|
||||
(
|
||||
result
|
||||
): result is PromiseFulfilledResult<{
|
||||
recipients: PeerId[];
|
||||
error: SendError | undefined;
|
||||
}> => result.status === "fulfilled"
|
||||
)
|
||||
.map((result) => result.value.error)
|
||||
.filter((error) => error !== undefined) as SendError[];
|
||||
|
||||
return {
|
||||
recipients,
|
||||
error
|
||||
errors
|
||||
};
|
||||
}
|
||||
}
|
||||
|
@ -1,5 +1,4 @@
|
||||
import type { Stream } from "@libp2p/interface/connection";
|
||||
import type { PeerId } from "@libp2p/interface/peer-id";
|
||||
import { sha256 } from "@noble/hashes/sha256";
|
||||
import {
|
||||
Cursor,
|
||||
@ -40,10 +39,6 @@ export interface TimeFilter {
|
||||
}
|
||||
|
||||
export interface QueryOptions {
|
||||
/**
|
||||
* The peer to query. If undefined, a pseudo-random peer is selected from the connected Waku Store peers.
|
||||
*/
|
||||
peerId?: PeerId;
|
||||
/**
|
||||
* The direction in which pages are retrieved:
|
||||
* - { @link PageDirection.BACKWARD }: Most recent page first.
|
||||
@ -80,6 +75,7 @@ export interface QueryOptions {
|
||||
*/
|
||||
class Store extends BaseProtocol implements IStore {
|
||||
options: ProtocolCreateOptions;
|
||||
private readonly NUM_PEERS_PROTOCOL = 1;
|
||||
|
||||
constructor(libp2p: Libp2p, options?: ProtocolCreateOptions) {
|
||||
super(StoreCodec, libp2p.components);
|
||||
@ -246,12 +242,14 @@ class Store extends BaseProtocol implements IStore {
|
||||
{ contentTopics, startTime, endTime }
|
||||
);
|
||||
|
||||
log("Querying history with the following options", {
|
||||
...options,
|
||||
peerId: options?.peerId?.toString()
|
||||
});
|
||||
log("Querying history with the following options", options);
|
||||
|
||||
const peer = await this.getPeer(options?.peerId);
|
||||
const peer = (
|
||||
await this.getPeers({
|
||||
numPeers: this.NUM_PEERS_PROTOCOL,
|
||||
maxBootstrapPeers: 1
|
||||
})
|
||||
)[0];
|
||||
|
||||
for await (const messages of paginate<T>(
|
||||
this.getStream.bind(this, peer),
|
||||
|
@ -54,13 +54,6 @@ export type ProtocolCreateOptions = {
|
||||
defaultBootstrap?: boolean;
|
||||
};
|
||||
|
||||
export type ProtocolOptions = {
|
||||
/**
|
||||
* Optionally specify an PeerId for the protocol request. If not included, will use a random peer.
|
||||
*/
|
||||
peerId?: PeerId;
|
||||
};
|
||||
|
||||
export type Callback<T extends IDecodedMessage> = (
|
||||
msg: T
|
||||
) => void | Promise<void>;
|
||||
@ -74,6 +67,6 @@ export enum SendError {
|
||||
}
|
||||
|
||||
export interface SendResult {
|
||||
error?: SendError;
|
||||
errors?: SendError[];
|
||||
recipients: PeerId[];
|
||||
}
|
||||
|
@ -1,6 +1,6 @@
|
||||
import type { IDecodedMessage, IDecoder } from "./message.js";
|
||||
import type { IAsyncIterator, PubSubTopic, Unsubscribe } from "./misc.js";
|
||||
import type { Callback, ProtocolOptions } from "./protocols.js";
|
||||
import type { Callback } from "./protocols.js";
|
||||
|
||||
type ContentTopic = string;
|
||||
|
||||
@ -8,12 +8,10 @@ export type ActiveSubscriptions = Map<PubSubTopic, ContentTopic[]>;
|
||||
|
||||
export interface IReceiver {
|
||||
toSubscriptionIterator: <T extends IDecodedMessage>(
|
||||
decoders: IDecoder<T> | IDecoder<T>[],
|
||||
opts?: ProtocolOptions
|
||||
decoders: IDecoder<T> | IDecoder<T>[]
|
||||
) => Promise<IAsyncIterator<T>>;
|
||||
subscribe: <T extends IDecodedMessage>(
|
||||
decoders: IDecoder<T> | IDecoder<T>[],
|
||||
callback: Callback<T>,
|
||||
opts?: ProtocolOptions
|
||||
callback: Callback<T>
|
||||
) => Unsubscribe | Promise<Unsubscribe>;
|
||||
}
|
||||
|
@ -1,10 +1,6 @@
|
||||
import type { IEncoder, IMessage } from "./message.js";
|
||||
import type { ProtocolOptions, SendResult } from "./protocols.js";
|
||||
import type { SendResult } from "./protocols.js";
|
||||
|
||||
export interface ISender {
|
||||
send: (
|
||||
encoder: IEncoder,
|
||||
message: IMessage,
|
||||
opts?: ProtocolOptions
|
||||
) => Promise<SendResult>;
|
||||
send: (encoder: IEncoder, message: IMessage) => Promise<SendResult>;
|
||||
}
|
||||
|
@ -1,5 +1,5 @@
|
||||
import type { IDecodedMessage, IDecoder } from "./message.js";
|
||||
import type { IBaseProtocol, ProtocolOptions } from "./protocols.js";
|
||||
import type { IBaseProtocol } from "./protocols.js";
|
||||
|
||||
export enum PageDirection {
|
||||
BACKWARD = "backward",
|
||||
@ -43,7 +43,7 @@ export type StoreQueryOptions = {
|
||||
* Message.
|
||||
*/
|
||||
cursor?: Cursor;
|
||||
} & ProtocolOptions;
|
||||
};
|
||||
|
||||
export interface IStore extends IBaseProtocol {
|
||||
queryWithOrderedCallback: <T extends IDecodedMessage>(
|
||||
|
@ -23,14 +23,11 @@ const log = debug("waku:peer-exchange");
|
||||
* Implementation of the Peer Exchange protocol (https://rfc.vac.dev/spec/34/)
|
||||
*/
|
||||
export class WakuPeerExchange extends BaseProtocol implements IPeerExchange {
|
||||
multicodec: string;
|
||||
|
||||
/**
|
||||
* @param components - libp2p components
|
||||
*/
|
||||
constructor(components: Libp2pComponents) {
|
||||
super(PeerExchangeCodec, components);
|
||||
this.multicodec = PeerExchangeCodec;
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -6,6 +6,7 @@ import {
|
||||
} from "@chainsafe/libp2p-gossipsub";
|
||||
import type { PeerIdStr, TopicStr } from "@chainsafe/libp2p-gossipsub/types";
|
||||
import { SignaturePolicy } from "@chainsafe/libp2p-gossipsub/types";
|
||||
import type { PeerId } from "@libp2p/interface/peer-id";
|
||||
import type { PubSub } from "@libp2p/interface/pubsub";
|
||||
import { sha256 } from "@noble/hashes/sha256";
|
||||
import { DefaultPubSubTopic } from "@waku/core";
|
||||
@ -20,7 +21,6 @@ import {
|
||||
IRelay,
|
||||
Libp2p,
|
||||
ProtocolCreateOptions,
|
||||
ProtocolOptions,
|
||||
SendError,
|
||||
SendResult
|
||||
} from "@waku/interfaces";
|
||||
@ -98,11 +98,12 @@ class Relay implements IRelay {
|
||||
* Send Waku message.
|
||||
*/
|
||||
public async send(encoder: IEncoder, message: IMessage): Promise<SendResult> {
|
||||
const recipients: PeerId[] = [];
|
||||
if (!isSizeValid(message.payload)) {
|
||||
log("Failed to send waku relay: message is bigger that 1MB");
|
||||
return {
|
||||
recipients: [],
|
||||
error: SendError.SIZE_TOO_BIG
|
||||
recipients,
|
||||
errors: [SendError.SIZE_TOO_BIG]
|
||||
};
|
||||
}
|
||||
|
||||
@ -110,8 +111,8 @@ class Relay implements IRelay {
|
||||
if (!msg) {
|
||||
log("Failed to encode message, aborting publish");
|
||||
return {
|
||||
recipients: [],
|
||||
error: SendError.ENCODE_FAILED
|
||||
recipients,
|
||||
errors: [SendError.ENCODE_FAILED]
|
||||
};
|
||||
}
|
||||
|
||||
@ -160,10 +161,9 @@ class Relay implements IRelay {
|
||||
}
|
||||
|
||||
public toSubscriptionIterator<T extends IDecodedMessage>(
|
||||
decoders: IDecoder<T> | IDecoder<T>[],
|
||||
opts?: ProtocolOptions | undefined
|
||||
decoders: IDecoder<T> | IDecoder<T>[]
|
||||
): Promise<IAsyncIterator<T>> {
|
||||
return toAsyncIterator(this, decoders, opts);
|
||||
return toAsyncIterator(this, decoders);
|
||||
}
|
||||
|
||||
public getActiveSubscriptions(): ActiveSubscriptions {
|
||||
|
@ -56,6 +56,7 @@
|
||||
"@waku/core": "0.0.22",
|
||||
"@waku/dns-discovery": "0.0.16",
|
||||
"@waku/interfaces": "0.0.17",
|
||||
"@waku/peer-exchange": "^0.0.15",
|
||||
"libp2p": "^0.46.8"
|
||||
},
|
||||
"devDependencies": {
|
||||
|
@ -21,6 +21,7 @@ import type {
|
||||
ProtocolCreateOptions,
|
||||
RelayNode
|
||||
} from "@waku/interfaces";
|
||||
import { wakuPeerExchangeDiscovery } from "@waku/peer-exchange";
|
||||
import { RelayCreateOptions, wakuGossipSub, wakuRelay } from "@waku/relay";
|
||||
import { createLibp2p, Libp2pOptions } from "libp2p";
|
||||
import { identifyService } from "libp2p/identify";
|
||||
@ -45,7 +46,7 @@ export async function createLightNode(
|
||||
const libp2pOptions = options?.libp2p ?? {};
|
||||
const peerDiscovery = libp2pOptions.peerDiscovery ?? [];
|
||||
if (options?.defaultBootstrap) {
|
||||
peerDiscovery.push(defaultPeerDiscovery());
|
||||
peerDiscovery.push(...defaultPeerDiscoveries());
|
||||
Object.assign(libp2pOptions, { peerDiscovery });
|
||||
}
|
||||
|
||||
@ -78,7 +79,7 @@ export async function createRelayNode(
|
||||
const libp2pOptions = options?.libp2p ?? {};
|
||||
const peerDiscovery = libp2pOptions.peerDiscovery ?? [];
|
||||
if (options?.defaultBootstrap) {
|
||||
peerDiscovery.push(defaultPeerDiscovery());
|
||||
peerDiscovery.push(...defaultPeerDiscoveries());
|
||||
Object.assign(libp2pOptions, { peerDiscovery });
|
||||
}
|
||||
|
||||
@ -119,7 +120,7 @@ export async function createFullNode(
|
||||
const libp2pOptions = options?.libp2p ?? {};
|
||||
const peerDiscovery = libp2pOptions.peerDiscovery ?? [];
|
||||
if (options?.defaultBootstrap) {
|
||||
peerDiscovery.push(defaultPeerDiscovery());
|
||||
peerDiscovery.push(...defaultPeerDiscoveries());
|
||||
Object.assign(libp2pOptions, { peerDiscovery });
|
||||
}
|
||||
|
||||
@ -144,10 +145,14 @@ export async function createFullNode(
|
||||
) as FullNode;
|
||||
}
|
||||
|
||||
export function defaultPeerDiscovery(): (
|
||||
export function defaultPeerDiscoveries(): ((
|
||||
components: Libp2pComponents
|
||||
) => PeerDiscovery {
|
||||
return wakuDnsDiscovery([enrTree["PROD"]], DEFAULT_NODE_REQUIREMENTS);
|
||||
) => PeerDiscovery)[] {
|
||||
const discoveries = [
|
||||
wakuDnsDiscovery([enrTree["PROD"]], DEFAULT_NODE_REQUIREMENTS),
|
||||
wakuPeerExchangeDiscovery()
|
||||
];
|
||||
return discoveries;
|
||||
}
|
||||
|
||||
type PubsubService = {
|
||||
|
@ -108,7 +108,7 @@ describe("Waku Light Push [node only]", () => {
|
||||
payload: generateRandomUint8Array(MB + 65536)
|
||||
});
|
||||
expect(pushResponse.recipients.length).to.eq(0);
|
||||
expect(pushResponse.error).to.eq(SendError.SIZE_TOO_BIG);
|
||||
expect(pushResponse.errors).to.include(SendError.SIZE_TOO_BIG);
|
||||
});
|
||||
});
|
||||
|
||||
@ -138,13 +138,9 @@ describe("Waku Light Push [node only] - custom pubsub topic", () => {
|
||||
const messageText = "Light Push works!";
|
||||
|
||||
log("Send message via lightpush");
|
||||
const pushResponse = await waku.lightPush.send(
|
||||
TestEncoder,
|
||||
{ payload: utf8ToBytes(messageText) },
|
||||
{
|
||||
peerId: nimPeerId
|
||||
}
|
||||
);
|
||||
const pushResponse = await waku.lightPush.send(TestEncoder, {
|
||||
payload: utf8ToBytes(messageText)
|
||||
});
|
||||
log("Ack received", pushResponse);
|
||||
expect(pushResponse.recipients[0].toString()).to.eq(nimPeerId.toString());
|
||||
|
||||
|
@ -382,13 +382,13 @@ describe("Waku Relay [node only]", () => {
|
||||
payload: generateRandomUint8Array(1 * MB + 65536)
|
||||
});
|
||||
expect(sendResult.recipients.length).to.eq(0);
|
||||
expect(sendResult.error).to.eq(SendError.SIZE_TOO_BIG);
|
||||
expect(sendResult.errors).to.include(SendError.SIZE_TOO_BIG);
|
||||
|
||||
sendResult = await waku1.relay.send(TestEncoder, {
|
||||
payload: generateRandomUint8Array(2 * MB)
|
||||
});
|
||||
expect(sendResult.recipients.length).to.eq(0);
|
||||
expect(sendResult.error).to.eq(SendError.SIZE_TOO_BIG);
|
||||
expect(sendResult.errors).to.include(SendError.SIZE_TOO_BIG);
|
||||
|
||||
const waku2ReceivedMsg = await waku2ReceivedMsgPromise;
|
||||
expect(waku2ReceivedMsg?.payload?.length).to.eq(0);
|
||||
|
@ -488,8 +488,6 @@ describe("Waku Store", () => {
|
||||
await waku.dial(await nwaku.getMultiaddrWithId());
|
||||
await waitForRemotePeer(waku, [Protocols.Store]);
|
||||
|
||||
const nwakuPeerId = await nwaku.getPeerId();
|
||||
|
||||
const firstMessages: IMessage[] = [];
|
||||
await waku.store.queryWithOrderedCallback(
|
||||
[TestDecoder],
|
||||
@ -499,7 +497,6 @@ describe("Waku Store", () => {
|
||||
}
|
||||
},
|
||||
{
|
||||
peerId: nwakuPeerId,
|
||||
timeFilter: { startTime, endTime: message1Timestamp }
|
||||
}
|
||||
);
|
||||
@ -511,7 +508,6 @@ describe("Waku Store", () => {
|
||||
bothMessages.push(msg);
|
||||
},
|
||||
{
|
||||
peerId: nwakuPeerId,
|
||||
timeFilter: {
|
||||
startTime,
|
||||
endTime
|
||||
|
@ -4,7 +4,7 @@ import {
|
||||
DefaultPubSubTopic,
|
||||
waitForRemotePeer
|
||||
} from "@waku/core";
|
||||
import type { LightNode } from "@waku/interfaces";
|
||||
import { LightNode } from "@waku/interfaces";
|
||||
import { Protocols } from "@waku/interfaces";
|
||||
import { createLightNode } from "@waku/sdk";
|
||||
import { toAsyncIterator } from "@waku/utils";
|
||||
@ -49,12 +49,9 @@ describe("Util: toAsyncIterator: Filter", () => {
|
||||
const messageText = "hey, what's up?";
|
||||
const sent = { payload: utf8ToBytes(messageText) };
|
||||
|
||||
const { iterator } = await toAsyncIterator(
|
||||
waku.filter,
|
||||
TestDecoder,
|
||||
{},
|
||||
{ timeoutMs: 1000 }
|
||||
);
|
||||
const { iterator } = await toAsyncIterator(waku.filter, TestDecoder, {
|
||||
timeoutMs: 1000
|
||||
});
|
||||
|
||||
await waku.lightPush.send(TestEncoder, sent);
|
||||
const { value } = await iterator.next();
|
||||
@ -66,12 +63,9 @@ describe("Util: toAsyncIterator: Filter", () => {
|
||||
|
||||
it("handles multiple messages", async function () {
|
||||
this.timeout(10000);
|
||||
const { iterator } = await toAsyncIterator(
|
||||
waku.filter,
|
||||
TestDecoder,
|
||||
{},
|
||||
{ timeoutMs: 1000 }
|
||||
);
|
||||
const { iterator } = await toAsyncIterator(waku.filter, TestDecoder, {
|
||||
timeoutMs: 1000
|
||||
});
|
||||
|
||||
await waku.lightPush.send(TestEncoder, {
|
||||
payload: utf8ToBytes("Filtering works!")
|
||||
@ -89,12 +83,9 @@ describe("Util: toAsyncIterator: Filter", () => {
|
||||
|
||||
it("unsubscribes", async function () {
|
||||
this.timeout(10000);
|
||||
const { iterator, stop } = await toAsyncIterator(
|
||||
waku.filter,
|
||||
TestDecoder,
|
||||
{},
|
||||
{ timeoutMs: 1000 }
|
||||
);
|
||||
const { iterator, stop } = await toAsyncIterator(waku.filter, TestDecoder, {
|
||||
timeoutMs: 1000
|
||||
});
|
||||
|
||||
await waku.lightPush.send(TestEncoder, {
|
||||
payload: utf8ToBytes("This should be received")
|
||||
|
@ -66,7 +66,8 @@
|
||||
},
|
||||
"dependencies": {
|
||||
"debug": "^4.3.4",
|
||||
"uint8arrays": "^4.0.4"
|
||||
"uint8arrays": "^4.0.4",
|
||||
"@waku/interfaces": "^0.0.17"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@rollup/plugin-commonjs": "^25.0.4",
|
||||
|
@ -1,5 +1,8 @@
|
||||
export * from "./is_defined.js";
|
||||
export * from "./random_subset.js";
|
||||
export * from "./group_by.js";
|
||||
export * from "./to_async_iterator.js";
|
||||
export * from "./is_size_valid.js";
|
||||
|
||||
export function removeItemFromArray(arr: unknown[], value: unknown): unknown[] {
|
||||
const index = arr.indexOf(value);
|
||||
@ -8,6 +11,3 @@ export function removeItemFromArray(arr: unknown[], value: unknown): unknown[] {
|
||||
}
|
||||
return arr;
|
||||
}
|
||||
export * from "./group_by.js";
|
||||
export * from "./to_async_iterator.js";
|
||||
export * from "./is_size_valid.js";
|
||||
|
@ -3,7 +3,6 @@ import type {
|
||||
IDecodedMessage,
|
||||
IDecoder,
|
||||
IReceiver,
|
||||
ProtocolOptions,
|
||||
Unsubscribe
|
||||
} from "@waku/interfaces";
|
||||
|
||||
@ -31,7 +30,6 @@ const FRAME_RATE = 60;
|
||||
export async function toAsyncIterator<T extends IDecodedMessage>(
|
||||
receiver: IReceiver,
|
||||
decoder: IDecoder<T> | IDecoder<T>[],
|
||||
options?: ProtocolOptions,
|
||||
iteratorOptions?: IteratorOptions
|
||||
): Promise<IAsyncIterator<T>> {
|
||||
const iteratorDelay = iteratorOptions?.iteratorDelay ?? FRAME_RATE;
|
||||
@ -39,13 +37,9 @@ export async function toAsyncIterator<T extends IDecodedMessage>(
|
||||
const messages: T[] = [];
|
||||
|
||||
let unsubscribe: undefined | Unsubscribe;
|
||||
unsubscribe = await receiver.subscribe(
|
||||
decoder,
|
||||
(message: T) => {
|
||||
messages.push(message);
|
||||
},
|
||||
options
|
||||
);
|
||||
unsubscribe = await receiver.subscribe(decoder, (message: T) => {
|
||||
messages.push(message);
|
||||
});
|
||||
|
||||
const isWithTimeout = Number.isInteger(iteratorOptions?.timeoutMs);
|
||||
const timeoutMs = iteratorOptions?.timeoutMs ?? 0;
|
||||
|
Loading…
x
Reference in New Issue
Block a user