@testgorilla/tgo-coding-test 0.0.1 → 1.0.0
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.
- package/README.md +49 -2
- package/esm2022/index.mjs +12 -0
- package/esm2022/lib/components/code-editor/code-editor.component.mjs +335 -0
- package/esm2022/lib/components/code-editor/code-editor.service.mjs +72 -0
- package/esm2022/lib/components/code-editor/helpers/c-helper.mjs +40 -0
- package/esm2022/lib/components/code-editor/helpers/code-editor-helper.base.mjs +11 -0
- package/esm2022/lib/components/code-editor/helpers/code-editor-helper.model.mjs +2 -0
- package/esm2022/lib/components/code-editor/helpers/cpp-helper.mjs +41 -0
- package/esm2022/lib/components/code-editor/helpers/csharp-helper.mjs +42 -0
- package/esm2022/lib/components/code-editor/helpers/go-helper.mjs +42 -0
- package/esm2022/lib/components/code-editor/helpers/index.mjs +16 -0
- package/esm2022/lib/components/code-editor/helpers/java-helper.mjs +42 -0
- package/esm2022/lib/components/code-editor/helpers/javascript-helper.mjs +26 -0
- package/esm2022/lib/components/code-editor/helpers/kotlin-helper.mjs +42 -0
- package/esm2022/lib/components/code-editor/helpers/php-helper.mjs +26 -0
- package/esm2022/lib/components/code-editor/helpers/python-helper.mjs +26 -0
- package/esm2022/lib/components/code-editor/helpers/r-helper.mjs +26 -0
- package/esm2022/lib/components/code-editor/helpers/ruby-helper.mjs +26 -0
- package/esm2022/lib/components/code-editor/helpers/scala-helper.mjs +41 -0
- package/esm2022/lib/components/code-editor/helpers/sql-helper.mjs +34 -0
- package/esm2022/lib/components/code-editor/helpers/swift-helper.mjs +40 -0
- package/esm2022/lib/components/code-editor/helpers/typescript-helper.mjs +41 -0
- package/esm2022/lib/components/code-editor/models/code-editor.model.mjs +2 -0
- package/esm2022/lib/components/code-editor/models/coding-snapshot.model.mjs +2 -0
- package/esm2022/lib/components/coding-question/coding-question.component.mjs +126 -0
- package/esm2022/lib/components/coding-section/coding-section.component.mjs +188 -0
- package/esm2022/lib/components/common/truncated-text/truncated-text.component.mjs +38 -0
- package/esm2022/lib/components/configurations/configurations.component.mjs +97 -0
- package/esm2022/lib/components/instructions/instructions.component.mjs +139 -0
- package/esm2022/lib/components/panel/panel.component.mjs +34 -0
- package/esm2022/lib/components/runnable-editor/runnable-editor.component.mjs +169 -0
- package/esm2022/lib/components/tests/test-cases/test-cases.component.mjs +198 -0
- package/esm2022/lib/components/tests/test-cases-content/test-cases-content.component.mjs +96 -0
- package/esm2022/lib/components/tests/test-cases-status/test-cases-status.component.mjs +21 -0
- package/esm2022/lib/components/tests/test-results.component.mjs +127 -0
- package/esm2022/lib/components/tgo-coding-test/tgo-coding-test.component.mjs +280 -0
- package/esm2022/lib/components/tgo-coding-test-candidate-view/tgo-coding-test-candidate-view.component.mjs +476 -0
- package/esm2022/lib/config/index.mjs +2 -0
- package/esm2022/lib/config/tgo-coding-test.config.mjs +2 -0
- package/esm2022/lib/config/tgo-coding-test.provider.mjs +34 -0
- package/esm2022/lib/config/tgo-coding-test.token.mjs +14 -0
- package/esm2022/lib/models/auto-saved-data.mjs +2 -0
- package/esm2022/lib/models/code-event.mjs +2 -0
- package/esm2022/lib/models/coderunner-execution-results.mjs +2 -0
- package/esm2022/lib/models/configs.mjs +2 -0
- package/esm2022/lib/models/language-change-action.mjs +2 -0
- package/esm2022/lib/models/lat-languages.mjs +3 -0
- package/esm2022/lib/models/mixpanel-events.mjs +2 -0
- package/esm2022/lib/models/mode.mjs +2 -0
- package/esm2022/lib/models/paste-data.mjs +2 -0
- package/esm2022/lib/models/programming-language.mjs +2 -0
- package/esm2022/lib/models/test-cases.mjs +7 -0
- package/esm2022/lib/models/theme.mjs +2 -0
- package/esm2022/lib/models/translations.mjs +2 -0
- package/esm2022/lib/models/view-mode.mjs +8 -0
- package/esm2022/lib/pipes/memoize-func.pipe.mjs +39 -0
- package/esm2022/lib/services/candidate-coding-test-services/candidature-api.service.mjs +19 -0
- package/esm2022/lib/services/candidate-coding-test-services/coderunner-api.service.mjs +58 -0
- package/esm2022/lib/services/candidate-coding-test-services/coding-test-tour.service.mjs +89 -0
- package/esm2022/lib/services/candidate-coding-test-services/coding-test.service.mjs +490 -0
- package/esm2022/lib/services/candidate-coding-test-services/index.mjs +5 -0
- package/esm2022/lib/services/coding-test-config.service.mjs +51 -0
- package/esm2022/lib/services/configurations.service.mjs +89 -0
- package/esm2022/lib/services/lib-coding-test.service.mjs +106 -0
- package/esm2022/lib/services/storage.service.mjs +624 -0
- package/esm2022/lib/services/test-cases.service.mjs +30 -0
- package/esm2022/lib/services/theme.service.mjs +36 -0
- package/esm2022/lib/utils/additional-languages/erlang.mjs +103 -0
- package/esm2022/lib/utils/resize-element.mjs +13 -0
- package/esm2022/lib/utils/time-to-ms.util.mjs +11 -0
- package/esm2022/shared/index.mjs +5 -0
- package/esm2022/shared/lib/components/audio-animation/audio-animation.component.mjs +114 -0
- package/esm2022/shared/lib/components/audio-animation/index.mjs +2 -0
- package/esm2022/shared/lib/components/index.mjs +3 -0
- package/esm2022/shared/lib/components/vimeo-video/index.mjs +2 -0
- package/esm2022/shared/lib/components/vimeo-video/vimeo-video.component.mjs +101 -0
- package/esm2022/shared/lib/models/answer.mjs +2 -0
- package/esm2022/shared/lib/models/assessment.mjs +2 -0
- package/esm2022/shared/lib/models/environment.mjs +2 -0
- package/esm2022/shared/lib/models/index.mjs +9 -0
- package/esm2022/shared/lib/models/question-component.mjs +2 -0
- package/esm2022/shared/lib/models/question.mjs +2 -0
- package/esm2022/shared/lib/models/test.mjs +2 -0
- package/esm2022/shared/lib/models/translations.mjs +2 -0
- package/esm2022/shared/lib/models/window.mjs +2 -0
- package/esm2022/shared/lib/services/api/api.service.mjs +97 -0
- package/esm2022/shared/lib/services/api/mocked-api.service.mjs +131 -0
- package/esm2022/shared/lib/services/environment/environment.service.mjs +13 -0
- package/esm2022/shared/lib/services/index.mjs +10 -0
- package/esm2022/shared/lib/services/localization/languages.model.mjs +19 -0
- package/esm2022/shared/lib/services/localization/transloco-lazy-module-utils.mjs +27 -0
- package/esm2022/shared/lib/services/localization/transloco-testing.module.mjs +11 -0
- package/esm2022/shared/lib/services/media/media.service.mjs +129 -0
- package/esm2022/shared/lib/services/mixpanel/mixpanel.service.mjs +30 -0
- package/esm2022/shared/lib/services/theme/theme.service.mjs +24 -0
- package/esm2022/shared/test-mocks/assessment-test.mock.mjs +112 -0
- package/esm2022/shared/test-mocks/index.mjs +3 -0
- package/esm2022/shared/test-mocks/tgo-ui.mock.mjs +39 -0
- package/esm2022/testgorilla-tgo-coding-test.mjs +5 -0
- package/fesm2022/testgorilla-tgo-coding-test.mjs +4526 -0
- package/fesm2022/testgorilla-tgo-coding-test.mjs.map +1 -0
- package/{src/index.ts → index.d.ts} +2 -19
- package/lib/components/code-editor/code-editor.component.d.ts +54 -0
- package/lib/components/code-editor/code-editor.service.d.ts +12 -0
- package/lib/components/code-editor/helpers/c-helper.d.ts +11 -0
- package/lib/components/code-editor/helpers/code-editor-helper.base.d.ts +8 -0
- package/lib/components/code-editor/helpers/code-editor-helper.model.d.ts +62 -0
- package/lib/components/code-editor/helpers/cpp-helper.d.ts +11 -0
- package/lib/components/code-editor/helpers/csharp-helper.d.ts +12 -0
- package/lib/components/code-editor/helpers/go-helper.d.ts +12 -0
- package/lib/components/code-editor/helpers/java-helper.d.ts +12 -0
- package/lib/components/code-editor/helpers/javascript-helper.d.ts +10 -0
- package/lib/components/code-editor/helpers/kotlin-helper.d.ts +12 -0
- package/lib/components/code-editor/helpers/php-helper.d.ts +10 -0
- package/lib/components/code-editor/helpers/python-helper.d.ts +10 -0
- package/lib/components/code-editor/helpers/r-helper.d.ts +10 -0
- package/lib/components/code-editor/helpers/ruby-helper.d.ts +10 -0
- package/lib/components/code-editor/helpers/scala-helper.d.ts +12 -0
- package/lib/components/code-editor/helpers/sql-helper.d.ts +10 -0
- package/lib/components/code-editor/helpers/swift-helper.d.ts +11 -0
- package/lib/components/code-editor/helpers/typescript-helper.d.ts +11 -0
- package/lib/components/code-editor/models/code-editor.model.d.ts +8 -0
- package/lib/components/code-editor/models/coding-snapshot.model.d.ts +4 -0
- package/lib/components/coding-question/coding-question.component.d.ts +45 -0
- package/lib/components/coding-section/coding-section.component.d.ts +52 -0
- package/lib/components/common/truncated-text/truncated-text.component.d.ts +13 -0
- package/lib/components/configurations/configurations.component.d.ts +40 -0
- package/lib/components/instructions/instructions.component.d.ts +48 -0
- package/lib/components/panel/panel.component.d.ts +12 -0
- package/lib/components/runnable-editor/runnable-editor.component.d.ts +53 -0
- package/lib/components/tests/test-cases/test-cases.component.d.ts +46 -0
- package/lib/components/tests/test-cases-content/test-cases-content.component.d.ts +26 -0
- package/lib/components/tests/test-cases-status/test-cases-status.component.d.ts +9 -0
- package/lib/components/tests/test-results.component.d.ts +33 -0
- package/lib/components/tgo-coding-test/tgo-coding-test.component.d.ts +69 -0
- package/lib/components/tgo-coding-test-candidate-view/tgo-coding-test-candidate-view.component.d.ts +101 -0
- package/{src/lib/config/index.ts → lib/config/index.d.ts} +0 -1
- package/lib/config/tgo-coding-test.config.d.ts +24 -0
- package/{src/lib/config/tgo-coding-test.provider.ts → lib/config/tgo-coding-test.provider.d.ts} +6 -18
- package/lib/config/tgo-coding-test.token.d.ts +9 -0
- package/lib/models/auto-saved-data.d.ts +42 -0
- package/lib/models/code-event.d.ts +15 -0
- package/lib/models/coderunner-execution-results.d.ts +51 -0
- package/lib/models/configs.d.ts +15 -0
- package/lib/models/language-change-action.d.ts +4 -0
- package/lib/models/lat-languages.d.ts +8 -0
- package/lib/models/mixpanel-events.d.ts +3 -0
- package/lib/models/mode.d.ts +5 -0
- package/lib/models/paste-data.d.ts +4 -0
- package/lib/models/programming-language.d.ts +8 -0
- package/lib/models/test-cases.d.ts +67 -0
- package/lib/models/theme.d.ts +4 -0
- package/lib/models/translations.d.ts +1 -0
- package/lib/models/view-mode.d.ts +6 -0
- package/{src/lib/pipes/memoize-func.pipe.ts → lib/pipes/memoize-func.pipe.d.ts} +6 -14
- package/lib/services/candidate-coding-test-services/candidature-api.service.d.ts +10 -0
- package/lib/services/candidate-coding-test-services/coderunner-api.service.d.ts +18 -0
- package/lib/services/candidate-coding-test-services/coding-test-tour.service.d.ts +22 -0
- package/lib/services/candidate-coding-test-services/coding-test.service.d.ts +68 -0
- package/{src/lib/services/candidate-coding-test-services/index.ts → lib/services/candidate-coding-test-services/index.d.ts} +1 -1
- package/lib/services/coding-test-config.service.d.ts +38 -0
- package/lib/services/configurations.service.d.ts +37 -0
- package/lib/services/lib-coding-test.service.d.ts +69 -0
- package/lib/services/storage.service.d.ts +83 -0
- package/lib/services/test-cases.service.d.ts +12 -0
- package/lib/services/theme.service.d.ts +17 -0
- package/lib/utils/additional-languages/erlang.d.ts +37 -0
- package/lib/utils/resize-element.d.ts +2 -0
- package/lib/utils/time-to-ms.util.d.ts +1 -0
- package/package.json +18 -7
- package/shared/index.d.ts +4 -0
- package/shared/lib/components/audio-animation/audio-animation.component.d.ts +27 -0
- package/shared/lib/components/audio-animation/index.d.ts +1 -0
- package/shared/lib/components/index.d.ts +2 -0
- package/shared/lib/components/vimeo-video/index.d.ts +1 -0
- package/shared/lib/components/vimeo-video/vimeo-video.component.d.ts +24 -0
- package/shared/lib/models/answer.d.ts +17 -0
- package/shared/lib/models/assessment.d.ts +80 -0
- package/shared/lib/models/environment.d.ts +1 -0
- package/shared/lib/models/index.d.ts +8 -0
- package/shared/lib/models/question-component.d.ts +54 -0
- package/shared/lib/models/question.d.ts +102 -0
- package/shared/lib/models/test.d.ts +81 -0
- package/shared/lib/models/translations.d.ts +1 -0
- package/shared/lib/models/window.d.ts +6 -0
- package/shared/lib/services/api/api.service.d.ts +25 -0
- package/shared/lib/services/api/mocked-api.service.d.ts +35 -0
- package/shared/lib/services/environment/environment.service.d.ts +6 -0
- package/shared/lib/services/index.d.ts +9 -0
- package/shared/lib/services/localization/languages.model.d.ts +15 -0
- package/shared/lib/services/localization/transloco-lazy-module-utils.d.ts +11 -0
- package/shared/lib/services/localization/transloco-testing.module.d.ts +2 -0
- package/shared/lib/services/media/media.service.d.ts +29 -0
- package/shared/lib/services/mixpanel/mixpanel.service.d.ts +10 -0
- package/shared/lib/services/theme/theme.service.d.ts +8 -0
- package/shared/test-mocks/assessment-test.mock.d.ts +21 -0
- package/shared/test-mocks/index.d.ts +2 -0
- package/shared/test-mocks/tgo-ui.mock.d.ts +21 -0
- package/.eslintrc.json +0 -45
- package/jest.config.ts +0 -21
- package/ng-package.json +0 -14
- package/project.json +0 -36
- package/src/lib/components/.gitkeep +0 -0
- package/src/lib/components/code-editor/code-editor.component.html +0 -10
- package/src/lib/components/code-editor/code-editor.component.scss +0 -21
- package/src/lib/components/code-editor/code-editor.component.spec.ts +0 -136
- package/src/lib/components/code-editor/code-editor.component.ts +0 -369
- package/src/lib/components/code-editor/code-editor.mocks.ts +0 -28
- package/src/lib/components/code-editor/code-editor.service.spec.ts +0 -160
- package/src/lib/components/code-editor/code-editor.service.ts +0 -94
- package/src/lib/components/code-editor/helpers/c-helper.spec.ts +0 -39
- package/src/lib/components/code-editor/helpers/c-helper.ts +0 -51
- package/src/lib/components/code-editor/helpers/code-editor-helper.base.spec.ts +0 -30
- package/src/lib/components/code-editor/helpers/code-editor-helper.base.ts +0 -16
- package/src/lib/components/code-editor/helpers/code-editor-helper.mocks.ts +0 -24
- package/src/lib/components/code-editor/helpers/code-editor-helper.model.ts +0 -67
- package/src/lib/components/code-editor/helpers/cpp-helper.spec.ts +0 -40
- package/src/lib/components/code-editor/helpers/cpp-helper.ts +0 -52
- package/src/lib/components/code-editor/helpers/csharp-helper.spec.ts +0 -42
- package/src/lib/components/code-editor/helpers/csharp-helper.ts +0 -55
- package/src/lib/components/code-editor/helpers/go-helper.spec.ts +0 -41
- package/src/lib/components/code-editor/helpers/go-helper.ts +0 -54
- package/src/lib/components/code-editor/helpers/java-helper.spec.ts +0 -41
- package/src/lib/components/code-editor/helpers/java-helper.ts +0 -54
- package/src/lib/components/code-editor/helpers/javascript-helper.spec.ts +0 -39
- package/src/lib/components/code-editor/helpers/javascript-helper.ts +0 -32
- package/src/lib/components/code-editor/helpers/kotlin-helper.spec.ts +0 -41
- package/src/lib/components/code-editor/helpers/kotlin-helper.ts +0 -54
- package/src/lib/components/code-editor/helpers/php-helper.spec.ts +0 -39
- package/src/lib/components/code-editor/helpers/php-helper.ts +0 -32
- package/src/lib/components/code-editor/helpers/python-helper.spec.ts +0 -39
- package/src/lib/components/code-editor/helpers/python-helper.ts +0 -32
- package/src/lib/components/code-editor/helpers/r-helper.spec.ts +0 -39
- package/src/lib/components/code-editor/helpers/r-helper.ts +0 -32
- package/src/lib/components/code-editor/helpers/ruby-helper.spec.ts +0 -39
- package/src/lib/components/code-editor/helpers/ruby-helper.ts +0 -32
- package/src/lib/components/code-editor/helpers/scala-helper.spec.ts +0 -41
- package/src/lib/components/code-editor/helpers/scala-helper.ts +0 -53
- package/src/lib/components/code-editor/helpers/sql-helper.spec.ts +0 -87
- package/src/lib/components/code-editor/helpers/sql-helper.ts +0 -44
- package/src/lib/components/code-editor/helpers/swift-helper.spec.ts +0 -40
- package/src/lib/components/code-editor/helpers/swift-helper.ts +0 -51
- package/src/lib/components/code-editor/helpers/typescript-helper.spec.ts +0 -40
- package/src/lib/components/code-editor/helpers/typescript-helper.ts +0 -52
- package/src/lib/components/code-editor/models/code-editor.model.ts +0 -9
- package/src/lib/components/code-editor/models/coding-snapshot.model.ts +0 -4
- package/src/lib/components/coding-question/coding-question.component.html +0 -78
- package/src/lib/components/coding-question/coding-question.component.scss +0 -76
- package/src/lib/components/coding-question/coding-question.component.spec.ts +0 -85
- package/src/lib/components/coding-question/coding-question.component.ts +0 -102
- package/src/lib/components/coding-section/coding-section.component.html +0 -82
- package/src/lib/components/coding-section/coding-section.component.scss +0 -64
- package/src/lib/components/coding-section/coding-section.component.spec.ts +0 -257
- package/src/lib/components/coding-section/coding-section.component.ts +0 -187
- package/src/lib/components/coding-test.module.ts +0 -124
- package/src/lib/components/common/truncated-text/truncated-text.component.html +0 -6
- package/src/lib/components/common/truncated-text/truncated-text.component.scss +0 -18
- package/src/lib/components/common/truncated-text/truncated-text.component.spec.ts +0 -84
- package/src/lib/components/common/truncated-text/truncated-text.component.ts +0 -37
- package/src/lib/components/configurations/configurations.component.html +0 -57
- package/src/lib/components/configurations/configurations.component.scss +0 -42
- package/src/lib/components/configurations/configurations.component.spec.ts +0 -186
- package/src/lib/components/configurations/configurations.component.ts +0 -98
- package/src/lib/components/instructions/instructions.component.html +0 -41
- package/src/lib/components/instructions/instructions.component.scss +0 -167
- package/src/lib/components/instructions/instructions.component.spec.ts +0 -106
- package/src/lib/components/instructions/instructions.component.ts +0 -138
- package/src/lib/components/panel/panel.component.html +0 -19
- package/src/lib/components/panel/panel.component.scss +0 -41
- package/src/lib/components/panel/panel.component.spec.ts +0 -40
- package/src/lib/components/panel/panel.component.ts +0 -34
- package/src/lib/components/runnable-editor/runnable-editor.component.html +0 -75
- package/src/lib/components/runnable-editor/runnable-editor.component.scss +0 -55
- package/src/lib/components/runnable-editor/runnable-editor.component.spec.ts +0 -124
- package/src/lib/components/runnable-editor/runnable-editor.component.ts +0 -155
- package/src/lib/components/tests/test-cases/test-cases.component.html +0 -135
- package/src/lib/components/tests/test-cases/test-cases.component.scss +0 -220
- package/src/lib/components/tests/test-cases/test-cases.component.spec.ts +0 -401
- package/src/lib/components/tests/test-cases/test-cases.component.ts +0 -205
- package/src/lib/components/tests/test-cases-content/test-cases-content.component.html +0 -94
- package/src/lib/components/tests/test-cases-content/test-cases-content.component.scss +0 -103
- package/src/lib/components/tests/test-cases-content/test-cases-content.component.spec.ts +0 -122
- package/src/lib/components/tests/test-cases-content/test-cases-content.component.ts +0 -102
- package/src/lib/components/tests/test-cases-status/test-cases-status.component.html +0 -16
- package/src/lib/components/tests/test-cases-status/test-cases-status.component.scss +0 -49
- package/src/lib/components/tests/test-cases-status/test-cases-status.component.spec.ts +0 -22
- package/src/lib/components/tests/test-cases-status/test-cases-status.component.ts +0 -18
- package/src/lib/components/tests/test-results.component.html +0 -119
- package/src/lib/components/tests/test-results.component.scss +0 -189
- package/src/lib/components/tests/test-results.component.spec.ts +0 -140
- package/src/lib/components/tests/test-results.component.ts +0 -98
- package/src/lib/components/tgo-coding-test/tgo-coding-test.component.html +0 -96
- package/src/lib/components/tgo-coding-test/tgo-coding-test.component.scss +0 -6
- package/src/lib/components/tgo-coding-test/tgo-coding-test.component.spec.ts +0 -599
- package/src/lib/components/tgo-coding-test/tgo-coding-test.component.ts +0 -279
- package/src/lib/components/tgo-coding-test-candidate-view/tgo-coding-test-candidate-view.component.html +0 -36
- package/src/lib/components/tgo-coding-test-candidate-view/tgo-coding-test-candidate-view.component.scss +0 -183
- package/src/lib/components/tgo-coding-test-candidate-view/tgo-coding-test-candidate-view.component.spec.ts +0 -883
- package/src/lib/components/tgo-coding-test-candidate-view/tgo-coding-test-candidate-view.component.ts +0 -575
- package/src/lib/config/tgo-coding-test.config.ts +0 -26
- package/src/lib/config/tgo-coding-test.token.ts +0 -21
- package/src/lib/models/.gitkeep +0 -0
- package/src/lib/models/auto-saved-data.ts +0 -51
- package/src/lib/models/code-event.ts +0 -17
- package/src/lib/models/coderunner-execution-results.ts +0 -58
- package/src/lib/models/coding-lib.mocks.ts +0 -246
- package/src/lib/models/configs.ts +0 -18
- package/src/lib/models/language-change-action.ts +0 -4
- package/src/lib/models/lat-languages.ts +0 -12
- package/src/lib/models/mixpanel-events.ts +0 -3
- package/src/lib/models/mode.ts +0 -5
- package/src/lib/models/paste-data.ts +0 -4
- package/src/lib/models/programming-language.ts +0 -9
- package/src/lib/models/test-cases.ts +0 -74
- package/src/lib/models/theme.ts +0 -5
- package/src/lib/models/translations.ts +0 -1
- package/src/lib/models/view-mode.ts +0 -6
- package/src/lib/services/.gitkeep +0 -0
- package/src/lib/services/candidate-coding-test-services/candidature-api.service.spec.ts +0 -40
- package/src/lib/services/candidate-coding-test-services/candidature-api.service.ts +0 -15
- package/src/lib/services/candidate-coding-test-services/coderunner-api.service.spec.ts +0 -134
- package/src/lib/services/candidate-coding-test-services/coderunner-api.service.ts +0 -105
- package/src/lib/services/candidate-coding-test-services/coding-test-tour.service.spec.ts +0 -161
- package/src/lib/services/candidate-coding-test-services/coding-test-tour.service.ts +0 -100
- package/src/lib/services/candidate-coding-test-services/coding-test.service.spec.ts +0 -1524
- package/src/lib/services/candidate-coding-test-services/coding-test.service.ts +0 -843
- package/src/lib/services/coding-test-config.service.ts +0 -48
- package/src/lib/services/configurations.service.mocks.ts +0 -77
- package/src/lib/services/configurations.service.spec.ts +0 -79
- package/src/lib/services/configurations.service.ts +0 -111
- package/src/lib/services/index.ts +0 -0
- package/src/lib/services/lib-coding-test.service.spec.ts +0 -265
- package/src/lib/services/lib-coding-test.service.ts +0 -157
- package/src/lib/services/local-storage.service.mocks.ts +0 -22
- package/src/lib/services/storage.service.spec.ts +0 -1120
- package/src/lib/services/storage.service.ts +0 -729
- package/src/lib/services/test-cases.service.spec.ts +0 -53
- package/src/lib/services/test-cases.service.ts +0 -29
- package/src/lib/services/theme.service.spec.ts +0 -76
- package/src/lib/services/theme.service.ts +0 -34
- package/src/lib/styles/mixins.scss +0 -86
- package/src/lib/styles/styles.scss +0 -112
- package/src/lib/styles/variables.scss +0 -105
- package/src/lib/utils/.gitkeep +0 -0
- package/src/lib/utils/additional-languages/erlang.ts +0 -115
- package/src/lib/utils/resize-element.ts +0 -15
- package/src/lib/utils/time-to-ms.util.ts +0 -10
- package/src/test-setup.ts +0 -1
- package/tsconfig.json +0 -16
- package/tsconfig.lib.json +0 -12
- package/tsconfig.lib.prod.json +0 -9
- package/tsconfig.spec.json +0 -13
- /package/{src/assets → assets}/i18n/en.json +0 -0
- /package/{src/lib/components/code-editor/helpers/index.ts → lib/components/code-editor/helpers/index.d.ts} +0 -0
|
@@ -0,0 +1,624 @@
|
|
|
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"]}
|