2021-04-22 04:47:43 +00:00
|
|
|
import Libp2p, { Libp2pConfig, Libp2pModules, Libp2pOptions } from 'libp2p';
|
2021-03-19 03:40:16 +00:00
|
|
|
import Mplex from 'libp2p-mplex';
|
2021-03-22 11:16:12 +00:00
|
|
|
import { bytes } from 'libp2p-noise/dist/src/@types/basic';
|
|
|
|
import { Noise } from 'libp2p-noise/dist/src/noise';
|
2021-04-15 01:19:26 +00:00
|
|
|
import Websockets from 'libp2p-websockets';
|
2021-04-22 00:39:05 +00:00
|
|
|
import filters from 'libp2p-websockets/src/filters';
|
2021-04-20 06:51:04 +00:00
|
|
|
import { Multiaddr } from 'multiaddr';
|
2021-03-23 00:14:51 +00:00
|
|
|
import PeerId from 'peer-id';
|
2021-03-19 03:40:16 +00:00
|
|
|
|
2021-04-16 01:25:08 +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
|
|
|
|
2021-04-22 00:39:05 +00:00
|
|
|
const transportKey = Websockets.prototype[Symbol.toStringTag];
|
|
|
|
|
2021-04-22 04:47:43 +00:00
|
|
|
export type CreateOptions =
|
|
|
|
| {
|
|
|
|
listenAddresses: string[] | undefined;
|
|
|
|
staticNoiseKey: bytes | undefined;
|
|
|
|
modules: Partial<Libp2pModules>;
|
|
|
|
config: Partial<Libp2pConfig>;
|
|
|
|
}
|
|
|
|
| (Libp2pOptions & import('libp2p').CreateOptions);
|
2021-03-29 02:56:17 +00:00
|
|
|
|
2021-03-19 03:40:16 +00:00
|
|
|
export default class Waku {
|
2021-04-16 01:25:08 +00:00
|
|
|
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
|
|
|
|
2021-03-22 11:16:12 +00:00
|
|
|
/**
|
|
|
|
* Create new waku node
|
2021-04-15 01:19:26 +00:00
|
|
|
* @param listenAddresses: Array of Multiaddrs on which the node should listen.
|
|
|
|
* If not present, the node is dial only.
|
2021-03-22 11:16:12 +00:00
|
|
|
* @param staticNoiseKey: A static key to use for noise,
|
|
|
|
* mainly used for test to reduce entropy usage.
|
2021-04-15 01:19:26 +00:00
|
|
|
* @throws If
|
2021-03-22 11:16:12 +00:00
|
|
|
* @returns {Promise<Waku>}
|
|
|
|
*/
|
2021-03-29 02:56:17 +00:00
|
|
|
static async create(options: Partial<CreateOptions>): Promise<Waku> {
|
|
|
|
const opts = Object.assign(
|
|
|
|
{
|
2021-04-15 01:19:26 +00:00
|
|
|
listenAddresses: [],
|
2021-03-29 02:56:17 +00:00
|
|
|
staticNoiseKey: undefined,
|
|
|
|
},
|
|
|
|
options
|
|
|
|
);
|
|
|
|
|
2021-04-22 04:47:43 +00:00
|
|
|
opts.config = Object.assign(
|
|
|
|
{
|
|
|
|
transport: {
|
|
|
|
[transportKey]: {
|
|
|
|
filter: filters.all,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
options.config
|
|
|
|
);
|
|
|
|
|
|
|
|
opts.modules = Object.assign({}, options.modules);
|
|
|
|
|
2021-04-15 01:19:26 +00:00
|
|
|
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: {
|
2021-04-15 01:19:26 +00:00
|
|
|
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
|
2021-04-16 01:25:08 +00:00
|
|
|
pubsub: WakuRelay,
|
2021-03-19 03:40:16 +00:00
|
|
|
},
|
2021-04-22 04:47:43 +00:00
|
|
|
config: opts.config,
|
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();
|
|
|
|
|
2021-04-16 01:25:08 +00:00
|
|
|
return new Waku(libp2p, wakuStore);
|
2021-03-19 03:40:16 +00:00
|
|
|
}
|
2021-03-19 05:07:56 +00:00
|
|
|
|
2021-03-23 00:14:51 +00:00
|
|
|
/**
|
2021-04-07 01:04:30 +00:00
|
|
|
* Dials to the provided peer.
|
2021-03-23 00:14:51 +00:00
|
|
|
* @param peer The peer to dial
|
|
|
|
*/
|
|
|
|
async dial(peer: PeerId | Multiaddr | string) {
|
2021-04-13 12:36:37 +00:00
|
|
|
await this.libp2p.dialProtocol(peer, [RelayCodec, StoreCodec]);
|
2021-03-23 00:14:51 +00:00
|
|
|
}
|
|
|
|
|
2021-05-04 00:23:31 +00:00
|
|
|
addPeerToAddressBook(peerId: PeerId, multiaddr: Multiaddr[]) {
|
2021-03-23 00:14:51 +00:00
|
|
|
this.libp2p.peerStore.addressBook.set(peerId, multiaddr);
|
|
|
|
}
|
|
|
|
|
2021-03-19 05:07:56 +00:00
|
|
|
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
|
|
|
}
|