UNPKG

@signalwire/realtime-api

Version:
208 lines (175 loc) 6.08 kB
import { EventEmitter } from '@signalwire/core' import { createClient } from '../../client/createClient' import { CallRecording } from './CallRecording' import { Call } from '../Call' import { Voice } from '../Voice' import { decorateRecordingPromise, methods, getters, } from './decorateRecordingPromise' describe('CallRecording', () => { let voice: Voice let call: Call let callRecording: CallRecording const userOptions = { host: 'example.com', project: 'example.project', token: 'example.token', } const swClientMock = { userOptions, client: createClient(userOptions), } beforeEach(() => { // @ts-expect-error voice = new Voice(swClientMock) call = new Call({ voice }) callRecording = new CallRecording({ call, // @ts-expect-error payload: { control_id: 'test_control_id', call_id: 'test_call_id', node_id: 'test_node_id', }, }) // @ts-expect-error callRecording._client.execute = jest.fn() }) it('should have an event emitter', () => { expect(callRecording['emitter']).toBeInstanceOf(EventEmitter) }) it('should declare the correct event map', () => { const expectedEventMap = { onStarted: 'recording.started', onUpdated: 'recording.updated', onFailed: 'recording.failed', onEnded: 'recording.ended', } expect(callRecording['_eventMap']).toEqual(expectedEventMap) }) it('should attach all listeners', () => { callRecording = new CallRecording({ call, // @ts-expect-error payload: {}, listeners: { onStarted: () => {}, onUpdated: () => {}, onFailed: () => {}, onEnded: () => {}, }, }) // @ts-expect-error expect(callRecording.emitter.eventNames()).toStrictEqual([ 'recording.started', 'recording.updated', 'recording.failed', 'recording.ended', ]) }) it('should control an active playback', async () => { const baseExecuteParams = { method: '', params: { control_id: 'test_control_id', call_id: 'test_call_id', node_id: 'test_node_id', }, } await callRecording.pause() // @ts-expect-error expect(callRecording._client.execute).toHaveBeenLastCalledWith({ method: 'calling.record.pause', params: { ...baseExecuteParams.params, behavior: 'silence', }, }) await callRecording.pause({ behavior: 'skip' }) // @ts-expect-error expect(callRecording._client.execute).toHaveBeenLastCalledWith({ method: 'calling.record.pause', params: { ...baseExecuteParams.params, behavior: 'skip', }, }) await callRecording.resume() // @ts-expect-error expect(callRecording._client.execute).toHaveBeenLastCalledWith({ ...baseExecuteParams, method: 'calling.record.resume', }) await callRecording.stop() // @ts-expect-error expect(callRecording._client.execute).toHaveBeenLastCalledWith({ ...baseExecuteParams, method: 'calling.record.stop', }) }) it('should throw an error on methods if recording has ended', async () => { // @ts-expect-error callRecording.setPayload({ control_id: 'test_control_id', call_id: 'test_call_id', node_id: 'test_node_id', state: 'finished', }) await expect(callRecording.pause()).rejects.toThrowError('Action has ended') await expect(callRecording.resume()).rejects.toThrowError( 'Action has ended' ) await expect(callRecording.stop()).rejects.toThrowError('Action has ended') }) describe('decorateRecordingPromise', () => { it('expose correct properties before resolve', () => { const innerPromise = Promise.resolve(callRecording) const decoratedPromise = decorateRecordingPromise.call(call, innerPromise) expect(decoratedPromise).toHaveProperty('onStarted', expect.any(Function)) expect(decoratedPromise.onStarted()).toBeInstanceOf(Promise) expect(decoratedPromise).toHaveProperty('onEnded', expect.any(Function)) expect(decoratedPromise.onEnded()).toBeInstanceOf(Promise) methods.forEach((method) => { expect(decoratedPromise).toHaveProperty(method, expect.any(Function)) // @ts-expect-error expect(decoratedPromise[method]()).toBeInstanceOf(Promise) }) getters.forEach((getter) => { expect(decoratedPromise).toHaveProperty(getter) // @ts-expect-error expect(decoratedPromise[getter]).toBeInstanceOf(Promise) }) }) it('expose correct properties after resolve', async () => { const innerPromise = Promise.resolve(callRecording) const decoratedPromise = decorateRecordingPromise.call(call, innerPromise) // Simulate the recording ended event call.emit('recording.ended', callRecording) const ended = await decoratedPromise expect(ended).not.toHaveProperty('onStarted', expect.any(Function)) expect(ended).not.toHaveProperty('onEnded', expect.any(Function)) methods.forEach((method) => { expect(ended).toHaveProperty(method, expect.any(Function)) }) getters.forEach((getter) => { expect(ended).toHaveProperty(getter) // @ts-expect-error expect(ended[getter]).not.toBeInstanceOf(Promise) }) }) it('resolves when recording ends', async () => { const innerPromise = Promise.resolve(callRecording) const decoratedPromise = decorateRecordingPromise.call(call, innerPromise) // Simulate the recording ended event call.emit('recording.ended', callRecording) await expect(decoratedPromise).resolves.toEqual(expect.any(CallRecording)) }) it('rejects on inner promise rejection', async () => { const innerPromise = Promise.reject(new Error('Recording failed')) const decoratedPromise = decorateRecordingPromise.call(call, innerPromise) await expect(decoratedPromise).rejects.toThrow('Recording failed') }) }) })