Add Executor and RelaySet
This commit is contained in:
@@ -0,0 +1,56 @@
|
||||
import type {EventBus} from './util/EventBus.ts'
|
||||
|
||||
type ExecutorTarget = {
|
||||
bus: EventBus
|
||||
send: (verb: string, ...args) => void
|
||||
}
|
||||
|
||||
export class Executor {
|
||||
target: ExecutorTarget
|
||||
constructor(target) {
|
||||
this.target = target
|
||||
}
|
||||
subscribe(filters, id, {onEvent, onEose}) {
|
||||
const [eventChannel, eoseChannel] = [
|
||||
this.target.bus.on("EVENT", (subid, e) => subid === id && onEvent?.(e)),
|
||||
this.target.bus.on("EOSE", subid => subid === id && onEose?.()),
|
||||
]
|
||||
|
||||
this.target.send("REQ", id, ...filters)
|
||||
|
||||
return {
|
||||
unsubscribe: () => {
|
||||
this.target.send("CLOSE", id)
|
||||
this.target.bus.off("EVENT", eventChannel)
|
||||
this.target.bus.off("EOSE", eoseChannel)
|
||||
},
|
||||
}
|
||||
}
|
||||
publish(event, {onOk, onError}) {
|
||||
const withCleanup = cb => (id, ...payload) => {
|
||||
if (id === event.id) {
|
||||
cb(id, ...payload)
|
||||
this.target.bus.off("OK", okChannel)
|
||||
this.target.bus.off("ERROR", errorChannel)
|
||||
}
|
||||
}
|
||||
|
||||
const [okChannel, errorChannel] = [
|
||||
this.target.bus.on("OK", withCleanup(onOk)),
|
||||
this.target.bus.on("ERROR", withCleanup(onError)),
|
||||
]
|
||||
|
||||
this.target.send("EVENT", event)
|
||||
}
|
||||
count(filter, id, {onCount}) {
|
||||
const channel = this.target.bus.on("COUNT", (subid, ...payload) => {
|
||||
if (subid === id) {
|
||||
onCount(...payload)
|
||||
|
||||
this.target.bus.off("COUNT", channel)
|
||||
}
|
||||
})
|
||||
|
||||
this.target.send("COUNT", id, ...filter)
|
||||
}
|
||||
}
|
||||
+12
-15
@@ -6,6 +6,11 @@ export class Pool {
|
||||
relays: Map<string, Relay>
|
||||
constructor() {
|
||||
this.relays = new Map()
|
||||
this.interval = setInterval(() => {
|
||||
for (const relay of this.relays) {
|
||||
relay.reconnect()
|
||||
}
|
||||
}, 30_000)
|
||||
}
|
||||
add(url) {
|
||||
url = normalizeUrl(url)
|
||||
@@ -22,6 +27,13 @@ export class Pool {
|
||||
this.relays.get(url)?.disconnect()
|
||||
this.relays.delete(url)
|
||||
}
|
||||
cleanup() {
|
||||
this.interval = clearInterval(this.interval)
|
||||
|
||||
for (const url of this.relays.keys()) {
|
||||
this.remove(url)
|
||||
}
|
||||
}
|
||||
async waitFor(url) {
|
||||
const relay = this.add(url)
|
||||
|
||||
@@ -29,19 +41,4 @@ export class Pool {
|
||||
|
||||
return relay.status === Relay.STATUS.READY ? relay : null
|
||||
}
|
||||
async execute(urls, callback) {
|
||||
const results = await Promise.all([
|
||||
urls.map(async url => {
|
||||
const relay = await this.waitFor(url)
|
||||
|
||||
if (!relay) {
|
||||
return null
|
||||
}
|
||||
|
||||
return [relay, callback(relay)]
|
||||
}),
|
||||
])
|
||||
|
||||
return results.filter(Boolean)
|
||||
}
|
||||
}
|
||||
|
||||
+8
-49
@@ -1,6 +1,6 @@
|
||||
import WebSocket from "isomorphic-ws"
|
||||
import {EventBus} from "./EventBus"
|
||||
import {Deferred, defer} from "./Deferred"
|
||||
import {EventBus} from "./util/EventBus"
|
||||
import {Deferred, defer} from "./util/Deferred"
|
||||
|
||||
export class Relay {
|
||||
ws?: WebSocket
|
||||
@@ -78,6 +78,12 @@ export class Relay {
|
||||
|
||||
await this.ready.catch(() => null)
|
||||
}
|
||||
reconnect() {
|
||||
if (this.status === Relay.STATUS.ERROR) {
|
||||
this.status = Relay.STATUS.NEW
|
||||
this.connect()
|
||||
}
|
||||
}
|
||||
disconnect() {
|
||||
if (this.ws) {
|
||||
console.log(`Disconnecting from ${this.url}`)
|
||||
@@ -109,51 +115,4 @@ export class Relay {
|
||||
|
||||
this.ws.send(JSON.stringify(payload))
|
||||
}
|
||||
subscribe(filters, id, {onEvent, onEose}) {
|
||||
const [eventChannel, eoseChannel] = [
|
||||
this.bus.on("EVENT", (subid, e) => subid === id && onEvent?.(e)),
|
||||
this.bus.on("EOSE", subid => subid === id && onEose?.()),
|
||||
]
|
||||
|
||||
this.send("REQ", id, ...filters)
|
||||
|
||||
return {
|
||||
conn: this,
|
||||
unsub: () => {
|
||||
if (this.status === Relay.STATUS.READY) {
|
||||
this.send("CLOSE", id)
|
||||
}
|
||||
|
||||
this.bus.off("EVENT", eventChannel)
|
||||
this.bus.off("EOSE", eoseChannel)
|
||||
},
|
||||
}
|
||||
}
|
||||
publish(event, {onOk, onError}) {
|
||||
const withCleanup = cb => k => {
|
||||
if (k === event.id) {
|
||||
cb()
|
||||
this.bus.off("OK", okChannel)
|
||||
this.bus.off("ERROR", errorChannel)
|
||||
}
|
||||
}
|
||||
|
||||
const [okChannel, errorChannel] = [
|
||||
this.bus.on("OK", withCleanup(onOk)),
|
||||
this.bus.on("ERROR", withCleanup(onError)),
|
||||
]
|
||||
|
||||
this.send("EVENT", event)
|
||||
}
|
||||
count(filter, id, {onCount}) {
|
||||
const channel = this.bus.on("COUNT", (subid, ...payload) => {
|
||||
if (subid === id) {
|
||||
onCount(...payload)
|
||||
|
||||
this.bus.off("COUNT", channel)
|
||||
}
|
||||
})
|
||||
|
||||
this.send("COUNT", id, ...filter)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,24 @@
|
||||
import type {Relay} from './Relay'
|
||||
import {EventBus} from './util/EventBus'
|
||||
|
||||
export class RelaySet {
|
||||
relays: Relay[]
|
||||
bus: EventBus
|
||||
constructor(relays) {
|
||||
this.relays = relays
|
||||
this.bus = new EventBus()
|
||||
|
||||
relays.forEach(relay => {
|
||||
relay.bus.pipe(EventBus.ANY, this.bus)
|
||||
})
|
||||
}
|
||||
send(...payload) {
|
||||
this.relays.forEach(async relay => {
|
||||
await relay.connect()
|
||||
|
||||
if (relay.status === Relay.STATUS.READY) {
|
||||
relay.send(...payload)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
+2
-2
@@ -1,4 +1,4 @@
|
||||
export * from "./EventBus"
|
||||
export * from "./Deferred"
|
||||
export * from "./util/EventBus"
|
||||
export * from "./util/Deferred"
|
||||
export * from "./Relay"
|
||||
export * from "./Pool"
|
||||
|
||||
@@ -10,5 +10,5 @@ export const defer = (): Deferred<any> => {
|
||||
reject = reject_
|
||||
})
|
||||
|
||||
return Object.assign(p, {resolve, reject})
|
||||
return Object.assign(p, {resolve, reject}) as any
|
||||
}
|
||||
@@ -1,11 +1,13 @@
|
||||
export type EventBusHandler = (...args: any[]) => void
|
||||
export type EventBusListener = {
|
||||
id: string
|
||||
handler: (...args: any[]) => void
|
||||
handler: EventBusHandler
|
||||
}
|
||||
|
||||
export class EventBus {
|
||||
static ANY = Math.random().toString().slice(2)
|
||||
listeners: Record<string, Array<EventBusListener>> = {}
|
||||
on(name, handler) {
|
||||
on(name: string, handler: EventBusHandler) {
|
||||
const id = Math.random().toString().slice(2)
|
||||
|
||||
this.listeners[name] = this.listeners[name] || ([] as Array<EventBusListener>)
|
||||
@@ -13,12 +15,22 @@ export class EventBus {
|
||||
|
||||
return id
|
||||
}
|
||||
off(name, id) {
|
||||
off(name: string, id: string) {
|
||||
this.listeners[name] = this.listeners[name].filter(l => l.id !== id)
|
||||
}
|
||||
handle(k, ...payload) {
|
||||
clear() {
|
||||
this.listeners = {}
|
||||
}
|
||||
handle(k: string, ...payload: any) {
|
||||
for (const {handler} of this.listeners[k] || []) {
|
||||
handler(...payload)
|
||||
}
|
||||
|
||||
for (const {handler} of this.listeners[EventBus.ANY] || []) {
|
||||
handler(k, ...payload)
|
||||
}
|
||||
}
|
||||
pipe(k: string, bus: EventBus) {
|
||||
this.on(k, (...payload: any[]) => bus.handle(k, ...payload))
|
||||
}
|
||||
}
|
||||
+1
-1
@@ -15,7 +15,7 @@
|
||||
"scripts": {
|
||||
"build": "node build.js",
|
||||
"pub": "npm i && node build.js && npm publish",
|
||||
"check:ts": "tsc --noEmit --esModuleInterop lib/*",
|
||||
"check:ts": "tsc --noEmit --esModuleInterop --strict lib/**/*",
|
||||
"check:es": "eslint lib/*",
|
||||
"check": "run-p check:*"
|
||||
},
|
||||
|
||||
Reference in New Issue
Block a user