mirror of
https://github.com/logos-messaging/logos-delivery-js.git
synced 2026-03-09 19:23:18 +00:00
146 lines
3.8 KiB
TypeScript
146 lines
3.8 KiB
TypeScript
import type { PeerId } from "@libp2p/interface";
|
|
import { LightPushCore } from "@waku/core";
|
|
import {
|
|
type IEncoder,
|
|
ILightPush,
|
|
type IMessage,
|
|
IProtoMessage,
|
|
type ISendOptions,
|
|
type Libp2p,
|
|
LightPushCoreResult,
|
|
LightPushError,
|
|
LightPushFailure,
|
|
type LightPushProtocolOptions,
|
|
LightPushSDKResult,
|
|
Protocols
|
|
} from "@waku/interfaces";
|
|
import { Logger } from "@waku/utils";
|
|
|
|
import { PeerManager } from "../peer_manager/index.js";
|
|
|
|
import { RetryManager } from "./retry_manager.js";
|
|
|
|
const log = new Logger("sdk:light-push");
|
|
|
|
const DEFAULT_MAX_ATTEMPTS = 3;
|
|
const DEFAULT_SEND_OPTIONS: LightPushProtocolOptions = {
|
|
autoRetry: true,
|
|
retryIntervalMs: 1000,
|
|
maxAttempts: DEFAULT_MAX_ATTEMPTS,
|
|
numPeersToUse: 1
|
|
};
|
|
|
|
type LightPushConstructorParams = {
|
|
peerManager: PeerManager;
|
|
libp2p: Libp2p;
|
|
options?: Partial<LightPushProtocolOptions>;
|
|
};
|
|
|
|
export class LightPush implements ILightPush {
|
|
private readonly config: LightPushProtocolOptions;
|
|
private readonly retryManager: RetryManager;
|
|
private readonly peerManager: PeerManager;
|
|
private readonly protocol: LightPushCore;
|
|
|
|
public constructor(params: LightPushConstructorParams) {
|
|
this.config = {
|
|
...DEFAULT_SEND_OPTIONS,
|
|
...(params.options || {})
|
|
} as LightPushProtocolOptions;
|
|
|
|
this.peerManager = params.peerManager;
|
|
this.protocol = new LightPushCore(params.libp2p);
|
|
this.retryManager = new RetryManager({
|
|
peerManager: params.peerManager,
|
|
retryIntervalMs: this.config.retryIntervalMs
|
|
});
|
|
}
|
|
|
|
public get multicodec(): string[] {
|
|
return this.protocol.multicodec;
|
|
}
|
|
|
|
public start(): void {
|
|
this.retryManager.start();
|
|
}
|
|
|
|
public stop(): void {
|
|
this.retryManager.stop();
|
|
}
|
|
|
|
public async send(
|
|
encoder: IEncoder,
|
|
message: IMessage,
|
|
options: ISendOptions = {}
|
|
): Promise<LightPushSDKResult> {
|
|
options = {
|
|
useLegacy: false,
|
|
...this.config,
|
|
...options
|
|
};
|
|
|
|
const { pubsubTopic } = encoder;
|
|
|
|
log.info("send: attempting to send a message to pubsubTopic:", pubsubTopic);
|
|
|
|
let peerIds = await this.peerManager.getPeers({
|
|
protocol: options.useLegacy ? "light-push-v2" : Protocols.LightPush,
|
|
pubsubTopic: encoder.pubsubTopic
|
|
});
|
|
peerIds = peerIds.slice(0, options.numPeersToUse);
|
|
|
|
const coreResults =
|
|
peerIds?.length > 0
|
|
? await Promise.all(
|
|
peerIds.map((peerId) =>
|
|
this.protocol
|
|
.send(encoder, message, peerId, options.useLegacy)
|
|
.catch(
|
|
(_e) =>
|
|
({
|
|
success: null,
|
|
failure: {
|
|
error: LightPushError.GENERIC_FAIL
|
|
}
|
|
}) as LightPushCoreResult
|
|
)
|
|
)
|
|
)
|
|
: [];
|
|
|
|
const results: LightPushSDKResult = coreResults.length
|
|
? {
|
|
successes: coreResults
|
|
.filter((v) => v.success)
|
|
.map((v) => v.success) as PeerId[],
|
|
failures: coreResults
|
|
.filter((v) => v.failure)
|
|
.map((v) => v.failure) as LightPushFailure[],
|
|
messages: coreResults
|
|
.filter((v) => v.message)
|
|
.map((v) => v.message) as IProtoMessage[]
|
|
}
|
|
: {
|
|
successes: [],
|
|
failures: [
|
|
{
|
|
error: LightPushError.NO_PEER_AVAILABLE
|
|
}
|
|
]
|
|
};
|
|
|
|
if (options.autoRetry && results.successes.length === 0) {
|
|
const sendCallback = (peerId: PeerId): Promise<LightPushCoreResult> =>
|
|
this.protocol.send(encoder, message, peerId, options.useLegacy);
|
|
|
|
this.retryManager.push(
|
|
sendCallback.bind(this),
|
|
options.maxAttempts || DEFAULT_MAX_ATTEMPTS,
|
|
encoder.routingInfo
|
|
);
|
|
}
|
|
|
|
return results;
|
|
}
|
|
}
|