@testgorilla/tgo-coding-test 1.0.0 → 2.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (102) hide show
  1. package/fesm2022/testgorilla-tgo-coding-test.mjs +94 -94
  2. package/fesm2022/testgorilla-tgo-coding-test.mjs.map +1 -1
  3. package/lib/components/configurations/configurations.component.d.ts +1 -1
  4. package/package.json +15 -17
  5. package/esm2022/index.mjs +0 -12
  6. package/esm2022/lib/components/code-editor/code-editor.component.mjs +0 -335
  7. package/esm2022/lib/components/code-editor/code-editor.service.mjs +0 -72
  8. package/esm2022/lib/components/code-editor/helpers/c-helper.mjs +0 -40
  9. package/esm2022/lib/components/code-editor/helpers/code-editor-helper.base.mjs +0 -11
  10. package/esm2022/lib/components/code-editor/helpers/code-editor-helper.model.mjs +0 -2
  11. package/esm2022/lib/components/code-editor/helpers/cpp-helper.mjs +0 -41
  12. package/esm2022/lib/components/code-editor/helpers/csharp-helper.mjs +0 -42
  13. package/esm2022/lib/components/code-editor/helpers/go-helper.mjs +0 -42
  14. package/esm2022/lib/components/code-editor/helpers/index.mjs +0 -16
  15. package/esm2022/lib/components/code-editor/helpers/java-helper.mjs +0 -42
  16. package/esm2022/lib/components/code-editor/helpers/javascript-helper.mjs +0 -26
  17. package/esm2022/lib/components/code-editor/helpers/kotlin-helper.mjs +0 -42
  18. package/esm2022/lib/components/code-editor/helpers/php-helper.mjs +0 -26
  19. package/esm2022/lib/components/code-editor/helpers/python-helper.mjs +0 -26
  20. package/esm2022/lib/components/code-editor/helpers/r-helper.mjs +0 -26
  21. package/esm2022/lib/components/code-editor/helpers/ruby-helper.mjs +0 -26
  22. package/esm2022/lib/components/code-editor/helpers/scala-helper.mjs +0 -41
  23. package/esm2022/lib/components/code-editor/helpers/sql-helper.mjs +0 -34
  24. package/esm2022/lib/components/code-editor/helpers/swift-helper.mjs +0 -40
  25. package/esm2022/lib/components/code-editor/helpers/typescript-helper.mjs +0 -41
  26. package/esm2022/lib/components/code-editor/models/code-editor.model.mjs +0 -2
  27. package/esm2022/lib/components/code-editor/models/coding-snapshot.model.mjs +0 -2
  28. package/esm2022/lib/components/coding-question/coding-question.component.mjs +0 -126
  29. package/esm2022/lib/components/coding-section/coding-section.component.mjs +0 -188
  30. package/esm2022/lib/components/common/truncated-text/truncated-text.component.mjs +0 -38
  31. package/esm2022/lib/components/configurations/configurations.component.mjs +0 -97
  32. package/esm2022/lib/components/instructions/instructions.component.mjs +0 -139
  33. package/esm2022/lib/components/panel/panel.component.mjs +0 -34
  34. package/esm2022/lib/components/runnable-editor/runnable-editor.component.mjs +0 -169
  35. package/esm2022/lib/components/tests/test-cases/test-cases.component.mjs +0 -198
  36. package/esm2022/lib/components/tests/test-cases-content/test-cases-content.component.mjs +0 -96
  37. package/esm2022/lib/components/tests/test-cases-status/test-cases-status.component.mjs +0 -21
  38. package/esm2022/lib/components/tests/test-results.component.mjs +0 -127
  39. package/esm2022/lib/components/tgo-coding-test/tgo-coding-test.component.mjs +0 -280
  40. package/esm2022/lib/components/tgo-coding-test-candidate-view/tgo-coding-test-candidate-view.component.mjs +0 -476
  41. package/esm2022/lib/config/index.mjs +0 -2
  42. package/esm2022/lib/config/tgo-coding-test.config.mjs +0 -2
  43. package/esm2022/lib/config/tgo-coding-test.provider.mjs +0 -34
  44. package/esm2022/lib/config/tgo-coding-test.token.mjs +0 -14
  45. package/esm2022/lib/models/auto-saved-data.mjs +0 -2
  46. package/esm2022/lib/models/code-event.mjs +0 -2
  47. package/esm2022/lib/models/coderunner-execution-results.mjs +0 -2
  48. package/esm2022/lib/models/configs.mjs +0 -2
  49. package/esm2022/lib/models/language-change-action.mjs +0 -2
  50. package/esm2022/lib/models/lat-languages.mjs +0 -3
  51. package/esm2022/lib/models/mixpanel-events.mjs +0 -2
  52. package/esm2022/lib/models/mode.mjs +0 -2
  53. package/esm2022/lib/models/paste-data.mjs +0 -2
  54. package/esm2022/lib/models/programming-language.mjs +0 -2
  55. package/esm2022/lib/models/test-cases.mjs +0 -7
  56. package/esm2022/lib/models/theme.mjs +0 -2
  57. package/esm2022/lib/models/translations.mjs +0 -2
  58. package/esm2022/lib/models/view-mode.mjs +0 -8
  59. package/esm2022/lib/pipes/memoize-func.pipe.mjs +0 -39
  60. package/esm2022/lib/services/candidate-coding-test-services/candidature-api.service.mjs +0 -19
  61. package/esm2022/lib/services/candidate-coding-test-services/coderunner-api.service.mjs +0 -58
  62. package/esm2022/lib/services/candidate-coding-test-services/coding-test-tour.service.mjs +0 -89
  63. package/esm2022/lib/services/candidate-coding-test-services/coding-test.service.mjs +0 -490
  64. package/esm2022/lib/services/candidate-coding-test-services/index.mjs +0 -5
  65. package/esm2022/lib/services/coding-test-config.service.mjs +0 -51
  66. package/esm2022/lib/services/configurations.service.mjs +0 -89
  67. package/esm2022/lib/services/lib-coding-test.service.mjs +0 -106
  68. package/esm2022/lib/services/storage.service.mjs +0 -624
  69. package/esm2022/lib/services/test-cases.service.mjs +0 -30
  70. package/esm2022/lib/services/theme.service.mjs +0 -36
  71. package/esm2022/lib/utils/additional-languages/erlang.mjs +0 -103
  72. package/esm2022/lib/utils/resize-element.mjs +0 -13
  73. package/esm2022/lib/utils/time-to-ms.util.mjs +0 -11
  74. package/esm2022/shared/index.mjs +0 -5
  75. package/esm2022/shared/lib/components/audio-animation/audio-animation.component.mjs +0 -114
  76. package/esm2022/shared/lib/components/audio-animation/index.mjs +0 -2
  77. package/esm2022/shared/lib/components/index.mjs +0 -3
  78. package/esm2022/shared/lib/components/vimeo-video/index.mjs +0 -2
  79. package/esm2022/shared/lib/components/vimeo-video/vimeo-video.component.mjs +0 -101
  80. package/esm2022/shared/lib/models/answer.mjs +0 -2
  81. package/esm2022/shared/lib/models/assessment.mjs +0 -2
  82. package/esm2022/shared/lib/models/environment.mjs +0 -2
  83. package/esm2022/shared/lib/models/index.mjs +0 -9
  84. package/esm2022/shared/lib/models/question-component.mjs +0 -2
  85. package/esm2022/shared/lib/models/question.mjs +0 -2
  86. package/esm2022/shared/lib/models/test.mjs +0 -2
  87. package/esm2022/shared/lib/models/translations.mjs +0 -2
  88. package/esm2022/shared/lib/models/window.mjs +0 -2
  89. package/esm2022/shared/lib/services/api/api.service.mjs +0 -97
  90. package/esm2022/shared/lib/services/api/mocked-api.service.mjs +0 -131
  91. package/esm2022/shared/lib/services/environment/environment.service.mjs +0 -13
  92. package/esm2022/shared/lib/services/index.mjs +0 -10
  93. package/esm2022/shared/lib/services/localization/languages.model.mjs +0 -19
  94. package/esm2022/shared/lib/services/localization/transloco-lazy-module-utils.mjs +0 -27
  95. package/esm2022/shared/lib/services/localization/transloco-testing.module.mjs +0 -11
  96. package/esm2022/shared/lib/services/media/media.service.mjs +0 -129
  97. package/esm2022/shared/lib/services/mixpanel/mixpanel.service.mjs +0 -30
  98. package/esm2022/shared/lib/services/theme/theme.service.mjs +0 -24
  99. package/esm2022/shared/test-mocks/assessment-test.mock.mjs +0 -112
  100. package/esm2022/shared/test-mocks/index.mjs +0 -3
  101. package/esm2022/shared/test-mocks/tgo-ui.mock.mjs +0 -39
  102. package/esm2022/testgorilla-tgo-coding-test.mjs +0 -5
@@ -1,624 +0,0 @@
1
- import { Injectable } from '@angular/core';
2
- import { LocalStorageService } from 'ngx-webstorage';
3
- import { filter } from 'rxjs';
4
- import { AssessmentIdPreview } from '../models/lat-languages';
5
- import { ConfigurationsService } from './configurations.service';
6
- import { LibCodingTestService } from './lib-coding-test.service';
7
- import { timeToMs } from '../utils/time-to-ms.util';
8
- import * as i0 from "@angular/core";
9
- import * as i1 from "ngx-webstorage";
10
- import * as i2 from "./lib-coding-test.service";
11
- import * as i3 from "./configurations.service";
12
- export class StorageCodingService {
13
- storage;
14
- libCodingTestService;
15
- configurationsService;
16
- CURRENT_STORAGE_VERSION = 2;
17
- autoSavedData = {
18
- version: this.CURRENT_STORAGE_VERSION,
19
- config: {},
20
- isFullscreen: false,
21
- assessments: {},
22
- lastModified: Date.now(),
23
- };
24
- key = 'savedData';
25
- assessmentId;
26
- questionId;
27
- language;
28
- codeSaveEnabled;
29
- testCasesSaveEnabled;
30
- isLAT;
31
- currentMode;
32
- isPreview;
33
- codeChangeSubscription;
34
- configSubscription;
35
- isFullscreenSubscription;
36
- currentLanguageSubscription;
37
- isCodeComponentAlive = filter(() => !this.configurationsService.getCodeComponentDestroyed());
38
- // Maximum age of entire storage data in days
39
- MAX_STORAGE_AGE_DAYS = 7;
40
- // Maximum age for assessment inner data in days
41
- MAX_QUESTION_DATA_AGE_DAYS = 1;
42
- // Maximum number of assessments to keep
43
- MAX_ASSESSMENTS = 10;
44
- // The default question ID used for migrated v1 data
45
- DEFAULT_MIGRATION_QUESTION_ID = 'default-question';
46
- // todo: remove
47
- // storage = {
48
- // retrieve: (key: string) => {
49
- // return {} as any;
50
- // },
51
- // store: (key: string, value: any) => {
52
- // return {} as any;
53
- // },
54
- // clear: (key: string) => {
55
- // return {} as any;
56
- // },
57
- // };
58
- constructor(
59
- // todo
60
- storage, libCodingTestService, configurationsService) {
61
- this.storage = storage;
62
- this.libCodingTestService = libCodingTestService;
63
- this.configurationsService = configurationsService;
64
- // Perform safety check on localStorage data
65
- this.performStorageSafetyCheck();
66
- }
67
- /**
68
- * Performs a safety check on localStorage data.
69
- * This is called at service initialization to ensure data integrity.
70
- */
71
- performStorageSafetyCheck() {
72
- try {
73
- const savedData = this.storage.retrieve(this.key);
74
- // If no data, nothing to check
75
- if (!savedData) {
76
- return;
77
- }
78
- // Check storage format and migrate if needed
79
- if (!('version' in savedData)) {
80
- // It's v1 format, so we'll migrate it when it's actually needed
81
- // No need to do anything right now
82
- }
83
- else {
84
- // It's v2 format, so we can perform cleanup right away
85
- this.autoSavedData = savedData;
86
- }
87
- }
88
- catch (error) {
89
- // If any error occurs during safety check, reset storage to avoid issues
90
- console.error('Error during storage safety check, resetting storage:', error);
91
- this.autoSavedData = {
92
- version: this.CURRENT_STORAGE_VERSION,
93
- config: {},
94
- isFullscreen: false,
95
- assessments: {},
96
- lastModified: Date.now(),
97
- };
98
- this.storeData();
99
- }
100
- }
101
- initializeSaving() {
102
- if (!this.codeChangeSubscription) {
103
- this.codeChangeSubscription = this.libCodingTestService.codeChange$
104
- .pipe(this.isCodeComponentAlive)
105
- .subscribe((code) => {
106
- if (this.codeSaveEnabled && this.assessmentId && this.questionId) {
107
- this.ensureDataStructure();
108
- const questionData = this.autoSavedData.assessments[this.assessmentId][this.questionId];
109
- const targetTestData = this.isPreview ? questionData.preview : questionData.actual;
110
- targetTestData.code[this.language] = code;
111
- this.updateLastModified(targetTestData);
112
- this.storeData();
113
- }
114
- });
115
- }
116
- if (!this.configSubscription) {
117
- this.configSubscription = this.configurationsService.config$
118
- .pipe(this.isCodeComponentAlive)
119
- .subscribe((config) => {
120
- this.autoSavedData.config = { ...config };
121
- this.updateLastModified(this.autoSavedData);
122
- this.storeData();
123
- });
124
- }
125
- if (!this.isFullscreenSubscription) {
126
- this.isFullscreenSubscription = this.libCodingTestService.isFullscreen$
127
- .pipe(this.isCodeComponentAlive)
128
- .subscribe((isFullscreen) => {
129
- this.autoSavedData.isFullscreen = isFullscreen;
130
- this.updateLastModified(this.autoSavedData);
131
- this.storeData();
132
- });
133
- }
134
- if (!this.currentLanguageSubscription && this.isLAT) {
135
- this.currentLanguageSubscription = this.libCodingTestService.currentLanguage$
136
- .pipe(this.isCodeComponentAlive)
137
- .subscribe((language) => {
138
- this.autoSavedData.lastLanguageLAT = {
139
- value: language,
140
- version: this.libCodingTestService.getVersion(),
141
- };
142
- this.updateLastModified(this.autoSavedData);
143
- this.storeData();
144
- });
145
- }
146
- }
147
- setupAutoSavedData(language, assessmentId, questionId, codeSaveEnabled, testCasesSaveEnabled, isLAT, mode) {
148
- this.language = language;
149
- this.assessmentId = assessmentId;
150
- this.questionId = questionId;
151
- this.codeSaveEnabled = codeSaveEnabled;
152
- this.testCasesSaveEnabled = testCasesSaveEnabled;
153
- this.isLAT = isLAT;
154
- this.currentMode = mode;
155
- this.isPreview = mode === "preview" /* Modes.Preview */ || mode === "non-assessment-preview" /* Modes.NonAssessmentPreview */;
156
- // Get saved data
157
- const savedData = this.storage.retrieve(this.key);
158
- // Handle migration if needed
159
- if (savedData) {
160
- // Check if it's v1 schema (no version field)
161
- if (!('version' in savedData)) {
162
- this.migrateFromV1ToV2(savedData);
163
- }
164
- else {
165
- this.autoSavedData = savedData;
166
- }
167
- // Apply loaded config and fullscreen setting
168
- this.configurationsService.setConfig(this.autoSavedData.config);
169
- this.libCodingTestService.setFullscreen(this.autoSavedData.isFullscreen);
170
- }
171
- }
172
- migrateFromV1ToV2(v1Data) {
173
- // Create a new v2 data structure
174
- const v2Data = {
175
- version: this.CURRENT_STORAGE_VERSION,
176
- config: v1Data.config,
177
- isFullscreen: v1Data.isFullscreen,
178
- lastLanguageLAT: v1Data.lastLanguageLAT,
179
- assessments: {},
180
- lastModified: Date.now(),
181
- };
182
- // Migrate assessment data
183
- // In v1, assessments were directly mapped to TestData
184
- // In v2, we need to organize by questionId
185
- for (const [assessmentId, testData] of Object.entries(v1Data.assessments)) {
186
- if (assessmentId === AssessmentIdPreview) {
187
- continue;
188
- }
189
- // We use a consistent default questionId for migration to ensure backward compatibility
190
- const defaultQuestionId = this.DEFAULT_MIGRATION_QUESTION_ID;
191
- // Initialize the assessment entry if it doesn't exist
192
- if (!v2Data.assessments[assessmentId]) {
193
- v2Data.assessments[assessmentId] = {};
194
- }
195
- // Create question data with the migrated test data
196
- const questionData = {};
197
- questionData.actual = {
198
- ...testData,
199
- lastModified: Date.now(),
200
- };
201
- // Set last modified time for the question data
202
- questionData.lastModified = Date.now();
203
- // Add to the v2 structure
204
- v2Data.assessments[assessmentId][defaultQuestionId] = questionData;
205
- }
206
- // Update our instance with migrated data
207
- this.autoSavedData = { ...v2Data };
208
- // Save the migrated data
209
- this.storeData();
210
- }
211
- ensureDataStructure() {
212
- // Make sure assessment exists
213
- if (!this.autoSavedData.assessments[this.assessmentId]) {
214
- this.autoSavedData.assessments[this.assessmentId] = {};
215
- }
216
- // Make sure question exists
217
- if (!this.autoSavedData.assessments[this.assessmentId][this.questionId]) {
218
- this.autoSavedData.assessments[this.assessmentId][this.questionId] = {
219
- lastModified: Date.now(),
220
- };
221
- }
222
- const questionData = this.autoSavedData.assessments[this.assessmentId][this.questionId];
223
- // Ensure preview or actual exists based on current mode
224
- if (this.isPreview && !questionData.preview) {
225
- questionData.preview = {
226
- code: {},
227
- testCasesLAT: [],
228
- lastModified: Date.now(),
229
- };
230
- }
231
- else if (!this.isPreview && !questionData.actual) {
232
- questionData.actual = {
233
- code: {},
234
- testCasesLAT: [],
235
- lastModified: Date.now(),
236
- };
237
- }
238
- }
239
- updateLastModified(data) {
240
- const now = Date.now();
241
- data.lastModified = now;
242
- // If updating a TestData, also update its parent QuestionData
243
- if ('code' in data && this.assessmentId && this.questionId) {
244
- this.autoSavedData.assessments[this.assessmentId][this.questionId].lastModified = now;
245
- }
246
- // Always update root timestamp regardless of the object being modified
247
- this.autoSavedData.lastModified = now;
248
- }
249
- /**
250
- * Cleanup with a temporary age threshold
251
- * @param maxAgeDays Maximum age in days for data to be considered valid
252
- */
253
- cleanupWithReducedThreshold(maxAgeDays) {
254
- // Call the unified cleanup method with more aggressive parameters
255
- this.cleanupStorageData({
256
- storageAgeMultiplier: 2, // More aggressive multiplier
257
- customStorageAgeDays: maxAgeDays, // Use the provided custom age
258
- minQuestionDataAgeDays: 0.5, // 12 hours minimum
259
- assessmentLimitMultiplier: 0.5, // Half the normal assessment limit
260
- });
261
- }
262
- /**
263
- * Unified method to clean up localStorage data with configurable intensity
264
- * @param options Cleanup options to control the aggressiveness of the cleanup
265
- */
266
- cleanupStorageData(options) {
267
- const now = Date.now();
268
- // Determine the storage age threshold
269
- const storageAgeDays = options.customStorageAgeDays ?? this.MAX_STORAGE_AGE_DAYS;
270
- const storageAgeThreshold = timeToMs(storageAgeDays); // Convert days to milliseconds
271
- const multiplier = options.storageAgeMultiplier ?? 1;
272
- // Check if the entire storage is too old and needs cleanup
273
- if (this.autoSavedData.lastModified && now - this.autoSavedData.lastModified > storageAgeThreshold * multiplier) {
274
- // If the entire storage is very old, just reset everything
275
- this.autoSavedData = {
276
- version: this.CURRENT_STORAGE_VERSION,
277
- config: this.autoSavedData.config, // Keep the config settings
278
- isFullscreen: this.autoSavedData.isFullscreen, // Keep fullscreen setting
279
- assessments: {},
280
- lastModified: now,
281
- };
282
- this.storeData();
283
- return;
284
- }
285
- const assessmentMetadata = [];
286
- // Determine the question data age threshold
287
- let questionDataAgeThreshold;
288
- if (options.useStandardQuestionDataAge) {
289
- // Use the standard threshold for normal cleanup
290
- questionDataAgeThreshold = timeToMs(this.MAX_QUESTION_DATA_AGE_DAYS);
291
- }
292
- else {
293
- // For maintenance/aggressive cleanup, use a custom threshold
294
- const minAgeHours = (options.minQuestionDataAgeDays ?? 0.5) * 24; // Convert days to hours
295
- questionDataAgeThreshold = Math.max(timeToMs(this.MAX_QUESTION_DATA_AGE_DAYS), // Standard age in milliseconds
296
- timeToMs(minAgeHours, 'hours') // Minimum age in milliseconds
297
- );
298
- }
299
- // Collect assessments with their latest modification times
300
- for (const [assessmentId, questions] of Object.entries(this.autoSavedData.assessments)) {
301
- let assessmentLastModified = 0;
302
- // Check all questions in this assessment
303
- let hasValidQuestions = false;
304
- for (const [questionId, questionData] of Object.entries(questions)) {
305
- // Skip if undefined
306
- if (!questionData) {
307
- continue;
308
- }
309
- // Get the latest modification time from preview and actual
310
- const previewTime = questionData.preview?.lastModified || 0;
311
- const actualTime = questionData.actual?.lastModified || 0;
312
- const questionLastModified = Math.max(previewTime, actualTime, questionData.lastModified || 0);
313
- // Check if data is too old - using configured threshold
314
- const isPreviewTooOld = previewTime > 0 && now - previewTime > questionDataAgeThreshold;
315
- const isActualTooOld = actualTime > 0 && now - actualTime > questionDataAgeThreshold;
316
- // Clean up old data within the question
317
- if (isPreviewTooOld && questionData.preview) {
318
- delete questionData.preview;
319
- }
320
- if (isActualTooOld && questionData.actual) {
321
- delete questionData.actual;
322
- }
323
- // If neither preview nor actual exists, remove the question
324
- if (!questionData.preview && !questionData.actual) {
325
- delete questions[questionId];
326
- }
327
- else {
328
- hasValidQuestions = true;
329
- assessmentLastModified = Math.max(assessmentLastModified, questionLastModified);
330
- }
331
- }
332
- // If no valid questions, remove the assessment
333
- if (!hasValidQuestions) {
334
- delete this.autoSavedData.assessments[assessmentId];
335
- }
336
- else {
337
- // Add to our list for potential size-based cleanup
338
- assessmentMetadata.push({
339
- id: assessmentId,
340
- lastModified: assessmentLastModified,
341
- });
342
- }
343
- }
344
- // Determine the assessment limit based on the multiplier
345
- const limitMultiplier = options.assessmentLimitMultiplier ?? 1;
346
- const assessmentLimit = Math.ceil(this.MAX_ASSESSMENTS * limitMultiplier);
347
- // If we have too many assessments, remove the oldest ones
348
- if (assessmentMetadata.length > assessmentLimit) {
349
- // Sort by lastModified (most recent first)
350
- assessmentMetadata.sort((a, b) => b.lastModified - a.lastModified);
351
- // Remove oldest assessments that exceed our limit
352
- for (let i = assessmentLimit; i < assessmentMetadata.length; i++) {
353
- delete this.autoSavedData.assessments[assessmentMetadata[i].id];
354
- }
355
- }
356
- // Save the cleaned data
357
- this.storeData();
358
- }
359
- changeLanguage(language) {
360
- this.language = language;
361
- }
362
- /**
363
- * Gets data from v1 schema format
364
- * Used for backward compatibility when retrieving data
365
- */
366
- getDataFromV1Schema(retrievedData) {
367
- // For v1, we need to handle the case where we don't have questionId
368
- // We're now using questionId in the app, but stored data might still be in v1 format
369
- // For preview data, the assessment ID would be AssessmentIdPreview
370
- if (this.assessmentId === AssessmentIdPreview) {
371
- if (this.isPreview) {
372
- // For preview mode on preview assessment, get the code directly
373
- const code = retrievedData.assessments[this.assessmentId]?.code;
374
- if (code && this.language in code) {
375
- return code[this.language];
376
- }
377
- }
378
- // We don't expect actual/non-preview data for AssessmentIdPreview
379
- return null;
380
- }
381
- else {
382
- // For regular assessments, v1 only had code in the "actual" section
383
- if (!this.isPreview) {
384
- const code = retrievedData.assessments[this.assessmentId]?.code;
385
- if (code && this.language in code) {
386
- return code[this.language];
387
- }
388
- }
389
- // We don't have preview data for regular assessments in v1
390
- return null;
391
- }
392
- }
393
- getSavedCode() {
394
- const retrievedData = this.retrieveData();
395
- if (!retrievedData) {
396
- return null;
397
- }
398
- // Check if using v2 schema
399
- if ('version' in retrievedData) {
400
- // Handle v2 schema
401
- const v2Data = retrievedData;
402
- const assessmentData = v2Data.assessments[this.assessmentId];
403
- if (!assessmentData) {
404
- return null;
405
- }
406
- // First try with provided questionId
407
- if (this.questionId && assessmentData[this.questionId]) {
408
- const questionData = assessmentData[this.questionId];
409
- const testData = this.isPreview ? questionData.preview : questionData.actual;
410
- if (testData?.code && this.language in testData.code) {
411
- return testData.code[this.language];
412
- }
413
- }
414
- if (this.questionId !== this.DEFAULT_MIGRATION_QUESTION_ID &&
415
- assessmentData[this.DEFAULT_MIGRATION_QUESTION_ID]) {
416
- const questionData = assessmentData[this.DEFAULT_MIGRATION_QUESTION_ID];
417
- const testData = this.isPreview ? questionData.preview : questionData.actual;
418
- if (testData?.code && this.language in testData.code) {
419
- return testData.code[this.language];
420
- }
421
- }
422
- return null;
423
- }
424
- else {
425
- // Handle v1 schema for backward compatibility
426
- return this.getDataFromV1Schema(retrievedData);
427
- }
428
- }
429
- getLastLanguageLAT() {
430
- const retrievedData = this.retrieveData();
431
- if (!retrievedData) {
432
- return null;
433
- }
434
- return retrievedData.lastLanguageLAT || null;
435
- }
436
- getSavedTestCases() {
437
- const retrievedData = this.retrieveData();
438
- if (!retrievedData) {
439
- return null;
440
- }
441
- // Check if using v2 schema
442
- if ('version' in retrievedData) {
443
- // Handle v2 schema
444
- const v2Data = retrievedData;
445
- const assessmentData = v2Data.assessments[this.assessmentId];
446
- if (!assessmentData) {
447
- return null;
448
- }
449
- // First try with provided questionId
450
- if (this.questionId && assessmentData[this.questionId]) {
451
- const questionData = assessmentData[this.questionId];
452
- const testData = this.isPreview ? questionData.preview : questionData.actual;
453
- if (testData?.testCasesLAT) {
454
- return testData.testCasesLAT;
455
- }
456
- }
457
- // If not found and questionId != DEFAULT_MIGRATION_QUESTION_ID,
458
- // also try with the default migration ID as a fallback
459
- if (this.questionId !== this.DEFAULT_MIGRATION_QUESTION_ID &&
460
- assessmentData[this.DEFAULT_MIGRATION_QUESTION_ID]) {
461
- const questionData = assessmentData[this.DEFAULT_MIGRATION_QUESTION_ID];
462
- const testData = this.isPreview ? questionData.preview : questionData.actual;
463
- if (testData?.testCasesLAT) {
464
- return testData.testCasesLAT;
465
- }
466
- }
467
- return null;
468
- }
469
- else {
470
- // Handle v1 schema test cases (test cases were only in the actual mode)
471
- const v1Data = retrievedData;
472
- if (!this.isPreview && v1Data.assessments[this.assessmentId]?.testCasesLAT) {
473
- return v1Data.assessments[this.assessmentId].testCasesLAT;
474
- }
475
- return null;
476
- }
477
- }
478
- getSavedConfig() {
479
- const retrievedData = this.retrieveData();
480
- return retrievedData?.config ? { ...retrievedData.config } : null;
481
- }
482
- getSavedFullScreenMode() {
483
- const retrievedData = this.retrieveData();
484
- return retrievedData?.isFullscreen !== undefined ? retrievedData.isFullscreen : null;
485
- }
486
- clearSavedCode() {
487
- this.ensureDataStructure();
488
- const questionData = this.autoSavedData.assessments[this.assessmentId][this.questionId];
489
- const targetTestData = this.isPreview ? [questionData.preview] : [questionData.preview, questionData.actual];
490
- let hasChanged = false;
491
- targetTestData.forEach(testData => {
492
- if (testData) {
493
- testData.code = {};
494
- this.updateLastModified(testData);
495
- hasChanged = true;
496
- }
497
- });
498
- if (hasChanged) {
499
- this.storeData();
500
- }
501
- }
502
- clearLastLang() {
503
- this.autoSavedData.lastLanguageLAT = null;
504
- this.updateLastModified(this.autoSavedData);
505
- this.storeData();
506
- }
507
- clearSavedTestCases() {
508
- this.ensureDataStructure();
509
- const questionData = this.autoSavedData.assessments[this.assessmentId][this.questionId];
510
- const targetTestData = this.isPreview ? questionData.preview : questionData.actual;
511
- if (targetTestData) {
512
- const hasNoCode = !Object.keys(targetTestData.code ?? {}).length;
513
- if (this.isPreview) {
514
- targetTestData.testCasesLAT = [];
515
- this.updateLastModified(targetTestData);
516
- }
517
- else if (hasNoCode) {
518
- const allQuestions = this.autoSavedData.assessments[this.assessmentId];
519
- delete allQuestions[this.questionId];
520
- this.autoSavedData.assessments[this.assessmentId] = { ...allQuestions };
521
- this.updateLastModified(this.autoSavedData);
522
- }
523
- this.storeData();
524
- }
525
- }
526
- updateVersion(version) {
527
- if (this.autoSavedData.lastLanguageLAT) {
528
- this.autoSavedData.lastLanguageLAT.version = version;
529
- this.updateLastModified(this.autoSavedData);
530
- this.storeData();
531
- }
532
- }
533
- updateTestCases(testCases) {
534
- if (this.assessmentId && this.questionId && this.testCasesSaveEnabled) {
535
- this.ensureDataStructure();
536
- const questionData = this.autoSavedData.assessments[this.assessmentId][this.questionId];
537
- const targetTestData = this.isPreview ? questionData.preview : questionData.actual;
538
- if (targetTestData) {
539
- targetTestData.testCasesLAT = testCases;
540
- this.updateLastModified(targetTestData);
541
- this.storeData();
542
- }
543
- }
544
- }
545
- storeData() {
546
- this.storage.store(this.key, this.autoSavedData);
547
- }
548
- retrieveData() {
549
- return this.storage.retrieve(this.key);
550
- }
551
- /**
552
- * Manually triggers data cleanup and migration.
553
- * Useful for handling large localStorage or migration when needed.
554
- * This can be called from application startup or after major events.
555
- */
556
- performMaintenanceAndCleanup() {
557
- // Get current data
558
- const savedData = this.storage.retrieve(this.key);
559
- // If no data, nothing to do
560
- if (!savedData) {
561
- return;
562
- }
563
- // Check if migration needed
564
- if (!('version' in savedData)) {
565
- this.migrateFromV1ToV2(savedData);
566
- }
567
- else {
568
- // Already migrated, just perform cleanup
569
- this.autoSavedData = savedData;
570
- // Perform a more aggressive cleanup for manual maintenance
571
- // by temporarily modifying the cleanup thresholds
572
- const originalMaxAge = this.MAX_STORAGE_AGE_DAYS;
573
- const halfAge = Math.floor(originalMaxAge / 2); // Cut threshold in half
574
- // Create a custom cleanup function with reduced threshold
575
- this.cleanupWithReducedThreshold(halfAge);
576
- }
577
- // Verify localStorage size is manageable
578
- try {
579
- const serializedData = JSON.stringify(this.autoSavedData);
580
- const sizeInKB = Math.round(serializedData.length / 1024);
581
- // If data is too large (over 2MB), perform more aggressive cleanup
582
- if (sizeInKB > 2048) {
583
- console.warn(`Local storage data is large (${sizeInKB}KB), performing deep cleanup`);
584
- // Use extremely aggressive cleanup settings
585
- this.cleanupStorageData({
586
- storageAgeMultiplier: 0.5, // Even more aggressive - half the normal age
587
- minQuestionDataAgeDays: 0.25, // 6 hours minimum
588
- assessmentLimitMultiplier: 0.2, // Keep only 20% of assessments
589
- });
590
- // If still too large, keep only a few most recent assessments
591
- const newSerializedData = JSON.stringify(this.autoSavedData);
592
- const newSizeInKB = Math.round(newSerializedData.length / 1024);
593
- if (newSizeInKB > 1024) {
594
- // Keep only recent assessments
595
- const assessments = this.autoSavedData.assessments;
596
- // Find the newest timestamp in each assessment
597
- const getNewestTimestamp = (obj) => Object.values(obj).reduce((max, questionData) => Math.max(max, questionData?.lastModified || 0), 0);
598
- const entries = Object.entries(assessments).sort(([, a], [, b]) => getNewestTimestamp(b) -
599
- getNewestTimestamp(a));
600
- // Keep only the 5 most recent assessments
601
- const keepCount = Math.min(5, entries.length);
602
- this.autoSavedData.assessments = {};
603
- for (let i = 0; i < keepCount; i++) {
604
- const [id, data] = entries[i];
605
- this.autoSavedData.assessments[id] = data;
606
- }
607
- this.storeData();
608
- }
609
- }
610
- }
611
- catch (error) {
612
- console.error('Error checking localStorage size', error);
613
- }
614
- }
615
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: StorageCodingService, deps: [{ token: i1.LocalStorageService }, { token: i2.LibCodingTestService }, { token: i3.ConfigurationsService }], target: i0.ɵɵFactoryTarget.Injectable });
616
- static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: StorageCodingService, providedIn: 'root' });
617
- }
618
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: StorageCodingService, decorators: [{
619
- type: Injectable,
620
- args: [{
621
- providedIn: 'root',
622
- }]
623
- }], ctorParameters: () => [{ type: i1.LocalStorageService }, { type: i2.LibCodingTestService }, { type: i3.ConfigurationsService }] });
624
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"storage.service.js","sourceRoot":"","sources":["../../../../../../packages/tgo-coding-test/src/lib/services/storage.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { LocalStorageService } from 'ngx-webstorage';\nimport { filter, Subscription } from 'rxjs';\nimport { CodeEditorLanguages } from '../components/code-editor/helpers/code-editor-helper.model';\nimport { AutoSavedData, AutoSavedDataV1, AutoSavedDataV2, QuestionData, TestData } from '../models/auto-saved-data';\nimport { Config } from '../models/configs';\nimport { AssessmentIdPreview } from '../models/lat-languages';\nimport { Modes } from '../models/mode';\nimport { CustomTestCase, ExampleTestCase } from '../models/test-cases';\nimport { ConfigurationsService } from './configurations.service';\nimport { LibCodingTestService } from './lib-coding-test.service';\nimport { timeToMs } from '../utils/time-to-ms.util';\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class StorageCodingService {\n  private readonly CURRENT_STORAGE_VERSION = 2;\n  private autoSavedData: AutoSavedData = {\n    version: this.CURRENT_STORAGE_VERSION,\n    config: {} as Config,\n    isFullscreen: false,\n    assessments: {},\n    lastModified: Date.now(),\n  };\n  private readonly key = 'savedData';\n  private assessmentId: string;\n  private questionId: string;\n  private language: CodeEditorLanguages;\n  private codeSaveEnabled: boolean;\n  private testCasesSaveEnabled: boolean;\n  private isLAT: boolean;\n  private currentMode: Modes;\n  private isPreview: boolean;\n  private codeChangeSubscription: Subscription;\n  private configSubscription: Subscription;\n  private isFullscreenSubscription: Subscription;\n  private currentLanguageSubscription: Subscription;\n  private isCodeComponentAlive = filter(() => !this.configurationsService.getCodeComponentDestroyed());\n\n  // Maximum age of entire storage data in days\n  private readonly MAX_STORAGE_AGE_DAYS = 7;\n  // Maximum age for assessment inner data in days\n  private readonly MAX_QUESTION_DATA_AGE_DAYS = 1;\n  // Maximum number of assessments to keep\n  private readonly MAX_ASSESSMENTS = 10;\n  // The default question ID used for migrated v1 data\n  private readonly DEFAULT_MIGRATION_QUESTION_ID = 'default-question';\n\n  // todo: remove\n  // storage = {\n  //   retrieve: (key: string) => {\n  //     return {} as any;\n  //   },\n  //   store: (key: string, value: any) => {\n  //     return {} as any;\n  //   },\n  //   clear: (key: string) => {\n  //     return {} as any;\n  //   },\n  // };\n\n  constructor(\n    // todo\n    private storage: LocalStorageService,\n    private libCodingTestService: LibCodingTestService,\n    private configurationsService: ConfigurationsService\n  ) {\n    // Perform safety check on localStorage data\n    this.performStorageSafetyCheck();\n  }\n\n  /**\n   * Performs a safety check on localStorage data.\n   * This is called at service initialization to ensure data integrity.\n   */\n  private performStorageSafetyCheck(): void {\n    try {\n      const savedData = this.storage.retrieve(this.key);\n\n      // If no data, nothing to check\n      if (!savedData) {\n        return;\n      }\n\n      // Check storage format and migrate if needed\n      if (!('version' in savedData)) {\n        // It's v1 format, so we'll migrate it when it's actually needed\n        // No need to do anything right now\n      } else {\n        // It's v2 format, so we can perform cleanup right away\n        this.autoSavedData = savedData as AutoSavedData;\n      }\n    } catch (error) {\n      // If any error occurs during safety check, reset storage to avoid issues\n      console.error('Error during storage safety check, resetting storage:', error);\n      this.autoSavedData = {\n        version: this.CURRENT_STORAGE_VERSION,\n        config: {} as Config,\n        isFullscreen: false,\n        assessments: {},\n        lastModified: Date.now(),\n      };\n      this.storeData();\n    }\n  }\n\n  initializeSaving(): void {\n    if (!this.codeChangeSubscription) {\n      this.codeChangeSubscription = this.libCodingTestService.codeChange$\n        .pipe(this.isCodeComponentAlive)\n        .subscribe((code: string) => {\n          if (this.codeSaveEnabled && this.assessmentId && this.questionId) {\n            this.ensureDataStructure();\n            const questionData = this.autoSavedData.assessments[this.assessmentId][this.questionId];\n            const targetTestData = this.isPreview ? questionData.preview : questionData.actual;\n            targetTestData.code[this.language] = code!;\n            this.updateLastModified(targetTestData);\n            this.storeData();\n          }\n        });\n    }\n\n    if (!this.configSubscription) {\n      this.configSubscription = this.configurationsService.config$\n        .pipe(this.isCodeComponentAlive)\n        .subscribe((config: Config) => {\n          this.autoSavedData.config = { ...config };\n          this.updateLastModified(this.autoSavedData);\n          this.storeData();\n        });\n    }\n\n    if (!this.isFullscreenSubscription) {\n      this.isFullscreenSubscription = this.libCodingTestService.isFullscreen$\n        .pipe(this.isCodeComponentAlive)\n        .subscribe((isFullscreen: boolean) => {\n          this.autoSavedData.isFullscreen = isFullscreen;\n          this.updateLastModified(this.autoSavedData);\n          this.storeData();\n        });\n    }\n\n    if (!this.currentLanguageSubscription && this.isLAT) {\n      this.currentLanguageSubscription = this.libCodingTestService.currentLanguage$\n        .pipe(this.isCodeComponentAlive)\n        .subscribe((language: CodeEditorLanguages) => {\n          this.autoSavedData.lastLanguageLAT = {\n            value: language,\n            version: this.libCodingTestService.getVersion(),\n          };\n          this.updateLastModified(this.autoSavedData);\n          this.storeData();\n        });\n    }\n  }\n\n  setupAutoSavedData(\n    language: CodeEditorLanguages,\n    assessmentId: string,\n    questionId: string,\n    codeSaveEnabled: boolean,\n    testCasesSaveEnabled: boolean,\n    isLAT: boolean,\n    mode: Modes\n  ) {\n    this.language = language;\n    this.assessmentId = assessmentId;\n    this.questionId = questionId;\n    this.codeSaveEnabled = codeSaveEnabled;\n    this.testCasesSaveEnabled = testCasesSaveEnabled;\n    this.isLAT = isLAT;\n    this.currentMode = mode;\n    this.isPreview = mode === Modes.Preview || mode === Modes.NonAssessmentPreview;\n\n    // Get saved data\n    const savedData = this.storage.retrieve(this.key);\n\n    // Handle migration if needed\n    if (savedData) {\n      // Check if it's v1 schema (no version field)\n      if (!('version' in savedData)) {\n        this.migrateFromV1ToV2(savedData as AutoSavedDataV1);\n      } else {\n        this.autoSavedData = savedData as AutoSavedData;\n      }\n\n      // Apply loaded config and fullscreen setting\n      this.configurationsService.setConfig(this.autoSavedData.config);\n      this.libCodingTestService.setFullscreen(this.autoSavedData.isFullscreen);\n    }\n  }\n\n  private migrateFromV1ToV2(v1Data: AutoSavedDataV1): void {\n    // Create a new v2 data structure\n    const v2Data: AutoSavedDataV2 = {\n      version: this.CURRENT_STORAGE_VERSION,\n      config: v1Data.config,\n      isFullscreen: v1Data.isFullscreen,\n      lastLanguageLAT: v1Data.lastLanguageLAT,\n      assessments: {},\n      lastModified: Date.now(),\n    };\n\n    // Migrate assessment data\n    // In v1, assessments were directly mapped to TestData\n    // In v2, we need to organize by questionId\n    for (const [assessmentId, testData] of Object.entries(v1Data.assessments)) {\n      if (assessmentId === AssessmentIdPreview) {\n        continue;\n      }\n\n      // We use a consistent default questionId for migration to ensure backward compatibility\n      const defaultQuestionId = this.DEFAULT_MIGRATION_QUESTION_ID;\n\n      // Initialize the assessment entry if it doesn't exist\n      if (!v2Data.assessments[assessmentId]) {\n        v2Data.assessments[assessmentId] = {};\n      }\n\n      // Create question data with the migrated test data\n      const questionData: QuestionData = {};\n\n      questionData.actual = {\n        ...testData,\n        lastModified: Date.now(),\n      };\n\n      // Set last modified time for the question data\n      questionData.lastModified = Date.now();\n\n      // Add to the v2 structure\n      v2Data.assessments[assessmentId][defaultQuestionId] = questionData;\n    }\n\n    // Update our instance with migrated data\n    this.autoSavedData = { ...v2Data };\n\n    // Save the migrated data\n    this.storeData();\n  }\n\n  private ensureDataStructure(): void {\n    // Make sure assessment exists\n    if (!this.autoSavedData.assessments[this.assessmentId]) {\n      this.autoSavedData.assessments[this.assessmentId] = {};\n    }\n\n    // Make sure question exists\n    if (!this.autoSavedData.assessments[this.assessmentId][this.questionId]) {\n      this.autoSavedData.assessments[this.assessmentId][this.questionId] = {\n        lastModified: Date.now(),\n      };\n    }\n\n    const questionData = this.autoSavedData.assessments[this.assessmentId][this.questionId];\n\n    // Ensure preview or actual exists based on current mode\n    if (this.isPreview && !questionData.preview) {\n      questionData.preview = {\n        code: {},\n        testCasesLAT: [],\n        lastModified: Date.now(),\n      };\n    } else if (!this.isPreview && !questionData.actual) {\n      questionData.actual = {\n        code: {},\n        testCasesLAT: [],\n        lastModified: Date.now(),\n      };\n    }\n  }\n\n  private updateLastModified(data: TestData | QuestionData | AutoSavedData): void {\n    const now = Date.now();\n    data.lastModified = now;\n\n    // If updating a TestData, also update its parent QuestionData\n    if ('code' in data && this.assessmentId && this.questionId) {\n      this.autoSavedData.assessments[this.assessmentId][this.questionId].lastModified = now;\n    }\n\n    // Always update root timestamp regardless of the object being modified\n    this.autoSavedData.lastModified = now;\n  }\n\n  /**\n   * Cleanup with a temporary age threshold\n   * @param maxAgeDays Maximum age in days for data to be considered valid\n   */\n  private cleanupWithReducedThreshold(maxAgeDays: number): void {\n    // Call the unified cleanup method with more aggressive parameters\n    this.cleanupStorageData({\n      storageAgeMultiplier: 2, // More aggressive multiplier\n      customStorageAgeDays: maxAgeDays, // Use the provided custom age\n      minQuestionDataAgeDays: 0.5, // 12 hours minimum\n      assessmentLimitMultiplier: 0.5, // Half the normal assessment limit\n    });\n  }\n\n  /**\n   * Unified method to clean up localStorage data with configurable intensity\n   * @param options Cleanup options to control the aggressiveness of the cleanup\n   */\n  private cleanupStorageData(options: {\n    storageAgeMultiplier?: number; // Multiplier for storage age threshold (default: 1)\n    customStorageAgeDays?: number; // Custom storage age in days (overrides MAX_STORAGE_AGE_DAYS if provided)\n    useStandardQuestionDataAge?: boolean; // Whether to use standard question data age (default: false)\n    minQuestionDataAgeDays?: number; // Minimum age for question data in days (default: 0.5 - 12 hours)\n    assessmentLimitMultiplier?: number; // Multiplier for maximum assessments (default: 1)\n  }): void {\n    const now = Date.now();\n\n    // Determine the storage age threshold\n    const storageAgeDays = options.customStorageAgeDays ?? this.MAX_STORAGE_AGE_DAYS;\n    const storageAgeThreshold = timeToMs(storageAgeDays); // Convert days to milliseconds\n    const multiplier = options.storageAgeMultiplier ?? 1;\n\n    // Check if the entire storage is too old and needs cleanup\n    if (this.autoSavedData.lastModified && now - this.autoSavedData.lastModified > storageAgeThreshold * multiplier) {\n      // If the entire storage is very old, just reset everything\n      this.autoSavedData = {\n        version: this.CURRENT_STORAGE_VERSION,\n        config: this.autoSavedData.config, // Keep the config settings\n        isFullscreen: this.autoSavedData.isFullscreen, // Keep fullscreen setting\n        assessments: {},\n        lastModified: now,\n      };\n      this.storeData();\n      return;\n    }\n\n    // Structure for collecting assessment data with metadata\n    interface AssessmentMetadata {\n      id: string;\n      lastModified: number;\n    }\n\n    const assessmentMetadata: AssessmentMetadata[] = [];\n\n    // Determine the question data age threshold\n    let questionDataAgeThreshold: number;\n    if (options.useStandardQuestionDataAge) {\n      // Use the standard threshold for normal cleanup\n      questionDataAgeThreshold = timeToMs(this.MAX_QUESTION_DATA_AGE_DAYS);\n    } else {\n      // For maintenance/aggressive cleanup, use a custom threshold\n      const minAgeHours = (options.minQuestionDataAgeDays ?? 0.5) * 24; // Convert days to hours\n      questionDataAgeThreshold = Math.max(\n        timeToMs(this.MAX_QUESTION_DATA_AGE_DAYS), // Standard age in milliseconds\n        timeToMs(minAgeHours, 'hours') // Minimum age in milliseconds\n      );\n    }\n\n    // Collect assessments with their latest modification times\n    for (const [assessmentId, questions] of Object.entries(this.autoSavedData.assessments)) {\n      let assessmentLastModified = 0;\n\n      // Check all questions in this assessment\n      let hasValidQuestions = false;\n\n      for (const [questionId, questionData] of Object.entries(questions)) {\n        // Skip if undefined\n        if (!questionData) {\n          continue;\n        }\n\n        // Get the latest modification time from preview and actual\n        const previewTime = questionData.preview?.lastModified || 0;\n        const actualTime = questionData.actual?.lastModified || 0;\n        const questionLastModified = Math.max(previewTime, actualTime, questionData.lastModified || 0);\n\n        // Check if data is too old - using configured threshold\n        const isPreviewTooOld = previewTime > 0 && now - previewTime > questionDataAgeThreshold;\n        const isActualTooOld = actualTime > 0 && now - actualTime > questionDataAgeThreshold;\n\n        // Clean up old data within the question\n        if (isPreviewTooOld && questionData.preview) {\n          delete questionData.preview;\n        }\n\n        if (isActualTooOld && questionData.actual) {\n          delete questionData.actual;\n        }\n\n        // If neither preview nor actual exists, remove the question\n        if (!questionData.preview && !questionData.actual) {\n          delete questions[questionId];\n        } else {\n          hasValidQuestions = true;\n          assessmentLastModified = Math.max(assessmentLastModified, questionLastModified);\n        }\n      }\n\n      // If no valid questions, remove the assessment\n      if (!hasValidQuestions) {\n        delete this.autoSavedData.assessments[assessmentId];\n      } else {\n        // Add to our list for potential size-based cleanup\n        assessmentMetadata.push({\n          id: assessmentId,\n          lastModified: assessmentLastModified,\n        });\n      }\n    }\n\n    // Determine the assessment limit based on the multiplier\n    const limitMultiplier = options.assessmentLimitMultiplier ?? 1;\n    const assessmentLimit = Math.ceil(this.MAX_ASSESSMENTS * limitMultiplier);\n\n    // If we have too many assessments, remove the oldest ones\n    if (assessmentMetadata.length > assessmentLimit) {\n      // Sort by lastModified (most recent first)\n      assessmentMetadata.sort((a, b) => b.lastModified - a.lastModified);\n\n      // Remove oldest assessments that exceed our limit\n      for (let i = assessmentLimit; i < assessmentMetadata.length; i++) {\n        delete this.autoSavedData.assessments[assessmentMetadata[i].id];\n      }\n    }\n\n    // Save the cleaned data\n    this.storeData();\n  }\n\n  changeLanguage(language: CodeEditorLanguages) {\n    this.language = language;\n  }\n\n  /**\n   * Gets data from v1 schema format\n   * Used for backward compatibility when retrieving data\n   */\n  private getDataFromV1Schema(retrievedData: AutoSavedDataV1): string | (ExampleTestCase | CustomTestCase)[] | null | undefined {\n    // For v1, we need to handle the case where we don't have questionId\n    // We're now using questionId in the app, but stored data might still be in v1 format\n\n    // For preview data, the assessment ID would be AssessmentIdPreview\n    if (this.assessmentId === AssessmentIdPreview) {\n      if (this.isPreview) {\n        // For preview mode on preview assessment, get the code directly\n        const code = retrievedData.assessments[this.assessmentId]?.code;\n        if (code && this.language in code) {\n          return code[this.language];\n        }\n      }\n      // We don't expect actual/non-preview data for AssessmentIdPreview\n      return null;\n    } else {\n      // For regular assessments, v1 only had code in the \"actual\" section\n      if (!this.isPreview) {\n        const code = retrievedData.assessments[this.assessmentId]?.code;\n        if (code && this.language in code) {\n          return code[this.language];\n        }\n      }\n      // We don't have preview data for regular assessments in v1\n      return null;\n    }\n  }\n\n  getSavedCode(): string | null | undefined {\n    const retrievedData = this.retrieveData();\n    if (!retrievedData) {\n      return null;\n    }\n\n    // Check if using v2 schema\n    if ('version' in retrievedData) {\n      // Handle v2 schema\n      const v2Data = retrievedData;\n      const assessmentData = v2Data.assessments[this.assessmentId];\n\n      if (!assessmentData) {\n        return null;\n      }\n\n      // First try with provided questionId\n      if (this.questionId && assessmentData[this.questionId]) {\n        const questionData = assessmentData[this.questionId];\n        const testData = this.isPreview ? questionData.preview : questionData.actual;\n        if (testData?.code && this.language in testData.code) {\n          return testData.code[this.language];\n        }\n      }\n\n      if (\n        this.questionId !== this.DEFAULT_MIGRATION_QUESTION_ID &&\n        assessmentData[this.DEFAULT_MIGRATION_QUESTION_ID]\n      ) {\n        const questionData = assessmentData[this.DEFAULT_MIGRATION_QUESTION_ID];\n        const testData = this.isPreview ? questionData.preview : questionData.actual;\n        if (testData?.code && this.language in testData.code) {\n          return testData.code[this.language];\n        }\n      }\n\n      return null;\n    } else {\n      // Handle v1 schema for backward compatibility\n      return this.getDataFromV1Schema(retrievedData) as string;\n    }\n  }\n\n  getLastLanguageLAT(): {\n    value: CodeEditorLanguages;\n    version: string;\n  } | null {\n    const retrievedData = this.retrieveData();\n    if (!retrievedData) {\n      return null;\n    }\n\n    return retrievedData.lastLanguageLAT || null;\n  }\n\n  getSavedTestCases(): (ExampleTestCase | CustomTestCase)[] | null | undefined {\n    const retrievedData = this.retrieveData();\n    if (!retrievedData) {\n      return null;\n    }\n\n    // Check if using v2 schema\n    if ('version' in retrievedData) {\n      // Handle v2 schema\n      const v2Data = retrievedData;\n      const assessmentData = v2Data.assessments[this.assessmentId];\n\n      if (!assessmentData) {\n        return null;\n      }\n\n      // First try with provided questionId\n      if (this.questionId && assessmentData[this.questionId]) {\n        const questionData = assessmentData[this.questionId];\n        const testData = this.isPreview ? questionData.preview : questionData.actual;\n        if (testData?.testCasesLAT) {\n          return testData.testCasesLAT;\n        }\n      }\n\n      // If not found and questionId != DEFAULT_MIGRATION_QUESTION_ID,\n      // also try with the default migration ID as a fallback\n      if (\n        this.questionId !== this.DEFAULT_MIGRATION_QUESTION_ID &&\n        assessmentData[this.DEFAULT_MIGRATION_QUESTION_ID]\n      ) {\n        const questionData = assessmentData[this.DEFAULT_MIGRATION_QUESTION_ID];\n        const testData = this.isPreview ? questionData.preview : questionData.actual;\n        if (testData?.testCasesLAT) {\n          return testData.testCasesLAT;\n        }\n      }\n\n      return null;\n    } else {\n      // Handle v1 schema test cases (test cases were only in the actual mode)\n      const v1Data = retrievedData;\n      if (!this.isPreview && v1Data.assessments[this.assessmentId]?.testCasesLAT) {\n        return v1Data.assessments[this.assessmentId].testCasesLAT;\n      }\n      return null;\n    }\n  }\n\n  getSavedConfig(): Config | null {\n    const retrievedData = this.retrieveData();\n    return retrievedData?.config ? { ...retrievedData.config } : null;\n  }\n\n  getSavedFullScreenMode(): boolean | null {\n    const retrievedData = this.retrieveData();\n    return retrievedData?.isFullscreen !== undefined ? retrievedData.isFullscreen : null;\n  }\n\n  clearSavedCode(): void {\n    this.ensureDataStructure();\n    const questionData = this.autoSavedData.assessments[this.assessmentId][this.questionId];\n    const targetTestData = this.isPreview ? [questionData.preview] : [questionData.preview, questionData.actual];\n\n    let hasChanged = false;\n    targetTestData.forEach(testData => {\n      if (testData) {\n        testData.code = {};\n        this.updateLastModified(testData);\n        hasChanged = true;\n      }\n    });\n    if (hasChanged) {\n      this.storeData();\n    }\n  }\n\n  clearLastLang(): void {\n    this.autoSavedData.lastLanguageLAT = null;\n    this.updateLastModified(this.autoSavedData);\n    this.storeData();\n  }\n\n  clearSavedTestCases(): void {\n    this.ensureDataStructure();\n    const questionData = this.autoSavedData.assessments[this.assessmentId][this.questionId];\n    const targetTestData = this.isPreview ? questionData.preview : questionData.actual;\n\n    if (targetTestData) {\n      const hasNoCode = !Object.keys(targetTestData.code ?? {}).length;\n\n      if (this.isPreview) {\n        targetTestData.testCasesLAT = [];\n        this.updateLastModified(targetTestData);\n      } else if (hasNoCode) {\n        const allQuestions = this.autoSavedData.assessments[this.assessmentId];\n        delete allQuestions[this.questionId];\n        this.autoSavedData.assessments[this.assessmentId] = { ...allQuestions };\n        this.updateLastModified(this.autoSavedData);\n      }\n      this.storeData();\n    }\n  }\n\n  updateVersion(version: string): void {\n    if (this.autoSavedData.lastLanguageLAT) {\n      this.autoSavedData.lastLanguageLAT.version = version;\n      this.updateLastModified(this.autoSavedData);\n      this.storeData();\n    }\n  }\n\n  updateTestCases(testCases: (ExampleTestCase | CustomTestCase)[]) {\n    if (this.assessmentId && this.questionId && this.testCasesSaveEnabled) {\n      this.ensureDataStructure();\n      const questionData = this.autoSavedData.assessments[this.assessmentId][this.questionId];\n      const targetTestData = this.isPreview ? questionData.preview : questionData.actual;\n\n      if (targetTestData) {\n        targetTestData.testCasesLAT = testCases;\n        this.updateLastModified(targetTestData);\n        this.storeData();\n      }\n    }\n  }\n\n  private storeData(): void {\n    this.storage.store(this.key, this.autoSavedData);\n  }\n\n  private retrieveData(): AutoSavedData | AutoSavedDataV1 {\n    return this.storage.retrieve(this.key) as AutoSavedData | AutoSavedDataV1;\n  }\n\n  /**\n   * Manually triggers data cleanup and migration.\n   * Useful for handling large localStorage or migration when needed.\n   * This can be called from application startup or after major events.\n   */\n  performMaintenanceAndCleanup(): void {\n    // Get current data\n    const savedData = this.storage.retrieve(this.key);\n\n    // If no data, nothing to do\n    if (!savedData) {\n      return;\n    }\n\n    // Check if migration needed\n    if (!('version' in savedData)) {\n      this.migrateFromV1ToV2(savedData as AutoSavedDataV1);\n    } else {\n      // Already migrated, just perform cleanup\n      this.autoSavedData = savedData as AutoSavedData;\n\n      // Perform a more aggressive cleanup for manual maintenance\n      // by temporarily modifying the cleanup thresholds\n      const originalMaxAge = this.MAX_STORAGE_AGE_DAYS;\n      const halfAge = Math.floor(originalMaxAge / 2); // Cut threshold in half\n\n      // Create a custom cleanup function with reduced threshold\n      this.cleanupWithReducedThreshold(halfAge);\n    }\n\n    // Verify localStorage size is manageable\n    try {\n      const serializedData = JSON.stringify(this.autoSavedData);\n      const sizeInKB = Math.round(serializedData.length / 1024);\n\n      // If data is too large (over 2MB), perform more aggressive cleanup\n      if (sizeInKB > 2048) {\n        console.warn(`Local storage data is large (${sizeInKB}KB), performing deep cleanup`);\n\n        // Use extremely aggressive cleanup settings\n        this.cleanupStorageData({\n          storageAgeMultiplier: 0.5, // Even more aggressive - half the normal age\n          minQuestionDataAgeDays: 0.25, // 6 hours minimum\n          assessmentLimitMultiplier: 0.2, // Keep only 20% of assessments\n        });\n\n        // If still too large, keep only a few most recent assessments\n        const newSerializedData = JSON.stringify(this.autoSavedData);\n        const newSizeInKB = Math.round(newSerializedData.length / 1024);\n\n        if (newSizeInKB > 1024) {\n          // Keep only recent assessments\n          const assessments = this.autoSavedData.assessments;\n          // Find the newest timestamp in each assessment\n          const getNewestTimestamp = (obj: Record<string, QuestionData>) =>\n            Object.values(obj).reduce((max, questionData) => Math.max(max, questionData?.lastModified || 0), 0);\n          const entries = Object.entries(assessments).sort(\n            ([, a], [, b]) =>\n              getNewestTimestamp(b as Record<string, QuestionData>) -\n              getNewestTimestamp(a as Record<string, QuestionData>)\n          );\n\n          // Keep only the 5 most recent assessments\n          const keepCount = Math.min(5, entries.length);\n          this.autoSavedData.assessments = {};\n\n          for (let i = 0; i < keepCount; i++) {\n            const [id, data] = entries[i];\n            this.autoSavedData.assessments[id] = data as Record<string, QuestionData>;\n          }\n\n          this.storeData();\n        }\n      }\n    } catch (error) {\n      console.error('Error checking localStorage size', error);\n    }\n  }\n}\n"]}