Implement queue work and listen commands
Some checks failed
continuous-integration/drone/push Build is failing
Some checks failed
continuous-integration/drone/push Build is failing
This commit is contained in:
parent
e098a5edb7
commit
16e5fa00aa
@ -468,4 +468,22 @@ export abstract class Directive extends AppClass {
|
|||||||
protected nativeOutput(...outputs: any[]): void {
|
protected nativeOutput(...outputs: any[]): void {
|
||||||
console.log(...outputs) // eslint-disable-line no-console
|
console.log(...outputs) // eslint-disable-line no-console
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get a promise that resolves after SIGINT is received, executing a
|
||||||
|
* callback beforehand.
|
||||||
|
* @param callback
|
||||||
|
* @protected
|
||||||
|
*/
|
||||||
|
protected async untilInterrupt(callback?: () => unknown): Promise<void> {
|
||||||
|
return new Promise<void>(res => {
|
||||||
|
process.on('SIGINT', async () => {
|
||||||
|
if ( callback ) {
|
||||||
|
await callback()
|
||||||
|
}
|
||||||
|
|
||||||
|
res()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
64
src/cli/directive/queue/ListenDirective.ts
Normal file
64
src/cli/directive/queue/ListenDirective.ts
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
import {Directive, OptionDefinition} from '../../Directive'
|
||||||
|
import {Inject, Injectable} from '../../../di'
|
||||||
|
import {Bus, PushedToQueue, Queue} from '../../../support/bus'
|
||||||
|
import {Queueables} from '../../../service/Queueables'
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class ListenDirective extends Directive {
|
||||||
|
@Inject()
|
||||||
|
protected readonly queue!: Queue
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
protected readonly queueables!: Queueables
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
protected readonly bus!: Bus
|
||||||
|
|
||||||
|
getDescription(): string {
|
||||||
|
return 'listen for jobs pushed to the queue and attempt to execute them'
|
||||||
|
}
|
||||||
|
|
||||||
|
getKeywords(): string | string[] {
|
||||||
|
return 'queue-listen'
|
||||||
|
}
|
||||||
|
|
||||||
|
getOptions(): OptionDefinition[] {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
|
||||||
|
async handle(): Promise<void> {
|
||||||
|
this.info('Subscribing to queue events...')
|
||||||
|
await this.bus.subscribe(PushedToQueue, async () => {
|
||||||
|
// A new job has been pushed to the queue, so try to pop it and execute it.
|
||||||
|
// We may get undefined if some other worker is running and picked up this job first.
|
||||||
|
await this.tryExecuteJob()
|
||||||
|
})
|
||||||
|
|
||||||
|
this.info('Setting periodic poll...')
|
||||||
|
const handle = setInterval(async () => {
|
||||||
|
await this.tryExecuteJob()
|
||||||
|
}, 5000)
|
||||||
|
|
||||||
|
this.info('Listening for jobs...')
|
||||||
|
await this.untilInterrupt()
|
||||||
|
|
||||||
|
this.info('Shutting down...')
|
||||||
|
clearInterval(handle)
|
||||||
|
}
|
||||||
|
|
||||||
|
protected async tryExecuteJob(): Promise<void> {
|
||||||
|
try {
|
||||||
|
const job = await this.queue.pop()
|
||||||
|
if ( !job ) {
|
||||||
|
return // Some other worker already picked up this job
|
||||||
|
}
|
||||||
|
|
||||||
|
this.info(`Executing: ${job.constructor?.name || 'unknown job'}`)
|
||||||
|
await job.execute()
|
||||||
|
this.success('Execution finished.')
|
||||||
|
} catch (e: unknown) {
|
||||||
|
this.error('Failed to execute job.')
|
||||||
|
this.error(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,12 +1,16 @@
|
|||||||
import {Directive, OptionDefinition} from '../../Directive'
|
import {Directive, OptionDefinition} from '../../Directive'
|
||||||
import {Inject, Injectable} from '../../../di'
|
import {Inject, Injectable} from '../../../di'
|
||||||
import {Queue} from '../../../support/bus'
|
import {Queue} from '../../../support/bus'
|
||||||
|
import {Queueables} from '../../../service/Queueables'
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class WorkDirective extends Directive {
|
export class WorkDirective extends Directive {
|
||||||
@Inject()
|
@Inject()
|
||||||
protected readonly queue!: Queue
|
protected readonly queue!: Queue
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
protected readonly queueables!: Queueables
|
||||||
|
|
||||||
getDescription(): string {
|
getDescription(): string {
|
||||||
return 'pop a single item from the queue and execute it'
|
return 'pop a single item from the queue and execute it'
|
||||||
}
|
}
|
||||||
|
@ -13,3 +13,6 @@ export * from './directive/TemplateDirective'
|
|||||||
export * from './directive/UsageDirective'
|
export * from './directive/UsageDirective'
|
||||||
|
|
||||||
export * from './decorators'
|
export * from './decorators'
|
||||||
|
|
||||||
|
export * from './directive/queue/ListenDirective'
|
||||||
|
export * from './directive/queue/WorkDirective'
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import {Unit} from '../../lifecycle/Unit'
|
import {Unit} from '../../lifecycle/Unit'
|
||||||
import {Logging} from '../../service/Logging'
|
import {Logging} from '../../service/Logging'
|
||||||
import {Singleton, Inject} from '../../di/decorator/injection'
|
import {Singleton, Inject} from '../../di'
|
||||||
import {CommandLine} from './CommandLine'
|
import {CommandLine} from './CommandLine'
|
||||||
import {UsageDirective} from '../directive/UsageDirective'
|
import {UsageDirective} from '../directive/UsageDirective'
|
||||||
import {Directive} from '../Directive'
|
import {Directive} from '../Directive'
|
||||||
@ -10,6 +10,7 @@ import {RunDirective} from '../directive/RunDirective'
|
|||||||
import {RoutesDirective} from '../directive/RoutesDirective'
|
import {RoutesDirective} from '../directive/RoutesDirective'
|
||||||
import {RouteDirective} from '../directive/RouteDirective'
|
import {RouteDirective} from '../directive/RouteDirective'
|
||||||
import {WorkDirective} from '../directive/queue/WorkDirective'
|
import {WorkDirective} from '../directive/queue/WorkDirective'
|
||||||
|
import {ListenDirective} from '../directive/queue/ListenDirective'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Unit that takes the place of the final unit in the application that handles
|
* Unit that takes the place of the final unit in the application that handles
|
||||||
@ -48,6 +49,7 @@ export class CommandLineApplication extends Unit {
|
|||||||
this.cli.registerDirective(RoutesDirective)
|
this.cli.registerDirective(RoutesDirective)
|
||||||
this.cli.registerDirective(RouteDirective)
|
this.cli.registerDirective(RouteDirective)
|
||||||
this.cli.registerDirective(WorkDirective)
|
this.cli.registerDirective(WorkDirective)
|
||||||
|
this.cli.registerDirective(ListenDirective)
|
||||||
|
|
||||||
const argv = process.argv.slice(2)
|
const argv = process.argv.slice(2)
|
||||||
const match = this.cli.getDirectives()
|
const match = this.cli.getDirectives()
|
||||||
|
@ -55,10 +55,6 @@ export class Container {
|
|||||||
.resolve()
|
.resolve()
|
||||||
.map(factory => container.registerFactory(factory))
|
.map(factory => container.registerFactory(factory))
|
||||||
|
|
||||||
ContainerBlueprint.getContainerBlueprint()
|
|
||||||
.resolveConstructable()
|
|
||||||
.map((factory: StaticClass<AbstractFactory<any>, any>) => console.log(factory)) // eslint-disable-line no-console
|
|
||||||
|
|
||||||
ContainerBlueprint.getContainerBlueprint()
|
ContainerBlueprint.getContainerBlueprint()
|
||||||
.resolveConstructable()
|
.resolveConstructable()
|
||||||
.map((factory: StaticClass<AbstractFactory<any>, any>) => container.registerFactory(container.make(factory)))
|
.map((factory: StaticClass<AbstractFactory<any>, any>) => container.registerFactory(container.make(factory)))
|
||||||
@ -335,7 +331,7 @@ export class Container {
|
|||||||
if ( factory ) {
|
if ( factory ) {
|
||||||
return factory
|
return factory
|
||||||
} else {
|
} else {
|
||||||
logIfDebugging('extollo.di.injector', 'unable to resolve factory', factory, this.factories)
|
logIfDebugging('extollo.di.injector', 'unable to resolve factory', key, factory, this.factories)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -388,6 +384,8 @@ export class Container {
|
|||||||
* @param {array} parameters
|
* @param {array} parameters
|
||||||
*/
|
*/
|
||||||
protected produceFactory<T>(factory: AbstractFactory<T>, parameters: any[]): T {
|
protected produceFactory<T>(factory: AbstractFactory<T>, parameters: any[]): T {
|
||||||
|
logIfDebugging('extollo.di.injector', 'Make stack', Container.makeStack)
|
||||||
|
|
||||||
// Create the dependencies for the factory
|
// Create the dependencies for the factory
|
||||||
const keys = factory.getDependencyKeys().filter(req => this.hasKey(req.key))
|
const keys = factory.getDependencyKeys().filter(req => this.hasKey(req.key))
|
||||||
const dependencies = keys.map<ResolvedDependency>(req => {
|
const dependencies = keys.map<ResolvedDependency>(req => {
|
||||||
@ -415,7 +413,9 @@ export class Container {
|
|||||||
// Produce a new instance
|
// Produce a new instance
|
||||||
const inst = factory.produce(constructorArguments, params.reverse().all())
|
const inst = factory.produce(constructorArguments, params.reverse().all())
|
||||||
|
|
||||||
|
logIfDebugging('extollo.di.injector', 'Resolving dependencies for factory', factory)
|
||||||
factory.getInjectedProperties().each(dependency => {
|
factory.getInjectedProperties().each(dependency => {
|
||||||
|
logIfDebugging('extollo.di.injector', 'Resolving injected dependency:', dependency)
|
||||||
if ( dependency.key && inst ) {
|
if ( dependency.key && inst ) {
|
||||||
(inst as any)[dependency.property] = this.resolveAndCreate(dependency.key)
|
(inst as any)[dependency.property] = this.resolveAndCreate(dependency.key)
|
||||||
}
|
}
|
||||||
|
@ -3,7 +3,8 @@ import {Singleton, Instantiable} from '../di'
|
|||||||
import {Queueable} from '../support/bus'
|
import {Queueable} from '../support/bus'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A canonical unit that resolves Queueable classes from `app/jobs`.
|
* A canonical unit that resolves Queueable classes from `app/jobs` and sets up
|
||||||
|
* any non-default queues.
|
||||||
*/
|
*/
|
||||||
@Singleton()
|
@Singleton()
|
||||||
export class Queueables extends CanonicalStatic<Queueables, Instantiable<Queueable>> {
|
export class Queueables extends CanonicalStatic<Queueables, Instantiable<Queueable>> {
|
||||||
|
@ -1,8 +1,19 @@
|
|||||||
import {Inject, Singleton, StaticInstantiable} from '../../di'
|
import {Inject, Singleton, StaticInstantiable} from '../../di'
|
||||||
import {BusSubscriber, Event, EventBus, EventHandler, EventHandlerReturn, EventHandlerSubscription} from './types'
|
import {
|
||||||
|
BusConnectorConfig,
|
||||||
|
BusSubscriber,
|
||||||
|
Event,
|
||||||
|
EventBus,
|
||||||
|
EventHandler,
|
||||||
|
EventHandlerReturn,
|
||||||
|
EventHandlerSubscription,
|
||||||
|
} from './types'
|
||||||
import {Awaitable, Collection, Pipeline, uuid4} from '../../util'
|
import {Awaitable, Collection, Pipeline, uuid4} from '../../util'
|
||||||
import {Logging} from '../../service/Logging'
|
import {Logging} from '../../service/Logging'
|
||||||
import {Unit} from '../../lifecycle/Unit'
|
import {Unit} from '../../lifecycle/Unit'
|
||||||
|
import {Config} from '../../service/Config'
|
||||||
|
import {RedisBus} from './RedisBus'
|
||||||
|
import {getEventName} from './getEventName'
|
||||||
|
|
||||||
export interface BusInternalSubscription {
|
export interface BusInternalSubscription {
|
||||||
busUuid: string
|
busUuid: string
|
||||||
@ -18,6 +29,9 @@ export class Bus<TEvent extends Event = Event> extends Unit implements EventBus<
|
|||||||
@Inject()
|
@Inject()
|
||||||
protected readonly logging!: Logging
|
protected readonly logging!: Logging
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
protected readonly config!: Config
|
||||||
|
|
||||||
public readonly uuid = uuid4()
|
public readonly uuid = uuid4()
|
||||||
|
|
||||||
/** Local listeners subscribed to events on this bus. */
|
/** Local listeners subscribed to events on this bus. */
|
||||||
@ -37,6 +51,7 @@ export class Bus<TEvent extends Event = Event> extends Unit implements EventBus<
|
|||||||
*/
|
*/
|
||||||
async push(event: TEvent): Promise<void> {
|
async push(event: TEvent): Promise<void> {
|
||||||
if ( event.originBusUuid === this.uuid ) {
|
if ( event.originBusUuid === this.uuid ) {
|
||||||
|
this.logging.verbose('Skipping propagation of event, because it originated from this bus.')
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -44,13 +59,19 @@ export class Bus<TEvent extends Event = Event> extends Unit implements EventBus<
|
|||||||
event.originBusUuid = this.uuid
|
event.originBusUuid = this.uuid
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.logging.verbose('Raising event on process-local bus:')
|
||||||
|
this.logging.verbose(event)
|
||||||
if ( await this.callSubscribers(event) ) {
|
if ( await this.callSubscribers(event) ) {
|
||||||
// One of the subscribers halted propagation of the event
|
// One of the subscribers halted propagation of the event
|
||||||
|
this.logging.verbose('Process-local subscriber halted propagation of event.')
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( await this.shouldBroadcast(event) ) {
|
if ( await this.shouldBroadcast(event) ) {
|
||||||
|
this.logging.verbose('Raising event on connected busses...')
|
||||||
await this.connectors.awaitMapCall('push', event)
|
await this.connectors.awaitMapCall('push', event)
|
||||||
|
} else {
|
||||||
|
this.logging.verbose('Will not broadcast event.')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -91,7 +112,7 @@ export class Bus<TEvent extends Event = Event> extends Unit implements EventBus<
|
|||||||
const uuid = uuid4()
|
const uuid = uuid4()
|
||||||
|
|
||||||
this.subscribers.push({
|
this.subscribers.push({
|
||||||
eventName: eventKey.prototype.eventName, // FIXME this is not working
|
eventName: getEventName(eventKey),
|
||||||
handler,
|
handler,
|
||||||
eventKey,
|
eventKey,
|
||||||
uuid,
|
uuid,
|
||||||
@ -99,6 +120,7 @@ export class Bus<TEvent extends Event = Event> extends Unit implements EventBus<
|
|||||||
|
|
||||||
this.subscriptions.concat(await this.connectors
|
this.subscriptions.concat(await this.connectors
|
||||||
.promiseMap<BusInternalSubscription>(async bus => {
|
.promiseMap<BusInternalSubscription>(async bus => {
|
||||||
|
this.logging.verbose(`Connecting subscriber to bus ${bus.constructor?.name}#${bus.uuid}...`)
|
||||||
return {
|
return {
|
||||||
busUuid: bus.uuid,
|
busUuid: bus.uuid,
|
||||||
subscriberUuid: uuid,
|
subscriberUuid: uuid,
|
||||||
@ -166,6 +188,17 @@ export class Bus<TEvent extends Event = Event> extends Unit implements EventBus<
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Read the connectors from the server.bus config and set them up
|
||||||
|
const config = this.config.get('server.bus', {})
|
||||||
|
if ( Array.isArray(config.connectors) ) {
|
||||||
|
for ( const connector of (config.connectors as BusConnectorConfig[]) ) {
|
||||||
|
this.logging.info(`Creating bus connection of type: ${connector.type}`)
|
||||||
|
if ( connector.type === 'redis' ) {
|
||||||
|
await this.connect(this.make(RedisBus))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
await this.connectors.awaitMapCall('up')
|
await this.connectors.awaitMapCall('up')
|
||||||
|
|
||||||
this.isUp = true
|
this.isUp = true
|
||||||
|
@ -4,6 +4,7 @@ import {Awaitable, Collection, Pipeline, uuid4} from '../../util'
|
|||||||
import {Logging} from '../../service/Logging'
|
import {Logging} from '../../service/Logging'
|
||||||
import {Bus, BusInternalSubscription} from './Bus'
|
import {Bus, BusInternalSubscription} from './Bus'
|
||||||
import {AppClass} from '../../lifecycle/AppClass'
|
import {AppClass} from '../../lifecycle/AppClass'
|
||||||
|
import {getEventName} from './getEventName'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Non-connectable event bus implementation. Can forward events to the main Bus instance.
|
* Non-connectable event bus implementation. Can forward events to the main Bus instance.
|
||||||
@ -86,7 +87,7 @@ export class LocalBus<TEvent extends Event = Event> extends AppClass implements
|
|||||||
const uuid = uuid4()
|
const uuid = uuid4()
|
||||||
|
|
||||||
this.subscribers.push({
|
this.subscribers.push({
|
||||||
eventName: eventKey.prototype.eventName,
|
eventName: getEventName(eventKey),
|
||||||
handler,
|
handler,
|
||||||
eventKey,
|
eventKey,
|
||||||
uuid,
|
uuid,
|
||||||
|
@ -1,9 +1,11 @@
|
|||||||
import {BusSubscriber, Event, EventBus, EventHandler, EventHandlerReturn, EventHandlerSubscription} from './types'
|
import {BusSubscriber, Event, EventBus, EventHandler, EventHandlerReturn, EventHandlerSubscription} from './types'
|
||||||
import {Inject, Injectable, StaticInstantiable} from '../../di'
|
import {Container, Inject, Injectable, StaticInstantiable} from '../../di'
|
||||||
import {Awaitable, Collection, Pipeline, uuid4} from '../../util'
|
import {Awaitable, Collection, Pipeline, uuid4} from '../../util'
|
||||||
import {Redis} from '../redis/Redis'
|
import {Redis} from '../redis/Redis'
|
||||||
import {Serialization} from './serial/Serialization'
|
import {Serialization} from './serial/Serialization'
|
||||||
import * as IORedis from 'ioredis'
|
import * as IORedis from 'ioredis'
|
||||||
|
import {Logging} from '../../service/Logging'
|
||||||
|
import {getEventName} from './getEventName'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Event bus implementation that does pub/sub over a Redis connection.
|
* Event bus implementation that does pub/sub over a Redis connection.
|
||||||
@ -16,6 +18,12 @@ export class RedisBus implements EventBus {
|
|||||||
@Inject()
|
@Inject()
|
||||||
protected readonly serial!: Serialization
|
protected readonly serial!: Serialization
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
protected readonly logging!: Logging
|
||||||
|
|
||||||
|
@Inject()
|
||||||
|
protected readonly injector!: Container
|
||||||
|
|
||||||
public readonly uuid = uuid4()
|
public readonly uuid = uuid4()
|
||||||
|
|
||||||
/** List of events for which we have created Redis channel subscriptions. */
|
/** List of events for which we have created Redis channel subscriptions. */
|
||||||
@ -40,18 +48,24 @@ export class RedisBus implements EventBus {
|
|||||||
const channel = `ex-event-${event.eventName}`
|
const channel = `ex-event-${event.eventName}`
|
||||||
const json = await this.serial.encodeJSON(event)
|
const json = await this.serial.encodeJSON(event)
|
||||||
|
|
||||||
|
this.logging.verbose(`Pushing event to channel: ${channel}`)
|
||||||
|
this.logging.verbose(json)
|
||||||
|
|
||||||
await this.publisherConnection.publish(channel, json)
|
await this.publisherConnection.publish(channel, json)
|
||||||
}
|
}
|
||||||
|
|
||||||
async subscribe<T extends Event>(eventKey: StaticInstantiable<T>, handler: EventHandler<T>): Promise<EventHandlerSubscription> {
|
async subscribe<T extends Event>(eventKey: StaticInstantiable<T>, handler: EventHandler<T>): Promise<EventHandlerSubscription> {
|
||||||
const uuid = uuid4()
|
const uuid = uuid4()
|
||||||
const subscriber: BusSubscriber<Event> = {
|
const subscriber: BusSubscriber<Event> = {
|
||||||
eventName: eventKey.prototype.eventName,
|
eventName: getEventName(eventKey),
|
||||||
eventKey,
|
eventKey,
|
||||||
handler,
|
handler,
|
||||||
uuid,
|
uuid,
|
||||||
} as unknown as BusSubscriber<Event>
|
} as unknown as BusSubscriber<Event>
|
||||||
|
|
||||||
|
this.logging.verbose(`Creating subscriber ${uuid}...`)
|
||||||
|
this.logging.verbose(subscriber)
|
||||||
|
|
||||||
if ( !this.internalSubscriptions.includes(subscriber.eventName) ) {
|
if ( !this.internalSubscriptions.includes(subscriber.eventName) ) {
|
||||||
await new Promise<void>((res, rej) => {
|
await new Promise<void>((res, rej) => {
|
||||||
if ( !this.subscriberConnection ) {
|
if ( !this.subscriberConnection ) {
|
||||||
@ -63,6 +77,7 @@ export class RedisBus implements EventBus {
|
|||||||
return rej(err)
|
return rej(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.logging.verbose('Successfully subscribed to channel on Redis...')
|
||||||
res()
|
res()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
@ -95,10 +110,12 @@ export class RedisBus implements EventBus {
|
|||||||
|
|
||||||
this.subscriberConnection.on('message', (channel: string, message: string) => {
|
this.subscriberConnection.on('message', (channel: string, message: string) => {
|
||||||
if ( !channel.startsWith('ex-event-') ) {
|
if ( !channel.startsWith('ex-event-') ) {
|
||||||
|
this.logging.debug(`Ignoring message on invalid channel: ${channel}`)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
const name = channel.substr('ex-event-'.length)
|
const name = channel.substr('ex-event-'.length)
|
||||||
|
this.logging.verbose(`Received event ${name} on channel ${channel}`)
|
||||||
this.handleEvent(name, message)
|
this.handleEvent(name, message)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
35
src/support/bus/getEventName.ts
Normal file
35
src/support/bus/getEventName.ts
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
import {Event} from './types'
|
||||||
|
import {Container, StaticInstantiable} from '../../di'
|
||||||
|
import {ErrorWithContext} from '../../util'
|
||||||
|
|
||||||
|
export function getEventName<T extends Event>(eventKey: StaticInstantiable<T>): string {
|
||||||
|
const protoName = eventKey.prototype.eventName
|
||||||
|
if ( protoName ) {
|
||||||
|
return protoName
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const inst = Container.getContainer().make<T>(eventKey)
|
||||||
|
if ( inst.eventName ) {
|
||||||
|
return inst.eventName
|
||||||
|
}
|
||||||
|
} catch (e: unknown) {} // eslint-disable-line no-empty
|
||||||
|
|
||||||
|
let stringParseName = eventKey.toString()
|
||||||
|
.split('\n')
|
||||||
|
.map(x => x.trim())
|
||||||
|
.filter(x => x.startsWith('this.eventName = \'') || x.startsWith('this.eventName = "'))?.[0]
|
||||||
|
?.split('=')?.[1]
|
||||||
|
?.trim()
|
||||||
|
|
||||||
|
if ( stringParseName ) {
|
||||||
|
stringParseName = stringParseName.endsWith(';') ? stringParseName.slice(1, -2) : stringParseName.slice(1, -1)
|
||||||
|
if ( stringParseName ) {
|
||||||
|
return stringParseName
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new ErrorWithContext('Unable to determine eventName from eventKey', {
|
||||||
|
eventKey,
|
||||||
|
})
|
||||||
|
}
|
@ -11,6 +11,8 @@ export * from './RedisBus'
|
|||||||
|
|
||||||
export * from './queue/event/PushingToQueue'
|
export * from './queue/event/PushingToQueue'
|
||||||
export * from './queue/event/PushedToQueue'
|
export * from './queue/event/PushedToQueue'
|
||||||
|
export * from './queue/event/QueueEventSerializer'
|
||||||
|
export * from './queue/BaseJob'
|
||||||
export * from './queue/Queue'
|
export * from './queue/Queue'
|
||||||
export * from './queue/CacheQueue'
|
export * from './queue/CacheQueue'
|
||||||
export * from './queue/SyncQueue'
|
export * from './queue/SyncQueue'
|
||||||
|
13
src/support/bus/queue/BaseJob.ts
Normal file
13
src/support/bus/queue/BaseJob.ts
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
import {Queueable} from '../types'
|
||||||
|
import {Awaitable} from '../../../util'
|
||||||
|
import {Inject, Injectable} from '../../../di'
|
||||||
|
import {Logging} from '../../../service/Logging'
|
||||||
|
import {CanonicalItemClass} from '../../CanonicalReceiver'
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export abstract class BaseJob extends CanonicalItemClass implements Queueable {
|
||||||
|
@Inject()
|
||||||
|
protected readonly logging!: Logging
|
||||||
|
|
||||||
|
abstract execute(): Awaitable<void>
|
||||||
|
}
|
@ -16,9 +16,9 @@ export abstract class Queue implements BusQueue {
|
|||||||
|
|
||||||
async dispatch<T extends Queueable>(item: T): Promise<void> {
|
async dispatch<T extends Queueable>(item: T): Promise<void> {
|
||||||
if ( shouldQueue(item) ) {
|
if ( shouldQueue(item) ) {
|
||||||
await this.bus.push(new PushingToQueue(item))
|
await this.bus.push(new PushingToQueue(item, this.name))
|
||||||
await this.push(item)
|
await this.push(item)
|
||||||
await this.bus.push(new PushedToQueue(item))
|
await this.bus.push(new PushedToQueue(item, this.name))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5,7 +5,7 @@ import {uuid4} from '../../../../util'
|
|||||||
* Event fired after an item is pushed to the queue.
|
* Event fired after an item is pushed to the queue.
|
||||||
*/
|
*/
|
||||||
export class PushedToQueue<T extends ShouldQueue<Queueable>> implements Event {
|
export class PushedToQueue<T extends ShouldQueue<Queueable>> implements Event {
|
||||||
public readonly eventName = '@extollo/lib:PushedToQueue'
|
public readonly eventName = '@extollo/lib.PushedToQueue'
|
||||||
|
|
||||||
public readonly eventUuid = uuid4()
|
public readonly eventUuid = uuid4()
|
||||||
|
|
||||||
@ -13,5 +13,6 @@ export class PushedToQueue<T extends ShouldQueue<Queueable>> implements Event {
|
|||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
public readonly item: T,
|
public readonly item: T,
|
||||||
|
public readonly queueName: string,
|
||||||
) {}
|
) {}
|
||||||
}
|
}
|
||||||
|
@ -5,7 +5,7 @@ import {uuid4} from '../../../../util'
|
|||||||
* Event fired before an item is pushed to the queue.
|
* Event fired before an item is pushed to the queue.
|
||||||
*/
|
*/
|
||||||
export class PushingToQueue<T extends ShouldQueue<Queueable>> implements Event {
|
export class PushingToQueue<T extends ShouldQueue<Queueable>> implements Event {
|
||||||
public readonly eventName = '@extollo/lib:PushingToQueue'
|
public readonly eventName = '@extollo/lib.PushingToQueue'
|
||||||
|
|
||||||
public readonly eventUuid = uuid4()
|
public readonly eventUuid = uuid4()
|
||||||
|
|
||||||
@ -13,5 +13,6 @@ export class PushingToQueue<T extends ShouldQueue<Queueable>> implements Event {
|
|||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
public readonly item: T,
|
public readonly item: T,
|
||||||
|
public readonly queueName: string,
|
||||||
) {}
|
) {}
|
||||||
}
|
}
|
||||||
|
43
src/support/bus/queue/event/QueueEventSerializer.ts
Normal file
43
src/support/bus/queue/event/QueueEventSerializer.ts
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
import {PushedToQueue} from './PushedToQueue'
|
||||||
|
import {Queueable, SerialPayload, ShouldQueue} from '../../types'
|
||||||
|
import {PushingToQueue} from './PushingToQueue'
|
||||||
|
import {BaseSerializer} from '../../serial/BaseSerializer'
|
||||||
|
import {JSONState} from '../../../../util'
|
||||||
|
import {ObjectSerializer} from '../../serial/decorators'
|
||||||
|
|
||||||
|
export type QueueEvent = PushedToQueue<ShouldQueue<Queueable>> | PushingToQueue<ShouldQueue<Queueable>>
|
||||||
|
|
||||||
|
export interface QueueEventSerialPayload extends JSONState {
|
||||||
|
eventName: '@extollo/lib.PushedToQueue' | '@extollo/lib.PushingToQueue'
|
||||||
|
itemPayload: SerialPayload<Queueable, JSONState>
|
||||||
|
queueName: string
|
||||||
|
}
|
||||||
|
|
||||||
|
@ObjectSerializer()
|
||||||
|
export class QueueEventSerializer extends BaseSerializer<QueueEvent, QueueEventSerialPayload> {
|
||||||
|
protected async decodeSerial(serial: QueueEventSerialPayload): Promise<QueueEvent> {
|
||||||
|
const item = await this.getSerialization().decode(serial.itemPayload)
|
||||||
|
|
||||||
|
if ( serial.eventName === '@extollo/lib.PushedToQueue' ) {
|
||||||
|
return new PushedToQueue(item as ShouldQueue<Queueable>, serial.queueName)
|
||||||
|
} else {
|
||||||
|
return new PushingToQueue(item as ShouldQueue<Queueable>, serial.queueName)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected async encodeActual(actual: QueueEvent): Promise<QueueEventSerialPayload> {
|
||||||
|
return {
|
||||||
|
eventName: actual.eventName,
|
||||||
|
queueName: actual.queueName,
|
||||||
|
itemPayload: await this.getSerialization().encode(actual.item),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected getName(): string {
|
||||||
|
return '@extollo/lib.QueueEventSerializer'
|
||||||
|
}
|
||||||
|
|
||||||
|
matchActual(some: QueueEvent): boolean {
|
||||||
|
return some instanceof PushedToQueue || some instanceof PushingToQueue
|
||||||
|
}
|
||||||
|
}
|
@ -3,6 +3,7 @@ import {BaseSerializer} from './BaseSerializer'
|
|||||||
import {Awaitable, ErrorWithContext, JSONState, Rehydratable} from '../../../util'
|
import {Awaitable, ErrorWithContext, JSONState, Rehydratable} from '../../../util'
|
||||||
import {Container, Inject, Injectable} from '../../../di'
|
import {Container, Inject, Injectable} from '../../../di'
|
||||||
import {Canon} from '../../../service/Canon'
|
import {Canon} from '../../../service/Canon'
|
||||||
|
import {ObjectSerializer} from './decorators'
|
||||||
|
|
||||||
/** State encoded by this class. */
|
/** State encoded by this class. */
|
||||||
export interface SimpleCanonicalItemSerialState extends JSONState {
|
export interface SimpleCanonicalItemSerialState extends JSONState {
|
||||||
@ -15,6 +16,7 @@ export interface SimpleCanonicalItemSerialState extends JSONState {
|
|||||||
* These instances must be CanonicalItemClass instances and take no constructor parameters.
|
* These instances must be CanonicalItemClass instances and take no constructor parameters.
|
||||||
* If the instance is Rehydratable, then the state will be (re-)stored.
|
* If the instance is Rehydratable, then the state will be (re-)stored.
|
||||||
*/
|
*/
|
||||||
|
@ObjectSerializer()
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class SimpleCanonicalItemSerializer<TActual extends CanonicalItemClass> extends BaseSerializer<TActual, SimpleCanonicalItemSerialState> {
|
export class SimpleCanonicalItemSerializer<TActual extends CanonicalItemClass> extends BaseSerializer<TActual, SimpleCanonicalItemSerialState> {
|
||||||
@Inject()
|
@Inject()
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import {Awaitable, JSONState, Maybe, Pipeline, TypeTag, uuid4} from '../../util'
|
import {Awaitable, JSONState, Maybe, Pipeline, TypeTag, uuid4} from '../../util'
|
||||||
import {StaticInstantiable} from '../../di'
|
import {Instantiable, StaticInstantiable} from '../../di'
|
||||||
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
||||||
export interface SerialPayload<TActual, TSerial extends JSONState> extends JSONState {
|
export interface SerialPayload<TActual, TSerial extends JSONState> extends JSONState {
|
||||||
@ -92,3 +92,17 @@ export interface BusQueue {
|
|||||||
|
|
||||||
pop(): Promise<Maybe<Queueable>>
|
pop(): Promise<Maybe<Queueable>>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface RedisBusConnectorConfig {
|
||||||
|
type: 'redis'
|
||||||
|
}
|
||||||
|
|
||||||
|
export type BusConnectorConfig = RedisBusConnectorConfig
|
||||||
|
|
||||||
|
export interface QueueConfig {
|
||||||
|
driver?: Instantiable<BusQueue>,
|
||||||
|
/* queues?: ({
|
||||||
|
name: string,
|
||||||
|
driver: Instantiable<BusQueue>,
|
||||||
|
})[] */
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user