feat(peer-exchange): support continuous peer information updates (#2088)

* feat(peer-exchange): update peer info if new is found

* chore: move diff checking logic to a private function

* chore: add tests

* chore: increase verbosity for mulltiaddr

* chore: use merge for metadata nad patch for multiaddrs

* chore: use peerId from peerInfo

* chore: remove unused import
This commit is contained in:
Danish Arora 2024-07-26 17:14:55 +05:30 committed by GitHub
parent 08fc2d133a
commit defe41bb9a
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
3 changed files with 208 additions and 4 deletions

View File

@ -41,12 +41,13 @@ export class WakuPeerExchange extends BaseProtocol implements IPeerExchange {
public async query( public async query(
params: PeerExchangeQueryParams params: PeerExchangeQueryParams
): Promise<PeerExchangeQueryResult> { ): Promise<PeerExchangeQueryResult> {
const { numPeers } = params; const { numPeers, peerId } = params;
const rpcQuery = PeerExchangeRPC.createRequest({ const rpcQuery = PeerExchangeRPC.createRequest({
numPeers: BigInt(numPeers) numPeers: BigInt(numPeers)
}); });
const peer = await this.peerStore.get(params.peerId); const peer = await this.peerStore.get(peerId);
if (!peer) { if (!peer) {
return { return {
peerInfos: null, peerInfos: null,

View File

@ -11,9 +11,10 @@ import {
type Libp2pComponents, type Libp2pComponents,
type PeerExchangeQueryResult, type PeerExchangeQueryResult,
PubsubTopic, PubsubTopic,
ShardInfo,
Tags Tags
} from "@waku/interfaces"; } from "@waku/interfaces";
import { encodeRelayShard, Logger } from "@waku/utils"; import { decodeRelayShard, encodeRelayShard, Logger } from "@waku/utils";
import { PeerExchangeCodec, WakuPeerExchange } from "./waku_peer_exchange.js"; import { PeerExchangeCodec, WakuPeerExchange } from "./waku_peer_exchange.js";
@ -198,7 +199,48 @@ export class PeerExchangeDiscovery
const hasPeer = await this.components.peerStore.has(peerId); const hasPeer = await this.components.peerStore.has(peerId);
if (hasPeer) { if (hasPeer) {
continue; const { hasMultiaddrDiff, hasShardDiff } = await this.checkPeerInfoDiff(
peerInfo,
shardInfo
);
if (hasMultiaddrDiff || hasShardDiff) {
log.info(
`Peer ${peerId.toString()} has updated multiaddrs or shardInfo, updating`
);
if (hasMultiaddrDiff) {
log.info(
`Peer ${peerId.toString()} has updated multiaddrs, updating`
);
await this.components.peerStore.patch(peerId, {
multiaddrs: peerInfo.multiaddrs
});
}
if (hasShardDiff && shardInfo) {
log.info(
`Peer ${peerId.toString()} has updated shardInfo, updating`
);
await this.components.peerStore.merge(peerId, {
metadata: {
shardInfo: encodeRelayShard(shardInfo)
}
});
this.dispatchEvent(
new CustomEvent<PeerInfo>("peer", {
detail: {
id: peerId,
multiaddrs: peerInfo.multiaddrs
}
})
);
}
continue;
}
} }
// update the tags for the peer // update the tags for the peer
@ -213,6 +255,9 @@ export class PeerExchangeDiscovery
metadata: { metadata: {
shardInfo: encodeRelayShard(shardInfo) shardInfo: encodeRelayShard(shardInfo)
} }
}),
...(peerInfo.multiaddrs && {
multiaddrs: peerInfo.multiaddrs
}) })
}); });
@ -236,6 +281,37 @@ export class PeerExchangeDiscovery
this.queryingPeers.delete(peerIdStr); this.queryingPeers.delete(peerIdStr);
this.queryAttempts.delete(peerIdStr); this.queryAttempts.delete(peerIdStr);
} }
private async checkPeerInfoDiff(
peerInfo: PeerInfo,
shardInfo?: ShardInfo
): Promise<{ hasMultiaddrDiff: boolean; hasShardDiff: boolean }> {
const { id: peerId } = peerInfo;
const peer = await this.components.peerStore.get(peerId);
const existingMultiaddrs = peer.addresses.map((a) =>
a.multiaddr.toString()
);
const newMultiaddrs = peerInfo.multiaddrs.map((ma) => ma.toString());
const hasMultiaddrDiff = existingMultiaddrs.some(
(ma) => !newMultiaddrs.includes(ma)
);
let hasShardDiff: boolean = false;
const existingShardInfoBytes = peer.metadata.get("shardInfo");
if (existingShardInfoBytes) {
const existingShardInfo = decodeRelayShard(existingShardInfoBytes);
if (existingShardInfo || shardInfo) {
hasShardDiff =
existingShardInfo.clusterId !== shardInfo?.clusterId ||
existingShardInfo.shards.some(
(shard) => !shardInfo?.shards.includes(shard)
);
}
}
return { hasMultiaddrDiff, hasShardDiff };
}
} }
export function wakuPeerExchangeDiscovery( export function wakuPeerExchangeDiscovery(

View File

@ -0,0 +1,127 @@
import { type PeerId } from "@libp2p/interface";
import { createSecp256k1PeerId } from "@libp2p/peer-id-factory";
import { multiaddr } from "@multiformats/multiaddr";
import { PeerExchangeDiscovery } from "@waku/discovery";
import { IEnr, LightNode } from "@waku/interfaces";
import { createLightNode, ShardInfo } from "@waku/sdk";
import { decodeRelayShard, shardInfoToPubsubTopics } from "@waku/utils";
import { expect } from "chai";
import Sinon from "sinon";
describe("Peer Exchange Continuous Discovery", () => {
let peerExchangeDiscovery: PeerExchangeDiscovery;
let queryStub: Sinon.SinonStub;
let peerId: PeerId;
let randomPeerId: PeerId;
let waku: LightNode;
const shardInfo: ShardInfo = {
clusterId: 1,
shards: [1, 2]
};
const multiaddrs = [multiaddr("/ip4/127.0.0.1/udp/1234")];
beforeEach(async () => {
waku = await createLightNode();
peerExchangeDiscovery = new PeerExchangeDiscovery(
waku.libp2p.components,
shardInfoToPubsubTopics(shardInfo)
);
queryStub = Sinon.stub(
(peerExchangeDiscovery as any).peerExchange,
"query" as any
);
await discoverPeerOnce();
});
it("Should update multiaddrs", async () => {
const newMultiaddrs = [multiaddr("/ip4/192.168.1.1/udp/1234")];
const newPeerInfo = {
ENR: {
peerId,
shardInfo,
peerInfo: {
multiaddrs: newMultiaddrs,
id: peerId
}
} as IEnr
};
queryStub.resolves({ error: null, peerInfos: [newPeerInfo] });
const newResult = await (peerExchangeDiscovery as any).query(randomPeerId);
expect(newResult.error).to.be.null;
const newPeers = await waku.libp2p.peerStore.all();
expect(newPeers.length).to.equal(1);
const newPeer = newPeers[0];
expect(newPeer.addresses.length).to.equal(1);
expect(newPeer.addresses[0].multiaddr.toString()).to.equal(
newMultiaddrs[0].toString()
);
});
it("Should update shard info", async () => {
const newShardInfo: ShardInfo = {
clusterId: 2,
shards: [1, 2, 3]
};
const newPeerInfo = {
ENR: {
peerId,
shardInfo: newShardInfo,
peerInfo: {
multiaddrs: multiaddrs,
id: peerId
}
} as IEnr
};
queryStub.resolves({ error: null, peerInfos: [newPeerInfo] });
const newResult = await (peerExchangeDiscovery as any).query(randomPeerId);
expect(newResult.error).to.be.null;
const newPeers = await waku.libp2p.peerStore.all();
expect(newPeers.length).to.equal(1);
const newPeer = newPeers[0];
expect(newPeer.addresses.length).to.equal(1);
expect(newPeer.addresses[0].multiaddr.toString()).to.equal(
multiaddrs[0].toString()
);
const _shardInfo = decodeRelayShard(newPeer.metadata.get("shardInfo")!);
expect(_shardInfo).to.deep.equal(newShardInfo);
});
async function discoverPeerOnce(): Promise<void> {
peerId = await createSecp256k1PeerId();
const enr: IEnr = {
peerId,
shardInfo,
peerInfo: {
multiaddrs: multiaddrs,
id: peerId
}
} as IEnr;
const peerInfo = {
ENR: enr
};
queryStub.resolves({ error: null, peerInfos: [peerInfo] });
randomPeerId = await createSecp256k1PeerId();
const result = await (peerExchangeDiscovery as any).query(randomPeerId);
expect(result.error).to.be.null;
const peers = await waku.libp2p.peerStore.all();
expect(peers.length).to.equal(1);
const peer = peers[0];
expect(peer.addresses.length).to.equal(1);
expect(peer.addresses[0].multiaddr.toString()).to.equal(
multiaddrs[0].toString()
);
const _shardInfo = decodeRelayShard(peer.metadata.get("shardInfo")!);
expect(_shardInfo).to.deep.equal(shardInfo);
}
});