js-waku/src/lib/waku.ts

124 lines
3.4 KiB
TypeScript
Raw Normal View History

2021-03-19 03:40:16 +00:00
import Libp2p from 'libp2p';
import Mplex from 'libp2p-mplex';
import { bytes } from 'libp2p-noise/dist/src/@types/basic';
import { Noise } from 'libp2p-noise/dist/src/noise';
import Websockets from 'libp2p-websockets';
import filters from 'libp2p-websockets/src/filters';
2021-04-20 06:51:04 +00:00
import { Multiaddr } from 'multiaddr';
import PeerId from 'peer-id';
2021-03-19 03:40:16 +00:00
import { RelayCodec, WakuRelay } from './waku_relay';
2021-04-07 01:04:30 +00:00
import { StoreCodec, WakuStore } from './waku_store';
2021-03-19 03:40:16 +00:00
const transportKey = Websockets.prototype[Symbol.toStringTag];
2021-03-29 02:56:17 +00:00
export interface CreateOptions {
listenAddresses: string[];
staticNoiseKey: bytes | undefined;
modules: {
transport: import('libp2p-interfaces/src/transport/types').TransportFactory<
any,
any
>[];
};
2021-03-29 02:56:17 +00:00
}
2021-03-19 03:40:16 +00:00
export default class Waku {
public libp2p: Libp2p;
public relay: WakuRelay;
public store: WakuStore;
private constructor(libp2p: Libp2p, store: WakuStore) {
this.libp2p = libp2p;
this.relay = (libp2p.pubsub as unknown) as WakuRelay;
this.store = store;
}
2021-03-19 03:40:16 +00:00
/**
* Create new waku node
* @param listenAddresses: Array of Multiaddrs on which the node should listen.
* If not present, the node is dial only.
* @param staticNoiseKey: A static key to use for noise,
* mainly used for test to reduce entropy usage.
* @throws If
* @returns {Promise<Waku>}
*/
2021-03-29 02:56:17 +00:00
static async create(options: Partial<CreateOptions>): Promise<Waku> {
const opts = Object.assign(
{
listenAddresses: [],
2021-03-29 02:56:17 +00:00
staticNoiseKey: undefined,
},
options
);
let transport = [Websockets];
if (opts.modules?.transport) {
transport = transport.concat(opts.modules?.transport);
}
// FIXME: By controlling the creation of libp2p we have to think about what
// needs to be exposed and what does not. Ideally, we should be able to let
// the user create the WakuStore, WakuRelay instances and pass them when
// creating the libp2p instance.
2021-03-19 03:40:16 +00:00
const libp2p = await Libp2p.create({
addresses: {
2021-03-29 02:56:17 +00:00
listen: opts.listenAddresses,
2021-03-19 03:40:16 +00:00
},
modules: {
transport,
2021-03-19 03:40:16 +00:00
streamMuxer: [Mplex],
2021-03-29 02:56:17 +00:00
connEncryption: [new Noise(opts.staticNoiseKey)],
2021-03-19 03:40:16 +00:00
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore: Type needs update
pubsub: WakuRelay,
2021-03-19 03:40:16 +00:00
},
config: {
transport: {
[transportKey]: {
filter: filters.all,
},
},
},
2021-03-19 03:40:16 +00:00
});
2021-04-07 01:04:30 +00:00
const wakuStore = new WakuStore(libp2p);
2021-03-19 03:40:16 +00:00
await libp2p.start();
return new Waku(libp2p, wakuStore);
2021-03-19 03:40:16 +00:00
}
/**
2021-04-07 01:04:30 +00:00
* Dials to the provided peer.
* @param peer The peer to dial
*/
async dial(peer: PeerId | Multiaddr | string) {
await this.libp2p.dialProtocol(peer, [RelayCodec, StoreCodec]);
}
async dialWithMultiAddr(peerId: PeerId, multiaddr: Multiaddr[]) {
this.libp2p.peerStore.addressBook.set(peerId, multiaddr);
}
async stop() {
await this.libp2p.stop();
}
2021-04-06 01:06:10 +00:00
/**
* Return the local multiaddr with peer id on which libp2p is listening.
* @throws if libp2p is not listening on localhost
*/
getLocalMultiaddrWithID(): string {
const localMultiaddr = this.libp2p.multiaddrs.find((addr) =>
addr.toString().match(/127\.0\.0\.1/)
);
if (!localMultiaddr || localMultiaddr.toString() === '') {
throw 'Not listening on localhost';
}
const multiAddrWithId =
localMultiaddr + '/p2p/' + this.libp2p.peerId.toB58String();
return multiAddrWithId;
}
2021-03-19 03:40:16 +00:00
}