mirror of
https://github.com/logos-messaging/js-waku.git
synced 2026-01-03 22:33:07 +00:00
* upgrade libp2p version, partially update protocols, rename to IBaseProtocol * complete transition for protocols * complete transition of connection maanger * finish sdk * complete core * complete relay * complete peer-exchange * complete dns-discovery * add components field to Libp2p interface and use it in core * add type hack for Libp2p creation: * finish waku node test * complete relay test * complete peer exchange * complete dns peer discovery test * add missing dependency to relay * fix new peer store integration * improve initialization of pubsub * add catch for missing peer * update test and remove extra dependency * prevent error throw * fix edge case with peerStore * fix peer exchange * fix protocols used * fix test with another evnet * bump libp2p and interfaces * add missing package * fix peer-exchange problem * prefer libp2p peerDiscovery for integration tests * fix import * increate timeout * return test against Test fleet * remove await for peer:update * increase timeout * add await for peerStore * comment event for testing * fix lint * remove bind * fix stub * decouple to separate test case * move back to explicit build * remove only * do not test event
171 lines
4.4 KiB
TypeScript
171 lines
4.4 KiB
TypeScript
import type {
|
|
PeerDiscovery,
|
|
PeerDiscoveryEvents,
|
|
} from "@libp2p/interface-peer-discovery";
|
|
import { peerDiscovery as symbol } from "@libp2p/interface-peer-discovery";
|
|
import type { PeerInfo } from "@libp2p/interface-peer-info";
|
|
import type { PeerStore } from "@libp2p/interface-peer-store";
|
|
import { CustomEvent, EventEmitter } from "@libp2p/interfaces/events";
|
|
import type { IEnr } from "@waku/interfaces";
|
|
import debug from "debug";
|
|
|
|
import { DnsNodeDiscovery, NodeCapabilityCount } from "./dns.js";
|
|
|
|
export { NodeCapabilityCount };
|
|
|
|
const log = debug("waku:peer-discovery-dns");
|
|
|
|
const enrTree = {
|
|
TEST: "enrtree://AOGECG2SPND25EEFMAJ5WF3KSGJNSGV356DSTL2YVLLZWIV6SAYBM@test.waku.nodes.status.im",
|
|
PROD: "enrtree://AOGECG2SPND25EEFMAJ5WF3KSGJNSGV356DSTL2YVLLZWIV6SAYBM@prod.waku.nodes.status.im",
|
|
};
|
|
|
|
const DEFAULT_BOOTSTRAP_TAG_NAME = "bootstrap";
|
|
const DEFAULT_BOOTSTRAP_TAG_VALUE = 50;
|
|
const DEFAULT_BOOTSTRAP_TAG_TTL = 120000;
|
|
|
|
export interface DnsDiscoveryComponents {
|
|
peerStore: PeerStore;
|
|
}
|
|
|
|
export interface Options {
|
|
/**
|
|
* ENR URL to use for DNS discovery
|
|
*/
|
|
enrUrls: string | string[];
|
|
/**
|
|
* Specifies what type of nodes are wanted from the discovery process
|
|
*/
|
|
wantedNodeCapabilityCount: Partial<NodeCapabilityCount>;
|
|
/**
|
|
* Tag a bootstrap peer with this name before "discovering" it (default: 'bootstrap')
|
|
*/
|
|
tagName?: string;
|
|
|
|
/**
|
|
* The bootstrap peer tag will have this value (default: 50)
|
|
*/
|
|
tagValue?: number;
|
|
|
|
/**
|
|
* Cause the bootstrap peer tag to be removed after this number of ms (default: 2 minutes)
|
|
*/
|
|
tagTTL?: number;
|
|
}
|
|
|
|
/**
|
|
* Parse options and expose function to return bootstrap peer addresses.
|
|
*/
|
|
export class PeerDiscoveryDns
|
|
extends EventEmitter<PeerDiscoveryEvents>
|
|
implements PeerDiscovery
|
|
{
|
|
private nextPeer: (() => AsyncGenerator<IEnr>) | undefined;
|
|
private _started: boolean;
|
|
private _components: DnsDiscoveryComponents;
|
|
private _options: Options;
|
|
|
|
constructor(components: DnsDiscoveryComponents, options: Options) {
|
|
super();
|
|
this._started = false;
|
|
this._components = components;
|
|
this._options = options;
|
|
|
|
const { enrUrls } = options;
|
|
log("Use following EIP-1459 ENR Tree URLs: ", enrUrls);
|
|
}
|
|
|
|
/**
|
|
* Start discovery process
|
|
*/
|
|
async start(): Promise<void> {
|
|
log("Starting peer discovery via dns");
|
|
|
|
this._started = true;
|
|
|
|
if (this.nextPeer === undefined) {
|
|
let { enrUrls } = this._options;
|
|
if (!Array.isArray(enrUrls)) enrUrls = [enrUrls];
|
|
|
|
const { wantedNodeCapabilityCount } = this._options;
|
|
const dns = await DnsNodeDiscovery.dnsOverHttp();
|
|
|
|
this.nextPeer = dns.getNextPeer.bind(
|
|
dns,
|
|
enrUrls,
|
|
wantedNodeCapabilityCount
|
|
);
|
|
}
|
|
|
|
for await (const peerEnr of this.nextPeer()) {
|
|
if (!this._started) {
|
|
return;
|
|
}
|
|
|
|
const peerInfo = peerEnr.peerInfo;
|
|
|
|
if (!peerInfo) {
|
|
continue;
|
|
}
|
|
|
|
const tagsToUpdate = {
|
|
tags: {
|
|
[DEFAULT_BOOTSTRAP_TAG_NAME]: {
|
|
value: this._options.tagValue ?? DEFAULT_BOOTSTRAP_TAG_VALUE,
|
|
ttl: this._options.tagTTL ?? DEFAULT_BOOTSTRAP_TAG_TTL,
|
|
},
|
|
},
|
|
};
|
|
|
|
let isPeerChanged = false;
|
|
const isPeerExists = await this._components.peerStore.has(peerInfo.id);
|
|
|
|
if (isPeerExists) {
|
|
const peer = await this._components.peerStore.get(peerInfo.id);
|
|
const hasBootstrapTag = peer.tags.has(DEFAULT_BOOTSTRAP_TAG_NAME);
|
|
|
|
if (!hasBootstrapTag) {
|
|
isPeerChanged = true;
|
|
await this._components.peerStore.merge(peerInfo.id, tagsToUpdate);
|
|
}
|
|
} else {
|
|
isPeerChanged = true;
|
|
await this._components.peerStore.save(peerInfo.id, tagsToUpdate);
|
|
}
|
|
|
|
if (isPeerChanged) {
|
|
this.dispatchEvent(
|
|
new CustomEvent<PeerInfo>("peer", { detail: peerInfo })
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Stop emitting events
|
|
*/
|
|
stop(): void {
|
|
this._started = false;
|
|
}
|
|
|
|
get [symbol](): true {
|
|
return true;
|
|
}
|
|
|
|
get [Symbol.toStringTag](): string {
|
|
return "@waku/bootstrap";
|
|
}
|
|
}
|
|
|
|
export function wakuDnsDiscovery(
|
|
enrUrls: string[],
|
|
wantedNodeCapabilityCount: Partial<NodeCapabilityCount>
|
|
): (components: DnsDiscoveryComponents) => PeerDiscoveryDns {
|
|
return (components: DnsDiscoveryComponents) =>
|
|
new PeerDiscoveryDns(components, { enrUrls, wantedNodeCapabilityCount });
|
|
}
|
|
|
|
export { DnsNodeDiscovery, SearchContext, DnsClient } from "./dns.js";
|
|
|
|
export { enrTree };
|