allow the client to react to ok

This commit is contained in:
ennmichael 2023-03-02 23:47:49 +01:00
parent 3c78f740e0
commit 191ec8d7f0
No known key found for this signature in database
GPG Key ID: 6E6E183431A26AF7
4 changed files with 102 additions and 36 deletions

View File

@ -32,16 +32,16 @@ export class Conn {
} }
constructor({ constructor({
endpoint, url,
onMessage, onMessage,
onError, onError,
}: { }: {
endpoint: string | URL url: URL
onMessage: (msg: IncomingMessage) => void onMessage: (msg: IncomingMessage) => void
onError: (err: unknown) => void onError: (err: unknown) => void
}) { }) {
this.#onError = onError this.#onError = onError
this.#socket = new WebSocket(endpoint) this.#socket = new WebSocket(url)
// Handle incoming messages. // Handle incoming messages.
this.#socket.addEventListener("message", async (msgData) => { this.#socket.addEventListener("message", async (msgData) => {

View File

@ -1,5 +1,6 @@
import Base from "events" import Base from "events"
import { EventParams, Nostr } from "." import { Nostr, SubscriptionId } from "."
import { EventId, RawEvent, SignedEvent } from "../event"
/** /**
* Overrides providing better types for EventEmitter methods. * Overrides providing better types for EventEmitter methods.
@ -11,6 +12,7 @@ export class EventEmitter extends Base {
listener: RemoveListener listener: RemoveListener
): this ): this
override addListener(eventName: "notice", listener: NoticeListener): this override addListener(eventName: "notice", listener: NoticeListener): this
override addListener(eventName: "ok", listener: OkListener): this
override addListener(eventName: "error", listener: ErrorListener): this override addListener(eventName: "error", listener: ErrorListener): this
override addListener(eventName: "newListener", listener: ErrorListener): this override addListener(eventName: "newListener", listener: ErrorListener): this
override addListener(eventName: EventName, listener: Listener): this { override addListener(eventName: EventName, listener: Listener): this {
@ -21,6 +23,7 @@ export class EventEmitter extends Base {
override emit(eventName: "removeListener", listener: RemoveListener): boolean override emit(eventName: "removeListener", listener: RemoveListener): boolean
override emit(eventName: "event", params: EventParams, nostr: Nostr): boolean override emit(eventName: "event", params: EventParams, nostr: Nostr): boolean
override emit(eventName: "notice", notice: string, nostr: Nostr): boolean override emit(eventName: "notice", notice: string, nostr: Nostr): boolean
override emit(eventName: "ok", params: OkParams, nostr: Nostr): boolean
override emit(eventName: "error", err: unknown, nostr: Nostr): boolean override emit(eventName: "error", err: unknown, nostr: Nostr): boolean
override emit(eventName: EventName, ...args: unknown[]): boolean { override emit(eventName: EventName, ...args: unknown[]): boolean {
return super.emit(eventName, ...args) return super.emit(eventName, ...args)
@ -34,6 +37,7 @@ export class EventEmitter extends Base {
override listeners(eventName: "removeListener"): EventListener[] override listeners(eventName: "removeListener"): EventListener[]
override listeners(eventName: "event"): EventListener[] override listeners(eventName: "event"): EventListener[]
override listeners(eventName: "notice"): NoticeListener[] override listeners(eventName: "notice"): NoticeListener[]
override listeners(eventName: "ok"): OkListener[]
override listeners(eventName: "error"): ErrorListener[] override listeners(eventName: "error"): ErrorListener[]
override listeners(eventName: EventName): Listener[] { override listeners(eventName: EventName): Listener[] {
return super.listeners(eventName) as Listener[] return super.listeners(eventName) as Listener[]
@ -43,6 +47,7 @@ export class EventEmitter extends Base {
override off(eventName: "removeListener", listener: RemoveListener): this override off(eventName: "removeListener", listener: RemoveListener): this
override off(eventName: "event", listener: EventListener): this override off(eventName: "event", listener: EventListener): this
override off(eventName: "notice", listener: NoticeListener): this override off(eventName: "notice", listener: NoticeListener): this
override off(eventName: "ok", listener: OkListener): this
override off(eventName: "error", listener: ErrorListener): this override off(eventName: "error", listener: ErrorListener): this
override off(eventName: EventName, listener: Listener): this { override off(eventName: EventName, listener: Listener): this {
return super.off(eventName, listener) return super.off(eventName, listener)
@ -52,6 +57,7 @@ export class EventEmitter extends Base {
override on(eventName: "removeListener", listener: RemoveListener): this override on(eventName: "removeListener", listener: RemoveListener): this
override on(eventName: "event", listener: EventListener): this override on(eventName: "event", listener: EventListener): this
override on(eventName: "notice", listener: NoticeListener): this override on(eventName: "notice", listener: NoticeListener): this
override on(eventName: "ok", listener: OkListener): this
override on(eventName: "error", listener: ErrorListener): this override on(eventName: "error", listener: ErrorListener): this
override on(eventName: EventName, listener: Listener): this { override on(eventName: EventName, listener: Listener): this {
return super.on(eventName, listener) return super.on(eventName, listener)
@ -61,6 +67,7 @@ export class EventEmitter extends Base {
override once(eventName: "removeListener", listener: RemoveListener): this override once(eventName: "removeListener", listener: RemoveListener): this
override once(eventName: "event", listener: EventListener): this override once(eventName: "event", listener: EventListener): this
override once(eventName: "notice", listener: NoticeListener): this override once(eventName: "notice", listener: NoticeListener): this
override once(eventName: "ok", listener: OkListener): this
override once(eventName: "error", listener: ErrorListener): this override once(eventName: "error", listener: ErrorListener): this
override once(eventName: EventName, listener: Listener): this { override once(eventName: EventName, listener: Listener): this {
return super.once(eventName, listener) return super.once(eventName, listener)
@ -76,6 +83,7 @@ export class EventEmitter extends Base {
): this ): this
override prependListener(eventName: "event", listener: EventListener): this override prependListener(eventName: "event", listener: EventListener): this
override prependListener(eventName: "notice", listener: NoticeListener): this override prependListener(eventName: "notice", listener: NoticeListener): this
override prependListener(eventName: "ok", listener: OkListener): this
override prependListener(eventName: "error", listener: ErrorListener): this override prependListener(eventName: "error", listener: ErrorListener): this
override prependListener(eventName: EventName, listener: Listener): this { override prependListener(eventName: EventName, listener: Listener): this {
return super.prependListener(eventName, listener) return super.prependListener(eventName, listener)
@ -97,6 +105,7 @@ export class EventEmitter extends Base {
eventName: "notice", eventName: "notice",
listener: NoticeListener listener: NoticeListener
): this ): this
override prependOnceListener(eventName: "ok", listener: OkListener): this
override prependOnceListener( override prependOnceListener(
eventName: "error", eventName: "error",
listener: ErrorListener listener: ErrorListener
@ -116,6 +125,7 @@ export class EventEmitter extends Base {
): this ): this
override removeListener(eventName: "event", listener: EventListener): this override removeListener(eventName: "event", listener: EventListener): this
override removeListener(eventName: "notice", listener: NoticeListener): this override removeListener(eventName: "notice", listener: NoticeListener): this
override removeListener(eventName: "ok", listener: OkListener): this
override removeListener(eventName: "error", listener: ErrorListener): this override removeListener(eventName: "error", listener: ErrorListener): this
override removeListener(eventName: EventName, listener: Listener): this { override removeListener(eventName: EventName, listener: Listener): this {
return super.removeListener(eventName, listener) return super.removeListener(eventName, listener)
@ -131,15 +141,40 @@ export class EventEmitter extends Base {
// TODO Also add on: ("subscribed", subscriptionId) which checks "OK"/"NOTICE" and makes a callback? // TODO Also add on: ("subscribed", subscriptionId) which checks "OK"/"NOTICE" and makes a callback?
// TODO Also add on: ("ok", boolean, eventId) which checks "OK"/"NOTICE" and makes a callback? // TODO Also add on: ("ok", boolean, eventId) which checks "OK"/"NOTICE" and makes a callback?
type EventName = "newListener" | "removeListener" | "event" | "notice" | "error" type EventName =
| "newListener"
| "removeListener"
| "event"
| "notice"
| "ok"
| "error"
type NewListener = (eventName: EventName, listener: Listener) => void type NewListener = (eventName: EventName, listener: Listener) => void
type RemoveListener = (eventName: EventName, listener: Listener) => void type RemoveListener = (eventName: EventName, listener: Listener) => void
type EventListener = (params: EventParams, nostr: Nostr) => void type EventListener = (params: EventParams, nostr: Nostr) => void
type OkListener = (params: OkParams, nostr: Nostr) => void
type NoticeListener = (notice: string, nostr: Nostr) => void type NoticeListener = (notice: string, nostr: Nostr) => void
type ErrorListener = (error: unknown, nostr: Nostr) => void type ErrorListener = (error: unknown, nostr: Nostr) => void
type Listener = type Listener =
| NewListener | NewListener
| RemoveListener | RemoveListener
| EventListener | EventListener
| NoticeListener | NoticeListener
| OkListener
| ErrorListener | ErrorListener
// TODO Document this
export interface EventParams {
signed: SignedEvent
subscriptionId: SubscriptionId
raw: RawEvent
}
// TODO Document this
export interface OkParams {
eventId: EventId
relay: URL
ok: boolean
message: string
}

View File

@ -46,9 +46,11 @@ export class Nostr extends EventEmitter {
return return
} }
const connUrl = new URL(url)
// If there is no existing connection, open a new one. // If there is no existing connection, open a new one.
const conn = new Conn({ const conn = new Conn({
endpoint: url, url: connUrl,
// Handle messages on this connection. // Handle messages on this connection.
onMessage: (msg) => { onMessage: (msg) => {
try { try {
@ -64,6 +66,17 @@ export class Nostr extends EventEmitter {
) )
} else if (msg.kind === "notice") { } else if (msg.kind === "notice") {
this.emit("notice", msg.notice, this) this.emit("notice", msg.notice, this)
} else if (msg.kind === "ok") {
this.emit(
"ok",
{
eventId: msg.eventId,
relay: connUrl,
ok: msg.ok,
message: msg.message,
},
this
)
} else { } else {
throw new ProtocolError(`invalid message ${msg}`) throw new ProtocolError(`invalid message ${msg}`)
} }
@ -287,10 +300,3 @@ export interface Filters {
until?: Date until?: Date
limit?: number limit?: number
} }
// TODO Document this
export interface EventParams {
signed: SignedEvent
subscriptionId: SubscriptionId
raw: RawEvent
}

View File

@ -1,26 +1,37 @@
import { EventParams, Nostr } from "../src/client" import { Nostr } from "../src/client"
import { EventKind } from "../src/event" import { EventKind, SignedEvent } from "../src/event"
import { PrivateKey } from "../src/keypair" import { PrivateKey } from "../src/keypair"
import assert from "assert" import assert from "assert"
import { EventParams } from "../src/client/emitter"
// TODO Switch out the relay implementation and see if the issue persists // TODO Switch out the relay implementation and see if the issue persists
// TODO Do on("error", done) for all of these
describe("single event communication", function () { describe("simple communication", function () {
it("ok", function (done) { const secret = new PrivateKey(
const secret = new PrivateKey( "nsec1xlu55y6fqfgrq448xslt6a8j2rh7lj08hyhgs94ryq04yf6surwsjl0kzh"
"nsec1xlu55y6fqfgrq448xslt6a8j2rh7lj08hyhgs94ryq04yf6surwsjl0kzh" )
) const pubkey = secret.pubkey
const pubkey = secret.pubkey const timestamp = new Date()
const note = "hello world"
const url = new URL("ws://localhost:12648")
const timestamp = new Date() const publisher = new Nostr()
const note = "hello world" const subscriber = new Nostr()
const publisher = new Nostr() beforeEach(() => {
publisher.open("ws://localhost:12648") publisher.open(url)
const subscriber = new Nostr() subscriber.open(url)
subscriber.open("ws://localhost:12648") })
function listener({ signed: { event } }: EventParams) { afterEach(() => {
publisher.close()
subscriber.close()
})
it("publish and receive", function (done) {
function listener({ signed: { event } }: EventParams, nostr: Nostr) {
assert.equal(nostr, subscriber)
assert.equal(event.kind, EventKind.TextNote) assert.equal(event.kind, EventKind.TextNote)
assert.equal(event.pubkey.toString(), pubkey.toString()) assert.equal(event.pubkey.toString(), pubkey.toString())
assert.equal(event.createdAt.toString(), timestamp.toString()) assert.equal(event.createdAt.toString(), timestamp.toString())
@ -34,19 +45,11 @@ describe("single event communication", function () {
// for future events. // for future events.
subscriber.off("event", listener) subscriber.off("event", listener)
subscriber.on("error", done)
publisher.on("error", done)
publisher.close()
subscriber.close()
done() done()
} }
subscriber.on("event", listener) subscriber.on("event", listener)
subscriber.subscribe([]) subscriber.subscribe([])
publisher.publish( publisher.publish(
{ {
kind: EventKind.TextNote, kind: EventKind.TextNote,
@ -57,4 +60,26 @@ describe("single event communication", function () {
secret secret
) )
}) })
it("publish and ok", function (done) {
SignedEvent.sign(
{
kind: EventKind.TextNote,
createdAt: timestamp,
content: note,
pubkey,
},
secret
).then((event) => {
publisher.on("ok", (params, nostr) => {
assert.equal(nostr, publisher)
assert.equal(params.eventId.toString(), event.eventId.toString())
assert.equal(params.relay.toString(), url.toString())
assert.equal(params.ok, true)
done()
})
publisher.on("error", done)
publisher.publish(event)
})
})
}) })