1// SPDX-License-Identifier: GPL-2.0-only
2// Copyright (c) 2026-present FERS Contributors (see AUTHORS.md).
4import { describe, expect, test } from 'bun:test';
5import { defaultGlobalParameters } from './defaults';
6import { buildHydratedScenarioState, parseScenarioData } from './hydration';
7import { ScenarioData, ScenarioState } from './types';
9function createScenarioData(): ScenarioData {
12 ...defaultGlobalParameters,
25 interpolation: 'static',
61function createState(overrides: Partial<ScenarioState> = {}): ScenarioState {
63 globalParameters: defaultGlobalParameters,
69 selectedComponentId: null,
73 targetPlaybackDuration: null,
74 isBackendSyncing: false,
76 scenarioFilePath: null,
77 outputDirectory: null,
78 antennaPreviewErrors: {},
79 notificationSnackbar: {
84 notificationQueue: [],
85 viewControlAction: { type: null, timestamp: 0 },
92 showLinkMonostatic: true,
93 showLinkIlluminator: true,
94 showLinkScattered: true,
98 showPlatformLabels: true,
99 showMotionPaths: true,
105describe('buildHydratedScenarioState', () => {
106 test('preserves selection and clamps current time during backend recovery', () => {
107 const hydrated = buildHydratedScenarioState(
109 selectedItemId: 'platform-1',
110 selectedComponentId: 'component-1',
113 createScenarioData(),
116 preserveSelection: true,
117 preserveCurrentTime: true,
121 expect(hydrated.selectedItemId).toBe('platform-1');
122 expect(hydrated.selectedComponentId).toBe('component-1');
123 expect(hydrated.currentTime).toBe(20);
124 expect(hydrated.isDirty).toBe(false);
127 test('clears stale selection and resets time for import-style loads', () => {
128 const hydrated = buildHydratedScenarioState(
130 selectedItemId: 'missing-platform',
131 selectedComponentId: 'missing-component',
134 createScenarioData(),
140 expect(hydrated.selectedItemId).toBeNull();
141 expect(hydrated.selectedComponentId).toBeNull();
142 expect(hydrated.currentTime).toBe(5);
143 expect(hydrated.isDirty).toBe(true);
147describe('parseScenarioData FMCW hydration', () => {
148 test('hydrates FMCW waveform and component mode from backend data', () => {
149 const scenario = parseScenarioData({
150 name: 'FMCW Scenario',
155 name: 'FMCW Down Chirp',
157 carrier_frequency: 10e9,
160 chirp_bandwidth: 20e6,
161 chirp_duration: 250e-6,
162 chirp_period: 300e-6,
163 start_frequency_offset: -10e6,
169 name: 'FMCW Triangle',
171 carrier_frequency: 10e9,
173 chirp_bandwidth: 20e6,
174 chirp_duration: 250e-6,
175 start_frequency_offset: -10e6,
185 name: 'Radar Platform',
187 interpolation: 'static',
189 { x: 0, y: 0, altitude: 0, time: 0 },
204 dechirp_mode: 'ideal',
207 waveform_name: 'FMCW Down Chirp',
210 if_filter_bandwidth: 4e5,
211 if_filter_transition_width: 1e5,
220 expect(scenario).not.toBeNull();
221 expect(scenario?.waveforms[0]).toMatchObject({
223 waveformType: 'fmcw_linear_chirp',
225 chirp_bandwidth: 20e6,
226 chirp_duration: 250e-6,
227 chirp_period: 300e-6,
228 start_frequency_offset: -10e6,
231 expect(scenario?.waveforms[1]).toMatchObject({
233 waveformType: 'fmcw_triangle',
234 chirp_bandwidth: 20e6,
235 chirp_duration: 250e-6,
236 start_frequency_offset: -10e6,
239 expect(scenario?.platforms[0].components[0]).toMatchObject({
244 dechirp_mode: 'ideal',
247 waveform_name: 'FMCW Down Chirp',
250 if_filter_bandwidth: 4e5,
251 if_filter_transition_width: 1e5,
256 test('hydrates receiver and monostatic dechirp references by backend names', () => {
257 const scenario = parseScenarioData({
258 name: 'FMCW Dechirp Scenario',
265 carrier_frequency: 10e9,
268 chirp_bandwidth: 20e6,
269 chirp_duration: 250e-6,
270 chirp_period: 300e-6,
279 name: 'Radar Platform',
281 interpolation: 'static',
283 { x: 0, y: 0, altitude: 0, time: 0 },
296 name: 'Reference TX',
306 dechirp_mode: 'physical',
308 source: 'transmitter',
309 transmitter_name: 'Reference TX',
322 dechirp_mode: 'ideal',
334 expect(scenario).not.toBeNull();
335 expect(scenario?.platforms[0].components[1]).toMatchObject({
340 dechirp_mode: 'physical',
342 source: 'transmitter',
343 transmitter_name: 'Reference TX',
347 expect(scenario?.platforms[0].components[2]).toMatchObject({
354 dechirp_mode: 'ideal',