unify mock-socket interface into a single implementation.

This commit is contained in:
fiatjaf
2024-01-19 16:01:06 -03:00
parent d4090dae2b
commit 8b2b050c0d
3 changed files with 77 additions and 69 deletions

BIN
bun.lockb

Binary file not shown.

View File

@@ -1,78 +1,44 @@
import { expect, test } from 'bun:test' import { expect, test } from 'bun:test'
import { Server } from 'mock-socket'
import { finalizeEvent, generateSecretKey, getPublicKey } from './pure.ts' import { finalizeEvent, generateSecretKey, getPublicKey } from './pure.ts'
import { Relay } from './relay.ts' import { Relay } from './relay.ts'
import { newMockRelay } from './test-helpers.ts'
test('connectivity', async () => { test('connectivity', async () => {
const mockRelayURL = 'wss://mock.relay.url' const { url } = newMockRelay()
const mockRelay = new Server(mockRelayURL) const relay = new Relay(url)
const relay = new Relay(mockRelayURL)
await relay.connect() await relay.connect()
expect(relay.connected).toBeTrue() expect(relay.connected).toBeTrue()
relay.close() relay.close()
mockRelay.stop()
}) })
test('connectivity, with Relay.connect()', async () => { test('connectivity, with Relay.connect()', async () => {
const mockRelayURL = 'wss://mock.relay.url' const { url } = newMockRelay()
const mockRelay = new Server(mockRelayURL) const relay = await Relay.connect(url)
const relay = await Relay.connect(mockRelayURL)
expect(relay.connected).toBeTrue() expect(relay.connected).toBeTrue()
relay.close() relay.close()
mockRelay.stop()
}) })
test('querying', async done => { test('querying', async done => {
const sk = generateSecretKey() const { url, authors } = newMockRelay()
const pk = getPublicKey(sk)
const kind = 0 const kind = 0
const mockRelayURL = 'wss://mock.relay.url' const relay = new Relay(url)
const mockRelay = new Server(mockRelayURL)
mockRelay.on('connection', socket => {
socket.on('message', message => {
const data = JSON.parse(message as string)
const event = finalizeEvent(
{
kind,
content: '',
created_at: Math.floor(Date.now() / 1000),
tags: [],
},
sk,
)
socket.send(JSON.stringify(['EVENT', data[1], event]))
})
})
const relay = new Relay(mockRelayURL)
await relay.connect() await relay.connect()
relay.subscribe( relay.subscribe(
[ [
{ {
authors: [pk], authors: authors,
kinds: [kind], kinds: [kind],
}, },
], ],
{ {
onevent(event) { onevent(event) {
expect(event).toHaveProperty('pubkey', pk) expect(authors).toContain(event.pubkey)
expect(event).toHaveProperty('kind', kind) expect(event).toHaveProperty('kind', kind)
relay.close() relay.close()
mockRelay.stop()
done() done()
}, },
}, },
@@ -84,27 +50,8 @@ test('listening and publishing and closing', async done => {
const pk = getPublicKey(sk) const pk = getPublicKey(sk)
const kind = 23571 const kind = 23571
const mockRelayURL = 'wss://mock.relay.url' const { url } = newMockRelay()
const mockRelay = new Server(mockRelayURL) const relay = new Relay(url)
mockRelay.on('connection', socket => {
let subId: string | null = null
socket.on('message', message => {
const data = JSON.parse(message as string)
if (data[0] === 'REQ') {
subId = data[1]
socket.send(JSON.stringify(['EOSE', data[1]]))
} else if (data[0] === 'EVENT') {
socket.send(JSON.stringify(['OK', data[1].id, 'true']))
socket.send(JSON.stringify(['EVENT', subId, data[1]]))
}
})
})
const relay = new Relay(mockRelayURL)
await relay.connect() await relay.connect()
let sub = relay.subscribe( let sub = relay.subscribe(
@@ -119,14 +66,11 @@ test('listening and publishing and closing', async done => {
expect(event).toHaveProperty('pubkey', pk) expect(event).toHaveProperty('pubkey', pk)
expect(event).toHaveProperty('kind', kind) expect(event).toHaveProperty('kind', kind)
expect(event).toHaveProperty('content', 'content') expect(event).toHaveProperty('content', 'content')
sub.close() sub.close()
}, },
oneose() {}, oneose() {},
onclose() { onclose() {
relay.close() relay.close()
mockRelay.stop()
done() done()
}, },
}, },

View File

@@ -1,6 +1,8 @@
import type { Event } from './pure.ts' import { Server } from 'mock-socket'
import { finalizeEvent, type Event, getPublicKey, generateSecretKey } from './pure.ts'
import { matchFilters, type Filter } from './filter.ts'
/** Build an event for testing purposes. */
export function buildEvent(params: Partial<Event>): Event { export function buildEvent(params: Partial<Event>): Event {
return { return {
id: '', id: '',
@@ -13,3 +15,65 @@ export function buildEvent(params: Partial<Event>): Event {
...params, ...params,
} }
} }
let serial = 0
// the mock relay will always return 3 events before eose and then do ok with everything
export function newMockRelay(): { url: string; authors: string[] } {
serial++
const url = `wss://mock.relay.url/${serial}`
const relay = new Server(url)
const secretKeys = [generateSecretKey(), generateSecretKey(), generateSecretKey(), generateSecretKey()]
relay.on('connection', (conn: any) => {
let subs: { [subId: string]: { conn: any; filters: Filter[] } } = {}
conn.on('message', (message: string) => {
const data = JSON.parse(message)
switch (data[0]) {
case 'REQ': {
let subId = data[1]
let filters = data.slice(2)
subs[subId] = { conn, filters }
filters.forEach((filter: Filter) => {
const kinds = filter.kinds?.length ? filter.kinds : [1]
kinds.forEach(kind => {
secretKeys.forEach(sk => {
const event = finalizeEvent(
{
kind,
content: '',
created_at: Math.floor(Date.now() / 1000),
tags: [],
},
sk,
)
conn.send(JSON.stringify(['EVENT', subId, event]))
})
})
})
conn.send(JSON.stringify(['EOSE', subId]))
break
}
case 'CLOSE': {
let subId = data[1]
delete subs[subId]
break
}
case 'EVENT': {
let event = data[1]
conn.send(JSON.stringify(['OK', event.id, 'true']))
for (let subId in subs) {
const { filters, conn: listener } = subs[subId]
if (matchFilters(filters, event)) {
listener.send(JSON.stringify(['EVENT', subId, event]))
}
}
break
}
}
})
})
return { url, authors: secretKeys.map(getPublicKey) }
}