From 184f71d9657e9d87bc73962568be5d6b7fccf6be Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Tue, 30 Apr 2024 10:31:57 +0200 Subject: [PATCH 01/17] chore: wip --- packages/quick-tsr/src/tsrHandler.ts | 121 ++---- .../timeline-state-resolver/src/conductor.ts | 395 +----------------- .../src/integrations/casparCG/index.ts | 46 +- .../src/integrations/sisyfos/index.ts | 6 +- .../src/integrations/sofieChef/index.ts | 4 +- .../src/integrations/vizMSE/index.ts | 4 +- .../src/service/ConnectionManager.ts | 395 ++++++++++++++++++ 7 files changed, 473 insertions(+), 498 deletions(-) create mode 100644 packages/timeline-state-resolver/src/service/ConnectionManager.ts diff --git a/packages/quick-tsr/src/tsrHandler.ts b/packages/quick-tsr/src/tsrHandler.ts index 3055eeba3..b8f5e41d2 100644 --- a/packages/quick-tsr/src/tsrHandler.ts +++ b/packages/quick-tsr/src/tsrHandler.ts @@ -8,7 +8,6 @@ import { Datastore, DeviceStatus, SlowSentCommandInfo, - DeviceOptionsBase, SlowFulfilledCommandInfo, DeviceType, CasparCGDevice, @@ -19,7 +18,6 @@ import { ThreadedClass } from 'threadedclass' import * as _ from 'underscore' import { TSRSettings } from './index' -import { BaseRemoteDeviceIntegration } from 'timeline-state-resolver/dist/service/remoteDeviceInstance' /** * Represents a connection between Gateway and TSR @@ -27,13 +25,9 @@ import { BaseRemoteDeviceIntegration } from 'timeline-state-resolver/dist/servic export class TSRHandler { private tsr!: Conductor - private _multiThreaded: boolean | null = null - // private _timeline: TSRTimeline // private _mappings: Mappings - private _devices: { [deviceId: string]: BaseRemoteDeviceIntegration> } = {} - constructor() { // nothing } @@ -76,6 +70,29 @@ export class TSRHandler { // todo ? }) + this.tsr.connectionManager.on('connectionEvent:connectionChanged', (deviceId: string, status: DeviceStatus) => { + console.log(`Device ${deviceId} status changed: ${JSON.stringify(status)}`) + }) + this.tsr.connectionManager.on( + 'connectionEvent:slowSentCommand', + (_deviceId: string, _info: SlowSentCommandInfo) => { + // console.log(`Device ${device.deviceId} slow sent command: ${_info}`) + } + ) + this.tsr.connectionManager.on( + 'connectionEvent:slowFulfilledCommand', + (_deviceId: string, _info: SlowFulfilledCommandInfo) => { + // console.log(`Device ${device.deviceId} slow fulfilled command: ${_info}`) + } + ) + this.tsr.connectionManager.on('connectionEvent:commandReport', (deviceId: string, command: any) => { + console.log(`Device ${deviceId} command: ${JSON.stringify(command)}`) + }) + this.tsr.connectionManager.on('connectionEvent:debug', (deviceId: string, ...args: any[]) => { + const data = args.map((arg) => (typeof arg === 'object' ? JSON.stringify(arg) : arg)) + console.log(`Device ${deviceId} debug: ${data}`) + }) + await this.tsr.init() // this._initialized = true @@ -90,7 +107,7 @@ export class TSRHandler { else return Promise.resolve() } async logMediaList(): Promise { - for (const deviceContainer of this.tsr.getDevices()) { + for (const deviceContainer of this.tsr.connectionManager.getConnections()) { if (deviceContainer.deviceType === DeviceType.CASPARCG) { const device = deviceContainer.device as ThreadedClass @@ -118,94 +135,6 @@ export class TSRHandler { this.tsr.setDatastore(store) } public async setDevices(devices: { [deviceId: string]: DeviceOptionsAny }): Promise { - const ps: Array> = [] - - _.each(devices, (deviceOptions: DeviceOptionsAny, deviceId: string) => { - const oldDevice = this.tsr.getDevice(deviceId) - - if (!oldDevice) { - if (deviceOptions.options) { - console.log('Initializing device: ' + deviceId) - ps.push(this._addDevice(deviceId, deviceOptions)) - } - } else { - if (this._multiThreaded !== null && deviceOptions.isMultiThreaded === undefined) { - deviceOptions.isMultiThreaded = this._multiThreaded - } - if (deviceOptions.options) { - let anyChanged = false - - // let oldOptions = (oldDevice.deviceOptions).options || {} - - if (!_.isEqual(oldDevice.deviceOptions, deviceOptions)) { - anyChanged = true - } - - if (anyChanged) { - console.log('Re-initializing device: ' + deviceId) - ps.push(this._removeDevice(deviceId).then(async () => this._addDevice(deviceId, deviceOptions))) - } - } - } - }) - - _.each(this.tsr.getDevices(), (oldDevice: BaseRemoteDeviceIntegration>) => { - const deviceId = oldDevice.deviceId - if (!devices[deviceId]) { - console.log('Un-initializing device: ' + deviceId) - ps.push(this._removeDevice(deviceId)) - } - }) - - await Promise.all(ps) - } - private async _addDevice(deviceId: string, options: DeviceOptionsAny) { - // console.log('Adding device ' + deviceId) - - if (!options.limitSlowSentCommand) options.limitSlowSentCommand = 40 - if (!options.limitSlowFulfilledCommand) options.limitSlowFulfilledCommand = 100 - - try { - const device = await this.tsr.addDevice(deviceId, options) - - this._devices[deviceId] = device - - await device.device.on('connectionChanged', ((status: DeviceStatus) => { - console.log(`Device ${device.deviceId} status changed: ${JSON.stringify(status)}`) - }) as () => void) - await device.device.on('slowSentCommand', ((_info: SlowSentCommandInfo) => { - // console.log(`Device ${device.deviceId} slow sent command: ${_info}`) - }) as () => void) - await device.device.on('slowFulfilledCommand', ((_info: SlowFulfilledCommandInfo) => { - // console.log(`Device ${device.deviceId} slow fulfilled command: ${_info}`) - }) as () => void) - await device.device.on('commandReport', ((command: any) => { - console.log(`Device ${device.deviceId} command: ${JSON.stringify(command)}`) - }) as () => void) - await device.device.on('debug', (...args: any[]) => { - const data = args.map((arg) => (typeof arg === 'object' ? JSON.stringify(arg) : arg)) - console.log(`Device ${device.deviceId} debug: ${data}`) - }) - // also ask for the status now, and update: - // onConnectionChanged(await device.device.getStatus()) - } catch (e) { - console.error(`Error when adding device "${deviceId}"`, e) - } - } - private async _removeDevice(deviceId: string) { - try { - await this.tsr.removeDevice(deviceId) - } catch (e) { - console.error('Error when removing tsr device: ' + e) - } - - if (this._devices[deviceId]) { - try { - await this._devices[deviceId].device.terminate() - } catch (e) { - console.error('Error when removing device: ' + e) - } - } - delete this._devices[deviceId] + this.tsr.connectionManager.setConnections(new Map(Object.entries(devices))) } } diff --git a/packages/timeline-state-resolver/src/conductor.ts b/packages/timeline-state-resolver/src/conductor.ts index 7bc432e04..7f267cea2 100644 --- a/packages/timeline-state-resolver/src/conductor.ts +++ b/packages/timeline-state-resolver/src/conductor.ts @@ -1,7 +1,7 @@ import * as _ from 'underscore' import { getResolvedState, ResolvedTimeline, ResolvedTimelineObjectInstance, TimelineObject } from 'superfly-timeline' import { EventEmitter } from 'eventemitter3' -import { MemUsageReport, threadedClass, ThreadedClass, ThreadedClassConfig, ThreadedClassManager } from 'threadedclass' +import { MemUsageReport, threadedClass, ThreadedClass, ThreadedClassManager } from 'threadedclass' import PQueue from 'p-queue' import * as PAll from 'p-all' import PTimeout from 'p-timeout' @@ -41,18 +41,17 @@ import { import { DoOnTime } from './devices/doOnTime' import { AsyncResolver } from './AsyncResolver' -import { assertNever, endTrace, fillStateFromDatastore, FinishedTrace, startTrace } from './lib' +import { endTrace, fillStateFromDatastore, FinishedTrace, startTrace } from './lib' import { CommandWithContext } from './devices/device' import { DeviceContainer } from './devices/deviceContainer' -import { CasparCGDevice, DeviceOptionsCasparCGInternal } from './integrations/casparCG' -import { SisyfosMessageDevice, DeviceOptionsSisyfosInternal } from './integrations/sisyfos' -import { VMixDevice, DeviceOptionsVMixInternal } from './integrations/vmix' -import { VizMSEDevice, DeviceOptionsVizMSEInternal } from './integrations/vizMSE' -import { BaseRemoteDeviceIntegration, RemoteDeviceInstance } from './service/remoteDeviceInstance' -import type { ImplementedServiceDeviceTypes } from './service/devices' -import { DeviceEvents } from './service/device' +import { DeviceOptionsCasparCGInternal } from './integrations/casparCG' +import { DeviceOptionsSisyfosInternal } from './integrations/sisyfos' +import { DeviceOptionsVMixInternal } from './integrations/vmix' +import { DeviceOptionsVizMSEInternal } from './integrations/vizMSE' +import { BaseRemoteDeviceIntegration } from './service/remoteDeviceInstance' +import { ConnectionManager } from './service/ConnectionManager' export { DeviceContainer } export { CommandWithContext } @@ -65,8 +64,6 @@ export const MINTIMEUNIT = 1 // Minimum unit of time /** When resolving and the timeline has repeating objects, only resolve this far into the future */ const RESOLVE_LIMIT_TIME = 10 * 1000 -const FREEZE_LIMIT = 5000 // how long to wait before considering the child to be unresponsive - export type TimelineTriggerTimeResult = Array<{ id: string; time: number }> export { Device } from './devices/device' @@ -97,7 +94,6 @@ interface TimelineCallback { } type TimelineCallbacks = { [key: string]: TimelineCallback } const CALLBACK_WAIT_TIME = 50 -const REMOVE_TIMEOUT = 5000 interface CallbackInstance { playing: boolean | undefined @@ -170,7 +166,7 @@ export class Conductor extends EventEmitter { private _options: ConductorOptions - private devices = new Map>>() + public readonly connectionManager = new ConnectionManager() private _getCurrentTime?: () => number @@ -332,309 +328,6 @@ export class Conductor extends EventEmitter { this._estimateResolveTimeMultiplier = value } - public getDevices(includeUninitialized = false): Array>> { - if (includeUninitialized) { - return Array.from(this.devices.values()) - } else { - return Array.from(this.devices.values()).filter((device) => device.initialized === true) - } - } - - public getDevice( - deviceId: string, - includeUninitialized = false - ): BaseRemoteDeviceIntegration> | undefined { - if (includeUninitialized) { - return this.devices.get(deviceId) - } else { - const device = this.devices.get(deviceId) - if (device?.initialized === true) { - return device - } else { - return undefined - } - } - } - - /** - * Adds a device that can be referenced by the timeline and mappings. - * NOTE: use this with caution! if a device fails to initialise (i.e. because the hardware is turned off) this may never resolve. It is preferred to use createDevice and initDevice separately for this reason. - * @param deviceId Id used by the mappings to reference the device. - * @param deviceOptions The options used to initalize the device - * @returns A promise that resolves with the created device, or rejects with an error message. - */ - public async addDevice( - deviceId: string, - deviceOptions: DeviceOptionsAnyInternal, - activeRundownPlaylistId?: string - ): Promise>> { - const newDevice = await this.createDevice(deviceId, deviceOptions) - - try { - // Temporary listening to events, these are removed after the devide has been initiated. - const instanceId = newDevice.instanceId - const onDeviceInfo: any = (...args: DeviceEvents['info']) => { - this.emit('info', instanceId, ...args) - } - const onDeviceWarning: any = (...args: DeviceEvents['warning']) => { - this.emit('warning', instanceId, ...args) - } - const onDeviceError: any = (...args: DeviceEvents['error']) => { - this.emit('error', instanceId, ...args) - } - const onDeviceDebug: any = (...args: DeviceEvents['debug']) => { - this.emit('debug', instanceId, ...args) - } - const onDeviceDebugState: any = (...args: DeviceEvents['debugState']) => { - this.emit('debugState', args) - } - - newDevice.device.on('info', onDeviceInfo).catch(console.error) - newDevice.device.on('warning', onDeviceWarning).catch(console.error) - newDevice.device.on('error', onDeviceError).catch(console.error) - newDevice.device.on('debug', onDeviceDebug).catch(console.error) - newDevice.device.on('debugState', onDeviceDebugState).catch(console.error) - - const device = await this.initDevice(deviceId, deviceOptions, activeRundownPlaylistId) - - // Remove listeners, expect consumer to subscribe to them now. - newDevice.device.removeListener('info', onDeviceInfo).catch(console.error) - newDevice.device.removeListener('warning', onDeviceWarning).catch(console.error) - newDevice.device.removeListener('error', onDeviceError).catch(console.error) - newDevice.device.removeListener('debug', onDeviceDebug).catch(console.error) - newDevice.device.removeListener('debugState', onDeviceDebugState).catch(console.error) - - return device - } catch (e) { - await this.terminateUnwantedDevice(newDevice) - this.devices.delete(deviceId) - this.emit('error', 'conductor.addDevice', e) - return Promise.reject(e) - } - } - - /** - * Creates an uninitialised device that can be referenced by the timeline and mappings. - * @param deviceId Id used by the mappings to reference the device. - * @param deviceOptions The options used to initalize the device - * @param options Additional options - * @returns A promise that resolves with the created device, or rejects with an error message. - */ - public async createDevice( - deviceId: string, - deviceOptions: DeviceOptionsAnyInternal, - options?: { signal?: AbortSignal } - ): Promise>> { - let newDevice: BaseRemoteDeviceIntegration> | undefined - try { - const throwIfAborted = () => this.throwIfAborted(options?.signal, deviceId, 'creation') - if (this.devices.has(deviceId)) { - throw new Error(`Device "${deviceId}" already exists when creating device`) - } - throwIfAborted() - - const threadedClassOptions: ThreadedClassConfig = { - threadUsage: deviceOptions.threadUsage || 1, - autoRestart: false, - disableMultithreading: !deviceOptions.isMultiThreaded, - instanceName: deviceId, - freezeLimit: FREEZE_LIMIT, - } - - const getCurrentTime = () => { - return this.getCurrentTime() - } - - const newDevicePromise = this.createDeviceContainer(deviceOptions, deviceId, getCurrentTime, threadedClassOptions) - - if (!newDevicePromise) { - const type: any = deviceOptions.type - throw new Error(`No matching device type for "${type}" ("${DeviceType[type]}") found in conductor`) - } - - newDevice = await makeImmediatelyAbortable(async () => { - throwIfAborted() - const newDevice = await newDevicePromise - if (options?.signal?.aborted) { - // if the promise above did not resolve before aborted, - // this executes some time after raceAbortable rejects, serving as a cleanup - await this.terminateUnwantedDevice(newDevice) - throw new AbortError(`Device "${deviceId}" creation aborted`) - } - return newDevice - }, options?.signal) - - newDevice.device.on('resetResolver', () => this.resetResolver()).catch(console.error) - newDevice.on('error', (context, e) => { - this.emit('error', `deviceContainer for "${newDevice?.deviceId}" emitted an error: ${context}, ${e}`) - }) - - // Double check that it hasnt been created while we were busy waiting - if (this.devices.has(deviceId)) { - throw new Error(`Device "${deviceId}" already exists when creating device`) - } - throwIfAborted() - } catch (e) { - await this.terminateUnwantedDevice(newDevice) - - this.emit('error', 'conductor.createDevice', e) - throw e - } - - this.devices.set(deviceId, newDevice) - - return newDevice - } - - private throwIfAborted(signal: AbortSignal | undefined, deviceId: string, action: string) { - if (signal?.aborted) { - throw new AbortError(`Device "${deviceId}" ${action} aborted`) - } - } - - private createDeviceContainer( - deviceOptions: DeviceOptionsAnyInternal, - deviceId: string, - getCurrentTime: () => number, - threadedClassOptions: ThreadedClassConfig - ): Promise>> | null { - switch (deviceOptions.type) { - case DeviceType.CASPARCG: - return DeviceContainer.create( - '../../dist/integrations/casparCG/index.js', - 'CasparCGDevice', - deviceId, - deviceOptions, - getCurrentTime, - threadedClassOptions - ) - case DeviceType.SISYFOS: - return DeviceContainer.create( - '../../dist/integrations/sisyfos/index.js', - 'SisyfosMessageDevice', - deviceId, - deviceOptions, - getCurrentTime, - threadedClassOptions - ) - case DeviceType.VIZMSE: - return DeviceContainer.create( - '../../dist/integrations/vizMSE/index.js', - 'VizMSEDevice', - deviceId, - deviceOptions, - getCurrentTime, - threadedClassOptions - ) - case DeviceType.VMIX: - return DeviceContainer.create( - '../../dist/integrations/vmix/index.js', - 'VMixDevice', - deviceId, - deviceOptions, - getCurrentTime, - threadedClassOptions - ) - case DeviceType.ABSTRACT: - case DeviceType.ATEM: - case DeviceType.HTTPSEND: - case DeviceType.HTTPWATCHER: - case DeviceType.HYPERDECK: - case DeviceType.LAWO: - case DeviceType.OBS: - case DeviceType.OSC: - case DeviceType.MULTI_OSC: - case DeviceType.PANASONIC_PTZ: - case DeviceType.PHAROS: - case DeviceType.SHOTOKU: - case DeviceType.SINGULAR_LIVE: - case DeviceType.SOFIE_CHEF: - case DeviceType.TCPSEND: - case DeviceType.TELEMETRICS: - case DeviceType.TRICASTER: - case DeviceType.QUANTEL: { - ensureIsImplementedAsService(deviceOptions.type) - - // presumably this device is implemented in the new service handler - return RemoteDeviceInstance.create(deviceId, deviceOptions, getCurrentTime, threadedClassOptions) - } - default: - assertNever(deviceOptions) - return null - } - } - - private async terminateUnwantedDevice(newDevice: BaseRemoteDeviceIntegration> | undefined) { - await newDevice - ?.terminate() - .catch((e) => this.emit('error', `Cleanup failed of aborted device "${newDevice.deviceId}": ${e}`)) - } - - /** - * Initialises an existing device that can be referenced by the timeline and mappings. - * @param deviceId Id used by the mappings to reference the device. - * @param deviceOptions The options used to initalize the device - * @param activeRundownPlaylistId Id of the current rundown playlist - * @param options Additional options - * @returns A promise that resolves with the initialised device, or rejects with an error message. - */ - public async initDevice( - deviceId: string, - deviceOptions: DeviceOptionsAnyInternal, - activeRundownPlaylistId?: string, - options?: { signal?: AbortSignal } - ): Promise>> { - const throwIfAborted = () => this.throwIfAborted(options?.signal, deviceId, 'initialisation') - - throwIfAborted() - - const newDevice = this.devices.get(deviceId) - - if (!newDevice) { - throw new Error('Could not find device ' + deviceId + ', has it been created?') - } - - if (newDevice.initialized === true) { - throw new Error('Device ' + deviceId + ' is already initialized!') - } - this.emit( - 'info', - `Initializing device ${newDevice.deviceId} (${newDevice.instanceId}) of type ${DeviceType[deviceOptions.type]}...` - ) - return makeImmediatelyAbortable(async () => { - throwIfAborted() - await newDevice.init(deviceOptions.options, activeRundownPlaylistId) - throwIfAborted() - await newDevice.reloadProps() - throwIfAborted() - this.emit('info', `Device ${newDevice.deviceId} (${newDevice.instanceId}) initialized!`) - return newDevice - }, options?.signal) - } - - /** - * Safely remove a device - * @param deviceId The id of the device to be removed - */ - public async removeDevice(deviceId: string): Promise { - const device = this.devices.get(deviceId) - if (device) { - try { - await Promise.race([ - device.device.terminate(), - new Promise((_, reject) => setTimeout(() => reject('Timeout'), REMOVE_TIMEOUT)), - ]) - } catch (e) { - // An error while terminating is probably not that important, since we'll kill the instance anyway - this.emit('warning', `Error when terminating device ${e}`) - } - await device.terminate() - this.devices.delete(deviceId) - } else { - return Promise.reject('No device found') - } - } - /** * Remove all devices */ @@ -643,7 +336,8 @@ export class Conductor extends EventEmitter { if (this._triggerSendStartStopCallbacksTimeout) clearTimeout(this._triggerSendStartStopCallbacksTimeout) - await this._mapAllDevices(true, async (d) => this.removeDevice(d.deviceId)) + // todo - reenable this + // await this._mapAllDevices(true, async (d) => this.removeDevice(d.deviceId)) } /** @@ -682,7 +376,7 @@ export class Conductor extends EventEmitter { }` ) await this._actionQueue.add(async () => { - await this._mapAllDevices(false, async (d) => + await this._mapAllConnections(false, async (d) => PTimeout( (async () => { const trace = startTrace('conductor:makeReady:' + d.deviceId) @@ -707,7 +401,7 @@ export class Conductor extends EventEmitter { this.activationId = undefined this.emit('debug', `devicesStandDown, ${okToDestroyStuff ? 'okToDestroyStuff' : 'undefined'}`) await this._actionQueue.add(async () => { - await this._mapAllDevices(false, async (d) => + await this._mapAllConnections(false, async (d) => PTimeout( (async () => { const trace = startTrace('conductor:standDown:' + d.deviceId) @@ -725,14 +419,12 @@ export class Conductor extends EventEmitter { return ThreadedClassManager.getThreadsMemoryUsage() } - private async _mapAllDevices( + private async _mapAllConnections( includeUninitialized: boolean, fcn: (d: BaseRemoteDeviceIntegration>) => Promise ): Promise { return PAll( - this.getDevices(true) - .filter((d) => includeUninitialized || d.initialized === true) - .map((d) => async () => fcn(d)), + this.connectionManager.getConnections(includeUninitialized).map((d) => async () => fcn(d)), { stopOnError: false, } @@ -840,8 +532,8 @@ export class Conductor extends EventEmitter { // TODO - the PAll way of doing this provokes https://github.com/nrkno/tv-automation-state-timeline-resolver/pull/139 // The doOnTime calls fire before this, meaning we cleanup the state for a time we have already sent commands for const pPrepareForHandleStates: Promise = Promise.all( - Array.from(this.devices.values()) - .filter((d) => d.initialized === true) + this.connectionManager + .getConnections(false) .map(async (device: BaseRemoteDeviceIntegration>): Promise => { await device.device.prepareForHandleState(resolveTime) }) @@ -926,11 +618,11 @@ export class Conductor extends EventEmitter { const layersPerDevice = this.filterLayersPerDevice( tlState.layers as Timeline.StateInTime, - Array.from(this.devices.values()).filter((d) => d.initialized === true) + this.connectionManager.getConnections(false) ) // Push state to the right device: - await this._mapAllDevices( + await this._mapAllConnections( false, async (device: BaseRemoteDeviceIntegration>): Promise => { if (this._options.optimizeForProduction) { @@ -963,7 +655,7 @@ export class Conductor extends EventEmitter { if (!nextEventTime && tlState.time < this._resolved.validTo) { // There's nothing ahead in the timeline (as far as we can see, ref: this._resolved.validTo) // Tell the devices that the future is clear: - await this._mapAllDevices(true, async (device: BaseRemoteDeviceIntegration>) => { + await this._mapAllConnections(true, async (device: BaseRemoteDeviceIntegration>) => { try { await device.device.clearFuture(tlState.time) } catch (e) { @@ -1113,7 +805,7 @@ export class Conductor extends EventEmitter { const filledState = fillStateFromDatastore(state, this._datastore) // send the filled state to the device handler - return this.getDevice(deviceId)?.device.handleState(filledState, mappings) + return this.connectionManager.getConnection(deviceId)?.device.handleState(filledState, mappings) } setDatastore(newStore: Datastore) { @@ -1145,7 +837,8 @@ export class Conductor extends EventEmitter { for (const s of toBeFilled) { const filledState = fillStateFromDatastore(s.state, this._datastore) - this.getDevice(deviceId) + this.connectionManager + .getConnection(deviceId) ?.device.handleState(filledState, s.mappings) .catch((e) => this.emit('error', 'resolveTimeline' + e + '\nStack: ' + (e as Error).stack)) } @@ -1490,45 +1183,3 @@ function removeParentFromState( } return o } - -/** - * If aborted, rejects as soon as possible, but lets the wraped function safely resolve or reject on its own - * @param func async function to wrap - * @param abortSignal the AbortSignal - * @returns Promise of the same type as `func` - */ -async function makeImmediatelyAbortable( - func: (abortSignal?: AbortSignal) => Promise, - abortSignal?: AbortSignal -): Promise { - const mainPromise = func(abortSignal) - if (!abortSignal) { - return mainPromise - } - let resolveAbortPromise: Function - const abortPromise = new Promise((resolve, reject) => { - resolveAbortPromise = () => { - resolve() - abortSignal.removeEventListener('abort', rejectPromise) - } - const rejectPromise = () => { - reject(new AbortError()) - } - abortSignal.addEventListener('abort', rejectPromise, { once: true }) - }) - return Promise.race([mainPromise, abortPromise]) - .then((result) => { - // only mainPromise could have resolved, so the result must be T - resolveAbortPromise() - return result as T - }) - .catch((reason) => { - // mainPromise or abortPromise might have rejected; calling resolveAbortPromise in the latter case is safe - resolveAbortPromise() - throw reason - }) -} - -function ensureIsImplementedAsService(_type: ImplementedServiceDeviceTypes): void { - // This is a type check -} diff --git a/packages/timeline-state-resolver/src/integrations/casparCG/index.ts b/packages/timeline-state-resolver/src/integrations/casparCG/index.ts index 1d7a951f1..1fa420a2e 100644 --- a/packages/timeline-state-resolver/src/integrations/casparCG/index.ts +++ b/packages/timeline-state-resolver/src/integrations/casparCG/index.ts @@ -130,8 +130,6 @@ export class CasparCGDevice extends DeviceWithState { - if (this.deviceOptions.skipVirginCheck) return false - // a "virgin server" was just restarted (so it is cleared & black). // Otherwise it was probably just a loss of connection @@ -143,18 +141,29 @@ export class CasparCGDevice extends DeviceWithState>[] = [] const channelLength: number = response?.data?.['length'] ?? 0 - // Issue commands - for (let i = 1; i <= channelLength; i++) { - // 1-based index for channels + for (let i = 0; i < channelLength; i++) { + const obj = response.data[i] + + if (!this._currentState.channels[i]) { + this._currentState.channels[obj.channel] = { + channelNo: obj.channel, + videoMode: this.getVideMode(obj), + fps: obj.frameRate, + layers: {}, + } + } + + if (this.deviceOptions.skipVirginCheck) continue + // Issue command const { error, request } = await this._ccg.executeCommand({ command: Commands.InfoChannel, - params: { channel: i }, + params: { channel: obj.channel }, }) if (error) { // We can't return here, as that will leave anything in channelPromises as potentially unhandled channelPromises.push(Promise.reject('execute failed')) - break + continue } channelPromises.push(request) } @@ -174,6 +183,8 @@ export class CasparCGDevice extends DeviceWithState { + if (this.deviceOptions.skipVirginCheck) return + // Finally we can report it as connected this._connected = true this._connectionChanged() @@ -197,25 +208,6 @@ export class CasparCGDevice extends DeviceWithState { - this._currentState.channels[obj.channel] = { - channelNo: obj.channel, - videoMode: this.getVideMode(obj), - fps: obj.frameRate, - layers: {}, - } - }) - } else { - return false // not being able to get channel count is a problem for us - } - if (typeof initOptions.retryInterval === 'number' && initOptions.retryInterval >= 0) { this._retryTime = initOptions.retryInterval || MEDIA_RETRY_INTERVAL this._retryTimeout = setTimeout(() => this._assertIntendedState(), this._retryTime) @@ -235,7 +227,7 @@ export class CasparCGDevice extends DeviceWithState { resolve() this._ccg.removeAllListeners() diff --git a/packages/timeline-state-resolver/src/integrations/sisyfos/index.ts b/packages/timeline-state-resolver/src/integrations/sisyfos/index.ts index 759f02dd1..dec0df22f 100644 --- a/packages/timeline-state-resolver/src/integrations/sisyfos/index.ts +++ b/packages/timeline-state-resolver/src/integrations/sisyfos/index.ts @@ -88,7 +88,11 @@ export class SisyfosMessageDevice extends DeviceWithState true) + this._sisyfos + .connect(initOptions.host, initOptions.port) + .catch((e) => this.emit('error', 'Failed to initialise Sisyfos connection', e)) + + return true } /** Called by the Conductor a bit before a .handleState is called */ prepareForHandleState(newStateTime: number) { diff --git a/packages/timeline-state-resolver/src/integrations/sofieChef/index.ts b/packages/timeline-state-resolver/src/integrations/sofieChef/index.ts index a40aada51..c36752d38 100644 --- a/packages/timeline-state-resolver/src/integrations/sofieChef/index.ts +++ b/packages/timeline-state-resolver/src/integrations/sofieChef/index.ts @@ -89,7 +89,9 @@ export class SofieChefDevice extends Device { // This is where we would do initialization, like connecting to the devices, etc this.initOptions = initOptions - await this._setupWSConnection() + + this._setupWSConnection().catch((e) => this.context.logger.error('Failed to initialise Sofie Chef connection', e)) + return true } private async _setupWSConnection() { diff --git a/packages/timeline-state-resolver/src/integrations/vizMSE/index.ts b/packages/timeline-state-resolver/src/integrations/vizMSE/index.ts index b965bb03c..aaf496ed4 100644 --- a/packages/timeline-state-resolver/src/integrations/vizMSE/index.ts +++ b/packages/timeline-state-resolver/src/integrations/vizMSE/index.ts @@ -142,7 +142,9 @@ export class VizMSEDevice extends DeviceWithState this.emit('error', 'VizMSE', typeof e === 'string' ? new Error(e) : e)) this._vizmseManager.on('debug', (...args) => this.emitDebug(...args)) - await this._vizmseManager.initializeRundown(activeRundownPlaylistId) + this._vizmseManager + .initializeRundown(activeRundownPlaylistId) + .catch((e) => this.emit('error', 'Failed to initialise Viz Rundown', e)) return true } diff --git a/packages/timeline-state-resolver/src/service/ConnectionManager.ts b/packages/timeline-state-resolver/src/service/ConnectionManager.ts new file mode 100644 index 000000000..e722f174c --- /dev/null +++ b/packages/timeline-state-resolver/src/service/ConnectionManager.ts @@ -0,0 +1,395 @@ +import { + DeviceOptionsBase, + DeviceOptionsMultiOSC, + DeviceOptionsTelemetrics, + DeviceType, +} from 'timeline-state-resolver-types' +import { BaseRemoteDeviceIntegration, RemoteDeviceInstance } from './remoteDeviceInstance' +import _ = require('underscore') +import { ThreadedClassConfig } from 'threadedclass' +import { DeviceOptionsAnyInternal } from '../conductor' +import { DeviceContainer } from '..//devices/deviceContainer' +import { assertNever } from 'atem-connection/dist/lib/atemUtil' +import { CasparCGDevice, DeviceOptionsCasparCGInternal } from '../integrations/casparCG' +import { MultiOSCMessageDevice } from '../integrations/multiOsc' +import { DeviceOptionsPharosInternal, PharosDevice } from '../integrations/pharos' +import { DeviceOptionsSingularLiveInternal, SingularLiveDevice } from '../integrations/singularLive' +import { DeviceOptionsSisyfosInternal, SisyfosMessageDevice } from '../integrations/sisyfos' +import { DeviceOptionsSofieChefInternal, SofieChefDevice } from '../integrations/sofieChef' +import { TelemetricsDevice } from '../integrations/telemetrics' +import { DeviceOptionsTriCasterInternal, TriCasterDevice } from '../integrations/tricaster' +import { DeviceOptionsVizMSEInternal, VizMSEDevice } from '../integrations/vizMSE' +import { DeviceOptionsVMixInternal, VMixDevice } from '../integrations/vmix' +import { ImplementedServiceDeviceTypes } from './devices' +import { EventEmitter } from 'eventemitter3' +import { DeviceEvents } from './device' + +interface Operation { + operation: 'create' | 'update' | 'delete' | 'setDebug' + id: string +} + +const FREEZE_LIMIT = 5000 // how long to wait before considering the child to be unresponsive + +export class ConnectionManager extends EventEmitter { + private _config: Map = new Map() + private _connections: Map> = new Map() + private _updating = false + + /** + * Set the config options for all connections + */ + public setConnections(connectionsConfig: Map) { + this._config = connectionsConfig + this._updateConnections() + } + + public getConnections(includeUninitialized = false): Array>> { + if (includeUninitialized) { + return Array.from(this._connections.values()) + } else { + return Array.from(this._connections.values()).filter((conn) => conn.initialized === true) + } + } + + public getConnection( + connectionId: string, + includeUninitialized = false + ): BaseRemoteDeviceIntegration> | undefined { + if (includeUninitialized) { + return this._connections.get(connectionId) + } else { + const device = this._connections.get(connectionId) + if (device?.initialized === true) { + return device + } else { + return undefined + } + } + } + + /** + * Iterate over config and check that the existing device has the right config, if + * not... recreate it + */ + private _updateConnections() { + if (this._updating) return + this._updating = true + + const operations: Operation[] = [] + + for (const [deviceId, config] of this._config.entries()) { + // find connection + const connection = this._connections.get(deviceId) + + if (connection) { + // see if it should be restarted because of an update + if (configHasChanged(connection, config)) { + operations.push({ operation: 'update', id: deviceId }) + } else if ( + connection.deviceOptions.debug !== config.debug || + connection.deviceOptions.debugState !== config.debugState + ) { + // see if we should set the debug params + operations.push({ operation: 'setDebug', id: deviceId }) + } + } else { + // create + operations.push({ operation: 'create', id: deviceId }) + } + } + + for (const deviceId of this._connections.keys()) { + // find if still in config + const config = this._config.get(deviceId) + + if (!config) { + // not found, so it should be closed + operations.push({ operation: 'delete', id: deviceId }) + } + } + + console.log('Ran update, got ops', operations) + + if (operations.length === 0) { + // no operations needed + this._updating = false + return + } + + Promise.allSettled(operations.map((op) => this.executeOperation(op))).then(() => { + this._updating = false + // todo - do another run to verify we have achieved creation of all devices (but prevent us from running into a death loop) + }) + } + + private async executeOperation({ operation, id }: Operation): Promise { + // todo: timeout here or log errors? + + switch (operation) { + case 'create': + await this.createConnection(id) + break + case 'delete': + await this.deleteConnection(id) + break + case 'update': + await this.deleteConnection(id) + await this.createConnection(id) + break + case 'setDebug': + await this.setDebugForConnection(id) + break + } + } + + private async createConnection(id: string): Promise { + const deviceOptions = this._config.get(id) + if (!deviceOptions) return // unexpected - throw error? + + const threadedClassOptions: ThreadedClassConfig = { + threadUsage: deviceOptions.threadUsage || 1, + autoRestart: false, + disableMultithreading: !deviceOptions.isMultiThreaded, + instanceName: id, + freezeLimit: FREEZE_LIMIT, + } + + const container = await createContainer(deviceOptions, id, () => Date.now(), threadedClassOptions) // time out if this gets el stucko + + if (!container) return // todo - log or throw error? + + // set up event handlers + this._setupDeviceListeners(id, container) + + this._connections.set(id, container) + this.emit('connectionAdded', id, container) + + // trigger device init + this._handleConnectionInitialisation(id, container).catch(() => { + this.emit('error', 'Device ' + id + ' failed to initialise') + this._connections.delete(id) // todo - this will cause device to be recreated next, which may cause a spiral? + }) + } + + private async deleteConnection(id: string): Promise { + const connection = this._connections.get(id) + if (!connection) return + + this._connections.delete(id) + this.emit('connectionRemoved', id) + + try { + await connection.device.terminate() + await connection.device.removeAllListeners() + await connection.terminate() + } catch { + await connection.terminate() + } + } + + private async setDebugForConnection(id: string): Promise { + const config = this._config.get(id) + const connection = this._connections.get(id) + if (!connection || !config) return + + try { + connection.device.setDebugLogging(config.debug ?? false) + connection.device.setDebugState(config.debugState ?? false) + } catch { + // log warning here + } + } + + private async _handleConnectionInitialisation( + id: string, + container: BaseRemoteDeviceIntegration + ) { + const deviceOptions = this._config.get(id) + if (!deviceOptions) return // unexpected - throw error? + + this.emit( + 'info', + `Initializing device ${id} (${container.instanceId}) of type ${DeviceType[deviceOptions.type]}...` + ) + await container.init(deviceOptions.options, undefined) + await container.reloadProps() + this.emit('info', `Device ${id} (${container.instanceId}) initialized!`) + } + + private async _setupDeviceListeners( + id: string, + container: BaseRemoteDeviceIntegration + ): Promise { + const passEvent = (ev: T) => { + const evHandler: any = (...args: DeviceEvents[T]) => this.emit('connectionEvent:' + ev, id, ...args) + container.device.on(ev, evHandler) + } + + passEvent('info') + passEvent('warning') + passEvent('error') + passEvent('debug') + passEvent('debugState') + passEvent('connectionChanged') + passEvent('resetResolver') + passEvent('slowCommand') + passEvent('slowSentCommand') + passEvent('slowFulfilledCommand') + passEvent('commandReport') + passEvent('commandError') + passEvent('updateMediaObject') + passEvent('clearMediaObjects') + passEvent('timeTrace') + } +} + +/** + * A config has changed if any of the options are no longer the same, taking default values into + * consideration. In addition, the debug logging flag should be ignored as that can be changed at runtime. + */ +function configHasChanged( + connection: BaseRemoteDeviceIntegration>, + config: DeviceOptionsBase +): boolean { + const oldConfig = connection.deviceOptions + + // first check common options + if ( + oldConfig.disable !== config.disable || + oldConfig.isMultiThreaded !== config.isMultiThreaded || + oldConfig.limitSlowFulfilledCommand !== config.limitSlowFulfilledCommand || + oldConfig.limitSlowSentCommand !== config.limitSlowSentCommand || + oldConfig.reportAllCommands !== config.reportAllCommands || + oldConfig.threadUsage !== config.threadUsage || + oldConfig.type !== config.type + ) + return true + + // now check device specific options + return _.isEqual(oldConfig.options, config.options) +} + +function createContainer( + deviceOptions: DeviceOptionsAnyInternal, + deviceId: string, + getCurrentTime: () => number, + threadedClassOptions: ThreadedClassConfig +): Promise>> | null { + switch (deviceOptions.type) { + case DeviceType.CASPARCG: + return DeviceContainer.create( + '../../dist/integrations/casparCG/index.js', + 'CasparCGDevice', + deviceId, + deviceOptions, + getCurrentTime, + threadedClassOptions + ) + case DeviceType.PHAROS: + return DeviceContainer.create( + '../../dist/integrations/pharos/index.js', + 'PharosDevice', + deviceId, + deviceOptions, + getCurrentTime, + threadedClassOptions + ) + case DeviceType.SISYFOS: + return DeviceContainer.create( + '../../dist/integrations/sisyfos/index.js', + 'SisyfosMessageDevice', + deviceId, + deviceOptions, + getCurrentTime, + threadedClassOptions + ) + case DeviceType.VIZMSE: + return DeviceContainer.create( + '../../dist/integrations/vizMSE/index.js', + 'VizMSEDevice', + deviceId, + deviceOptions, + getCurrentTime, + threadedClassOptions + ) + case DeviceType.SINGULAR_LIVE: + return DeviceContainer.create( + '../../dist/integrations/singularLive/index.js', + 'SingularLiveDevice', + deviceId, + deviceOptions, + getCurrentTime, + threadedClassOptions + ) + case DeviceType.VMIX: + return DeviceContainer.create( + '../../dist/integrations/vmix/index.js', + 'VMixDevice', + deviceId, + deviceOptions, + getCurrentTime, + threadedClassOptions + ) + case DeviceType.TELEMETRICS: + return DeviceContainer.create( + '../../dist/integrations/telemetrics/index.js', + 'TelemetricsDevice', + deviceId, + deviceOptions, + getCurrentTime, + threadedClassOptions + ) + case DeviceType.SOFIE_CHEF: + return DeviceContainer.create( + '../../dist/integrations/sofieChef/index.js', + 'SofieChefDevice', + deviceId, + deviceOptions, + getCurrentTime, + threadedClassOptions + ) + case DeviceType.TRICASTER: + return DeviceContainer.create( + '../../dist/integrations/tricaster/index.js', + 'TriCasterDevice', + deviceId, + deviceOptions, + getCurrentTime, + threadedClassOptions + ) + case DeviceType.MULTI_OSC: + return DeviceContainer.create( + '../../dist/integrations/multiOsc/index.js', + 'MultiOSCMessageDevice', + deviceId, + deviceOptions, + getCurrentTime, + threadedClassOptions + ) + case DeviceType.ABSTRACT: + case DeviceType.ATEM: + case DeviceType.HTTPSEND: + case DeviceType.HTTPWATCHER: + case DeviceType.HYPERDECK: + case DeviceType.LAWO: + case DeviceType.OBS: + case DeviceType.OSC: + case DeviceType.PANASONIC_PTZ: + case DeviceType.SHOTOKU: + case DeviceType.TCPSEND: + case DeviceType.QUANTEL: { + ensureIsImplementedAsService(deviceOptions.type) + + // presumably this device is implemented in the new service handler + return RemoteDeviceInstance.create(deviceId, deviceOptions, getCurrentTime, threadedClassOptions) + } + default: + assertNever(deviceOptions) + return null + } +} + +function ensureIsImplementedAsService(_type: ImplementedServiceDeviceTypes): void { + // This is a type check +} From 81d52ff8d23546686b3317ea2e454bb9e12be2ed Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Fri, 31 May 2024 10:59:25 +0200 Subject: [PATCH 02/17] chore: wip --- .../timeline-state-resolver/src/service/ConnectionManager.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/timeline-state-resolver/src/service/ConnectionManager.ts b/packages/timeline-state-resolver/src/service/ConnectionManager.ts index e722f174c..b1452f83f 100644 --- a/packages/timeline-state-resolver/src/service/ConnectionManager.ts +++ b/packages/timeline-state-resolver/src/service/ConnectionManager.ts @@ -267,7 +267,7 @@ function configHasChanged( return true // now check device specific options - return _.isEqual(oldConfig.options, config.options) + return !_.isEqual(oldConfig.options, config.options) } function createContainer( From 24816066108ccbf91ec942c7020af6c9d1644bb1 Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Tue, 9 Jul 2024 09:15:45 +0200 Subject: [PATCH 03/17] chore: wip: add retrying to connection mgmt --- .../src/service/ConnectionManager.ts | 126 ++++++++++++------ 1 file changed, 85 insertions(+), 41 deletions(-) diff --git a/packages/timeline-state-resolver/src/service/ConnectionManager.ts b/packages/timeline-state-resolver/src/service/ConnectionManager.ts index b1452f83f..fbfb77cb0 100644 --- a/packages/timeline-state-resolver/src/service/ConnectionManager.ts +++ b/packages/timeline-state-resolver/src/service/ConnectionManager.ts @@ -31,11 +31,27 @@ interface Operation { const FREEZE_LIMIT = 5000 // how long to wait before considering the child to be unresponsive -export class ConnectionManager extends EventEmitter { +export type ConnectionManagerEvents = ConnectionManagerIntEvents & MappedDeviceEvents +export interface ConnectionManagerIntEvents { + info: [info: string] + warning: [warning: string] + error: [context: string, err?: Error] + debug: [...debug: any[]] + + connectionAdded: [id: string, container: BaseRemoteDeviceIntegration>] + connectionRemoved: [id: string] +} +export type MappedDeviceEvents = { + [T in keyof DeviceEvents as `connectionEvent:${T}`]: [id: string, ...DeviceEvents[T]] +} + +export class ConnectionManager extends EventEmitter { private _config: Map = new Map() private _connections: Map> = new Map() private _updating = false + private _connectionAttempts = new Map() + /** * Set the config options for all connections */ @@ -110,42 +126,72 @@ export class ConnectionManager extends EventEmitter { } console.log('Ran update, got ops', operations) + const isAllowedOp = (op: Operation): boolean => { + if (op.operation !== 'create') return true // allow non-create ops + + const nextCreate = this._connectionAttempts.get(op.id) + if (!nextCreate || nextCreate.next < Date.now()) return true + + return false + } + const allowedOperations = operations.filter(isAllowedOp) if (operations.length === 0) { - // no operations needed + // no operations needed, so stop the loop + this._updating = false + return + } else if (allowedOperations.length === 0) { this._updating = false + // wait until next + const nextTime = Array.from(this._connectionAttempts.values()).reduce((a, b) => (a.next < b.next ? a : b)) + // todo - keep track of these + setTimeout(() => { + this._updateConnections() + }, nextTime.next - Date.now()) + // there's nothing to execute right now so return return } - Promise.allSettled(operations.map((op) => this.executeOperation(op))).then(() => { + Promise.allSettled(allowedOperations.map((op) => this.executeOperation(op))).then(() => { this._updating = false - // todo - do another run to verify we have achieved creation of all devices (but prevent us from running into a death loop) + // rerun the algorithm once to make sure we have no missed operations in the meanwhile + this._updateConnections() }) } private async executeOperation({ operation, id }: Operation): Promise { - // todo: timeout here or log errors? - - switch (operation) { - case 'create': - await this.createConnection(id) - break - case 'delete': - await this.deleteConnection(id) - break - case 'update': - await this.deleteConnection(id) - await this.createConnection(id) - break - case 'setDebug': - await this.setDebugForConnection(id) - break + // todo - timeout? + try { + switch (operation) { + case 'create': + await this.createConnection(id) + break + case 'delete': + await this.deleteConnection(id) + break + case 'update': + await this.deleteConnection(id) + await this.createConnection(id) + break + case 'setDebug': + await this.setDebugForConnection(id) + break + } + } catch { + this.emit('warning', `Failed to execute "${operation} for ${id}"`) } } private async createConnection(id: string): Promise { const deviceOptions = this._config.get(id) - if (!deviceOptions) return // unexpected - throw error? + if (!deviceOptions) return // has been removed since, so do not create + + const lastAttempt = this._connectionAttempts.get(id) + const last = lastAttempt?.last ?? Date.now() + this._connectionAttempts.set(id, { + last: Date.now(), + next: Date.now() + Math.min(Math.max(Date.now() - last, 2000) * 2, 60 * 1000), + }) // first retry after 4secs, double it every try, max 60s const threadedClassOptions: ThreadedClassConfig = { threadUsage: deviceOptions.threadUsage || 1, @@ -166,15 +212,24 @@ export class ConnectionManager extends EventEmitter { this.emit('connectionAdded', id, container) // trigger device init - this._handleConnectionInitialisation(id, container).catch(() => { - this.emit('error', 'Device ' + id + ' failed to initialise') - this._connections.delete(id) // todo - this will cause device to be recreated next, which may cause a spiral? - }) + this._handleConnectionInitialisation(id, container) + .then(() => { + this._connectionAttempts.delete(id) + // todo - if this triggers false, shell we restart the connection? + }) + .catch((e) => { + this.emit('error', 'Device ' + id + ' failed to initialise') + this._connections.delete(id) + + container.terminate().catch(() => this.emit('warning', `Failed to initialise ${id} (${e})`)) + // todo - find a good point to retrigger _updateConnections + this._updateConnections() // this can't be the right place... right?? it was not :kekw:... was it not?? + }) } private async deleteConnection(id: string): Promise { const connection = this._connections.get(id) - if (!connection) return + if (!connection) return // already removed / never existed this._connections.delete(id) this.emit('connectionRemoved', id) @@ -197,7 +252,7 @@ export class ConnectionManager extends EventEmitter { connection.device.setDebugLogging(config.debug ?? false) connection.device.setDebugState(config.debugState ?? false) } catch { - // log warning here + this.emit('warning', 'Failed to update debug values for ' + id) } } @@ -222,7 +277,8 @@ export class ConnectionManager extends EventEmitter { container: BaseRemoteDeviceIntegration ): Promise { const passEvent = (ev: T) => { - const evHandler: any = (...args: DeviceEvents[T]) => this.emit('connectionEvent:' + ev, id, ...args) + const evHandler: any = (...args: DeviceEvents[T]) => + this.emit(('connectionEvent:' + ev) as `connectionEvent:${keyof DeviceEvents}`, id, ...args) container.device.on(ev, evHandler) } @@ -254,20 +310,8 @@ function configHasChanged( ): boolean { const oldConfig = connection.deviceOptions - // first check common options - if ( - oldConfig.disable !== config.disable || - oldConfig.isMultiThreaded !== config.isMultiThreaded || - oldConfig.limitSlowFulfilledCommand !== config.limitSlowFulfilledCommand || - oldConfig.limitSlowSentCommand !== config.limitSlowSentCommand || - oldConfig.reportAllCommands !== config.reportAllCommands || - oldConfig.threadUsage !== config.threadUsage || - oldConfig.type !== config.type - ) - return true - // now check device specific options - return !_.isEqual(oldConfig.options, config.options) + return !_.isEqual(_.omit(oldConfig, 'debug', 'debugState'), _.omit(config, 'debug', 'debugState')) } function createContainer( From 14e1a606b1cc79410fadbe10c6a36b21f61eb218 Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Mon, 26 Aug 2024 07:50:13 +0200 Subject: [PATCH 04/17] chore: wip --- .../src/service/ConnectionManager.ts | 45 +++++++++++-------- 1 file changed, 27 insertions(+), 18 deletions(-) diff --git a/packages/timeline-state-resolver/src/service/ConnectionManager.ts b/packages/timeline-state-resolver/src/service/ConnectionManager.ts index fbfb77cb0..b7bdd079c 100644 --- a/packages/timeline-state-resolver/src/service/ConnectionManager.ts +++ b/packages/timeline-state-resolver/src/service/ConnectionManager.ts @@ -22,7 +22,7 @@ import { DeviceOptionsVizMSEInternal, VizMSEDevice } from '../integrations/vizMS import { DeviceOptionsVMixInternal, VMixDevice } from '../integrations/vmix' import { ImplementedServiceDeviceTypes } from './devices' import { EventEmitter } from 'eventemitter3' -import { DeviceEvents } from './device' +import { DeviceInstanceEvents } from './DeviceInstance' interface Operation { operation: 'create' | 'update' | 'delete' | 'setDebug' @@ -42,7 +42,7 @@ export interface ConnectionManagerIntEvents { connectionRemoved: [id: string] } export type MappedDeviceEvents = { - [T in keyof DeviceEvents as `connectionEvent:${T}`]: [id: string, ...DeviceEvents[T]] + [T in keyof DeviceInstanceEvents as `connectionEvent:${T}`]: [deviceId: string, ...DeviceInstanceEvents[T]] } export class ConnectionManager extends EventEmitter { @@ -51,6 +51,7 @@ export class ConnectionManager extends EventEmitter { private _updating = false private _connectionAttempts = new Map() + private _nextAttempt: NodeJS.Timeout | undefined /** * Set the config options for all connections @@ -75,9 +76,9 @@ export class ConnectionManager extends EventEmitter { if (includeUninitialized) { return this._connections.get(connectionId) } else { - const device = this._connections.get(connectionId) - if (device?.initialized === true) { - return device + const connection = this._connections.get(connectionId) + if (connection?.initialized === true) { + return connection } else { return undefined } @@ -85,13 +86,18 @@ export class ConnectionManager extends EventEmitter { } /** - * Iterate over config and check that the existing device has the right config, if + * Iterate over config and check that the existing connection has the right config, if * not... recreate it */ private _updateConnections() { if (this._updating) return this._updating = true + if (this._nextAttempt) { + clearTimeout(this._nextAttempt) + this._nextAttempt = undefined + } + const operations: Operation[] = [] for (const [deviceId, config] of this._config.entries()) { @@ -125,7 +131,6 @@ export class ConnectionManager extends EventEmitter { } } - console.log('Ran update, got ops', operations) const isAllowedOp = (op: Operation): boolean => { if (op.operation !== 'create') return true // allow non-create ops @@ -142,18 +147,20 @@ export class ConnectionManager extends EventEmitter { return } else if (allowedOperations.length === 0) { this._updating = false + // wait until next const nextTime = Array.from(this._connectionAttempts.values()).reduce((a, b) => (a.next < b.next ? a : b)) - // todo - keep track of these - setTimeout(() => { + this._nextAttempt = setTimeout(() => { this._updateConnections() }, nextTime.next - Date.now()) + // there's nothing to execute right now so return return } Promise.allSettled(allowedOperations.map((op) => this.executeOperation(op))).then(() => { this._updating = false + // rerun the algorithm once to make sure we have no missed operations in the meanwhile this._updateConnections() }) @@ -203,7 +210,10 @@ export class ConnectionManager extends EventEmitter { const container = await createContainer(deviceOptions, id, () => Date.now(), threadedClassOptions) // time out if this gets el stucko - if (!container) return // todo - log or throw error? + if (!container) { + this.emit('warning', 'Failed to create container for ' + id) + return + } // set up event handlers this._setupDeviceListeners(id, container) @@ -211,14 +221,13 @@ export class ConnectionManager extends EventEmitter { this._connections.set(id, container) this.emit('connectionAdded', id, container) - // trigger device init + // trigger conenction init this._handleConnectionInitialisation(id, container) .then(() => { this._connectionAttempts.delete(id) - // todo - if this triggers false, shell we restart the connection? }) .catch((e) => { - this.emit('error', 'Device ' + id + ' failed to initialise') + this.emit('error', 'Connection ' + id + ' failed to initialise') this._connections.delete(id) container.terminate().catch(() => this.emit('warning', `Failed to initialise ${id} (${e})`)) @@ -265,20 +274,20 @@ export class ConnectionManager extends EventEmitter { this.emit( 'info', - `Initializing device ${id} (${container.instanceId}) of type ${DeviceType[deviceOptions.type]}...` + `Initializing connection ${id} (${container.instanceId}) of type ${DeviceType[deviceOptions.type]}...` ) await container.init(deviceOptions.options, undefined) await container.reloadProps() - this.emit('info', `Device ${id} (${container.instanceId}) initialized!`) + this.emit('info', `Connection ${id} (${container.instanceId}) initialized!`) } private async _setupDeviceListeners( id: string, container: BaseRemoteDeviceIntegration ): Promise { - const passEvent = (ev: T) => { - const evHandler: any = (...args: DeviceEvents[T]) => - this.emit(('connectionEvent:' + ev) as `connectionEvent:${keyof DeviceEvents}`, id, ...args) + const passEvent = (ev: T) => { + const evHandler: any = (...args: DeviceInstanceEvents[T]) => + this.emit(('connectionEvent:' + ev) as `connectionEvent:${keyof DeviceInstanceEvents}`, id, ...args) container.device.on(ev, evHandler) } From 34cad9ebc50e1156195fb7aaeeb79b70442da58f Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Mon, 26 Aug 2024 08:49:52 +0200 Subject: [PATCH 05/17] chore: add basic test --- .../src/service/ConnectionManager.ts | 2 +- .../__tests__/ConnectionManager.spec.ts | 54 +++++++++++++++++++ 2 files changed, 55 insertions(+), 1 deletion(-) create mode 100644 packages/timeline-state-resolver/src/service/__tests__/ConnectionManager.spec.ts diff --git a/packages/timeline-state-resolver/src/service/ConnectionManager.ts b/packages/timeline-state-resolver/src/service/ConnectionManager.ts index b7bdd079c..b016068c0 100644 --- a/packages/timeline-state-resolver/src/service/ConnectionManager.ts +++ b/packages/timeline-state-resolver/src/service/ConnectionManager.ts @@ -42,7 +42,7 @@ export interface ConnectionManagerIntEvents { connectionRemoved: [id: string] } export type MappedDeviceEvents = { - [T in keyof DeviceInstanceEvents as `connectionEvent:${T}`]: [deviceId: string, ...DeviceInstanceEvents[T]] + [T in keyof DeviceInstanceEvents as `connectionEvent:${T}`]: [string, ...DeviceInstanceEvents[T]] } export class ConnectionManager extends EventEmitter { diff --git a/packages/timeline-state-resolver/src/service/__tests__/ConnectionManager.spec.ts b/packages/timeline-state-resolver/src/service/__tests__/ConnectionManager.spec.ts new file mode 100644 index 000000000..440651a4b --- /dev/null +++ b/packages/timeline-state-resolver/src/service/__tests__/ConnectionManager.spec.ts @@ -0,0 +1,54 @@ +import { DeviceType, OSCDeviceType } from 'timeline-state-resolver-types' +import { ConstructedMockDevices, MockDeviceInstanceWrapper } from '../../__tests__/mockDeviceInstanceWrapper' +import { ConnectionManager } from '../ConnectionManager' + +// Mock explicitly the 'dist' version, as that is what threadedClass is being told to load +jest.mock('../../../dist/service/DeviceInstance', () => ({ + DeviceInstanceWrapper: MockDeviceInstanceWrapper, +})) +jest.mock('../DeviceInstance', () => ({ + DeviceInstanceWrapper: MockDeviceInstanceWrapper, +})) + +describe('ConnectionManager', () => { + const connManager = new ConnectionManager() + + test('adding/removing a device', async () => { + let resolveAdded: undefined | (() => void) = undefined + const psAdded = new Promise((resolveCb) => (resolveAdded = resolveCb)) + connManager.on('connectionAdded', () => { + if (resolveAdded) resolveAdded() + }) + + let resolveRemoved: undefined | (() => void) = undefined + const psRemoved = new Promise((resolveCb) => (resolveRemoved = resolveCb)) + connManager.on('connectionRemoved', () => { + if (resolveRemoved) resolveRemoved() + }) + + connManager.setConnections( + new Map( + Object.entries({ + osc0: { + type: DeviceType.OSC, + options: { + host: '127.0.0.1', + port: 5250, + type: OSCDeviceType.UDP, + }, + }, + }) + ) + ) + + await psAdded + + expect(ConstructedMockDevices['osc0']).toBeTruthy() + + connManager.setConnections(new Map()) + + await psRemoved + + expect(ConstructedMockDevices['osc0']).toBeFalsy() + }) +}) From eb03434c1d8a05c33f165d100c0799121a71e8c7 Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Wed, 4 Sep 2024 10:36:59 +0200 Subject: [PATCH 06/17] feat: add option to resync states without resolving tl --- .../timeline-state-resolver/src/conductor.ts | 26 ++++++++++++ .../src/service/DeviceInstance.ts | 4 +- .../src/service/device.ts | 2 + .../src/service/stateHandler.ts | 40 +++++++++++++++++++ 4 files changed, 70 insertions(+), 2 deletions(-) diff --git a/packages/timeline-state-resolver/src/conductor.ts b/packages/timeline-state-resolver/src/conductor.ts index 7f267cea2..a327319b5 100644 --- a/packages/timeline-state-resolver/src/conductor.ts +++ b/packages/timeline-state-resolver/src/conductor.ts @@ -235,6 +235,9 @@ export class Conductor extends EventEmitter { this.emit('error', 'Error during auto-init: ', e) }) } + + this.connectionManager.on('error', (e) => this.emit('error', e)) + this.connectionManager.on('connectionEvent:resyncStates', (deviceId: string) => this.resyncDeviceStates(deviceId)) } /** * Initializates the resolver, with optional multithreading @@ -849,6 +852,29 @@ export class Conductor extends EventEmitter { }) } + private resyncDeviceStates(deviceId: string) { + this._actionQueue + .add(() => { + const toBeFilled = _.compact([ + // shallow clone so we don't reverse the array in place + [...this._deviceStates[deviceId]].reverse().find((s) => s.time <= this.getCurrentTime()), // one state before now + ...this._deviceStates[deviceId].filter((s) => s.time > this.getCurrentTime()), // all states after now + ]) + + for (const s of toBeFilled) { + const filledState = fillStateFromDatastore(s.state, this._datastore) + + this.connectionManager + .getConnection(deviceId) + ?.device.handleState(filledState, s.mappings) + .catch((e) => this.emit('error', 'resolveTimeline' + e + '\nStack: ' + (e as Error).stack)) + } + }) + .catch((e) => { + this.emit('error', 'Caught error in resyncDeviceStates' + e) + }) + } + getTimelineSize(): number { if (this._timelineSize === undefined) { // Update the cache: diff --git a/packages/timeline-state-resolver/src/service/DeviceInstance.ts b/packages/timeline-state-resolver/src/service/DeviceInstance.ts index 5e8554e88..71a2a5f6c 100644 --- a/packages/timeline-state-resolver/src/service/DeviceInstance.ts +++ b/packages/timeline-state-resolver/src/service/DeviceInstance.ts @@ -264,13 +264,13 @@ export class DeviceInstanceWrapper extends EventEmitter { resetState: async () => { await this._stateHandler.setCurrentState(undefined) await this._stateHandler.clearFutureStates() - this.emit('resetResolver') + this.emit('resyncStates') }, resetToState: async (state: any) => { await this._stateHandler.setCurrentState(state) await this._stateHandler.clearFutureStates() - this.emit('resetResolver') + this.emit('resyncStates') }, } } diff --git a/packages/timeline-state-resolver/src/service/device.ts b/packages/timeline-state-resolver/src/service/device.ts index 7884faf3e..05313fcd5 100644 --- a/packages/timeline-state-resolver/src/service/device.ts +++ b/packages/timeline-state-resolver/src/service/device.ts @@ -111,6 +111,8 @@ export interface DeviceEvents { connectionChanged: [status: Omit] /** A message to the resolver that something has happened that warrants a reset of the resolver (to re-run it again) */ resetResolver: [] + /** A message to the resolver that the device needs to receive all known states */ + resyncStates: [] /** @deprecated replaced by slowSentCommand & slowFulfilledCommand */ slowCommand: [commandInfo: string] diff --git a/packages/timeline-state-resolver/src/service/stateHandler.ts b/packages/timeline-state-resolver/src/service/stateHandler.ts index 25f99e70c..586051575 100644 --- a/packages/timeline-state-resolver/src/service/stateHandler.ts +++ b/packages/timeline-state-resolver/src/service/stateHandler.ts @@ -105,6 +105,9 @@ export class StateHandler { } } + /** + * Sets the current state and makes sure the commands to get to the next state are still corrects + **/ async setCurrentState(state: DeviceState | undefined) { this.currentState = { commands: [], @@ -115,6 +118,43 @@ export class StateHandler { await this.calculateNextStateChange() } + /** + * This takes in a DeviceState and then updates the commands such that the device + * will be put back into its intended state as designated by the timeline + * @todo: this may need to be tied into _executingStateChange variable + */ + async updateStateFromDeviceState(state: DeviceState | undefined) { + // update the current state to the state we received + const timelineState = this.currentState?.state || { + time: this.context.getCurrentTime(), + layers: {}, + nextEvents: [], + } + const currentMappings = this.currentState?.mappings || {} + + this.currentState = { + commands: [], + deviceState: state, + state: timelineState, + mappings: currentMappings, + } + + // calculate how to get to the timeline state (because the device state may have changed based on device config changes or something) + const trace = startTrace('device:convertTimelineStateToDeviceState', { deviceId: this.context.deviceId }) + const deviceState = this.device.convertTimelineStateToDeviceState(timelineState, currentMappings) // @todo - we should probably be recalculating all of these :x + this.context.emitTimeTrace(endTrace(trace)) + + // push a new state + this.stateQueue.unshift({ + deviceState: deviceState, + state: this.currentState?.state || { time: this.context.getCurrentTime(), layers: {}, nextEvents: [] }, + mappings: this.currentState?.mappings || {}, + }) + + // now we let it calculate commands to get into the right state, which should be executed immediately given this state is from the past + await this.calculateNextStateChange() + } + clearFutureAfterTimestamp(t: number) { this.stateQueue = this.stateQueue.filter((s) => s.state.time <= t) } From a7ae8732b6b18c881d93a8ba39b872c735792907 Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Wed, 4 Sep 2024 10:39:34 +0200 Subject: [PATCH 07/17] chore: update integrations to resync state at first connection --- .../src/integrations/casparCG/index.ts | 8 ++--- .../src/integrations/multiOsc/index.ts | 3 ++ .../src/integrations/osc/index.ts | 23 ++++++++++++++- .../src/integrations/pharos/index.ts | 1 + .../src/integrations/quantel/index.ts | 29 ++++++++++++++----- .../src/integrations/shotoku/index.ts | 9 ++++++ .../src/integrations/sisyfos/index.ts | 5 ++++ .../src/integrations/sofieChef/index.ts | 7 ++++- .../src/integrations/tcpSend/index.ts | 11 +++++++ .../src/integrations/vizMSE/index.ts | 5 ++++ .../src/integrations/vmix/index.ts | 4 +++ 11 files changed, 92 insertions(+), 13 deletions(-) diff --git a/packages/timeline-state-resolver/src/integrations/casparCG/index.ts b/packages/timeline-state-resolver/src/integrations/casparCG/index.ts index 1fa420a2e..101b3655b 100644 --- a/packages/timeline-state-resolver/src/integrations/casparCG/index.ts +++ b/packages/timeline-state-resolver/src/integrations/casparCG/index.ts @@ -123,6 +123,7 @@ export class CasparCGDevice extends DeviceWithState { this.makeReady(false) // always make sure timecode is correct, setting it can never do bad @@ -183,16 +184,15 @@ export class CasparCGDevice extends DeviceWithState { - if (this.deviceOptions.skipVirginCheck) return - // Finally we can report it as connected this._connected = true this._connectionChanged() - if (doResync) { + if (firstConnect || doResync) { + firstConnect = false this._currentState = { channels: {} } this.clearStates() - this.emit('resetResolver') + this.emit('resyncStates') } }) .catch((e) => { diff --git a/packages/timeline-state-resolver/src/integrations/multiOsc/index.ts b/packages/timeline-state-resolver/src/integrations/multiOsc/index.ts index 46135ad59..1dfe18bf8 100644 --- a/packages/timeline-state-resolver/src/integrations/multiOsc/index.ts +++ b/packages/timeline-state-resolver/src/integrations/multiOsc/index.ts @@ -76,6 +76,9 @@ export class MultiOSCMessageDevice extends Device [id, {}]))) + return true } diff --git a/packages/timeline-state-resolver/src/integrations/osc/index.ts b/packages/timeline-state-resolver/src/integrations/osc/index.ts index aa791a11b..13bb71a1f 100644 --- a/packages/timeline-state-resolver/src/integrations/osc/index.ts +++ b/packages/timeline-state-resolver/src/integrations/osc/index.ts @@ -56,9 +56,21 @@ export class OscDevice extends Device { this._oscClientStatus = 'connected' - this.context.connectionChanged(this.getStatus()) + if (firstConnect) { + // note - perhaps we could resend the commands every time we reconnect? or that could be a device option + firstConnect = false + this.context.connectionChanged(this.getStatus()) + this.context + .resetToState({}) + .catch((e) => + this.context.logger.warning( + 'Failed to reset to state after first connection, device may be in unknown state (reason: ' + e + ')' + ) + ) + } }) client.socket.on('close', () => { this._oscClientStatus = 'disconnected' @@ -74,6 +86,15 @@ export class OscDevice extends Device { + this.context + .resetToState({}) + .catch((e) => + this.context.logger.warning( + 'Failed to reset to state after first connection, device may be in unknown state (reason: ' + e + ')' + ) + ) + }) this._oscClient.open() } else { assertNever(options.type) diff --git a/packages/timeline-state-resolver/src/integrations/pharos/index.ts b/packages/timeline-state-resolver/src/integrations/pharos/index.ts index c1e025075..0f4ab00fd 100644 --- a/packages/timeline-state-resolver/src/integrations/pharos/index.ts +++ b/packages/timeline-state-resolver/src/integrations/pharos/index.ts @@ -58,6 +58,7 @@ export class PharosDevice extends Device { this.context.logger.info(`Current project: ${info.name}`) + this.context.resetToState({}) }) .catch((e) => this.context.logger.error('Failed to query project', e)) }) diff --git a/packages/timeline-state-resolver/src/integrations/quantel/index.ts b/packages/timeline-state-resolver/src/integrations/quantel/index.ts index 3b3b95dcd..6bdc0e090 100644 --- a/packages/timeline-state-resolver/src/integrations/quantel/index.ts +++ b/packages/timeline-state-resolver/src/integrations/quantel/index.ts @@ -74,16 +74,31 @@ export class QuantelDevice extends Device { this._quantel.monitorServerStatus((connected: boolean) => { - if (!this._disconnectedSince && connected === false && options.suppressDisconnectTime) { + if (!this._disconnectedSince && connected === false) { this._disconnectedSince = Date.now() - // trigger another update after debounce - setTimeout(() => { - if (!this._quantel.connected) { - this.context.connectionChanged(this.getStatus()) - } - }, options.suppressDisconnectTime) + if (options.suppressDisconnectTime) { + // trigger another update after debounce + setTimeout(() => { + if (!this._quantel.connected) { + this.context.connectionChanged(this.getStatus()) + } + }, options.suppressDisconnectTime) + } } else if (connected === true) { + if (!this._disconnectedSince) { + // this must be our first time connecting, so let's resend any commands we missed + this.context + .resetToState({ time: 0, port: {} }) + .catch((e) => + this.context.logger.warning( + 'Failed to reset to state after first connection, device may be in unknown state (reason: ' + + e + + ')' + ) + ) + } + this._disconnectedSince = undefined } diff --git a/packages/timeline-state-resolver/src/integrations/shotoku/index.ts b/packages/timeline-state-resolver/src/integrations/shotoku/index.ts index 73842947a..a0cac93d7 100644 --- a/packages/timeline-state-resolver/src/integrations/shotoku/index.ts +++ b/packages/timeline-state-resolver/src/integrations/shotoku/index.ts @@ -48,6 +48,15 @@ export class ShotokuDevice extends Device { + this.context + .resetToState({ shots: {}, sequences: {} }) + .catch((e) => + this.context.logger.warning( + 'Failed to reset to state after first connection, device may be in unknown state (reason: ' + e + ')' + ) + ) + }) .catch((e) => this.context.logger.debug('Shotoku device failed initial connection attempt', e)) return true diff --git a/packages/timeline-state-resolver/src/integrations/sisyfos/index.ts b/packages/timeline-state-resolver/src/integrations/sisyfos/index.ts index dec0df22f..8d427f6d1 100644 --- a/packages/timeline-state-resolver/src/integrations/sisyfos/index.ts +++ b/packages/timeline-state-resolver/src/integrations/sisyfos/index.ts @@ -90,6 +90,11 @@ export class SisyfosMessageDevice extends DeviceWithState { + // process any states that we missed + this.clearStates() + this.emit('resyncStates') + }) .catch((e) => this.emit('error', 'Failed to initialise Sisyfos connection', e)) return true diff --git a/packages/timeline-state-resolver/src/integrations/sofieChef/index.ts b/packages/timeline-state-resolver/src/integrations/sofieChef/index.ts index c36752d38..5a67b91c4 100644 --- a/packages/timeline-state-resolver/src/integrations/sofieChef/index.ts +++ b/packages/timeline-state-resolver/src/integrations/sofieChef/index.ts @@ -90,7 +90,12 @@ export class SofieChefDevice extends Device this.context.logger.error('Failed to initialise Sofie Chef connection', e)) + this._setupWSConnection() + .then(() => { + // assume empty state on start (would be nice if we could get the url for each window on connection) + this.context.resetToState({ windows: {} }) + }) + .catch((e) => this.context.logger.error('Failed to initialise Sofie Chef connection', e)) return true } diff --git a/packages/timeline-state-resolver/src/integrations/tcpSend/index.ts b/packages/timeline-state-resolver/src/integrations/tcpSend/index.ts index 1a5e2dad1..b4fb99f03 100644 --- a/packages/timeline-state-resolver/src/integrations/tcpSend/index.ts +++ b/packages/timeline-state-resolver/src/integrations/tcpSend/index.ts @@ -30,6 +30,17 @@ export class TcpSendDevice extends Device { + this.tcpConnection.once('connectionChanged', (connected) => { + if (connected) { + this.context + .resetState() + .catch((e) => + this.context.logger.warning( + 'Failed to reset state after first connection, device may be in unknown state (reason: ' + e + ')' + ) + ) + } + }) this.tcpConnection.activate(options) return true } diff --git a/packages/timeline-state-resolver/src/integrations/vizMSE/index.ts b/packages/timeline-state-resolver/src/integrations/vizMSE/index.ts index aaf496ed4..80957c513 100644 --- a/packages/timeline-state-resolver/src/integrations/vizMSE/index.ts +++ b/packages/timeline-state-resolver/src/integrations/vizMSE/index.ts @@ -144,6 +144,11 @@ export class VizMSEDevice extends DeviceWithState { + // reset any states we had to re-enforce them + this.clearStates() + this.emit('resyncStates') + }) .catch((e) => this.emit('error', 'Failed to initialise Viz Rundown', e)) return true diff --git a/packages/timeline-state-resolver/src/integrations/vmix/index.ts b/packages/timeline-state-resolver/src/integrations/vmix/index.ts index 7897cb60e..2b49fd98e 100644 --- a/packages/timeline-state-resolver/src/integrations/vmix/index.ts +++ b/packages/timeline-state-resolver/src/integrations/vmix/index.ts @@ -161,6 +161,10 @@ export class VMixDevice extends DeviceWithState Date: Fri, 6 Sep 2024 10:04:46 +0200 Subject: [PATCH 08/17] chore: wip --- .../src/service/ConnectionManager.ts | 79 +++++++++++++------ 1 file changed, 55 insertions(+), 24 deletions(-) diff --git a/packages/timeline-state-resolver/src/service/ConnectionManager.ts b/packages/timeline-state-resolver/src/service/ConnectionManager.ts index b016068c0..22546dd5a 100644 --- a/packages/timeline-state-resolver/src/service/ConnectionManager.ts +++ b/packages/timeline-state-resolver/src/service/ConnectionManager.ts @@ -23,6 +23,7 @@ import { DeviceOptionsVMixInternal, VMixDevice } from '../integrations/vmix' import { ImplementedServiceDeviceTypes } from './devices' import { EventEmitter } from 'eventemitter3' import { DeviceInstanceEvents } from './DeviceInstance' +import { deferAsync } from '../lib' interface Operation { operation: 'create' | 'update' | 'delete' | 'setDebug' @@ -158,16 +159,19 @@ export class ConnectionManager extends EventEmitter { return } - Promise.allSettled(allowedOperations.map((op) => this.executeOperation(op))).then(() => { - this._updating = false + Promise.allSettled(allowedOperations.map(async (op) => this.executeOperation(op))) + .then(() => { + this._updating = false - // rerun the algorithm once to make sure we have no missed operations in the meanwhile - this._updateConnections() - }) + // rerun the algorithm once to make sure we have no missed operations in the meanwhile + this._updateConnections() + }) + .catch((e) => { + this.emit('warning', 'Error encountered while updating connections: ' + e) + }) } private async executeOperation({ operation, id }: Operation): Promise { - // todo - timeout? try { switch (operation) { case 'create': @@ -208,7 +212,7 @@ export class ConnectionManager extends EventEmitter { freezeLimit: FREEZE_LIMIT, } - const container = await createContainer(deviceOptions, id, () => Date.now(), threadedClassOptions) // time out if this gets el stucko + const container = await createContainer(deviceOptions, id, () => Date.now(), threadedClassOptions) // we rely on threadedclass to timeout if this fails if (!container) { this.emit('warning', 'Failed to create container for ' + id) @@ -216,12 +220,12 @@ export class ConnectionManager extends EventEmitter { } // set up event handlers - this._setupDeviceListeners(id, container) + await this._setupDeviceListeners(id, container) this._connections.set(id, container) this.emit('connectionAdded', id, container) - // trigger conenction init + // trigger connection init this._handleConnectionInitialisation(id, container) .then(() => { this._connectionAttempts.delete(id) @@ -230,26 +234,51 @@ export class ConnectionManager extends EventEmitter { this.emit('error', 'Connection ' + id + ' failed to initialise') this._connections.delete(id) - container.terminate().catch(() => this.emit('warning', `Failed to initialise ${id} (${e})`)) - // todo - find a good point to retrigger _updateConnections - this._updateConnections() // this can't be the right place... right?? it was not :kekw:... was it not?? + container + .terminate() + .catch(() => this.emit('warning', `Failed to initialise ${id} (${e})`)) + .finally(() => { + this._updateConnections() + }) }) } private async deleteConnection(id: string): Promise { const connection = this._connections.get(id) - if (!connection) return // already removed / never existed + if (!connection) return Promise.resolve() // already removed / never existed this._connections.delete(id) this.emit('connectionRemoved', id) - try { - await connection.device.terminate() - await connection.device.removeAllListeners() - await connection.terminate() - } catch { - await connection.terminate() - } + return new Promise((resolve) => + deferAsync( + async () => { + let finished = false + setTimeout(() => { + if (!finished) { + resolve() + this.emit('warning', 'Failed to delete connection in time') + + connection.terminate().catch((e) => this.emit('error', 'Failed to terminate connection: ' + e)) + } + }, 30000) + + try { + await connection.device.terminate() + await connection.device.removeAllListeners() + await connection.terminate() + } catch { + await connection.terminate() + } + + finished = true + resolve() + }, + (e) => { + this.emit('warning', 'Error encountered trying to delete connection: ' + e) + } + ) + ) } private async setDebugForConnection(id: string): Promise { @@ -258,8 +287,8 @@ export class ConnectionManager extends EventEmitter { if (!connection || !config) return try { - connection.device.setDebugLogging(config.debug ?? false) - connection.device.setDebugState(config.debugState ?? false) + await connection.device.setDebugLogging(config.debug ?? false) + await connection.device.setDebugState(config.debugState ?? false) } catch { this.emit('warning', 'Failed to update debug values for ' + id) } @@ -270,7 +299,7 @@ export class ConnectionManager extends EventEmitter { container: BaseRemoteDeviceIntegration ) { const deviceOptions = this._config.get(id) - if (!deviceOptions) return // unexpected - throw error? + if (!deviceOptions) return // if the config has been removed, the connection should be removed as well so no need to init this.emit( 'info', @@ -288,7 +317,9 @@ export class ConnectionManager extends EventEmitter { const passEvent = (ev: T) => { const evHandler: any = (...args: DeviceInstanceEvents[T]) => this.emit(('connectionEvent:' + ev) as `connectionEvent:${keyof DeviceInstanceEvents}`, id, ...args) - container.device.on(ev, evHandler) + container.device + .on(ev, evHandler) + .catch((e) => this.emit('error', 'Failed to attach listener for device: ' + id + ' ' + ev, e)) } passEvent('info') From f6b8bd5c0e62374ece598643d1468997476c0a41 Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Fri, 6 Sep 2024 10:06:37 +0200 Subject: [PATCH 09/17] chore: update tests --- .../src/__tests__/conductor.spec.ts | 95 +++--- .../src/__tests__/lib.ts | 42 +++ .../timeline-state-resolver/src/conductor.ts | 6 +- .../casparCG/__tests__/casparcg.spec.ts | 302 ++++++++++-------- .../sisyfos/__tests__/sisyfos.spec.ts | 98 +++--- .../vizMSE/__tests__/vizMSE.spec.ts | 132 ++++---- .../src/service/ConnectionManager.ts | 71 +--- 7 files changed, 393 insertions(+), 353 deletions(-) diff --git a/packages/timeline-state-resolver/src/__tests__/conductor.spec.ts b/packages/timeline-state-resolver/src/__tests__/conductor.spec.ts index cc23db37c..86054999e 100644 --- a/packages/timeline-state-resolver/src/__tests__/conductor.spec.ts +++ b/packages/timeline-state-resolver/src/__tests__/conductor.spec.ts @@ -12,7 +12,7 @@ import { } from 'timeline-state-resolver-types' import { MockTime } from './mockTime' import { ThreadedClass } from 'threadedclass' -import { getMockCall } from './lib' +import { addConnections, getMockCall, removeConnections } from './lib' import { setupAllMocks } from '../__mocks__/_setup-all-mocks' import { Commands } from 'casparcg-connection' import { MockDeviceInstanceWrapper, ConstructedMockDevices, DiscardAllMockDevices } from './mockDeviceInstanceWrapper' @@ -27,6 +27,8 @@ jest.mock('../service/DeviceInstance', () => ({ import { Conductor, TimelineTriggerTimeResult } from '../conductor' import type { DeviceInstanceWrapper } from '../service/DeviceInstance' +import { DeviceOptionsAnyInternal } from '..' +import { ConnectionManager } from '../service/ConnectionManager' describe('Conductor', () => { const mockTime = new MockTime() @@ -38,11 +40,11 @@ describe('Conductor', () => { DiscardAllMockDevices() }) - async function getMockDeviceWrapper(conductor: Conductor, deviceId: string): Promise { - const deviceContainer = conductor.getDevice(deviceId) + async function getMockDeviceWrapper(conductor: Conductor, connectionId: string): Promise { + const deviceContainer = conductor.connectionManager.getConnection(connectionId) expect(deviceContainer).toBeTruthy() - const mockDevice = ConstructedMockDevices[deviceId] + const mockDevice = ConstructedMockDevices[connectionId] expect(mockDevice).toBeTruthy() return mockDevice } @@ -70,13 +72,15 @@ describe('Conductor', () => { try { await conductor.init() - await conductor.addDevice('device0', { - type: DeviceType.ABSTRACT, - options: {}, - }) - await conductor.addDevice('device1', { - type: DeviceType.ABSTRACT, - options: {}, + await addConnections(conductor.connectionManager, { + device0: { + type: DeviceType.ABSTRACT, + options: {}, + }, + device1: { + type: DeviceType.ABSTRACT, + options: {}, + }, }) // add something that will play in a seconds time @@ -191,13 +195,18 @@ describe('Conductor', () => { ) // Remove the device - await conductor.removeDevice('device1') - expect(conductor.getDevice('device1')).toBeFalsy() + await removeConnections( + conductor.connectionManager, + { + device0: { + type: DeviceType.ABSTRACT, + options: {}, + }, + }, + ['device1'] + ) + expect(conductor.connectionManager.getConnection('device1')).toBeFalsy() expect(ConstructedMockDevices['device1']).toBeFalsy() - - // Re-add a device - const addedDevice = await conductor.addDevice('device1', { type: DeviceType.ABSTRACT, options: {} }) - expect(addedDevice).toBeTruthy() } finally { await conductor.destroy() } @@ -220,9 +229,11 @@ describe('Conductor', () => { try { await conductor.init() - await conductor.addDevice('device0', { - type: DeviceType.ABSTRACT, - options: {}, + await addConnections(conductor.connectionManager, { + device0: { + type: DeviceType.ABSTRACT, + options: {}, + }, }) // add something that will play "now" @@ -387,13 +398,15 @@ describe('Conductor', () => { try { await conductor.init() - await conductor.addDevice('device0', { - type: DeviceType.ABSTRACT, - options: {}, - }) - await conductor.addDevice('device1', { - type: DeviceType.HTTPSEND, - options: {}, + await addConnections(conductor.connectionManager, { + device0: { + type: DeviceType.ABSTRACT, + options: {}, + }, + device1: { + type: DeviceType.ABSTRACT, + options: {}, + }, }) const device0 = await getMockDeviceWrapper(conductor, 'device0') @@ -437,15 +450,17 @@ describe('Conductor', () => { try { await conductor.init() - await conductor.addDevice('device0', { - type: DeviceType.ABSTRACT, - options: {}, - isMultiThreaded: true, + await addConnections(conductor.connectionManager, { + device0: { + type: DeviceType.ABSTRACT, + options: {}, + isMultiThreaded: true, + }, }) conductor.setTimelineAndMappings([], myLayerMapping) - const device = conductor.getDevice('device0')!.device - expect(await device.getCurrentTime()).toBeTruthy() + const connection = conductor.connectionManager.getConnection('device0')!.device + expect(await connection.getCurrentTime()).toBeTruthy() } finally { await conductor.destroy() } @@ -475,12 +490,14 @@ describe('Conductor', () => { }) await conductor.init() - await conductor.addDevice('device0', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(conductor.connectionManager, { + device0: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }) conductor.setTimelineAndMappings([], myLayerMapping) @@ -505,7 +522,7 @@ describe('Conductor', () => { const timeline: TSRTimeline = [video0] - const device0Container = conductor.getDevice('device0') + const device0Container = conductor.connectionManager.getConnection('device0') const device0 = device0Container!.device as ThreadedClass expect(device0).toBeTruthy() diff --git a/packages/timeline-state-resolver/src/__tests__/lib.ts b/packages/timeline-state-resolver/src/__tests__/lib.ts index 5aec3bc7d..c9a7e5076 100644 --- a/packages/timeline-state-resolver/src/__tests__/lib.ts +++ b/packages/timeline-state-resolver/src/__tests__/lib.ts @@ -1,9 +1,51 @@ +import { DeviceOptionsAnyInternal } from '../conductor' +import { ConnectionManager } from '../service/ConnectionManager' + /** * Just a wrapper to :any type, to be used in tests only */ export function getMockCall(fcn: jest.Mock, callIndex: number, paramIndex: number): any { return fcn.mock.calls[callIndex][paramIndex] } +export async function addConnections( + connManager: ConnectionManager, + connections: Record +): Promise { + const connectionIds = Object.keys(connections) + const addedConns: string[] = [] + + let resolveAdded: undefined | (() => void) = undefined + const psAdded = new Promise((resolveCb) => (resolveAdded = resolveCb)) + connManager.on('connectionAdded', (id) => { + addedConns.push(id) + + if (resolveAdded && addedConns.length === connectionIds.length) resolveAdded() + }) + + connManager.setConnections(new Map(Object.entries(connections))) + + await psAdded +} + +export async function removeConnections( + connManager: ConnectionManager, + connections: Record, + toBeRemoved: string[] +): Promise { + const addedConns: string[] = [] + + let resolveAdded: undefined | (() => void) = undefined + const psAdded = new Promise((resolveCb) => (resolveAdded = resolveCb)) + connManager.on('connectionRemoved', (id) => { + addedConns.push(id) + + if (resolveAdded && addedConns.length === toBeRemoved.length) resolveAdded() + }) + + connManager.setConnections(new Map(Object.entries(connections))) + + await psAdded +} // Excend jest.expect in functionality and typings expect.extend({ diff --git a/packages/timeline-state-resolver/src/conductor.ts b/packages/timeline-state-resolver/src/conductor.ts index a327319b5..60b33572a 100644 --- a/packages/timeline-state-resolver/src/conductor.ts +++ b/packages/timeline-state-resolver/src/conductor.ts @@ -332,15 +332,15 @@ export class Conductor extends EventEmitter { } /** - * Remove all devices + * Remove all connections */ public async destroy(): Promise { clearTimeout(this._interval) if (this._triggerSendStartStopCallbacksTimeout) clearTimeout(this._triggerSendStartStopCallbacksTimeout) - // todo - reenable this - // await this._mapAllDevices(true, async (d) => this.removeDevice(d.deviceId)) + // remove all connections: + this.connectionManager.setConnections(new Map()) } /** diff --git a/packages/timeline-state-resolver/src/integrations/casparCG/__tests__/casparcg.spec.ts b/packages/timeline-state-resolver/src/integrations/casparCG/__tests__/casparcg.spec.ts index edb05af02..c36812df5 100644 --- a/packages/timeline-state-resolver/src/integrations/casparCG/__tests__/casparcg.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/casparCG/__tests__/casparcg.spec.ts @@ -13,7 +13,7 @@ import { MappingCasparCGType, } from 'timeline-state-resolver-types' import { MockTime } from '../../../__tests__/mockTime' -import { getMockCall } from '../../../__tests__/lib' +import { addConnections, getMockCall } from '../../../__tests__/lib' import { Commands } from 'casparcg-connection' // usage logCalls(commandReceiver0) @@ -51,13 +51,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() // we cannot do an await, because setTimeout will never call without jest moving on. - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) await mockTime.advanceTimeToTicks(10100) @@ -129,13 +131,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() // we cannot do an await, because setTimeout will never call without jest moving on. - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) await mockTime.advanceTimeToTicks(10100) @@ -196,14 +200,16 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() // we cannot do an await, because setTimeout will never call without jest moving on. - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', - fps: 50, + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + fps: 50, + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) await mockTime.advanceTimeToTicks(10100) @@ -278,13 +284,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) myConductor.setTimelineAndMappings( @@ -351,13 +359,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) // await mockTime.advanceTimeToTicks(10050) @@ -440,13 +450,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) await mockTime.advanceTimeToTicks(10050) @@ -519,13 +531,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) await mockTime.advanceTimeToTicks(10050) @@ -599,18 +613,20 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) - const deviceContainer = myConductor.getDevice('myCCG') - const device = deviceContainer!.device - await device['_ccgState'] + const connContainer = myConductor.connectionManager.getConnection('myCCG') + const conn = connContainer!.device + await conn['_ccgState'] await mockTime.advanceTimeToTicks(10050) expect(commandReceiver0).toHaveBeenCalledTimes(0) @@ -715,13 +731,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) // Check that no commands has been sent: @@ -834,13 +852,15 @@ describe('CasparCG', () => { console.warn(msg) }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) // Check that no commands has been sent: @@ -972,13 +992,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) expect(mockTime.getCurrentTime()).toEqual(10000) @@ -1077,13 +1099,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) expect(mockTime.getCurrentTime()).toEqual(10000) @@ -1181,13 +1205,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) await mockTime.advanceTimeToTicks(10050) @@ -1281,13 +1307,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) await mockTime.advanceTimeToTicks(10050) @@ -1370,13 +1398,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() // we cannot do an await, because setTimeout will never call without jest moving on. - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) await mockTime.advanceTimeToTicks(10100) @@ -1454,14 +1484,16 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', - retryInterval: undefined, // disable retries explicitly, we will manually trigger them + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + retryInterval: undefined, // disable retries explicitly, we will manually trigger them + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) myConductor.setTimelineAndMappings([], myLayerMapping) await mockTime.advanceTimeToTicks(10100) @@ -1470,8 +1502,8 @@ describe('CasparCG', () => { commandReceiver0.mockClear() - const deviceContainer = myConductor.getDevice('myCCG') - const device = deviceContainer!.device + const connContainer = myConductor.connectionManager.getConnection('myCCG') + const conn = connContainer!.device myConductor.setTimelineAndMappings([ { @@ -1513,7 +1545,7 @@ describe('CasparCG', () => { // advance to half way await mockTime.advanceTimeToTicks(10700) // call the retry mechanism - await (device as any)._assertIntendedState() + await (conn as any)._assertIntendedState() await mockTime.advanceTimeToTicks(10800) expect(commandReceiver0).toHaveBeenCalledTimes(2) @@ -1530,13 +1562,13 @@ describe('CasparCG', () => { // apply command to internal ccg-state const resCommand = getMockCall(commandReceiver0, 1, 1) // @ts-ignore - await device._changeTrackedStateFromCommand( + await conn._changeTrackedStateFromCommand( resCommand, { responseCode: 202, command: resCommand.command }, mockTime.getCurrentTime() ) // trigger retry mechanism - await (device as any)._assertIntendedState() + await (conn as any)._assertIntendedState() await mockTime.advanceTimeToTicks(10900) // no retries done expect(commandReceiver0).toHaveBeenCalledTimes(2) @@ -1553,7 +1585,7 @@ describe('CasparCG', () => { // advance time to after clip: await mockTime.advanceTimeToTicks(11700) // call the retry mechanism - await (device as any)._assertIntendedState() + await (conn as any)._assertIntendedState() await mockTime.advanceTimeToTicks(11800) // no retries issued expect(commandReceiver0).toHaveBeenCalledTimes(3) @@ -1581,14 +1613,16 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', - retryInterval: undefined, // disable retries explicitly, we will manually trigger them + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + retryInterval: undefined, // disable retries explicitly, we will manually trigger them + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) myConductor.setTimelineAndMappings([], myLayerMapping) await mockTime.advanceTimeToTicks(10100) @@ -1597,8 +1631,8 @@ describe('CasparCG', () => { commandReceiver0.mockClear() - const deviceContainer = myConductor.getDevice('myCCG') - const device = deviceContainer!.device + const connContainer = myConductor.connectionManager.getConnection('myCCG') + const conn = connContainer!.device myConductor.setTimelineAndMappings([ { @@ -1658,7 +1692,7 @@ describe('CasparCG', () => { // @ts-ignore const resCommand = getMockCall(commandReceiver0, 0, 1) // @ts-ignore - await device._changeTrackedStateFromCommand( + await conn._changeTrackedStateFromCommand( resCommand, { responseCode: 202, command: resCommand.command }, mockTime.getCurrentTime() @@ -1672,7 +1706,7 @@ describe('CasparCG', () => { // advance to half way await mockTime.advanceTimeToTicks(10700) // call the retry mechanism - await (device as any)._assertIntendedState() + await (conn as any)._assertIntendedState() // still no retries as empty always plays expect(commandReceiver0).toHaveBeenCalledTimes(1) @@ -1681,7 +1715,7 @@ describe('CasparCG', () => { // advance time to after clip: await mockTime.advanceTimeToTicks(20700) // call the retry mechanism - await (device as any)._assertIntendedState() + await (conn as any)._assertIntendedState() await mockTime.advanceTimeToTicks(20800) // no retries issued expect(commandReceiver0).toHaveBeenCalledTimes(1) @@ -1710,13 +1744,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() // we cannot do an await, because setTimeout will never call without jest moving on. - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) await mockTime.advanceTimeToTicks(10100) @@ -1806,13 +1842,15 @@ describe('CasparCG', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() // we cannot do an await, because setTimeout will never call without jest moving on. - await myConductor.addDevice('myCCG', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', + await addConnections(myConductor.connectionManager, { + myCCG: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + }, + commandReceiver: commandReceiver0, + skipVirginCheck: true, }, - commandReceiver: commandReceiver0, - skipVirginCheck: true, }) await mockTime.advanceTimeToTicks(10100) diff --git a/packages/timeline-state-resolver/src/integrations/sisyfos/__tests__/sisyfos.spec.ts b/packages/timeline-state-resolver/src/integrations/sisyfos/__tests__/sisyfos.spec.ts index f4014276f..8f14a8442 100644 --- a/packages/timeline-state-resolver/src/integrations/sisyfos/__tests__/sisyfos.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/sisyfos/__tests__/sisyfos.spec.ts @@ -13,7 +13,7 @@ const MockOSC = OSC.MockOSC import { MockTime } from '../../../__tests__/mockTime' import { ThreadedClass } from 'threadedclass' import { SisyfosMessageDevice } from '../../../integrations/sisyfos' -import { getMockCall } from '../../../__tests__/lib' +import { addConnections, getMockCall } from '../../../__tests__/lib' describe('Sisyfos', () => { jest.mock('osc', () => OSC) @@ -82,18 +82,20 @@ describe('Sisyfos', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() // we cannot do an await, because setTimeout will never call without jest moving on. - await myConductor.addDevice('mySisyfos', { - type: DeviceType.SISYFOS, - options: { - host: '192.168.0.10', - port: 8900, + await addConnections(myConductor.connectionManager, { + mySisyfos: { + type: DeviceType.SISYFOS, + options: { + host: '192.168.0.10', + port: 8900, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }) myConductor.setTimelineAndMappings([], myChannelMapping) await mockTime.advanceTimeToTicks(10100) - const deviceContainer = myConductor.getDevice('mySisyfos') + const deviceContainer = myConductor.connectionManager.getConnection('mySisyfos') const device = deviceContainer!.device as ThreadedClass // Check that no commands has been scheduled: @@ -309,18 +311,20 @@ describe('Sisyfos', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() // we cannot do an await, because setTimeout will never call without jest moving on. - await myConductor.addDevice('mySisyfos', { - type: DeviceType.SISYFOS, - options: { - host: '192.168.0.10', - port: 8900, + await addConnections(myConductor.connectionManager, { + mySisyfos: { + type: DeviceType.SISYFOS, + options: { + host: '192.168.0.10', + port: 8900, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }) myConductor.setTimelineAndMappings([], myChannelMapping) await mockTime.advanceTimeToTicks(10100) - const deviceContainer = myConductor.getDevice('mySisyfos') + const deviceContainer = myConductor.connectionManager.getConnection('mySisyfos') const device = deviceContainer!.device as ThreadedClass // Check that no commands has been scheduled: @@ -535,18 +539,20 @@ describe('Sisyfos', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() // we cannot do an await, because setTimeout will never call without jest moving on. - await myConductor.addDevice('mySisyfos', { - type: DeviceType.SISYFOS, - options: { - host: '192.168.0.10', - port: 8900, + await addConnections(myConductor.connectionManager, { + mySisyfos: { + type: DeviceType.SISYFOS, + options: { + host: '192.168.0.10', + port: 8900, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }) myConductor.setTimelineAndMappings([], myChannelMapping) await mockTime.advanceTimeToTicks(10100) - const deviceContainer = myConductor.getDevice('mySisyfos') + const deviceContainer = myConductor.connectionManager.getConnection('mySisyfos') const device = deviceContainer!.device as ThreadedClass // Check that no commands has been scheduled: @@ -673,18 +679,20 @@ describe('Sisyfos', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() // we cannot do an await, because setTimeout will never call without jest moving on. - await myConductor.addDevice('mySisyfos', { - type: DeviceType.SISYFOS, - options: { - host: '192.168.0.10', - port: 8900, + await addConnections(myConductor.connectionManager, { + mySisyfos: { + type: DeviceType.SISYFOS, + options: { + host: '192.168.0.10', + port: 8900, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }) myConductor.setTimelineAndMappings([], myChannelMapping) await mockTime.advanceTimeToTicks(10100) - const deviceContainer = myConductor.getDevice('mySisyfos') + const deviceContainer = myConductor.connectionManager.getConnection('mySisyfos') const device = deviceContainer!.device as ThreadedClass // Check that no commands has been scheduled: @@ -891,18 +899,20 @@ describe('Sisyfos', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() // we cannot do an await, because setTimeout will never call without jest moving on. - await myConductor.addDevice('mySisyfos', { - type: DeviceType.SISYFOS, - options: { - host: '192.168.0.10', - port: 8900, + await addConnections(myConductor.connectionManager, { + mySisyfos: { + type: DeviceType.SISYFOS, + options: { + host: '192.168.0.10', + port: 8900, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }) myConductor.setTimelineAndMappings([], myChannelMapping) await mockTime.advanceTimeToTicks(10100) - const deviceContainer = myConductor.getDevice('mySisyfos') + const deviceContainer = myConductor.connectionManager.getConnection('mySisyfos') const device = deviceContainer!.device as ThreadedClass // Check that no commands has been scheduled: @@ -1041,17 +1051,19 @@ describe('Sisyfos', () => { }) // myConductor.setTimelineAndMappings([], myChannelMapping) await myConductor.init() - await myConductor.addDevice('mySisyfos', { - type: DeviceType.SISYFOS, - options: { - host: '127.0.0.1', - port: 1234, + await addConnections(myConductor.connectionManager, { + mySisyfos: { + type: DeviceType.SISYFOS, + options: { + host: '192.168.0.10', + port: 8900, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }) await mockTime.advanceTimeToTicks(10100) - const deviceContainer = myConductor.getDevice('mySisyfos') + const deviceContainer = myConductor.connectionManager.getConnection('mySisyfos') const device = deviceContainer!.device as ThreadedClass const onConnectionChanged = jest.fn() diff --git a/packages/timeline-state-resolver/src/integrations/vizMSE/__tests__/vizMSE.spec.ts b/packages/timeline-state-resolver/src/integrations/vizMSE/__tests__/vizMSE.spec.ts index 19c095974..bdeaae6b6 100644 --- a/packages/timeline-state-resolver/src/integrations/vizMSE/__tests__/vizMSE.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/vizMSE/__tests__/vizMSE.spec.ts @@ -13,7 +13,7 @@ import { } from 'timeline-state-resolver-types' import { MockTime } from '../../../__tests__/mockTime' import { ThreadedClass } from 'threadedclass' -import { getMockCall } from '../../../__tests__/lib' +import { addConnections, getMockCall } from '../../../__tests__/lib' import { VizMSEDevice } from '..' import * as vConnection from '../../../__mocks__/v-connection' import * as net from '../../../__mocks__/net' @@ -64,18 +64,20 @@ async function setupDevice() { myConductor.setTimelineAndMappings([], myChannelMapping) await myConductor.init() - await myConductor.addDevice('myViz', { - type: DeviceType.VIZMSE, - options: { - host: '127.0.0.1', - preloadAllElements: true, - playlistID: 'my-super-playlist-id', - profile: 'profile9999', - showDirectoryPath: 'SOFIE', + await addConnections(myConductor.connectionManager, { + myViz: { + type: DeviceType.VIZMSE, + options: { + host: '127.0.0.1', + preloadAllElements: true, + playlistID: 'my-super-playlist-id', + profile: 'profile9999', + showDirectoryPath: 'SOFIE', + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }) - const deviceContainer = myConductor.getDevice('myViz') + const deviceContainer = myConductor.connectionManager.getConnection('myViz') device = deviceContainer!.device as ThreadedClass return { device, myConductor, onError, commandReceiver0 } @@ -482,7 +484,7 @@ describe('vizMSE', () => { expect(onError).toHaveBeenCalledTimes(0) }) - test('bad init options & basic functionality', async () => { + test('basic functionality', async () => { const myConductor = new Conductor({ multiThreadedResolver: false, getCurrentTime: mockTime.getCurrentTime, @@ -494,37 +496,19 @@ describe('vizMSE', () => { await myConductor.init() - await expect( - myConductor.addDevice('myViz', { - type: DeviceType.VIZMSE, - options: literal>({ - // host: '127.0.0.1', - profile: 'myProfile', - }) as any, - }) - ).rejects.toMatch(/bad option/) - await expect( - // @ts-ignore - myConductor.addDevice('myViz', { + expect(onError).toHaveBeenCalledTimes(2) + onError.mockClear() + + await addConnections(myConductor.connectionManager, { + myViz: { type: DeviceType.VIZMSE, options: { host: '127.0.0.1', - // profile: 'myProfile' + profile: 'myProfile', }, - }) - ).rejects.toMatch(/bad option/) - - expect(onError).toHaveBeenCalledTimes(2) - onError.mockClear() - - const deviceContainer = await myConductor.addDevice('myViz', { - type: DeviceType.VIZMSE, - options: { - host: '127.0.0.1', - profile: 'myProfile', }, }) - const device = deviceContainer.device + const device = myConductor.connectionManager.getConnection('myViz')!.device const connectionChanged = jest.fn() await device.on('connectionChanged', connectionChanged) @@ -587,22 +571,24 @@ describe('vizMSE', () => { }) myConductor.setTimelineAndMappings([], myChannelMapping) await myConductor.init() - await myConductor.addDevice('myViz', { - type: DeviceType.VIZMSE, - options: { - host: '127.0.0.1', - preloadAllElements: true, - playlistID: 'my-super-playlist-id', - profile: 'profile9999', - clearAllTemplateName: 'clear_all_of_them', - clearAllCommands: ['RENDERER*FRONT_LAYER SET_OBJECT ', 'RENDERER SET_OBJECT '], - showDirectoryPath: 'SOFIE', + await addConnections(myConductor.connectionManager, { + myViz: { + type: DeviceType.VIZMSE, + options: { + host: '127.0.0.1', + preloadAllElements: true, + playlistID: 'my-super-playlist-id', + profile: 'profile9999', + clearAllTemplateName: 'clear_all_of_them', + clearAllCommands: ['RENDERER*FRONT_LAYER SET_OBJECT ', 'RENDERER SET_OBJECT '], + showDirectoryPath: 'SOFIE', + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }) await mockTime.advanceTimeToTicks(10100) - const deviceContainer = myConductor.getDevice('myViz') + const deviceContainer = myConductor.connectionManager.getConnection('myViz') const device = deviceContainer!.device as ThreadedClass await device.ignoreWaitsInTests() @@ -1040,20 +1026,22 @@ describe('vizMSE', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myViz', { - type: DeviceType.VIZMSE, - options: { - host: '127.0.0.1', - preloadAllElements: true, - playlistID: 'my-super-playlist-id', - profile: PROFILE_NAME, - clearAllOnMakeReady: true, - clearAllTemplateName: 'clear_all_of_them', - clearAllCommands: [CLEAR_COMMAND], + await addConnections(myConductor.connectionManager, { + myViz: { + type: DeviceType.VIZMSE, + options: { + host: '127.0.0.1', + preloadAllElements: true, + playlistID: 'my-super-playlist-id', + profile: PROFILE_NAME, + clearAllOnMakeReady: true, + clearAllTemplateName: 'clear_all_of_them', + clearAllCommands: [CLEAR_COMMAND], + }, }, }) - const deviceContainer = myConductor.getDevice('myViz') + const deviceContainer = myConductor.connectionManager.getConnection('myViz') const device = deviceContainer!.device as ThreadedClass await device.ignoreWaitsInTests() @@ -1111,20 +1099,22 @@ describe('vizMSE', () => { getCurrentTime: mockTime.getCurrentTime, }) await myConductor.init() - await myConductor.addDevice('myViz', { - type: DeviceType.VIZMSE, - options: { - host: '127.0.0.1', - preloadAllElements: true, - playlistID: 'my-super-playlist-id', - profile: PROFILE_NAME, - clearAllOnMakeReady: false, - clearAllTemplateName: 'clear_all_of_them', - clearAllCommands: [CLEAR_COMMAND], + await addConnections(myConductor.connectionManager, { + myViz: { + type: DeviceType.VIZMSE, + options: { + host: '127.0.0.1', + preloadAllElements: true, + playlistID: 'my-super-playlist-id', + profile: PROFILE_NAME, + clearAllOnMakeReady: false, + clearAllTemplateName: 'clear_all_of_them', + clearAllCommands: [CLEAR_COMMAND], + }, }, }) - const deviceContainer = myConductor.getDevice('myViz') + const deviceContainer = myConductor.connectionManager.getConnection('myViz') const device = deviceContainer!.device as ThreadedClass await device.ignoreWaitsInTests() diff --git a/packages/timeline-state-resolver/src/service/ConnectionManager.ts b/packages/timeline-state-resolver/src/service/ConnectionManager.ts index 22546dd5a..19309c2ec 100644 --- a/packages/timeline-state-resolver/src/service/ConnectionManager.ts +++ b/packages/timeline-state-resolver/src/service/ConnectionManager.ts @@ -1,9 +1,4 @@ -import { - DeviceOptionsBase, - DeviceOptionsMultiOSC, - DeviceOptionsTelemetrics, - DeviceType, -} from 'timeline-state-resolver-types' +import { DeviceOptionsBase, DeviceType } from 'timeline-state-resolver-types' import { BaseRemoteDeviceIntegration, RemoteDeviceInstance } from './remoteDeviceInstance' import _ = require('underscore') import { ThreadedClassConfig } from 'threadedclass' @@ -11,13 +6,7 @@ import { DeviceOptionsAnyInternal } from '../conductor' import { DeviceContainer } from '..//devices/deviceContainer' import { assertNever } from 'atem-connection/dist/lib/atemUtil' import { CasparCGDevice, DeviceOptionsCasparCGInternal } from '../integrations/casparCG' -import { MultiOSCMessageDevice } from '../integrations/multiOsc' -import { DeviceOptionsPharosInternal, PharosDevice } from '../integrations/pharos' -import { DeviceOptionsSingularLiveInternal, SingularLiveDevice } from '../integrations/singularLive' import { DeviceOptionsSisyfosInternal, SisyfosMessageDevice } from '../integrations/sisyfos' -import { DeviceOptionsSofieChefInternal, SofieChefDevice } from '../integrations/sofieChef' -import { TelemetricsDevice } from '../integrations/telemetrics' -import { DeviceOptionsTriCasterInternal, TriCasterDevice } from '../integrations/tricaster' import { DeviceOptionsVizMSEInternal, VizMSEDevice } from '../integrations/vizMSE' import { DeviceOptionsVMixInternal, VMixDevice } from '../integrations/vmix' import { ImplementedServiceDeviceTypes } from './devices' @@ -370,15 +359,6 @@ function createContainer( getCurrentTime, threadedClassOptions ) - case DeviceType.PHAROS: - return DeviceContainer.create( - '../../dist/integrations/pharos/index.js', - 'PharosDevice', - deviceId, - deviceOptions, - getCurrentTime, - threadedClassOptions - ) case DeviceType.SISYFOS: return DeviceContainer.create( '../../dist/integrations/sisyfos/index.js', @@ -397,15 +377,6 @@ function createContainer( getCurrentTime, threadedClassOptions ) - case DeviceType.SINGULAR_LIVE: - return DeviceContainer.create( - '../../dist/integrations/singularLive/index.js', - 'SingularLiveDevice', - deviceId, - deviceOptions, - getCurrentTime, - threadedClassOptions - ) case DeviceType.VMIX: return DeviceContainer.create( '../../dist/integrations/vmix/index.js', @@ -415,53 +386,23 @@ function createContainer( getCurrentTime, threadedClassOptions ) + case DeviceType.SINGULAR_LIVE: case DeviceType.TELEMETRICS: - return DeviceContainer.create( - '../../dist/integrations/telemetrics/index.js', - 'TelemetricsDevice', - deviceId, - deviceOptions, - getCurrentTime, - threadedClassOptions - ) - case DeviceType.SOFIE_CHEF: - return DeviceContainer.create( - '../../dist/integrations/sofieChef/index.js', - 'SofieChefDevice', - deviceId, - deviceOptions, - getCurrentTime, - threadedClassOptions - ) - case DeviceType.TRICASTER: - return DeviceContainer.create( - '../../dist/integrations/tricaster/index.js', - 'TriCasterDevice', - deviceId, - deviceOptions, - getCurrentTime, - threadedClassOptions - ) - case DeviceType.MULTI_OSC: - return DeviceContainer.create( - '../../dist/integrations/multiOsc/index.js', - 'MultiOSCMessageDevice', - deviceId, - deviceOptions, - getCurrentTime, - threadedClassOptions - ) + case DeviceType.PHAROS: case DeviceType.ABSTRACT: case DeviceType.ATEM: case DeviceType.HTTPSEND: case DeviceType.HTTPWATCHER: case DeviceType.HYPERDECK: case DeviceType.LAWO: + case DeviceType.MULTI_OSC: case DeviceType.OBS: case DeviceType.OSC: case DeviceType.PANASONIC_PTZ: case DeviceType.SHOTOKU: + case DeviceType.SOFIE_CHEF: case DeviceType.TCPSEND: + case DeviceType.TRICASTER: case DeviceType.QUANTEL: { ensureIsImplementedAsService(deviceOptions.type) From 81d6afc0293a5ee2c30cece42eec27ce8d21d974 Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Fri, 6 Sep 2024 15:41:24 +0200 Subject: [PATCH 10/17] chore: wip --- .../src/__tests__/lib.ts | 2 +- .../casparCG/__tests__/casparcg.spec.ts | 2 + .../src/integrations/pharos/index.ts | 2 +- .../src/integrations/sofieChef/index.ts | 2 +- .../vizMSE/__tests__/vizMSE.spec.ts | 1 - .../integrations/vmix/__tests__/vmix.spec.ts | 397 ++++++++++-------- .../src/service/ConnectionManager.ts | 2 + .../__tests__/ConnectionManager.spec.ts | 3 +- 8 files changed, 230 insertions(+), 181 deletions(-) diff --git a/packages/timeline-state-resolver/src/__tests__/lib.ts b/packages/timeline-state-resolver/src/__tests__/lib.ts index c9a7e5076..86bd7c84d 100644 --- a/packages/timeline-state-resolver/src/__tests__/lib.ts +++ b/packages/timeline-state-resolver/src/__tests__/lib.ts @@ -16,7 +16,7 @@ export async function addConnections( let resolveAdded: undefined | (() => void) = undefined const psAdded = new Promise((resolveCb) => (resolveAdded = resolveCb)) - connManager.on('connectionAdded', (id) => { + connManager.on('connectionInitialised', (id) => { addedConns.push(id) if (resolveAdded && addedConns.length === connectionIds.length) resolveAdded() diff --git a/packages/timeline-state-resolver/src/integrations/casparCG/__tests__/casparcg.spec.ts b/packages/timeline-state-resolver/src/integrations/casparCG/__tests__/casparcg.spec.ts index c36812df5..a65b5c87a 100644 --- a/packages/timeline-state-resolver/src/integrations/casparCG/__tests__/casparcg.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/casparCG/__tests__/casparcg.spec.ts @@ -863,6 +863,8 @@ describe('CasparCG', () => { }, }) + await mockTime.advanceTimeTicks(100) // let the device settle + // Check that no commands has been sent: expect(commandReceiver0).toHaveBeenCalledTimes(0) diff --git a/packages/timeline-state-resolver/src/integrations/pharos/index.ts b/packages/timeline-state-resolver/src/integrations/pharos/index.ts index 0f4ab00fd..7ff4a0004 100644 --- a/packages/timeline-state-resolver/src/integrations/pharos/index.ts +++ b/packages/timeline-state-resolver/src/integrations/pharos/index.ts @@ -58,7 +58,7 @@ export class PharosDevice extends Device { this.context.logger.info(`Current project: ${info.name}`) - this.context.resetToState({}) + this.context.resetToState({}).catch((e) => this.context.logger.error('Failed to reset state', e)) }) .catch((e) => this.context.logger.error('Failed to query project', e)) }) diff --git a/packages/timeline-state-resolver/src/integrations/sofieChef/index.ts b/packages/timeline-state-resolver/src/integrations/sofieChef/index.ts index 5a67b91c4..5f0d886e0 100644 --- a/packages/timeline-state-resolver/src/integrations/sofieChef/index.ts +++ b/packages/timeline-state-resolver/src/integrations/sofieChef/index.ts @@ -93,7 +93,7 @@ export class SofieChefDevice extends Device { // assume empty state on start (would be nice if we could get the url for each window on connection) - this.context.resetToState({ windows: {} }) + this.context.resetToState({ windows: {} }).catch((e) => this.context.logger.error('Failed to reset state', e)) }) .catch((e) => this.context.logger.error('Failed to initialise Sofie Chef connection', e)) diff --git a/packages/timeline-state-resolver/src/integrations/vizMSE/__tests__/vizMSE.spec.ts b/packages/timeline-state-resolver/src/integrations/vizMSE/__tests__/vizMSE.spec.ts index bdeaae6b6..265ef2fc1 100644 --- a/packages/timeline-state-resolver/src/integrations/vizMSE/__tests__/vizMSE.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/vizMSE/__tests__/vizMSE.spec.ts @@ -7,7 +7,6 @@ import { SomeMappingVizMSE, TimelineContentTypeVizMSE, VIZMSETransitionType, - VizMSEOptions, VIZMSEPlayoutItemContentExternal, VIZMSEPlayoutItemContentInternal, } from 'timeline-state-resolver-types' diff --git a/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vmix.spec.ts b/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vmix.spec.ts index 83efaf278..1787b57c1 100644 --- a/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vmix.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vmix.spec.ts @@ -29,6 +29,7 @@ import { MockTime } from '../../../__tests__/mockTime' import '../../../__tests__/lib' import { CommandContext } from '../vMixCommands' import { prefixAddedInput } from './mockState' +import { addConnections } from '../../../__tests__/lib' const orgSetTimeout = setTimeout @@ -94,19 +95,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -273,19 +276,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -585,19 +590,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -829,19 +836,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -1144,19 +1153,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -1495,19 +1506,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -1722,19 +1735,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -1851,19 +1866,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -1977,19 +1994,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -2103,19 +2122,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -2230,19 +2251,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -2361,19 +2384,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -2493,19 +2518,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -2619,19 +2646,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 8099, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 8099, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -2747,19 +2776,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 9999, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 9999, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -2852,19 +2883,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 9999, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 9999, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -2977,19 +3010,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 9999, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 9999, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -3056,19 +3091,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 9999, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 9999, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -3157,19 +3194,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 9999, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 9999, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -3259,19 +3298,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 9999, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 9999, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -3378,19 +3419,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 9999, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 9999, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) @@ -3507,19 +3550,21 @@ describe('vMix', () => { await myConductor.init() await runPromise( - myConductor.addDevice('myvmix', { - type: DeviceType.VMIX, - options: { - host: '127.0.0.1', - port: 9999, - pollInterval: 0, + addConnections(myConductor.connectionManager, { + myvmix: { + type: DeviceType.VMIX, + options: { + host: '127.0.0.1', + port: 9999, + pollInterval: 0, + }, + commandReceiver: commandReceiver0, }, - commandReceiver: commandReceiver0, }), mockTime ) - const deviceContainer = myConductor.getDevice('myvmix') + const deviceContainer = myConductor.connectionManager.getConnection('myvmix') device = deviceContainer!.device as ThreadedClass const deviceErrorHandler = jest.fn((...args) => console.log('Error in device', ...args)) device.on('error', deviceErrorHandler) diff --git a/packages/timeline-state-resolver/src/service/ConnectionManager.ts b/packages/timeline-state-resolver/src/service/ConnectionManager.ts index 19309c2ec..83724a11c 100644 --- a/packages/timeline-state-resolver/src/service/ConnectionManager.ts +++ b/packages/timeline-state-resolver/src/service/ConnectionManager.ts @@ -29,6 +29,7 @@ export interface ConnectionManagerIntEvents { debug: [...debug: any[]] connectionAdded: [id: string, container: BaseRemoteDeviceIntegration>] + connectionInitialised: [id: string] connectionRemoved: [id: string] } export type MappedDeviceEvents = { @@ -218,6 +219,7 @@ export class ConnectionManager extends EventEmitter { this._handleConnectionInitialisation(id, container) .then(() => { this._connectionAttempts.delete(id) + this.emit('connectionInitialised', id) }) .catch((e) => { this.emit('error', 'Connection ' + id + ' failed to initialise') diff --git a/packages/timeline-state-resolver/src/service/__tests__/ConnectionManager.spec.ts b/packages/timeline-state-resolver/src/service/__tests__/ConnectionManager.spec.ts index 440651a4b..ab846c7eb 100644 --- a/packages/timeline-state-resolver/src/service/__tests__/ConnectionManager.spec.ts +++ b/packages/timeline-state-resolver/src/service/__tests__/ConnectionManager.spec.ts @@ -1,6 +1,7 @@ import { DeviceType, OSCDeviceType } from 'timeline-state-resolver-types' import { ConstructedMockDevices, MockDeviceInstanceWrapper } from '../../__tests__/mockDeviceInstanceWrapper' import { ConnectionManager } from '../ConnectionManager' +import { DeviceOptionsAnyInternal } from '../..' // Mock explicitly the 'dist' version, as that is what threadedClass is being told to load jest.mock('../../../dist/service/DeviceInstance', () => ({ @@ -28,7 +29,7 @@ describe('ConnectionManager', () => { connManager.setConnections( new Map( - Object.entries({ + Object.entries({ osc0: { type: DeviceType.OSC, options: { From bd71aafcd689c6f8015e1fa23e3512f48dc48a2f Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Mon, 9 Sep 2024 12:55:41 +0200 Subject: [PATCH 11/17] chore: fix some tests --- .../src/__mocks__/osc.ts | 13 +++++-- .../src/__tests__/lib.ts | 30 +++++++++++++-- .../integrations/osc/__tests__/osc.spec.ts | 6 +++ .../src/integrations/sisyfos/index.ts | 11 ++---- .../vizMSE/__tests__/vizMSE.spec.ts | 37 +++++++++++++++++-- .../integrations/vmix/__tests__/vmix.spec.ts | 4 +- .../src/service/ConnectionManager.ts | 18 ++++++++- 7 files changed, 97 insertions(+), 22 deletions(-) diff --git a/packages/timeline-state-resolver/src/__mocks__/osc.ts b/packages/timeline-state-resolver/src/__mocks__/osc.ts index 411de3936..803225f2e 100644 --- a/packages/timeline-state-resolver/src/__mocks__/osc.ts +++ b/packages/timeline-state-resolver/src/__mocks__/osc.ts @@ -9,7 +9,7 @@ export class UDPPort extends EventEmitter { this.emit('ready') } - send({ address }) { + send({ address }: { address: string }) { orgSetTimeout(() => { if (MockOSC.connectionIsGood) { if (address === '/state/full') { @@ -21,11 +21,18 @@ export class UDPPort extends EventEmitter { value: `{ "channel": [{ "faderLevel": 0.75, "pgmOn": false, - "pstOn": false + "pstOn": false, + "showChannel": true }, { "faderLevel": 0.75, "pgmOn": false, - "pstOn": false + "pstOn": false, + "showChannel": true + }, { + "faderLevel": 0.75, + "pgmOn": false, + "pstOn": false, + "showChannel": true }] }`, }, ], diff --git a/packages/timeline-state-resolver/src/__tests__/lib.ts b/packages/timeline-state-resolver/src/__tests__/lib.ts index 86bd7c84d..3dc1dc9f1 100644 --- a/packages/timeline-state-resolver/src/__tests__/lib.ts +++ b/packages/timeline-state-resolver/src/__tests__/lib.ts @@ -9,18 +9,32 @@ export function getMockCall(fcn: jest.Mock, callIndex: number, paramIndex: } export async function addConnections( connManager: ConnectionManager, - connections: Record + connections: Record, + waitForInit = true ): Promise { const connectionIds = Object.keys(connections) const addedConns: string[] = [] let resolveAdded: undefined | (() => void) = undefined const psAdded = new Promise((resolveCb) => (resolveAdded = resolveCb)) - connManager.on('connectionInitialised', (id) => { + const cb = (id: string) => { + console.log('got ' + id, 'expect ' + connectionIds) addedConns.push(id) - if (resolveAdded && addedConns.length === connectionIds.length) resolveAdded() - }) + if (resolveAdded && addedConns.length === connectionIds.length) { + resolveAdded() + if (waitForInit) { + connManager.removeListener('connectionInitialised', cb) + } else { + connManager.removeListener('connectionAdded', cb) + } + } + } + if (waitForInit) { + connManager.on('connectionInitialised', cb) + } else { + connManager.on('connectionAdded', cb) + } connManager.setConnections(new Map(Object.entries(connections))) @@ -47,6 +61,14 @@ export async function removeConnections( await psAdded } +export async function awaitNextRemoval(connManager: ConnectionManager): Promise { + return new Promise((resolve) => { + connManager.once('connectionRemoved', () => { + resolve() + }) + }) +} + // Excend jest.expect in functionality and typings expect.extend({ toBeCloseTo(received: number, target: number, diff: number) { diff --git a/packages/timeline-state-resolver/src/integrations/osc/__tests__/osc.spec.ts b/packages/timeline-state-resolver/src/integrations/osc/__tests__/osc.spec.ts index d2dfa725d..66f86b745 100644 --- a/packages/timeline-state-resolver/src/integrations/osc/__tests__/osc.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/osc/__tests__/osc.spec.ts @@ -24,6 +24,12 @@ jest.mock('osc', () => { on: (event: string, listener: (...args: any[]) => void) => { SOCKET_EVENTS.set(event, listener) }, + once: (event: string, listener: (...args: any[]) => void) => { + SOCKET_EVENTS.set(event, (...args: any[]) => { + SOCKET_EVENTS.delete(event) + return listener(...args) + }) + }, close: jest.fn(), } }), diff --git a/packages/timeline-state-resolver/src/integrations/sisyfos/index.ts b/packages/timeline-state-resolver/src/integrations/sisyfos/index.ts index 8d427f6d1..88803b36d 100644 --- a/packages/timeline-state-resolver/src/integrations/sisyfos/index.ts +++ b/packages/timeline-state-resolver/src/integrations/sisyfos/index.ts @@ -85,16 +85,11 @@ export class SisyfosMessageDevice extends DeviceWithState { this._sisyfos.once('initialized', () => { this.setState(this.getDeviceState(false), this.getCurrentTime()) - this.emit('resetResolver') + this.emit('resyncStates') }) this._sisyfos .connect(initOptions.host, initOptions.port) - .then(() => { - // process any states that we missed - this.clearStates() - this.emit('resyncStates') - }) .catch((e) => this.emit('error', 'Failed to initialise Sisyfos connection', e)) return true @@ -199,7 +194,7 @@ export class SisyfosMessageDevice extends DeviceWithState { + this._sisyfos.once('initialized', () => { if (resync) { this._resyncing = false const targetState = this.getState(this.getCurrentTime()) @@ -209,7 +204,7 @@ export class SisyfosMessageDevice extends DeviceWithState { await myConductor.init() + await addConnections( + myConductor.connectionManager, + { + myViz: { + type: DeviceType.VIZMSE, + options: literal>({ + // host: '127.0.0.1', + profile: 'myProfile', + }) as any, + }, + }, + false + ) + await awaitNextRemoval(myConductor.connectionManager) + await addConnections( + myConductor.connectionManager, + { + myViz: { + type: DeviceType.VIZMSE, + options: literal>({ + host: '127.0.0.1', + // profile: 'myProfile', + }) as any, + }, + }, + false + ) + await awaitNextRemoval(myConductor.connectionManager) + expect(onError).toHaveBeenCalledTimes(2) onError.mockClear() @@ -648,7 +679,7 @@ describe('vizMSE', () => { expect(getMockCall(commandReceiver0, 0, 1)).toMatchObject({ timelineObjId: 'obj0', - time: 10105, + time: 10100, content: { instanceName: expect.stringContaining('myInternalElement'), templateName: 'myInternalElement', diff --git a/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vmix.spec.ts b/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vmix.spec.ts index 1787b57c1..8915ec990 100644 --- a/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vmix.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vmix.spec.ts @@ -224,7 +224,7 @@ describe('vMix', () => { expect(commandReceiver0).toHaveBeenCalledTimes(1) expect(commandReceiver0).toHaveBeenNthCalledWith( 1, - 17000, + 17001, expect.objectContaining({ command: { command: VMixCommand.REMOVE_INPUT, @@ -1079,7 +1079,7 @@ describe('vMix', () => { ) expect(commandReceiver0).toHaveBeenNthCalledWith( 5, - 17000, + 17001, expect.objectContaining({ command: { command: VMixCommand.REMOVE_INPUT, diff --git a/packages/timeline-state-resolver/src/service/ConnectionManager.ts b/packages/timeline-state-resolver/src/service/ConnectionManager.ts index 83724a11c..9166d54bd 100644 --- a/packages/timeline-state-resolver/src/service/ConnectionManager.ts +++ b/packages/timeline-state-resolver/src/service/ConnectionManager.ts @@ -48,6 +48,15 @@ export class ConnectionManager extends EventEmitter { * Set the config options for all connections */ public setConnections(connectionsConfig: Map) { + // run through and see if we need to reset any of the counters + this._config.forEach((conf, id) => { + const newConf = connectionsConfig.get(id) + if (newConf && configHasChanged(conf, newConf)) { + // new conf warrants an immediate retry + this._connectionAttempts.delete(id) + } + }) + this._config = connectionsConfig this._updateConnections() } @@ -97,7 +106,7 @@ export class ConnectionManager extends EventEmitter { if (connection) { // see if it should be restarted because of an update - if (configHasChanged(connection, config)) { + if (connectionConfigHasChanged(connection, config)) { operations.push({ operation: 'update', id: deviceId }) } else if ( connection.deviceOptions.debug !== config.debug || @@ -224,6 +233,7 @@ export class ConnectionManager extends EventEmitter { .catch((e) => { this.emit('error', 'Connection ' + id + ' failed to initialise') this._connections.delete(id) + this.emit('connectionRemoved', id) container .terminate() @@ -335,12 +345,16 @@ export class ConnectionManager extends EventEmitter { * A config has changed if any of the options are no longer the same, taking default values into * consideration. In addition, the debug logging flag should be ignored as that can be changed at runtime. */ -function configHasChanged( +function connectionConfigHasChanged( connection: BaseRemoteDeviceIntegration>, config: DeviceOptionsBase ): boolean { const oldConfig = connection.deviceOptions + // now check device specific options + return configHasChanged(oldConfig, config) +} +function configHasChanged(oldConfig: DeviceOptionsBase, config: DeviceOptionsBase): boolean { // now check device specific options return !_.isEqual(_.omit(oldConfig, 'debug', 'debugState'), _.omit(config, 'debug', 'debugState')) } From 35f4b88c98768202d83a8b9d95d3ab625ef7d90a Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Mon, 9 Sep 2024 13:19:32 +0200 Subject: [PATCH 12/17] chore: lint/test --- packages/quick-tsr/src/tsrHandler.ts | 2 +- .../src/__tests__/conductor.spec.ts | 2 -- .../src/__tests__/lib.ts | 5 ++-- .../timeline-state-resolver/src/conductor.ts | 2 +- .../src/integrations/multiOsc/index.ts | 4 ++- .../vizMSE/__tests__/vizMSE.spec.ts | 5 ++-- .../src/service/ConnectionManager.ts | 6 ++--- .../__tests__/ConnectionManager.spec.ts | 27 ++++++++----------- 8 files changed, 23 insertions(+), 30 deletions(-) diff --git a/packages/quick-tsr/src/tsrHandler.ts b/packages/quick-tsr/src/tsrHandler.ts index b8f5e41d2..0fbd355e4 100644 --- a/packages/quick-tsr/src/tsrHandler.ts +++ b/packages/quick-tsr/src/tsrHandler.ts @@ -135,6 +135,6 @@ export class TSRHandler { this.tsr.setDatastore(store) } public async setDevices(devices: { [deviceId: string]: DeviceOptionsAny }): Promise { - this.tsr.connectionManager.setConnections(new Map(Object.entries(devices))) + this.tsr.connectionManager.setConnections(devices) } } diff --git a/packages/timeline-state-resolver/src/__tests__/conductor.spec.ts b/packages/timeline-state-resolver/src/__tests__/conductor.spec.ts index 86054999e..6c8d3990a 100644 --- a/packages/timeline-state-resolver/src/__tests__/conductor.spec.ts +++ b/packages/timeline-state-resolver/src/__tests__/conductor.spec.ts @@ -27,8 +27,6 @@ jest.mock('../service/DeviceInstance', () => ({ import { Conductor, TimelineTriggerTimeResult } from '../conductor' import type { DeviceInstanceWrapper } from '../service/DeviceInstance' -import { DeviceOptionsAnyInternal } from '..' -import { ConnectionManager } from '../service/ConnectionManager' describe('Conductor', () => { const mockTime = new MockTime() diff --git a/packages/timeline-state-resolver/src/__tests__/lib.ts b/packages/timeline-state-resolver/src/__tests__/lib.ts index 3dc1dc9f1..a37012565 100644 --- a/packages/timeline-state-resolver/src/__tests__/lib.ts +++ b/packages/timeline-state-resolver/src/__tests__/lib.ts @@ -18,7 +18,6 @@ export async function addConnections( let resolveAdded: undefined | (() => void) = undefined const psAdded = new Promise((resolveCb) => (resolveAdded = resolveCb)) const cb = (id: string) => { - console.log('got ' + id, 'expect ' + connectionIds) addedConns.push(id) if (resolveAdded && addedConns.length === connectionIds.length) { @@ -36,7 +35,7 @@ export async function addConnections( connManager.on('connectionAdded', cb) } - connManager.setConnections(new Map(Object.entries(connections))) + connManager.setConnections(connections) await psAdded } @@ -56,7 +55,7 @@ export async function removeConnections( if (resolveAdded && addedConns.length === toBeRemoved.length) resolveAdded() }) - connManager.setConnections(new Map(Object.entries(connections))) + connManager.setConnections(connections) await psAdded } diff --git a/packages/timeline-state-resolver/src/conductor.ts b/packages/timeline-state-resolver/src/conductor.ts index 60b33572a..a747c7354 100644 --- a/packages/timeline-state-resolver/src/conductor.ts +++ b/packages/timeline-state-resolver/src/conductor.ts @@ -340,7 +340,7 @@ export class Conductor extends EventEmitter { if (this._triggerSendStartStopCallbacksTimeout) clearTimeout(this._triggerSendStartStopCallbacksTimeout) // remove all connections: - this.connectionManager.setConnections(new Map()) + this.connectionManager.setConnections({}) } /** diff --git a/packages/timeline-state-resolver/src/integrations/multiOsc/index.ts b/packages/timeline-state-resolver/src/integrations/multiOsc/index.ts index 1dfe18bf8..56480cbff 100644 --- a/packages/timeline-state-resolver/src/integrations/multiOsc/index.ts +++ b/packages/timeline-state-resolver/src/integrations/multiOsc/index.ts @@ -77,7 +77,9 @@ export class MultiOSCMessageDevice extends Device [id, {}]))) + this.context + .resetToState(Object.fromEntries(Object.keys(this._connections).map((id) => [id, {}]))) + .catch((e) => this.context.logger.warning('Failed to reset state: ' + e)) return true } diff --git a/packages/timeline-state-resolver/src/integrations/vizMSE/__tests__/vizMSE.spec.ts b/packages/timeline-state-resolver/src/integrations/vizMSE/__tests__/vizMSE.spec.ts index df5131b5b..a5283e05c 100644 --- a/packages/timeline-state-resolver/src/integrations/vizMSE/__tests__/vizMSE.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/vizMSE/__tests__/vizMSE.spec.ts @@ -1,4 +1,4 @@ -import { Conductor, DeviceOptionsAnyInternal } from '../../../conductor' +import { Conductor } from '../../../conductor' import { Mappings, DeviceType, @@ -13,7 +13,7 @@ import { } from 'timeline-state-resolver-types' import { MockTime } from '../../../__tests__/mockTime' import { ThreadedClass } from 'threadedclass' -import { addConnections, awaitNextRemoval, getMockCall, removeConnections } from '../../../__tests__/lib' +import { addConnections, awaitNextRemoval, getMockCall } from '../../../__tests__/lib' import { VizMSEDevice } from '..' import * as vConnection from '../../../__mocks__/v-connection' import * as net from '../../../__mocks__/net' @@ -25,7 +25,6 @@ import _ = require('underscore') import { StatusCode } from '../../../devices/device' import { MOCK_SHOWS } from '../../../__mocks__/v-connection' import { literal } from '../../../lib' -import { ConnectionManager } from '../../../service/ConnectionManager' const orgSetTimeout = setTimeout diff --git a/packages/timeline-state-resolver/src/service/ConnectionManager.ts b/packages/timeline-state-resolver/src/service/ConnectionManager.ts index 9166d54bd..951137ebe 100644 --- a/packages/timeline-state-resolver/src/service/ConnectionManager.ts +++ b/packages/timeline-state-resolver/src/service/ConnectionManager.ts @@ -47,17 +47,17 @@ export class ConnectionManager extends EventEmitter { /** * Set the config options for all connections */ - public setConnections(connectionsConfig: Map) { + public setConnections(connectionsConfig: Record) { // run through and see if we need to reset any of the counters this._config.forEach((conf, id) => { - const newConf = connectionsConfig.get(id) + const newConf = connectionsConfig[id] if (newConf && configHasChanged(conf, newConf)) { // new conf warrants an immediate retry this._connectionAttempts.delete(id) } }) - this._config = connectionsConfig + this._config = new Map(Object.entries(connectionsConfig)) this._updateConnections() } diff --git a/packages/timeline-state-resolver/src/service/__tests__/ConnectionManager.spec.ts b/packages/timeline-state-resolver/src/service/__tests__/ConnectionManager.spec.ts index ab846c7eb..9c47cb149 100644 --- a/packages/timeline-state-resolver/src/service/__tests__/ConnectionManager.spec.ts +++ b/packages/timeline-state-resolver/src/service/__tests__/ConnectionManager.spec.ts @@ -1,7 +1,6 @@ import { DeviceType, OSCDeviceType } from 'timeline-state-resolver-types' import { ConstructedMockDevices, MockDeviceInstanceWrapper } from '../../__tests__/mockDeviceInstanceWrapper' import { ConnectionManager } from '../ConnectionManager' -import { DeviceOptionsAnyInternal } from '../..' // Mock explicitly the 'dist' version, as that is what threadedClass is being told to load jest.mock('../../../dist/service/DeviceInstance', () => ({ @@ -27,26 +26,22 @@ describe('ConnectionManager', () => { if (resolveRemoved) resolveRemoved() }) - connManager.setConnections( - new Map( - Object.entries({ - osc0: { - type: DeviceType.OSC, - options: { - host: '127.0.0.1', - port: 5250, - type: OSCDeviceType.UDP, - }, - }, - }) - ) - ) + connManager.setConnections({ + osc0: { + type: DeviceType.OSC, + options: { + host: '127.0.0.1', + port: 5250, + type: OSCDeviceType.UDP, + }, + }, + }) await psAdded expect(ConstructedMockDevices['osc0']).toBeTruthy() - connManager.setConnections(new Map()) + connManager.setConnections({}) await psRemoved From d6c93972f856063a88d48efe6d8605ed61150c2d Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Mon, 9 Sep 2024 13:27:01 +0200 Subject: [PATCH 13/17] chore: lint --- packages/quick-tsr/src/tsrHandler.ts | 1 - .../examples/CasparcgVideoPlayES6example.js | 12 +++++++----- .../examples/playVideoInCaspar.ts | 12 +++++++----- .../examples/testChangeTimelineQuickly.ts | 12 +++++++----- 4 files changed, 21 insertions(+), 16 deletions(-) diff --git a/packages/quick-tsr/src/tsrHandler.ts b/packages/quick-tsr/src/tsrHandler.ts index 0fbd355e4..d1a95cb4b 100644 --- a/packages/quick-tsr/src/tsrHandler.ts +++ b/packages/quick-tsr/src/tsrHandler.ts @@ -16,7 +16,6 @@ import { } from 'timeline-state-resolver' import { ThreadedClass } from 'threadedclass' -import * as _ from 'underscore' import { TSRSettings } from './index' /** diff --git a/packages/timeline-state-resolver/examples/CasparcgVideoPlayES6example.js b/packages/timeline-state-resolver/examples/CasparcgVideoPlayES6example.js index 2516772cf..0d4ef2753 100644 --- a/packages/timeline-state-resolver/examples/CasparcgVideoPlayES6example.js +++ b/packages/timeline-state-resolver/examples/CasparcgVideoPlayES6example.js @@ -14,11 +14,13 @@ tsrConductor .init() .then(() => { // Add devices to the TSR-conductor: - return tsrConductor.addDevice('casparcg0', { - type: DeviceType.CASPARCG, - options: { - host: 'localhost', - port: 5250, + return tsr.connectionManager.setConnections({ + casparcg0: { + type: DeviceType.CASPARCG, + options: { + host: 'localhost', + port: 5250, + }, }, }) }) diff --git a/packages/timeline-state-resolver/examples/playVideoInCaspar.ts b/packages/timeline-state-resolver/examples/playVideoInCaspar.ts index 0ce4c4372..c5f3e562b 100644 --- a/packages/timeline-state-resolver/examples/playVideoInCaspar.ts +++ b/packages/timeline-state-resolver/examples/playVideoInCaspar.ts @@ -12,11 +12,13 @@ tsr.on('debug', (deviceId, cmd) => console.log('debug', deviceId, cmd)) const a = async function () { await tsr.init() - await tsr.addDevice('casparcg0', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', - // port: 5250 + tsr.connectionManager.setConnections({ + casparcg0: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + // port: 5250 + }, }, }) diff --git a/packages/timeline-state-resolver/examples/testChangeTimelineQuickly.ts b/packages/timeline-state-resolver/examples/testChangeTimelineQuickly.ts index c75082fa3..863a9e8c1 100644 --- a/packages/timeline-state-resolver/examples/testChangeTimelineQuickly.ts +++ b/packages/timeline-state-resolver/examples/testChangeTimelineQuickly.ts @@ -10,11 +10,13 @@ tsr.on('error', (e) => console.log('error', e)) const a = async function () { await tsr.init() - await tsr.addDevice('casparcg0', { - type: DeviceType.CASPARCG, - options: { - host: '127.0.0.1', - // port: 5250 + tsr.connectionManager.setConnections({ + casparcg0: { + type: DeviceType.CASPARCG, + options: { + host: '127.0.0.1', + // port: 5250 + }, }, }) From 004243e0299e071a36965e1463448444b8be9951 Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Mon, 9 Sep 2024 13:33:33 +0200 Subject: [PATCH 14/17] chore: typo --- .../examples/CasparcgVideoPlayES6example.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/timeline-state-resolver/examples/CasparcgVideoPlayES6example.js b/packages/timeline-state-resolver/examples/CasparcgVideoPlayES6example.js index 0d4ef2753..4d64e6ffb 100644 --- a/packages/timeline-state-resolver/examples/CasparcgVideoPlayES6example.js +++ b/packages/timeline-state-resolver/examples/CasparcgVideoPlayES6example.js @@ -14,7 +14,7 @@ tsrConductor .init() .then(() => { // Add devices to the TSR-conductor: - return tsr.connectionManager.setConnections({ + return tsrConductor.connectionManager.setConnections({ casparcg0: { type: DeviceType.CASPARCG, options: { From cf0e82beac96267f793f63f6e1259064bf8fb318 Mon Sep 17 00:00:00 2001 From: Mint de Wit Date: Wed, 11 Sep 2024 15:50:14 +0200 Subject: [PATCH 15/17] chore: add onChildClose handler --- .../src/service/ConnectionManager.ts | 17 +++++++++++++++-- 1 file changed, 15 insertions(+), 2 deletions(-) diff --git a/packages/timeline-state-resolver/src/service/ConnectionManager.ts b/packages/timeline-state-resolver/src/service/ConnectionManager.ts index 951137ebe..c05f445eb 100644 --- a/packages/timeline-state-resolver/src/service/ConnectionManager.ts +++ b/packages/timeline-state-resolver/src/service/ConnectionManager.ts @@ -221,6 +221,19 @@ export class ConnectionManager extends EventEmitter { // set up event handlers await this._setupDeviceListeners(id, container) + container.onChildClose = () => { + this.emit('error', 'Connection ' + id + ' closed') + this._connections.delete(id) + this.emit('connectionRemoved', id) + + container + .terminate() + .catch((e) => this.emit('warning', `Failed to initialise ${id} (${e})`)) + .finally(() => { + this._updateConnections() + }) + } + this._connections.set(id, container) this.emit('connectionAdded', id, container) @@ -231,13 +244,13 @@ export class ConnectionManager extends EventEmitter { this.emit('connectionInitialised', id) }) .catch((e) => { - this.emit('error', 'Connection ' + id + ' failed to initialise') + this.emit('error', 'Connection ' + id + ' failed to initialise', e) this._connections.delete(id) this.emit('connectionRemoved', id) container .terminate() - .catch(() => this.emit('warning', `Failed to initialise ${id} (${e})`)) + .catch((e) => this.emit('warning', `Failed to initialise ${id} (${e})`)) .finally(() => { this._updateConnections() }) From 34bf23014deb549a737182969d5b4d388c97a0d4 Mon Sep 17 00:00:00 2001 From: Johan Nyman Date: Fri, 20 Sep 2024 09:34:50 +0200 Subject: [PATCH 16/17] chore: cleanup & fix test --- .../src/__tests__/lib.ts | 52 +++++++++++++++++++ .../src/integrations/lawo/index.ts | 1 - .../quantel/__tests__/quantelGatewayMock.ts | 5 +- .../sisyfos/__tests__/sisyfos.spec.ts | 20 +++++-- 4 files changed, 68 insertions(+), 10 deletions(-) diff --git a/packages/timeline-state-resolver/src/__tests__/lib.ts b/packages/timeline-state-resolver/src/__tests__/lib.ts index a37012565..341a44965 100644 --- a/packages/timeline-state-resolver/src/__tests__/lib.ts +++ b/packages/timeline-state-resolver/src/__tests__/lib.ts @@ -1,5 +1,6 @@ import { DeviceOptionsAnyInternal } from '../conductor' import { ConnectionManager } from '../service/ConnectionManager' +import { MockTime } from './mockTime' /** * Just a wrapper to :any type, to be used in tests only @@ -85,3 +86,54 @@ declare global { } } } +/** setTimeout (not affected by jest.fakeTimers) */ +const setTimeoutOrg = setTimeout +/** Sleep for a */ +export function waitTime(ms: number): Promise { + return new Promise((resolve) => setTimeoutOrg(resolve, ms)) +} + +/** An actual monotonic time, not affected by jest.fakeTimers */ +export const realTimeNow = performance.now +/** + * Executes {expectFcn} intermittently until it doesn't throw anymore. + * Waits up to {maxWaitTime} ms, then throws the latest error. + * Useful in unit-tests as a way to wait until a predicate is fulfilled. + */ +export async function waitUntil( + expectFcn: () => void | Promise, + maxWaitTime: number, + mockTime?: MockTime +): Promise { + const startTime = realTimeNow() + + const previousErrors: string[] = [] + + while (true) { + mockTime?.advanceTimeTicks(100) + await waitTime(100) + try { + await Promise.resolve(expectFcn()) + return + } catch (err) { + const errorStr = `${err}` + if (previousErrors.length) { + const previousError = previousErrors[previousErrors.length - 1] + if (errorStr !== previousError) { + previousErrors.push(errorStr) + } + } else { + previousErrors.push(errorStr) + } + + const waitedTime = realTimeNow() - startTime + if (waitedTime > maxWaitTime) { + console.log(`waitUntil: waited for ${waitedTime} ms, giving up (maxWaitTime: ${maxWaitTime}).`) + console.log(`Previous errors: \n${previousErrors.join('\n')}`) + + throw err + } + // else ignore error and try again later + } + } +} diff --git a/packages/timeline-state-resolver/src/integrations/lawo/index.ts b/packages/timeline-state-resolver/src/integrations/lawo/index.ts index 380bd90da..6e5065c6a 100644 --- a/packages/timeline-state-resolver/src/integrations/lawo/index.ts +++ b/packages/timeline-state-resolver/src/integrations/lawo/index.ts @@ -23,7 +23,6 @@ export class LawoDevice extends Device this.context.logger.error('Lawo.LawoConnection', e)) this._lawo.on('debug', (...debug) => this.context.logger.debug('Lawo.LawoConnection', ...debug)) - this._lawo.on('debug', (...debug) => console.log('Lawo.LawoConnection', ...debug)) this._lawo.on('connected', (firstConnection) => { if (firstConnection) { // reset state diff --git a/packages/timeline-state-resolver/src/integrations/quantel/__tests__/quantelGatewayMock.ts b/packages/timeline-state-resolver/src/integrations/quantel/__tests__/quantelGatewayMock.ts index c9ce6754e..74206c5ea 100644 --- a/packages/timeline-state-resolver/src/integrations/quantel/__tests__/quantelGatewayMock.ts +++ b/packages/timeline-state-resolver/src/integrations/quantel/__tests__/quantelGatewayMock.ts @@ -29,9 +29,8 @@ export function setupQuantelGatewayMock() { }, } - // @ts-ignore: not logging const onRequest = jest.fn((_type: string, _url: string) => { - // console.log('onRequest', type, url) + // nothing }) const onRequestRaw = jest.fn((type: string, url: string) => { @@ -223,7 +222,6 @@ async function handleRequest( message: `ISA URL not provided`, stack: '', } - // console.log(type, resource) urlRoute(type, resource, { // @ts-ignore: no need for params @@ -837,7 +835,6 @@ async function handleRequest( }, }) .then((body) => { - // console.log('got responding:', type, resource, body) resolve({ statusCode: quantelServer.requestReturnsOK ? 200 : 500, // body: JSON.stringify(body) diff --git a/packages/timeline-state-resolver/src/integrations/sisyfos/__tests__/sisyfos.spec.ts b/packages/timeline-state-resolver/src/integrations/sisyfos/__tests__/sisyfos.spec.ts index 8f14a8442..90c9a0055 100644 --- a/packages/timeline-state-resolver/src/integrations/sisyfos/__tests__/sisyfos.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/sisyfos/__tests__/sisyfos.spec.ts @@ -13,7 +13,7 @@ const MockOSC = OSC.MockOSC import { MockTime } from '../../../__tests__/mockTime' import { ThreadedClass } from 'threadedclass' import { SisyfosMessageDevice } from '../../../integrations/sisyfos' -import { addConnections, getMockCall } from '../../../__tests__/lib' +import { addConnections, getMockCall, waitUntil } from '../../../__tests__/lib' describe('Sisyfos', () => { jest.mock('osc', () => OSC) @@ -1071,19 +1071,29 @@ describe('Sisyfos', () => { // Check that no commands has been scheduled: expect(await device.queue).toHaveLength(0) - expect(await device.connected).toEqual(true) - expect(onConnectionChanged).toHaveBeenCalledTimes(0) + + // Wait for the connection to be initialized: + await waitUntil( + async () => { + expect(await device.connected).toEqual(true) + }, + 1000, + mockTime + ) // Simulate a connection loss: MockOSC.connectionIsGood = false + // Wait for the OSC timeout to trigger: await mockTime.advanceTimeTicks(3000) await wait(1) await mockTime.advanceTimeTicks(3000) await wait(1) expect(await device.connected).toEqual(false) - expect(onConnectionChanged).toHaveBeenCalledTimes(1) + + expect(onConnectionChanged.mock.calls.length).toBeGreaterThanOrEqual(1) + onConnectionChanged.mockClear() // Simulate a connection regain: MockOSC.connectionIsGood = true @@ -1093,6 +1103,6 @@ describe('Sisyfos', () => { await wait(1) expect(await device.connected).toEqual(true) - expect(onConnectionChanged).toHaveBeenCalledTimes(4) + expect(onConnectionChanged.mock.calls.length).toBeGreaterThanOrEqual(1) }) }) From 3d5ee7e35e12243b6a477906e67e75842b4fc3d6 Mon Sep 17 00:00:00 2001 From: Johan Nyman Date: Fri, 20 Sep 2024 09:46:02 +0200 Subject: [PATCH 17/17] chore: lint --- packages/timeline-state-resolver/src/__tests__/lib.ts | 9 +++++---- .../src/integrations/vmix/__tests__/vmix.spec.ts | 1 - .../src/service/ConnectionManager.ts | 5 ++++- 3 files changed, 9 insertions(+), 6 deletions(-) diff --git a/packages/timeline-state-resolver/src/__tests__/lib.ts b/packages/timeline-state-resolver/src/__tests__/lib.ts index 341a44965..9416bcafb 100644 --- a/packages/timeline-state-resolver/src/__tests__/lib.ts +++ b/packages/timeline-state-resolver/src/__tests__/lib.ts @@ -89,12 +89,12 @@ declare global { /** setTimeout (not affected by jest.fakeTimers) */ const setTimeoutOrg = setTimeout /** Sleep for a */ -export function waitTime(ms: number): Promise { +export async function waitTime(ms: number): Promise { return new Promise((resolve) => setTimeoutOrg(resolve, ms)) } -/** An actual monotonic time, not affected by jest.fakeTimers */ -export const realTimeNow = performance.now +/** The current time, not affected by jest.fakeTimers */ +export const realTimeNow = Date.now.bind(Date) /** * Executes {expectFcn} intermittently until it doesn't throw anymore. * Waits up to {maxWaitTime} ms, then throws the latest error. @@ -109,8 +109,9 @@ export async function waitUntil( const previousErrors: string[] = [] + // eslint-disable-next-line no-constant-condition while (true) { - mockTime?.advanceTimeTicks(100) + await mockTime?.advanceTimeTicks(100) await waitTime(100) try { await Promise.resolve(expectFcn()) diff --git a/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vmix.spec.ts b/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vmix.spec.ts index 8915ec990..3a8bf0699 100644 --- a/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vmix.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/vmix/__tests__/vmix.spec.ts @@ -26,7 +26,6 @@ import { import { ThreadedClass } from 'threadedclass' import { VMixDevice } from '..' import { MockTime } from '../../../__tests__/mockTime' -import '../../../__tests__/lib' import { CommandContext } from '../vMixCommands' import { prefixAddedInput } from './mockState' import { addConnections } from '../../../__tests__/lib' diff --git a/packages/timeline-state-resolver/src/service/ConnectionManager.ts b/packages/timeline-state-resolver/src/service/ConnectionManager.ts index c05f445eb..0cc202cc0 100644 --- a/packages/timeline-state-resolver/src/service/ConnectionManager.ts +++ b/packages/timeline-state-resolver/src/service/ConnectionManager.ts @@ -149,7 +149,10 @@ export class ConnectionManager extends EventEmitter { this._updating = false // wait until next - const nextTime = Array.from(this._connectionAttempts.values()).reduce((a, b) => (a.next < b.next ? a : b)) + const nextTime = Array.from(this._connectionAttempts.values()).reduce((a, b) => (a.next < b.next ? a : b), { + last: Date.now(), // not used + next: Date.now() + 4000, // in 4 seconds + }) this._nextAttempt = setTimeout(() => { this._updateConnections() }, nextTime.next - Date.now())