mtcute/packages/dispatcher/tests/dispatcher.test.ts

275 lines
8.4 KiB
TypeScript
Raw Normal View History

2024-05-01 20:13:22 +03:00
import { Message, PeersIndex } from '@mtcute/core'
import { TelegramClient } from '@mtcute/core/client.js'
import { createStub, StubTelegramClient } from '@mtcute/test'
import { describe, expect, it } from 'vitest'
import { Dispatcher, PropagationAction } from '../src/index.js'
describe('Dispatcher', () => {
// todo: replace with proper mocked TelegramClient
const client = new TelegramClient({ client: new StubTelegramClient({ disableUpdates: false }) })
const emptyPeers = new PeersIndex()
describe('Raw updates', () => {
it('registers and unregisters handlers for raw updates', async () => {
2023-10-26 23:54:04 +03:00
const dp = Dispatcher.for(client)
const log: string[] = []
dp.onRawUpdate((cl, upd) => {
log.push(`(first) received ${upd._}`)
return PropagationAction.Continue
})
dp.onRawUpdate((cl, upd) => {
log.push(`(second) received ${upd._}`)
return PropagationAction.Continue
})
await dp.dispatchRawUpdateNow({ _: 'updateConfig' }, emptyPeers)
dp.removeUpdateHandler('raw')
await dp.dispatchRawUpdateNow({ _: 'updateConfig' }, emptyPeers)
2023-09-24 01:32:22 +03:00
expect(log).eql(['(first) received updateConfig', '(second) received updateConfig'])
})
it('supports filters for raw updates', async () => {
2023-10-26 23:54:04 +03:00
const dp = Dispatcher.for(client)
const log: string[] = []
dp.onRawUpdate((cl, upd) => {
log.push(`(no) received ${upd._}`)
return PropagationAction.Continue
})
dp.onRawUpdate(
() => true,
(cl, upd) => {
log.push(`(true) received ${upd._}`)
return PropagationAction.Continue
},
)
dp.onRawUpdate(
() => false,
(cl, upd) => {
log.push(`(false) received ${upd._}`)
return PropagationAction.Continue
},
)
await dp.dispatchRawUpdateNow({ _: 'updateConfig' }, emptyPeers)
2023-09-24 01:32:22 +03:00
expect(log).eql(['(no) received updateConfig', '(true) received updateConfig'])
})
})
// todo: other update types
describe('Filter groups', () => {
it('does separate propagation for filter groups', async () => {
2023-10-26 23:54:04 +03:00
const dp = Dispatcher.for(client)
const log: string[] = []
dp.onRawUpdate((cl, upd) => {
log.push(`(grp0) received ${upd._}`)
}, 0)
dp.onRawUpdate((cl, upd) => {
log.push(`(grp0 2) received ${upd._}`)
}, 0)
dp.onRawUpdate((cl, upd) => {
log.push(`(grp1) received ${upd._}`)
}, 1)
dp.onRawUpdate((cl, upd) => {
log.push(`(grp2) received ${upd._}`)
}, 2)
await dp.dispatchRawUpdateNow({ _: 'updateConfig' }, emptyPeers)
expect(log).eql([
'(grp0) received updateConfig',
'(grp1) received updateConfig',
'(grp2) received updateConfig',
])
})
it('allows continuing propagation in the same group with ContinuePropagation', async () => {
2023-10-26 23:54:04 +03:00
const dp = Dispatcher.for(client)
const log: string[] = []
dp.onRawUpdate((cl, upd) => {
log.push(`(grp0) received ${upd._}`)
return PropagationAction.Continue
}, 0)
dp.onRawUpdate((cl, upd) => {
log.push(`(grp0 2) received ${upd._}`)
}, 0)
dp.onRawUpdate((cl, upd) => {
log.push(`(grp1) received ${upd._}`)
}, 1)
dp.onRawUpdate((cl, upd) => {
log.push(`(grp1 2) received ${upd._}`)
}, 1)
dp.onRawUpdate((cl, upd) => {
log.push(`(grp2) received ${upd._}`)
}, 2)
await dp.dispatchRawUpdateNow({ _: 'updateConfig' }, emptyPeers)
expect(log).eql([
'(grp0) received updateConfig',
'(grp0 2) received updateConfig',
'(grp1) received updateConfig',
'(grp2) received updateConfig',
])
})
it('allows stopping any further propagation with Stop', async () => {
2023-10-26 23:54:04 +03:00
const dp = Dispatcher.for(client)
const log: string[] = []
dp.onRawUpdate((cl, upd) => {
log.push(`(grp0) received ${upd._}`)
return PropagationAction.Continue
}, 0)
dp.onRawUpdate((cl, upd) => {
log.push(`(grp0 2) received ${upd._}`)
}, 0)
dp.onRawUpdate((cl, upd) => {
log.push(`(grp1) received ${upd._}`)
return PropagationAction.Stop
}, 1)
dp.onRawUpdate((cl, upd) => {
log.push(`(grp1 2) received ${upd._}`)
}, 1)
dp.onRawUpdate((cl, upd) => {
log.push(`(grp2) received ${upd._}`)
}, 2)
await dp.dispatchRawUpdateNow({ _: 'updateConfig' }, emptyPeers)
expect(log).eql([
'(grp0) received updateConfig',
'(grp0 2) received updateConfig',
'(grp1) received updateConfig',
])
})
})
describe('Children', () => {
it('should call children handlers after own handlers', async () => {
2023-10-26 23:54:04 +03:00
const dp = Dispatcher.for(client)
const child = Dispatcher.child()
dp.addChild(child)
const log: string[] = []
dp.onRawUpdate((cl, upd) => {
log.push(`(parent) received ${upd._}`)
})
child.onRawUpdate((cl, upd) => {
log.push(`(child) received ${upd._}`)
})
await dp.dispatchRawUpdateNow({ _: 'updateConfig' }, emptyPeers)
2023-09-24 01:32:22 +03:00
expect(log).eql(['(parent) received updateConfig', '(child) received updateConfig'])
})
it('should have separate handler groups for children', async () => {
2023-10-26 23:54:04 +03:00
const dp = Dispatcher.for(client)
const child = Dispatcher.child()
dp.addChild(child)
const log: string[] = []
dp.onRawUpdate((cl, upd) => {
log.push(`(parent 0) received ${upd._}`)
return PropagationAction.Continue
}, 0)
dp.onRawUpdate((cl, upd) => {
log.push(`(parent 1) received ${upd._}`)
return PropagationAction.Stop
}, 1)
dp.onRawUpdate((cl, upd) => {
log.push(`(parent 2) received ${upd._}`)
}, 1)
child.onRawUpdate((cl, upd) => {
log.push(`(child 0) received ${upd._}`)
return PropagationAction.Continue
}, 0)
child.onRawUpdate((cl, upd) => {
log.push(`(child 1) received ${upd._}`)
return PropagationAction.Stop
}, 1)
child.onRawUpdate((cl, upd) => {
log.push(`(child 2) received ${upd._}`)
}, 1)
await dp.dispatchRawUpdateNow({ _: 'updateConfig' }, emptyPeers)
expect(log).eql([
'(parent 0) received updateConfig',
'(parent 1) received updateConfig',
'(child 0) received updateConfig',
'(child 1) received updateConfig',
])
})
})
2024-05-01 20:13:22 +03:00
describe('Dependency injection', () => {
it('should inject dependencies into update contexts', async () => {
const dp = Dispatcher.for(client)
dp.inject('foo' as never, 'foo' as never)
const log: string[] = []
dp.onNewMessage(() => {
log.push(`received ${(dp.deps as any).foo}`)
})
const dp2 = Dispatcher.child()
dp2.onNewMessage(() => {
log.push(`received ${(dp.deps as any).foo} (child)`)
})
dp.addChild(dp2)
await dp.dispatchUpdateNow({
name: 'new_message',
data: new Message(
createStub('message'),
emptyPeers,
),
})
expect(log).eql([
'received foo',
'received foo (child)',
])
})
})
})