mirror of
https://github.com/cerc-io/watcher-ts
synced 2025-01-23 11:39:05 +00:00
Restrict number of relay connections for a peer (#317)
* Restrict number of relay connections for a peer * Add a log message when ignoring a relay node
This commit is contained in:
parent
582a8e10e3
commit
197aec74aa
@ -32,6 +32,9 @@ export const PING_TIMEOUT = 5000; // 5 seconds
|
|||||||
// Delay time in ms to redial relay node on failing to connect
|
// Delay time in ms to redial relay node on failing to connect
|
||||||
export const RELAY_REDIAL_DELAY = 5000; // 5 sconds
|
export const RELAY_REDIAL_DELAY = 5000; // 5 sconds
|
||||||
|
|
||||||
|
// Max number of relay node connections for a peer after which it starts igoring them
|
||||||
|
export const DEFAULT_MAX_RELAY_CONNECTIONS = 2;
|
||||||
|
|
||||||
// Peer connection manager config constants
|
// Peer connection manager config constants
|
||||||
|
|
||||||
// Number of max concurrent dials per peer
|
// Number of max concurrent dials per peer
|
||||||
|
@ -25,9 +25,10 @@ import { multiaddr, Multiaddr } from '@multiformats/multiaddr';
|
|||||||
import { floodsub } from '@libp2p/floodsub';
|
import { floodsub } from '@libp2p/floodsub';
|
||||||
import { pubsubPeerDiscovery } from '@libp2p/pubsub-peer-discovery';
|
import { pubsubPeerDiscovery } from '@libp2p/pubsub-peer-discovery';
|
||||||
|
|
||||||
import { MAX_CONCURRENT_DIALS_PER_PEER, MAX_CONNECTIONS, MIN_CONNECTIONS, PUBSUB_DISCOVERY_INTERVAL, PUBSUB_SIGNATURE_POLICY, RELAY_TAG, RELAY_REDIAL_DELAY, PING_TIMEOUT } from './constants.js';
|
import { MAX_CONCURRENT_DIALS_PER_PEER, MAX_CONNECTIONS, MIN_CONNECTIONS, PUBSUB_DISCOVERY_INTERVAL, PUBSUB_SIGNATURE_POLICY, RELAY_TAG, RELAY_REDIAL_DELAY, PING_TIMEOUT, DEFAULT_MAX_RELAY_CONNECTIONS } from './constants.js';
|
||||||
import { PeerHearbeatChecker } from './peer-heartbeat-checker.js';
|
import { PeerHearbeatChecker } from './peer-heartbeat-checker.js';
|
||||||
|
|
||||||
|
const P2P_CIRCUIT_ID = 'p2p-circuit';
|
||||||
export const CHAT_PROTOCOL = '/chat/1.0.0';
|
export const CHAT_PROTOCOL = '/chat/1.0.0';
|
||||||
|
|
||||||
export const ERR_PROTOCOL_SELECTION = 'protocol selection failed';
|
export const ERR_PROTOCOL_SELECTION = 'protocol selection failed';
|
||||||
@ -37,6 +38,7 @@ export class Peer {
|
|||||||
_peerHeartbeatChecker?: PeerHearbeatChecker
|
_peerHeartbeatChecker?: PeerHearbeatChecker
|
||||||
_wrtcTransport: (components: WebRTCDirectComponents) => Transport
|
_wrtcTransport: (components: WebRTCDirectComponents) => Transport
|
||||||
_relayNodeMultiaddr: Multiaddr
|
_relayNodeMultiaddr: Multiaddr
|
||||||
|
_numRelayConnections = 0
|
||||||
|
|
||||||
_peerStreamMap: Map<string, Pushable<any>> = new Map()
|
_peerStreamMap: Map<string, Pushable<any>> = new Map()
|
||||||
_messageHandlers: Array<(peerId: PeerId, message: any) => void> = []
|
_messageHandlers: Array<(peerId: PeerId, message: any) => void> = []
|
||||||
@ -69,7 +71,7 @@ export class Peer {
|
|||||||
return this._node;
|
return this._node;
|
||||||
}
|
}
|
||||||
|
|
||||||
async init (): Promise<void> {
|
async init (maxRelayConnections = DEFAULT_MAX_RELAY_CONNECTIONS): Promise<void> {
|
||||||
try {
|
try {
|
||||||
this._node = await createLibp2p({
|
this._node = await createLibp2p({
|
||||||
addresses: {
|
addresses: {
|
||||||
@ -139,7 +141,7 @@ export class Peer {
|
|||||||
// Listen for peers discovery
|
// Listen for peers discovery
|
||||||
this._node.addEventListener('peer:discovery', (evt) => {
|
this._node.addEventListener('peer:discovery', (evt) => {
|
||||||
// console.log('event peer:discovery', evt);
|
// console.log('event peer:discovery', evt);
|
||||||
this._handleDiscovery(evt.detail);
|
this._handleDiscovery(evt.detail, maxRelayConnections);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Listen for peers connection
|
// Listen for peers connection
|
||||||
@ -285,9 +287,28 @@ export class Peer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
_handleDiscovery (peer: PeerInfo): void {
|
_isRelayPeerMultiaddr (multiaddrString: string): boolean {
|
||||||
|
// Multiaddr not having p2p-circuit id or webrtc-star id is of a relay node
|
||||||
|
return !(multiaddrString.includes(P2P_CIRCUIT_ID) || multiaddrString.includes(P2P_WEBRTC_STAR_ID));
|
||||||
|
}
|
||||||
|
|
||||||
|
_handleDiscovery (peer: PeerInfo, maxRelayConnections: number): void {
|
||||||
// Check connected peers as they are discovered repeatedly.
|
// Check connected peers as they are discovered repeatedly.
|
||||||
if (!this._node?.getPeers().some(remotePeerId => remotePeerId.toString() === peer.id.toString())) {
|
if (!this._node?.getPeers().some(remotePeerId => remotePeerId.toString() === peer.id.toString())) {
|
||||||
|
let isRelayPeer = false;
|
||||||
|
for (const multiaddr of peer.multiaddrs) {
|
||||||
|
if (this._isRelayPeerMultiaddr(multiaddr.toString())) {
|
||||||
|
isRelayPeer = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check relay connections limit if it's a relay peer
|
||||||
|
if (isRelayPeer && this._numRelayConnections >= maxRelayConnections) {
|
||||||
|
// console.log(`Ignoring discovered relay node ${peer.id.toString()} as max relay connections limit reached`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
console.log(`Discovered peer ${peer.id.toString()} with multiaddrs`, peer.multiaddrs.map(addr => addr.toString()));
|
console.log(`Discovered peer ${peer.id.toString()} with multiaddrs`, peer.multiaddrs.map(addr => addr.toString()));
|
||||||
this._connectPeer(peer);
|
this._connectPeer(peer);
|
||||||
}
|
}
|
||||||
@ -296,9 +317,14 @@ export class Peer {
|
|||||||
async _handleConnect (connection: Connection): Promise<void> {
|
async _handleConnect (connection: Connection): Promise<void> {
|
||||||
assert(this._node);
|
assert(this._node);
|
||||||
const remotePeerId = connection.remotePeer;
|
const remotePeerId = connection.remotePeer;
|
||||||
|
const remoteAddrString = connection.remoteAddr.toString();
|
||||||
|
|
||||||
// Log connected peer
|
// Log connected peer
|
||||||
console.log(`Connected to ${remotePeerId.toString()} using multiaddr ${connection.remoteAddr.toString()}`);
|
console.log(`Connected to ${remotePeerId.toString()} using multiaddr ${remoteAddrString}`);
|
||||||
|
|
||||||
|
if (this._isRelayPeerMultiaddr(remoteAddrString)) {
|
||||||
|
this._numRelayConnections++;
|
||||||
|
}
|
||||||
|
|
||||||
// Manage connections and stream if peer id is smaller to break symmetry
|
// Manage connections and stream if peer id is smaller to break symmetry
|
||||||
if (this._node.peerId.toString() < remotePeerId.toString()) {
|
if (this._node.peerId.toString() < remotePeerId.toString()) {
|
||||||
@ -307,7 +333,7 @@ export class Peer {
|
|||||||
// Keep only one connection with a peer
|
// Keep only one connection with a peer
|
||||||
if (remoteConnections.length > 1) {
|
if (remoteConnections.length > 1) {
|
||||||
// Close new connection if using relayed multiaddr
|
// Close new connection if using relayed multiaddr
|
||||||
if (connection.remoteAddr.protoNames().includes('p2p-circuit')) {
|
if (connection.remoteAddr.protoNames().includes(P2P_CIRCUIT_ID)) {
|
||||||
console.log('Closing new connection for already connected peer');
|
console.log('Closing new connection for already connected peer');
|
||||||
await connection.close();
|
await connection.close();
|
||||||
console.log('Closed');
|
console.log('Closed');
|
||||||
@ -358,11 +384,16 @@ export class Peer {
|
|||||||
async _handleDisconnect (connection: Connection): Promise<void> {
|
async _handleDisconnect (connection: Connection): Promise<void> {
|
||||||
assert(this._node);
|
assert(this._node);
|
||||||
const disconnectedPeerId = connection.remotePeer;
|
const disconnectedPeerId = connection.remotePeer;
|
||||||
|
const remoteAddrString = connection.remoteAddr.toString();
|
||||||
|
|
||||||
// Log disconnected peer
|
// Log disconnected peer
|
||||||
console.log(`Disconnected from ${disconnectedPeerId.toString()} using multiaddr ${connection.remoteAddr.toString()}`);
|
console.log(`Disconnected from ${disconnectedPeerId.toString()} using multiaddr ${remoteAddrString}`);
|
||||||
console.log(`Current number of peers connected: ${this._node?.getPeers().length}`);
|
console.log(`Current number of peers connected: ${this._node?.getPeers().length}`);
|
||||||
|
|
||||||
|
if (this._isRelayPeerMultiaddr(remoteAddrString)) {
|
||||||
|
this._numRelayConnections--;
|
||||||
|
}
|
||||||
|
|
||||||
const peerConnections = this._node.getConnections(disconnectedPeerId);
|
const peerConnections = this._node.getConnections(disconnectedPeerId);
|
||||||
|
|
||||||
if (!peerConnections.length) {
|
if (!peerConnections.length) {
|
||||||
|
Loading…
Reference in New Issue
Block a user