diff --git a/lib/ops/report-store-ops.js b/lib/ops/report-store-ops.js new file mode 100644 --- /dev/null +++ b/lib/ops/report-store-ops.js @@ -0,0 +1,49 @@ +// @flow + +import type { ClientReportCreationRequest } from '../types/report-types.js'; + +export type ReplaceQueuedReportOperation = { + +type: 'replace_report', + +payload: { report: ClientReportCreationRequest }, +}; + +export type RemoveQueuedReportsOperation = { + +type: 'remove_reports', + +payload: { +ids: $ReadOnlyArray }, +}; + +export type RemoveAllQueuedReportsOperation = { + +type: 'remove_all_reports', +}; + +export type ReportStoreOperation = + | ReplaceQueuedReportOperation + | RemoveQueuedReportsOperation + | RemoveAllQueuedReportsOperation; + +function processReportStoreOperations( + queuedReports: $ReadOnlyArray, + reportStoreOps: $ReadOnlyArray, +): $ReadOnlyArray { + if (reportStoreOps.length === 0) { + return queuedReports; + } + let processedReports = [...queuedReports]; + for (const operation of reportStoreOps) { + if (operation.type === 'replace_report') { + const filteredReports = processedReports.filter( + report => report.id !== operation.payload.report.id, + ); + processedReports = [...filteredReports, { ...operation.payload.report }]; + } else if (operation.type === 'remove_reports') { + processedReports = processedReports.filter( + report => !operation.payload.ids.includes(report.id), + ); + } else if (operation.type === 'remove_all_reports') { + processedReports = []; + } + } + return processedReports; +} + +export { processReportStoreOperations }; diff --git a/lib/ops/report-store-ops.test.js b/lib/ops/report-store-ops.test.js new file mode 100644 --- /dev/null +++ b/lib/ops/report-store-ops.test.js @@ -0,0 +1,180 @@ +// @flow + +import { processReportStoreOperations } from './report-store-ops.js'; +import { + type ClientReportCreationRequest, + reportTypes, +} from '../types/report-types.js'; + +const defaultReport = { + type: reportTypes.MEDIA_MISSION, + time: Date.now(), + mediaMission: { + steps: [], + result: { success: true }, + totalTime: 0, + userTime: 0, + }, +}; + +const defaultMockReportWeb = { + platformDetails: { platform: 'web' }, + ...defaultReport, +}; + +const defaultMockReportIos = { + platformDetails: { platform: 'ios' }, + ...defaultReport, +}; + +const mockReports: $ReadOnlyArray = [ + { + ...defaultMockReportWeb, + id: '1', + }, + { + ...defaultMockReportWeb, + id: '2', + }, + { + ...defaultMockReportWeb, + id: '3', + }, +]; + +const sortReports = ( + queuedReports: $ReadOnlyArray, +): $ReadOnlyArray => { + return [...queuedReports].sort( + (a: ClientReportCreationRequest, b: ClientReportCreationRequest) => + a.id.localeCompare(b.id), + ); +}; + +describe('processReportStoreOperations', () => { + it('should return the original reports if no operations are processed', () => { + const reportStoreOps = []; + const processedReports = processReportStoreOperations( + mockReports, + reportStoreOps, + ); + expect(processedReports).toEqual(mockReports); + }); + + it('should replace the report with the given id', () => { + const reportStoreOps = [ + { + type: 'replace_report', + payload: { + report: { + ...defaultMockReportIos, + id: '2', + }, + }, + }, + ]; + const expectedReports = [ + { + ...defaultMockReportWeb, + id: '1', + }, + { + ...defaultMockReportIos, + id: '2', + }, + { + ...defaultMockReportWeb, + id: '3', + }, + ]; + const processedReports = processReportStoreOperations( + mockReports, + reportStoreOps, + ); + + expect(sortReports(processedReports)).toEqual(sortReports(expectedReports)); + }); + + it('should handle an empty reports with replace operation', () => { + const reportStoreOps = [ + { + type: 'replace_report', + payload: { + report: { + ...defaultMockReportWeb, + id: '1', + }, + }, + }, + ]; + const expectedReports = [ + { + ...defaultMockReportWeb, + id: '1', + }, + ]; + + const processedReports = processReportStoreOperations([], reportStoreOps); + expect(processedReports).toEqual(expectedReports); + }); + + it('should remove reports with given ids', () => { + const reportStoreOps = [ + { + type: 'remove_reports', + payload: { + ids: ['1', '2'], + }, + }, + ]; + const expectedReports = [ + { + ...defaultMockReportWeb, + id: '3', + }, + ]; + const processedReports = processReportStoreOperations( + mockReports, + reportStoreOps, + ); + expect(processedReports).toEqual(expectedReports); + }); + + it('should handle empty reports with remove_reports operation', () => { + const reportStoreOps = [ + { + type: 'remove_reports', + payload: { + ids: ['1', '2'], + }, + }, + ]; + + const processedReports = processReportStoreOperations([], reportStoreOps); + expect(processedReports).toEqual([]); + }); + + it('should remove all reports', () => { + const reportStoreOps = [ + { + type: 'remove_all_reports', + }, + ]; + const processedReports = processReportStoreOperations( + mockReports, + reportStoreOps, + ); + expect(processedReports).toEqual([]); + }); + + it('should handle empty reports with remove_all_reports operation', () => { + const reportStoreOps = [ + { + type: 'remove_all_reports', + }, + ]; + + const processedReports = processReportStoreOperations([], reportStoreOps); + expect(processedReports).toEqual([]); + }); +});