diff --git a/packages/timeline-state-resolver/src/__tests__/mockTime.ts b/packages/timeline-state-resolver/src/__tests__/mockTime.ts index 3c9ad8c86..338e5f08b 100644 --- a/packages/timeline-state-resolver/src/__tests__/mockTime.ts +++ b/packages/timeline-state-resolver/src/__tests__/mockTime.ts @@ -22,6 +22,9 @@ export class MockTime { this._now = 10000 jest.useFakeTimers({ now: this._now }) } + reset = () => { + jest.useRealTimers() + } advanceTime = (advanceTime: number) => { this._now += advanceTime jest.advanceTimersByTime(advanceTime) diff --git a/packages/timeline-state-resolver/src/integrations/quantel/__tests__/quantel.spec.ts b/packages/timeline-state-resolver/src/integrations/quantel/__tests__/quantel.spec.ts index ff0946ae5..b8960fdd9 100644 --- a/packages/timeline-state-resolver/src/integrations/quantel/__tests__/quantel.spec.ts +++ b/packages/timeline-state-resolver/src/integrations/quantel/__tests__/quantel.spec.ts @@ -8,6 +8,7 @@ import { SomeMappingQuantel, Timeline, TimelineContentQuantelAny, + TSRTimeline, TSRTimelineContent, } from 'timeline-state-resolver-types' import { QuantelCommandWithContext, QuantelDevice } from '..' @@ -15,6 +16,12 @@ import { QuantelCommandType, QuantelState } from '../types' import { setupQuantelGatewayMock } from './quantelGatewayMock' import { MockTime } from '../../../__tests__/mockTime' import { getDeviceContext } from '../../../integrations/__tests__/testlib' +import { StateHandler } from '../../../service/stateHandler' +import { CommandWithContext } from '../../..' +import { getResolvedState, resolveTimeline } from 'superfly-timeline' +import { DevicesDict } from '../../../service/devices' +import { setSoftJumpWaitTime } from '../connection' +import { ExecuteMode } from '../../../service/device' async function getInitialisedQuantelDevice(clearMock?: jest.Mock) { const dev = new QuantelDevice(getDeviceContext()) @@ -415,6 +422,16 @@ describe('Quantel Device', () => { }, }, [ + { + command: { + type: QuantelCommandType.CANCELWAITING, + portId: 'port0', + timelineObjId: '', + }, + context: 'Clear all delayed out-transitions', + timelineObjId: '', + executeMode: ExecuteMode.SALVO, + }, { command: { type: QuantelCommandType.SETUPPORT, @@ -473,6 +490,16 @@ describe('Quantel Device', () => { }, }, [ + { + command: { + type: QuantelCommandType.CANCELWAITING, + portId: 'port0', + timelineObjId: '', + }, + context: 'Clear all delayed out-transitions', + timelineObjId: '', + executeMode: ExecuteMode.SALVO, + }, { command: { type: QuantelCommandType.LOADCLIPFRAGMENTS, @@ -505,6 +532,7 @@ describe('Quantel Device', () => { mode: QuantelControlMode.QUALITY, transition: undefined, }, + preliminary: undefined, context: 'New clip is paused', timelineObjId: 'obj1', }, @@ -544,6 +572,16 @@ describe('Quantel Device', () => { }, }, [ + { + command: { + type: QuantelCommandType.CANCELWAITING, + portId: 'port0', + timelineObjId: '', + }, + context: 'Clear all delayed out-transitions', + timelineObjId: '', + executeMode: ExecuteMode.SALVO, + }, { command: { type: QuantelCommandType.LOADCLIPFRAGMENTS, @@ -578,6 +616,7 @@ describe('Quantel Device', () => { }, context: 'New clip is paused', timelineObjId: 'obj1', + preliminary: undefined, }, ] ) @@ -620,6 +659,16 @@ describe('Quantel Device', () => { }, }, [ + { + command: { + type: QuantelCommandType.CANCELWAITING, + portId: 'port0', + timelineObjId: '', + }, + context: 'Clear all delayed out-transitions', + timelineObjId: '', + executeMode: ExecuteMode.SALVO, + }, { command: { type: QuantelCommandType.LOADCLIPFRAGMENTS, @@ -654,6 +703,7 @@ describe('Quantel Device', () => { }, context: 'New clip is playing', timelineObjId: 'obj1', + preliminary: undefined, }, ] ) @@ -696,6 +746,16 @@ describe('Quantel Device', () => { }, }, [ + { + command: { + type: QuantelCommandType.CANCELWAITING, + portId: 'port0', + timelineObjId: '', + }, + context: 'Clear all delayed out-transitions', + timelineObjId: '', + executeMode: ExecuteMode.SALVO, + }, { command: { type: QuantelCommandType.LOADCLIPFRAGMENTS, @@ -730,6 +790,7 @@ describe('Quantel Device', () => { }, context: 'New clip is playing', timelineObjId: 'obj1', + preliminary: undefined, }, ] ) @@ -756,6 +817,16 @@ describe('Quantel Device', () => { }, { time: 3000, port: {} }, [ + { + command: { + type: QuantelCommandType.CANCELWAITING, + portId: 'port0', + timelineObjId: '', + }, + context: 'Clear all delayed out-transitions', + timelineObjId: '', + executeMode: ExecuteMode.SALVO, + }, { command: { type: QuantelCommandType.RELEASEPORT, @@ -808,6 +879,16 @@ describe('Quantel Device', () => { }, }, [ + { + command: { + type: QuantelCommandType.CANCELWAITING, + portId: 'port0', + timelineObjId: '', + }, + context: 'Clear all delayed out-transitions', + timelineObjId: '', + executeMode: ExecuteMode.SALVO, + }, { command: { type: QuantelCommandType.LOADCLIPFRAGMENTS, @@ -842,6 +923,7 @@ describe('Quantel Device', () => { }, context: 'New clip is playing', timelineObjId: 'obj2', + preliminary: undefined, }, ], 15020 @@ -886,6 +968,16 @@ describe('Quantel Device', () => { }, }, [ + { + command: { + type: QuantelCommandType.CANCELWAITING, + portId: 'port0', + timelineObjId: '', + }, + context: 'Clear all delayed out-transitions', + timelineObjId: '', + executeMode: ExecuteMode.SALVO, + }, { command: { type: QuantelCommandType.CLEARCLIP, @@ -946,6 +1038,16 @@ describe('Quantel Device', () => { }, }, [ + { + command: { + type: QuantelCommandType.CANCELWAITING, + portId: 'port0', + timelineObjId: '', + }, + context: 'Clear all delayed out-transitions', + timelineObjId: '', + executeMode: ExecuteMode.SALVO, + }, { command: { type: QuantelCommandType.LOADCLIPFRAGMENTS, @@ -983,6 +1085,7 @@ describe('Quantel Device', () => { }, context: 'New clip is paused', timelineObjId: 'obj2', + preliminary: undefined, }, ], 11500 @@ -1031,6 +1134,16 @@ describe('Quantel Device', () => { }, }, [ + { + command: { + type: QuantelCommandType.CANCELWAITING, + portId: 'port0', + timelineObjId: '', + }, + context: 'Clear all delayed out-transitions', + timelineObjId: '', + executeMode: ExecuteMode.SALVO, + }, { command: { type: QuantelCommandType.LOADCLIPFRAGMENTS, @@ -1065,6 +1178,7 @@ describe('Quantel Device', () => { }, context: 'New clip is paused', timelineObjId: 'obj2', + preliminary: undefined, }, ], 11500 @@ -1097,6 +1211,26 @@ describe('Quantel Device', () => { }, }, [ + { + command: { + type: QuantelCommandType.CANCELWAITING, + portId: 'port0_renamed', + timelineObjId: '', + }, + context: 'Clear all delayed out-transitions', + timelineObjId: '', + executeMode: ExecuteMode.SALVO, + }, + { + command: { + type: QuantelCommandType.CANCELWAITING, + portId: 'port0', + timelineObjId: '', + }, + context: 'Clear all delayed out-transitions', + timelineObjId: '', + executeMode: ExecuteMode.SALVO, + }, { command: { type: QuantelCommandType.RELEASEPORT, @@ -1141,6 +1275,9 @@ describe('Quantel Device', () => { beforeAll(() => { mockTime.init() }) + afterAll(() => { + mockTime.reset() + }) test('sequence of commands', async () => { // note - the internals of the QuantelManager class are state-based so it's easier to do all of this in one long test @@ -1304,6 +1441,536 @@ describe('Quantel Device', () => { expect(onRequest).toHaveBeenNthCalledWith(1, 'post', expect.stringContaining('port/my_port/reset')) }) }) + describe('with StateHandler', () => { + const MOCK_SEND_COMMAND = jest.fn() + const CONTEXT = { + deviceId: 'unitTests0', + logger: { + debug: console.log, + info: console.log, + warn: console.log, + error: console.log, + }, + emitTimeTrace: () => null, + reportStateChangeMeasurement: () => null, + getCurrentTime: () => Date.now(), + } + function getNewStateHandler(dev: QuantelDevice): StateHandler { + // eslint-disable-next-line @typescript-eslint/unbound-method + const orgSendCommand = dev.sendCommand + dev.sendCommand = async (...args) => { + MOCK_SEND_COMMAND(...args) + return orgSendCommand.apply(dev, args) + } + + const deviceSpecs = DevicesDict[DeviceType.QUANTEL] + return new StateHandler( + CONTEXT, + { + executionType: deviceSpecs.executionMode({}), + }, + dev + ) + } + function clearMocks() { + MOCK_SEND_COMMAND.mockClear() + onRequest.mockClear() + } + beforeAll(() => { + setSoftJumpWaitTime(0) + }) + + test('outTransition to clear, cancel, then play another', async () => { + const dev = await getInitialisedQuantelDevice() + + // give it some time to finish the init + await sleep(10) + + const stateHandler = getNewStateHandler(dev) + + const timeline: TSRTimeline = [ + { + id: 'obj0', + enable: { + start: 1000, + end: 2000, + }, + content: { + deviceType: DeviceType.QUANTEL, + title: 'myClip0', + outTransition: { + type: QuantelTransitionType.DELAY, + delay: 500, // 2500 + }, + }, + layer: 'layer0', + }, + { + id: 'obj1', + enable: { + start: 2100, + end: 10000, + }, + content: { + deviceType: DeviceType.QUANTEL, + title: 'myClip1', + }, + layer: 'layer0', + }, + ] + const mappings: Mappings = { + layer0: { + device: DeviceType.QUANTEL, + deviceId: 'quantel0', + options: { + mappingType: MappingQuantelType.Port, + portId: 'my_port', + channelId: 1, + }, + }, + } + const resolved = resolveTimeline(timeline, { + time: 0, + }) + + // Handle state at time 0 (nothing is playing) + { + const state = getResolvedState(resolved, 0) + await stateHandler.handleState(state, mappings) + + // Give QuantelManager some time to process the commands + await sleep(10) + expect(MOCK_SEND_COMMAND).toHaveBeenCalledTimes(3) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 1, + expect.objectContaining({ + command: expect.objectContaining({ type: QuantelCommandType.CANCELWAITING, portId: 'my_port' }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 2, + expect.objectContaining({ + command: expect.objectContaining({ type: QuantelCommandType.SETUPPORT, portId: 'my_port', channel: 1 }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 3, + expect.objectContaining({ + command: expect.objectContaining({ type: QuantelCommandType.CLEARCLIP, portId: 'my_port' }), + }) + ) + + expect(onRequest).toHaveBeenCalledWith('post', 'http://localhost:3000/connect/myISA%3A8000') + expect(onRequest).toHaveBeenCalledWith('get', 'http://localhost:3000/default/server') + expect(onRequest).toHaveBeenCalledWith('get', 'http://localhost:3000/default/server/1100/port/my_port') + expect(onRequest).toHaveBeenCalledWith( + 'put', + 'http://localhost:3000/default/server/1100/port/my_port/channel/1' + ) + expect(onRequest).toHaveBeenCalledWith('post', 'http://localhost:3000/default/server/1100/port/my_port/reset') + clearMocks() + } + + // Handle state at time 1000 (myClip0 starts to play) + { + const state = getResolvedState(resolved, 1000) + await stateHandler.handleState(state, mappings) + // Give QuantelManager some time to process the commands + await sleep(100) + + expect(MOCK_SEND_COMMAND).toHaveBeenCalledTimes(3) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 1, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.CANCELWAITING, + portId: 'my_port', + }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 2, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.LOADCLIPFRAGMENTS, + portId: 'my_port', + clip: expect.objectContaining({ title: 'myClip0' }), + }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 3, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.PLAYCLIP, + portId: 'my_port', + clip: expect.objectContaining({ title: 'myClip0' }), + }), + }) + ) + + expect(onRequest).toHaveBeenCalledWith('post', expect.stringContaining('/1100/port/my_port/fragments?offset=0')) + expect(onRequest).toHaveBeenCalledWith('put', expect.stringContaining('/1100/port/my_port/jump?offset=')) + expect(onRequest).toHaveBeenCalledWith('post', expect.stringContaining('/1100/port/my_port/trigger/JUMP')) + expect(onRequest).toHaveBeenCalledWith('post', expect.stringContaining('/1100/port/my_port/trigger/START')) + expect(onRequest).toHaveBeenCalledWith('get', expect.stringContaining('/1100/port/my_port')) + expect(onRequest).toHaveBeenCalledWith( + 'post', + expect.stringContaining('/1100/port/my_port/trigger/STOP?offset=1999') + ) + + clearMocks() + } + // Handle state at time 2000 (myClip0 should stop (but is delayed due to outTransition)) + { + const state = getResolvedState(resolved, 2000) + await stateHandler.handleState(state, mappings) + // Give QuantelManager some time to process the commands + await sleep(10) + + expect(MOCK_SEND_COMMAND).toHaveBeenCalledTimes(2) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 1, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.CANCELWAITING, + portId: 'my_port', + }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 2, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.CLEARCLIP, + portId: 'my_port', + transition: { type: QuantelTransitionType.DELAY, delay: 500 }, + }), + }) + ) + + // Since the clear is delayed, we should not have sent any commands: + + expect(onRequest).toHaveBeenCalledTimes(0) + + clearMocks() + } + // Handle state at time 2100 (myClip1 starts playing) + { + const state = getResolvedState(resolved, 2100) + await stateHandler.handleState(state, mappings) + + // Wait enough time to ensure that the outTransition from previous clip would have finished (had it not been cancelled) + await sleep(500) + + expect(MOCK_SEND_COMMAND).toHaveBeenCalledTimes(3) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 1, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.CANCELWAITING, + portId: 'my_port', + }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 2, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.LOADCLIPFRAGMENTS, + portId: 'my_port', + clip: expect.objectContaining({ title: 'myClip1' }), + }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 3, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.PLAYCLIP, + portId: 'my_port', + clip: expect.objectContaining({ title: 'myClip1' }), + }), + }) + ) + // Start playing of next clip: + expect(onRequest).toHaveBeenCalledWith( + 'post', + expect.stringContaining('/1100/port/my_port/fragments?offset=2000') + ) + expect(onRequest).toHaveBeenCalledWith('put', expect.stringContaining('/1100/port/my_port/jump?offset=')) + expect(onRequest).toHaveBeenCalledWith('post', expect.stringContaining('/1100/port/my_port/trigger/JUMP')) + expect(onRequest).toHaveBeenCalledWith('post', expect.stringContaining('/1100/port/my_port/trigger/START')) + expect(onRequest).toHaveBeenCalledWith('get', expect.stringContaining('/1100/port/my_port')) + expect(onRequest).toHaveBeenCalledWith( + 'post', + expect.stringContaining('/1100/port/my_port/trigger/STOP?offset=3233') + ) + // The first clip should NOT have stopped, as it was delayed and cancelled: + expect(onRequest).not.toHaveBeenCalledWith('post', expect.stringMatching(/trigger\/STOP$/)) + + clearMocks() + } + await dev.terminate() + }) + test('outTransition to lookahead, cancel, then play another', async () => { + const dev = await getInitialisedQuantelDevice() + + // give it some time to finish the init + await sleep(10) + + const stateHandler = getNewStateHandler(dev) + + const timeline: TSRTimeline = [ + { + id: 'obj0', + enable: { + start: 1000, + end: 2000, + }, + content: { + deviceType: DeviceType.QUANTEL, + title: 'myClip0', + outTransition: { + type: QuantelTransitionType.DELAY, + delay: 500, // 2500 + }, + }, + layer: 'layer0', + }, + { + id: 'obj0_lookahead', + enable: { + start: '#obj0.end', + end: '#obj1.start', + }, + content: { + deviceType: DeviceType.QUANTEL, + title: 'myClip0', + // outTransition: { + // type: QuantelTransitionType.DELAY, + // delay: 1000, // 3000 + // }, + }, + layer: 'layer0_lookahead', + lookaheadForLayer: 'layer0', + isLookahead: true, + }, + { + id: 'obj1', + enable: { + start: 2100, + end: 10000, + }, + content: { + deviceType: DeviceType.QUANTEL, + title: 'myClip1', + }, + layer: 'layer0', + }, + ] + const mappings: Mappings = { + layer0: { + device: DeviceType.QUANTEL, + deviceId: 'quantel0', + options: { + mappingType: MappingQuantelType.Port, + portId: 'my_port', + channelId: 1, + }, + }, + } + const resolved = resolveTimeline(timeline, { + time: 0, + }) + + // Handle state at time 0 (nothing is playing) + { + const state = getResolvedState(resolved, 0) + + await stateHandler.handleState(state, mappings) + + // Give QuantelManager some time to process the commands + + await sleep(10) + + expect(MOCK_SEND_COMMAND).toHaveBeenCalledTimes(3) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 1, + expect.objectContaining({ + command: expect.objectContaining({ type: QuantelCommandType.CANCELWAITING, portId: 'my_port' }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 2, + expect.objectContaining({ + command: expect.objectContaining({ type: QuantelCommandType.SETUPPORT, portId: 'my_port', channel: 1 }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 3, + expect.objectContaining({ + command: expect.objectContaining({ type: QuantelCommandType.CLEARCLIP, portId: 'my_port' }), + }) + ) + + expect(onRequest).toHaveBeenCalledWith('post', 'http://localhost:3000/connect/myISA%3A8000') + expect(onRequest).toHaveBeenCalledWith('get', 'http://localhost:3000/default/server') + expect(onRequest).toHaveBeenCalledWith('get', 'http://localhost:3000/default/server/1100/port/my_port') + expect(onRequest).toHaveBeenCalledWith( + 'put', + 'http://localhost:3000/default/server/1100/port/my_port/channel/1' + ) + expect(onRequest).toHaveBeenCalledWith('post', 'http://localhost:3000/default/server/1100/port/my_port/reset') + clearMocks() + } + + // Handle state at time 1000 (myClip0 starts to play) + { + const state = getResolvedState(resolved, 1000) + await stateHandler.handleState(state, mappings) + // Give QuantelManager some time to process the commands + await sleep(100) + + expect(MOCK_SEND_COMMAND).toHaveBeenCalledTimes(3) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 1, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.CANCELWAITING, + portId: 'my_port', + }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 2, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.LOADCLIPFRAGMENTS, + portId: 'my_port', + clip: expect.objectContaining({ title: 'myClip0' }), + }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 3, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.PLAYCLIP, + portId: 'my_port', + clip: expect.objectContaining({ title: 'myClip0' }), + }), + }) + ) + expect(onRequest).toHaveBeenCalledWith('post', expect.stringContaining('/1100/port/my_port/fragments?offset=0')) + expect(onRequest).toHaveBeenCalledWith('put', expect.stringContaining('/1100/port/my_port/jump?offset=')) + expect(onRequest).toHaveBeenCalledWith('post', expect.stringContaining('/1100/port/my_port/trigger/JUMP')) + expect(onRequest).toHaveBeenCalledWith('post', expect.stringContaining('/1100/port/my_port/trigger/START')) + expect(onRequest).toHaveBeenCalledWith('get', expect.stringContaining('/1100/port/my_port')) + expect(onRequest).toHaveBeenCalledWith( + 'post', + expect.stringContaining('/1100/port/my_port/trigger/STOP?offset=1999') + ) + + clearMocks() + } + // Handle state at time 2000 (myClip0 should stop (but is delayed due to outTransition)) + { + const state = getResolvedState(resolved, 2000) + await stateHandler.handleState(state, mappings) + // Give QuantelManager some time to process the commands + await sleep(10) + + expect(MOCK_SEND_COMMAND).toHaveBeenCalledTimes(2) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 1, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.LOADCLIPFRAGMENTS, + portId: 'my_port', + clip: expect.objectContaining({ title: 'myClip0' }), + }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 2, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.PAUSECLIP, + portId: 'my_port', + clip: expect.objectContaining({ title: 'myClip0' }), + transition: { type: QuantelTransitionType.DELAY, delay: 500 }, + }), + }) + ) + + // Since the pause is delayed, we should not have sent any reset or stop commands: + + expect(onRequest).not.toHaveBeenCalledWith('post', expect.stringContaining('reset')) + expect(onRequest).not.toHaveBeenCalledWith('post', expect.stringContaining('trigger/STOP')) + + clearMocks() + } + // Handle state at time 2500 (myClip1 starts playing) + { + const state = getResolvedState(resolved, 2500) + await stateHandler.handleState(state, mappings) + + // Wait enough time to ensure that the outTransition from previous clip would have finished (had it not been cancelled) + await sleep(500) + + expect(MOCK_SEND_COMMAND).toHaveBeenCalledTimes(3) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 1, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.CANCELWAITING, + portId: 'my_port', + }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 2, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.LOADCLIPFRAGMENTS, + portId: 'my_port', + clip: expect.objectContaining({ title: 'myClip1' }), + }), + }) + ) + expect(MOCK_SEND_COMMAND).toHaveBeenNthCalledWith( + 3, + expect.objectContaining({ + command: expect.objectContaining({ + type: QuantelCommandType.PLAYCLIP, + portId: 'my_port', + clip: expect.objectContaining({ title: 'myClip1' }), + }), + }) + ) + + // Start playing of next clip: + expect(onRequest).toHaveBeenCalledWith( + 'post', + expect.stringContaining('/1100/port/my_port/fragments?offset=2000') + ) + expect(onRequest).toHaveBeenCalledWith('put', expect.stringContaining('/1100/port/my_port/jump?offset=')) + expect(onRequest).toHaveBeenCalledWith('post', expect.stringContaining('/1100/port/my_port/trigger/JUMP')) + expect(onRequest).toHaveBeenCalledWith('post', expect.stringContaining('/1100/port/my_port/trigger/START')) + expect(onRequest).toHaveBeenCalledWith('get', expect.stringContaining('/1100/port/my_port')) + expect(onRequest).toHaveBeenCalledWith( + 'post', + expect.stringContaining('/1100/port/my_port/trigger/STOP?offset=3233') + ) + // The first clip should NOT have stopped, as it was delayed and cancelled: + expect(onRequest).not.toHaveBeenCalledWith('post', expect.stringMatching(/trigger\/STOP$/)) + + clearMocks() + } + await dev.terminate() + }) + }) }) function createTimelineState( @@ -1324,3 +1991,7 @@ function createTimelineState( nextEvents: [], } } + +async function sleep(ms: number) { + return new Promise((resolve) => setTimeout(resolve, ms)) +} 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 74206c5ea..7b02aa7fc 100644 --- a/packages/timeline-state-resolver/src/integrations/quantel/__tests__/quantelGatewayMock.ts +++ b/packages/timeline-state-resolver/src/integrations/quantel/__tests__/quantelGatewayMock.ts @@ -311,7 +311,7 @@ async function handleRequest( 'delete /:zoneID/server/:serverID/port/:portID': async (params): Promise => { if (!quantelServer.ISAOptionHasBeenProvided) return noIsaSetupResponse - await sleep(100) + await sleep(1) const port = quantelServer.ports[params.portID] diff --git a/packages/timeline-state-resolver/src/integrations/quantel/connection.ts b/packages/timeline-state-resolver/src/integrations/quantel/connection.ts index 937fa8b4a..9e4ffc968 100644 --- a/packages/timeline-state-resolver/src/integrations/quantel/connection.ts +++ b/packages/timeline-state-resolver/src/integrations/quantel/connection.ts @@ -17,7 +17,7 @@ import { } from './types' import { WaitGroup } from '../../waitGroup' -const SOFT_JUMP_WAIT_TIME = 250 +let SOFT_JUMP_WAIT_TIME = 250 // Is a constant, but can be changed during unit tests const DEFAULT_FPS = 25 // frames per second const JUMP_ERROR_MARGIN = 10 // frames @@ -26,7 +26,15 @@ interface QuantelManagerOptions { /** If set: If a clip turns out to be on the wrong server, an attempt to copy the clip will be done. */ allowCloneClips?: boolean } -export class QuantelManager extends EventEmitter { + +interface QuantelManagerEvents { + info: [arg0: any] + warning: [arg0: any] + error: [err: any] + debug: [...arg: any] +} + +export class QuantelManager extends EventEmitter { private _quantelState: QuantelTrackedState = { port: {}, } @@ -47,7 +55,7 @@ export class QuantelManager extends EventEmitter { private options: QuantelManagerOptions ) { super() - this._quantel.on('error', (...args) => this.emit('error', ...args)) + this._quantel.on('error', (...args) => this.emit('error', args)) this._quantel.on('debug', (...args) => this.emit('debug', ...args)) } @@ -348,12 +356,20 @@ export class QuantelManager extends EventEmitter { private async prepareClipJump(cmd: QuantelCommandClip, alsoDoAction: 'play' | 'pause'): Promise { // Fetch tracked reference to the loaded clip: const trackedPort = this.getTrackedPort(cmd.portId) + const cmdStr = JSON.stringify(cmd) + this.emit('debug', `prepareClipJump: cmd=${cmdStr} ${alsoDoAction}`) + if (cmd.transition) { if (cmd.transition.type === QuantelTransitionType.DELAY) { + this.emit( + 'debug', + `prepareClipJump: cmd=${cmdStr} ${alsoDoAction}, waiting with port ${cmd.portId} for ${cmd.transition.delay}ms` + ) if (await this.waitWithPort(cmd.portId, cmd.transition.delay)) { // at this point, the wait aws aborted by someone else. Do nothing then. return } + this.emit('debug', `prepareClipJump: cmd=${cmdStr} ${alsoDoAction}, waiting done`) } } @@ -373,9 +389,9 @@ export class QuantelManager extends EventEmitter { ) this.emit( 'warning', - `prepareClipJump: cmd=${JSON.stringify( - cmd - )}: ${alsoDoAction}: clipId=${clipId}: jumpToOffset=${jumpToOffset}: trackedPort=${JSON.stringify(trackedPort)}` + `prepareClipJump: cmd=${cmdStr}: ${alsoDoAction}: clipId=${clipId}: jumpToOffset=${jumpToOffset}: trackedPort=${JSON.stringify( + trackedPort + )}` ) if ( (jumpToOffset === trackedPort.offset && trackedPort.playing === false) || // On request to play clip again, prepare jump // We're already there @@ -451,6 +467,7 @@ export class QuantelManager extends EventEmitter { if (alsoDoAction === 'play') { // Start playing: + this.emit('debug', `prepareClipJump: cmd=${cmdStr} ${alsoDoAction}, portPlay`) await this._quantel.portPlay(cmd.portId) await this.wait(60) @@ -496,6 +513,7 @@ export class QuantelManager extends EventEmitter { trackedPort.scheduledStop = loadedFragments.portOutPoint } } else if (alsoDoAction === 'pause' && trackedPort.playing) { + this.emit('debug', `prepareClipJump: cmd=${cmdStr} ${alsoDoAction}, portHardJump ${jumpToOffset}`) await this._quantel.portHardJump(cmd.portId, jumpToOffset) trackedPort.offset = jumpToOffset @@ -652,3 +670,10 @@ class Cache { }, 1) } } + +/** + * USED IN UNIT TESTS ONLY + */ +export function setSoftJumpWaitTime(time: number) { + SOFT_JUMP_WAIT_TIME = time +} diff --git a/packages/timeline-state-resolver/src/integrations/quantel/diff.ts b/packages/timeline-state-resolver/src/integrations/quantel/diff.ts index c83431ea7..641ec2c55 100644 --- a/packages/timeline-state-resolver/src/integrations/quantel/diff.ts +++ b/packages/timeline-state-resolver/src/integrations/quantel/diff.ts @@ -2,6 +2,7 @@ import { QuantelOutTransition } from 'timeline-state-resolver-types' import { QuantelCommandWithContext } from '.' import { QuantelCommand, QuantelCommandType, QuantelState, QuantelStatePort, QuantelStatePortClip } from './types' import _ = require('underscore') +import { ExecuteMode } from '../../service/device' const IDEAL_PREPARE_TIME = 1000 const PREPARE_TIME_WAIT = 50 @@ -104,6 +105,36 @@ export function diffStates( if (a.command.type !== QuantelCommandType.RELEASEPORT && b.command.type === QuantelCommandType.RELEASEPORT) return 1 return 0 }) + + // If we run any play-command, we will need to cancel any delayed/waiting out-transitions for that port: + const portIdsToCancelWaiting: Set = new Set() + for (const cmd of allCommands) { + if ( + (cmd.command.type === QuantelCommandType.PLAYCLIP || + cmd.command.type === QuantelCommandType.PAUSECLIP || + cmd.command.type === QuantelCommandType.CLEARCLIP || + cmd.command.type === QuantelCommandType.RELEASEPORT) && + !cmd.command.fromLookahead + ) { + // We should clear any delayed out-transitions for this port + portIdsToCancelWaiting.add(cmd.command.portId) + } + } + + for (const portId of portIdsToCancelWaiting.values()) { + allCommands.unshift({ + command: { + type: QuantelCommandType.CANCELWAITING, + portId: portId, + timelineObjId: '', + }, + timelineObjId: '', + context: 'Clear all delayed out-transitions', + // These must be SALVO, so that they are executed first thing, and not get stuck behind a delayed sequential command. + executeMode: ExecuteMode.SALVO, + }) + } + return allCommands } interface LookaheadPreloadClip { diff --git a/packages/timeline-state-resolver/src/integrations/quantel/index.ts b/packages/timeline-state-resolver/src/integrations/quantel/index.ts index e49225cec..bd8c7102f 100644 --- a/packages/timeline-state-resolver/src/integrations/quantel/index.ts +++ b/packages/timeline-state-resolver/src/integrations/quantel/index.ts @@ -150,6 +150,8 @@ export class QuantelDevice extends Device { + test('Sequential order', async () => { + // CommandExecutor + + const logger = { + debug: jest.fn((...args: any[]) => console.log('debug', ...args)), + info: jest.fn((info: string) => console.log('info', info)), + warn: jest.fn((warn: string) => console.log('warning', warn)), + error: jest.fn((context: string, e: Error) => console.log('error', context, e)), + } + const sendCommand = jest.fn() + + let sequence = 0 + + const executor = new CommandExecutor(logger, ExecuteMode.SEQUENTIAL, async (c) => { + await sleep(10) + sendCommand(c, sequence++) + }) + + await executor.executeCommands([makeCommand('a'), makeCommand('b'), makeCommand('c')]) + + expect(sendCommand).toHaveBeenCalledTimes(3) + expect(sendCommand).toHaveBeenNthCalledWith(1, makeCommand('a'), 0) + expect(sendCommand).toHaveBeenNthCalledWith(2, makeCommand('b'), 1) + expect(sendCommand).toHaveBeenNthCalledWith(3, makeCommand('c'), 2) + + sendCommand.mockClear() + sequence = 0 + + // If calling executeCommands several times synchronously, the commands should + // be executed in the order they are given: + + await Promise.all([ + executor.executeCommands([makeCommand('a'), makeCommand('b'), makeCommand('c')]), + executor.executeCommands([makeCommand('x'), makeCommand('y'), makeCommand('z')]), + ]) + + expect(sendCommand).toHaveBeenCalledTimes(6) + expect(sendCommand).toHaveBeenNthCalledWith(1, makeCommand('a'), 0) + expect(sendCommand).toHaveBeenNthCalledWith(2, makeCommand('b'), 1) + expect(sendCommand).toHaveBeenNthCalledWith(3, makeCommand('c'), 2) + expect(sendCommand).toHaveBeenNthCalledWith(4, makeCommand('x'), 3) + expect(sendCommand).toHaveBeenNthCalledWith(5, makeCommand('y'), 4) + expect(sendCommand).toHaveBeenNthCalledWith(6, makeCommand('z'), 5) + }) + test('Salvo order', async () => { + // CommandExecutor + + const logger = { + debug: jest.fn((...args: any[]) => console.log('debug', ...args)), + info: jest.fn((info: string) => console.log('info', info)), + warn: jest.fn((warn: string) => console.log('warning', warn)), + error: jest.fn((context: string, e: Error) => console.log('error', context, e)), + } + const sendCommandStart = jest.fn() + const sendCommandEnd = jest.fn() + + let sequence = 1 + + const executor = new CommandExecutor(logger, ExecuteMode.SALVO, async (c) => { + sendCommandStart(c, sequence++) + await sleep(10) + sendCommandEnd(c, sequence++) + }) + + await Promise.all([ + executor.executeCommands([makeCommand('a'), makeCommand('b'), makeCommand('c')]), + executor.executeCommands([makeCommand('d'), makeCommand('e'), makeCommand('f')]), + ]) + + expect(sendCommandStart).toHaveBeenCalledTimes(6) + expect(sendCommandEnd).toHaveBeenCalledTimes(6) + + // The Salvos in batch 1 are executed in parallel: + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('a'), 1) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('b'), 2) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('c'), 3) + + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('a'), 4) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('b'), 5) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('c'), 6) + + // The Salvos in batch 2 are executed in parallel: + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('d'), 7) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('e'), 8) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('f'), 9) + + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('d'), 10) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('e'), 11) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('f'), 12) + }) + test('Mixed order', async () => { + // CommandExecutor + + const logger = { + debug: jest.fn((...args: any[]) => console.log('debug', ...args)), + info: jest.fn((info: string) => console.log('info', info)), + warn: jest.fn((warn: string) => console.log('warning', warn)), + error: jest.fn((context: string, e: Error) => console.log('error', context, e)), + } + const sendCommandStart = jest.fn() + const sendCommandEnd = jest.fn() + + let sequence = 1 + + const executor = new CommandExecutor(logger, ExecuteMode.SEQUENTIAL, async (c) => { + sendCommandStart(c, sequence++) + await sleep(10) + sendCommandEnd(c, sequence++) + }) + + await Promise.all([ + executor.executeCommands([ + makeCommand('1_a1', ExecuteMode.SALVO), + makeCommand('1_b1', ExecuteMode.SEQUENTIAL), + makeCommand('1_a2', ExecuteMode.SALVO), + makeCommand('1_b2', ExecuteMode.SEQUENTIAL), + makeCommand('1_a3', ExecuteMode.SALVO), + makeCommand('1_b3', ExecuteMode.SEQUENTIAL), + ]), + + executor.executeCommands([ + makeCommand('2_a1', ExecuteMode.SALVO), + makeCommand('2_b1', ExecuteMode.SEQUENTIAL), + makeCommand('2_a2', ExecuteMode.SALVO), + makeCommand('2_b2', ExecuteMode.SEQUENTIAL), + makeCommand('2_a3', ExecuteMode.SALVO), + makeCommand('2_b3', ExecuteMode.SEQUENTIAL), + ]), + ]) + + expect(sendCommandStart).toHaveBeenCalledTimes(12) + expect(sendCommandEnd).toHaveBeenCalledTimes(12) + + // The Salvos in batch 1 are executed in parallel: + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('1_a1', ExecuteMode.SALVO), 1) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('1_a2', ExecuteMode.SALVO), 2) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('1_a3', ExecuteMode.SALVO), 3) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('1_a1', ExecuteMode.SALVO), 4) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('1_a2', ExecuteMode.SALVO), 5) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('1_a3', ExecuteMode.SALVO), 6) + + // The Sequentials in batch 1 are executed in order, but interleaved with the Salvos in batch 2: + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('1_b1', ExecuteMode.SEQUENTIAL), 7) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('2_a1', ExecuteMode.SALVO), 8) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('2_a2', ExecuteMode.SALVO), 9) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('2_a3', ExecuteMode.SALVO), 10) + + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('1_b1', ExecuteMode.SEQUENTIAL), 11) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('1_b2', ExecuteMode.SEQUENTIAL), 12) + + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('2_a1', ExecuteMode.SALVO), 13) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('2_a2', ExecuteMode.SALVO), 14) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('2_a3', ExecuteMode.SALVO), 15) + + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('1_b2', ExecuteMode.SEQUENTIAL), 16) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('1_b3', ExecuteMode.SEQUENTIAL), 17) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('1_b3', ExecuteMode.SEQUENTIAL), 18) + + // The Sequentials in batch 2 are executed in order: + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('2_b1', ExecuteMode.SEQUENTIAL), 19) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('2_b1', ExecuteMode.SEQUENTIAL), 20) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('2_b2', ExecuteMode.SEQUENTIAL), 21) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('2_b2', ExecuteMode.SEQUENTIAL), 22) + expect(sendCommandStart).toHaveBeenCalledWith(makeCommand('2_b3', ExecuteMode.SEQUENTIAL), 23) + expect(sendCommandEnd).toHaveBeenCalledWith(makeCommand('2_b3', ExecuteMode.SEQUENTIAL), 24) + + sendCommandStart.mockClear() + sequence = 0 + }) +}) + +function makeCommand(cmd: string, executeMode?: ExecuteMode) { + return { command: cmd, context: undefined, timelineObjId: '', executeMode } +} +async function sleep(time: number) { + return new Promise((resolve) => { + setTimeout(resolve, time) + }) +} diff --git a/packages/timeline-state-resolver/src/service/__tests__/stateHandler.spec.ts b/packages/timeline-state-resolver/src/service/__tests__/stateHandler.spec.ts index a33aba3e0..39b44d4b0 100644 --- a/packages/timeline-state-resolver/src/service/__tests__/stateHandler.spec.ts +++ b/packages/timeline-state-resolver/src/service/__tests__/stateHandler.spec.ts @@ -1,6 +1,7 @@ import { Timeline, TSRTimelineContent } from 'timeline-state-resolver-types' import { StateHandler } from '../stateHandler' import { MockTime } from '../../__tests__/mockTime' +import { ExecuteMode } from '../device' interface DeviceState { [prop: string]: { @@ -44,7 +45,7 @@ describe('stateHandler', () => { return new StateHandler( CONTEXT, { - executionType: 'salvo', + executionType: ExecuteMode.SALVO, }, { convertTimelineStateToDeviceState: (s) => s.layers as unknown as DeviceState, diff --git a/packages/timeline-state-resolver/src/service/commandExecutor.ts b/packages/timeline-state-resolver/src/service/commandExecutor.ts index ba3b03137..5d7198070 100644 --- a/packages/timeline-state-resolver/src/service/commandExecutor.ts +++ b/packages/timeline-state-resolver/src/service/commandExecutor.ts @@ -1,14 +1,18 @@ import * as _ from 'underscore' -import { BaseDeviceAPI, CommandWithContext } from './device' +import { BaseDeviceAPI, CommandWithContext, ExecuteMode } from './device' import { Measurement } from './measure' import { StateHandlerContext } from './stateHandler' +import PQueue from 'p-queue' const wait = async (t: number) => new Promise((r) => setTimeout(() => r(), t)) export class CommandExecutor { + private commandQueueSalvo = new PQueue({ concurrency: 1 }) + private commandQueueSequential = new PQueue({ concurrency: 1 }) + constructor( private logger: StateHandlerContext['logger'], - private mode: 'salvo' | 'sequential', + private defaultMode: ExecuteMode, private sendCommand: BaseDeviceAPI['sendCommand'] ) {} @@ -18,11 +22,44 @@ export class CommandExecutor { commands.sort((a, b) => (b.preliminary ?? 0) - (a.preliminary ?? 0)) const totalTime = commands[0].preliminary ?? 0 - if (this.mode === 'salvo') { - return this._executeCommandsSalvo(totalTime, commands, measurement) - } else { - return this._executeCommandsSequential(totalTime, commands, measurement) + const salvoCommands: Command[] = [] + const sequentialCommands: Command[] = [] + + for (const command of commands) { + const mode = command.executeMode || this.defaultMode + if (mode === ExecuteMode.SEQUENTIAL) { + sequentialCommands.push(command) + } else { + salvoCommands.push(command) + } } + + // The execution logic is as follows: + // * When the commands sent into executeCommands() is called a BATCH. + // + // * Salvos should wait for Salvos from previous BATCH to have finished before executing + // * Salvos within the same BATCH are executed in parallel. + // + // * Sequentials should wait for all Salvos and Sequentials from previous BATCH to have finished before executing + // * Sequentials should be executed after all Salvos from the same BATCH have been executed + // * Sequentials within the same BATCH are executed in order + + let pSequential: Promise | null = null + + // Wait for Salvos from previous batch to finish: + await this.commandQueueSalvo.add(async () => { + await this._executeCommandsSalvo(totalTime, salvoCommands, measurement) + + // Salvos should not have to wait for Sequentials to finish, so we don't await the pSequential here: + // eslint-disable-next-line @typescript-eslint/await-thenable + pSequential = this.commandQueueSequential.add(async () => { + await this._executeCommandsSequential(totalTime, sequentialCommands, measurement) + }) + }) + + if (!pSequential) throw new Error('Internal error in CommandExecutor: pSequential not set') + // eslint-disable-next-line @typescript-eslint/await-thenable + await pSequential } private async _executeCommandsSalvo( diff --git a/packages/timeline-state-resolver/src/service/device.ts b/packages/timeline-state-resolver/src/service/device.ts index 1d01a185f..79888fd7b 100644 --- a/packages/timeline-state-resolver/src/service/device.ts +++ b/packages/timeline-state-resolver/src/service/device.ts @@ -18,8 +18,15 @@ export type CommandWithContext = { timelineObjId: string /** this command is to be executed x ms _before_ the scheduled time */ preliminary?: number - /** commands with different queueId's can be executed in parallel in sequential mode */ + /** sequential mode only: commands with different queueId's can be executed in parallel */ queueId?: string + /** If the command should be executed in sequential order or as a salvo. If not set, falls back to device default */ + executeMode?: ExecuteMode +} + +export enum ExecuteMode { + SALVO = 'salvo', + SEQUENTIAL = 'sequential', } /** diff --git a/packages/timeline-state-resolver/src/service/devices.ts b/packages/timeline-state-resolver/src/service/devices.ts index ac5eecff5..e4a8f3259 100644 --- a/packages/timeline-state-resolver/src/service/devices.ts +++ b/packages/timeline-state-resolver/src/service/devices.ts @@ -1,6 +1,6 @@ import { OscDevice } from '../integrations/osc' import { DeviceType } from 'timeline-state-resolver-types' -import { Device, DeviceContextAPI } from './device' +import { Device, DeviceContextAPI, ExecuteMode } from './device' import { AuthenticatedHTTPSendDevice } from '../integrations/httpSend/AuthenticatedHTTPSendDevice' import { ShotokuDevice } from '../integrations/shotoku' import { HTTPWatcherDevice } from '../integrations/httpWatcher' @@ -23,7 +23,7 @@ export interface DeviceEntry { deviceClass: new (context: DeviceContextAPI) => Device canConnect: boolean deviceName: (deviceId: string, options: any) => string - executionMode: (options: any) => 'salvo' | 'sequential' + executionMode: (options: any) => ExecuteMode } export type ImplementedServiceDeviceTypes = @@ -52,108 +52,108 @@ export const DevicesDict: Record = { deviceClass: AbstractDevice, canConnect: false, deviceName: (deviceId: string) => 'Abstract ' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.ATEM]: { deviceClass: AtemDevice, canConnect: true, deviceName: (deviceId: string) => 'Atem ' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.HTTPSEND]: { deviceClass: AuthenticatedHTTPSendDevice, canConnect: false, deviceName: (deviceId: string) => 'HTTPSend ' + deviceId, - executionMode: () => 'sequential', // todo - config? + executionMode: () => ExecuteMode.SEQUENTIAL, // todo - config? }, [DeviceType.HTTPWATCHER]: { deviceClass: HTTPWatcherDevice, canConnect: false, deviceName: (deviceId: string) => 'HTTP-Watch ' + deviceId, - executionMode: () => 'sequential', + executionMode: () => ExecuteMode.SEQUENTIAL, }, [DeviceType.HYPERDECK]: { deviceClass: HyperdeckDevice, canConnect: true, deviceName: (deviceId: string) => 'Hyperdeck ' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.LAWO]: { deviceClass: LawoDevice, canConnect: true, deviceName: (deviceId: string) => 'Lawo ' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.OBS]: { deviceClass: OBSDevice, canConnect: true, deviceName: (deviceId: string) => 'OBS ' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.OSC]: { deviceClass: OscDevice, canConnect: true, deviceName: (deviceId: string) => 'OSC ' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.MULTI_OSC]: { deviceClass: MultiOSCMessageDevice, canConnect: false, deviceName: (deviceId: string) => 'MultiOSC ' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.PANASONIC_PTZ]: { deviceClass: PanasonicPtzDevice, canConnect: true, deviceName: (deviceId: string) => 'Panasonic PTZ ' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.PHAROS]: { deviceClass: PharosDevice, canConnect: true, deviceName: (deviceId: string) => 'Pharos ' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.SOFIE_CHEF]: { deviceClass: SofieChefDevice, canConnect: true, deviceName: (deviceId: string) => 'SofieChef ' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.SHOTOKU]: { deviceClass: ShotokuDevice, canConnect: true, deviceName: (deviceId: string) => 'SHOTOKU' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.SINGULAR_LIVE]: { deviceClass: SingularLiveDevice, canConnect: false, deviceName: (deviceId: string) => 'Singular.Live ' + deviceId, - executionMode: () => 'sequential', + executionMode: () => ExecuteMode.SEQUENTIAL, }, [DeviceType.TCPSEND]: { deviceClass: TcpSendDevice, canConnect: true, deviceName: (deviceId: string) => 'TCP' + deviceId, - executionMode: () => 'sequential', // todo: should this be configurable? + executionMode: () => ExecuteMode.SEQUENTIAL, // todo: should this be configurable? }, [DeviceType.TELEMETRICS]: { deviceClass: TelemetricsDevice, canConnect: true, deviceName: (deviceId: string) => 'Telemetrics ' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.TRICASTER]: { deviceClass: TriCasterDevice, canConnect: true, deviceName: (deviceId: string) => 'TriCaster ' + deviceId, - executionMode: () => 'salvo', + executionMode: () => ExecuteMode.SALVO, }, [DeviceType.QUANTEL]: { deviceClass: QuantelDevice, canConnect: true, deviceName: (deviceId: string) => 'Quantel' + deviceId, - executionMode: () => 'sequential', + executionMode: () => ExecuteMode.SEQUENTIAL, }, } diff --git a/packages/timeline-state-resolver/src/service/stateHandler.ts b/packages/timeline-state-resolver/src/service/stateHandler.ts index 586051575..9951185c4 100644 --- a/packages/timeline-state-resolver/src/service/stateHandler.ts +++ b/packages/timeline-state-resolver/src/service/stateHandler.ts @@ -1,6 +1,6 @@ import { FinishedTrace, startTrace, endTrace } from '../lib' import { Mappings, Timeline, TSRTimelineContent } from 'timeline-state-resolver-types' -import { BaseDeviceAPI, CommandWithContext } from './device' +import { BaseDeviceAPI, CommandWithContext, ExecuteMode } from './device' import { Measurement, StateChangeReport } from './measure' import { CommandExecutor } from './commandExecutor' @@ -228,7 +228,7 @@ export class StateHandler { } export interface StateHandlerConfig { - executionType: 'salvo' | 'sequential' + executionType: ExecuteMode } export interface StateHandlerContext {