@wise/dynamic-flow-client 5.8.1 → 5.9.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/build/DynamicFlowCore.js +5 -0
- package/build/common/errorBoundary/ErrorBoundary.js +26 -0
- package/build/common/errorBoundary/ErrorBoundary.test.js +29 -0
- package/build/common/errorBoundary/ErrorBoundaryAlert.js +11 -0
- package/build/common/makeHttpClient/api-utils.js +3 -0
- package/build/common/makeHttpClient/index.js +1 -0
- package/build/common/makeHttpClient/makeHttpClient.js +10 -0
- package/build/common/makeHttpClient/makeHttpClient.test.js +186 -0
- package/build/common/messages/external-confirmation.messages.js +23 -0
- package/build/common/messages/file-upload.messages.js +13 -0
- package/build/common/messages/generic-error.messages.js +18 -0
- package/build/common/messages/help.messages.js +8 -0
- package/build/common/messages/multi-file-upload.messages.js +18 -0
- package/build/common/messages/multi-select.messages.js +8 -0
- package/build/common/messages/paragraph.messages.js +13 -0
- package/build/common/messages/repeatable.messages.js +23 -0
- package/build/common/messages/search.messages.js +8 -0
- package/build/common/messages/validation.array.messages.js +13 -0
- package/build/common/messages/validation.messages.js +53 -0
- package/build/controller/FlowController.js +368 -0
- package/build/controller/executePoll.js +49 -0
- package/build/controller/executeRefresh.js +39 -0
- package/build/controller/executeRequest.js +77 -0
- package/build/controller/executeSubmission.js +69 -0
- package/build/controller/getErrorMessage.js +7 -0
- package/build/controller/getRequestAbortController.js +13 -0
- package/build/controller/getResponseType.js +35 -0
- package/build/controller/getSafeHttpClient.js +8 -0
- package/build/controller/getStepCounter.js +16 -0
- package/build/controller/handleErrorResponse.js +26 -0
- package/build/controller/makeSafeHttpClient.js +8 -0
- package/build/controller/response-utils.js +72 -0
- package/build/domain/components/AlertComponent.js +1 -0
- package/build/domain/components/AllOfComponent.js +40 -0
- package/build/domain/components/BooleanInputComponent.js +50 -0
- package/build/domain/components/BoxComponent.js +3 -0
- package/build/domain/components/ButtonComponent.js +1 -0
- package/build/domain/components/ColumnsComponent.js +3 -0
- package/build/domain/components/ConstComponent.js +18 -0
- package/build/domain/components/ContainerComponent.js +3 -0
- package/build/domain/components/DateInputComponent.js +75 -0
- package/build/domain/components/DecisionComponent.js +1 -0
- package/build/domain/components/DividerComponent.js +1 -0
- package/build/domain/components/FormComponent.js +3 -0
- package/build/domain/components/FormattedValueComponent.js +44 -0
- package/build/domain/components/HeadingComponent.js +1 -0
- package/build/domain/components/ImageComponent.js +1 -0
- package/build/domain/components/InstructionsComponent.js +1 -0
- package/build/domain/components/IntegerInputComponent.js +74 -0
- package/build/domain/components/ListComponent.js +1 -0
- package/build/domain/components/LoadingIndicatorComponent.js +1 -0
- package/build/domain/components/MarkdownComponent.js +1 -0
- package/build/domain/components/MediaComponent.js +1 -0
- package/build/domain/components/ModalComponent.js +16 -0
- package/build/domain/components/ModalLayoutComponent.js +3 -0
- package/build/domain/components/MoneyInputComponent.js +57 -0
- package/build/domain/components/MultiSelectInputComponent.js +81 -0
- package/build/domain/components/MultiUploadInputComponent.js +88 -0
- package/build/domain/components/NumberInputComponent.js +73 -0
- package/build/domain/components/ObjectComponent.js +45 -0
- package/build/domain/components/ParagraphComponent.js +1 -0
- package/build/domain/components/PersistAsyncComponent.js +92 -0
- package/build/domain/components/ProgressComponent.js +1 -0
- package/build/domain/components/RepeatableComponent.js +103 -0
- package/build/domain/components/ReviewComponent.js +1 -0
- package/build/domain/components/RootDomainComponent.js +173 -0
- package/build/domain/components/SectionComponent.js +5 -0
- package/build/domain/components/SelectInputComponent.js +88 -0
- package/build/domain/components/StatusListComponent.js +1 -0
- package/build/domain/components/SubflowDomainComponent.js +9 -0
- package/build/domain/components/TabsComponent.js +1 -0
- package/build/domain/components/TextInputComponent.js +76 -0
- package/build/domain/components/TupleComponent.js +41 -0
- package/build/domain/components/UploadInputComponent.js +83 -0
- package/build/domain/components/UpsellComponent.js +25 -0
- package/build/domain/components/searchComponent/SearchComponent.js +92 -0
- package/build/domain/components/searchComponent/SearchComponent.test.js +190 -0
- package/build/domain/components/step/ExternalConfirmationComponent.js +28 -0
- package/build/domain/components/step/StepDomainComponent.js +73 -0
- package/build/domain/components/step/ToolbarComponent.js +1 -0
- package/build/domain/components/utils/WithUpdate.js +1 -0
- package/build/domain/components/utils/component-utils.js +12 -0
- package/build/domain/components/utils/debounce.js +34 -0
- package/build/domain/components/utils/debounce.test.js +67 -0
- package/build/domain/components/utils/file-utils.js +21 -0
- package/build/domain/components/utils/file-utils.test.js +27 -0
- package/build/domain/components/utils/getRandomId.js +1 -0
- package/build/domain/components/utils/isExactLocalValueMatch.js +14 -0
- package/build/domain/components/utils/isOrWasValid.js +5 -0
- package/build/domain/components/utils/isPartialModelMatch.js +18 -0
- package/build/domain/components/utils/isPartialModelMatch.test.js +74 -0
- package/build/domain/features/eventNames.js +24 -0
- package/build/domain/features/events.js +1 -0
- package/build/domain/features/persistAsync/getComponentMultiPersistAsync.js +50 -0
- package/build/domain/features/persistAsync/getInitialPersistedState.js +7 -0
- package/build/domain/features/persistAsync/getPerformPersistAsync.js +43 -0
- package/build/domain/features/persistAsync/getPerformPersistAsync.test.js +139 -0
- package/build/domain/features/polling/getStepPolling.js +43 -0
- package/build/domain/features/polling/getStepPolling.test.js +90 -0
- package/build/domain/features/prefetch/getStepPrefetch.js +43 -0
- package/build/domain/features/prefetch/request-cache.js +49 -0
- package/build/domain/features/prefetch/request-cache.test.js +70 -0
- package/build/domain/features/refreshAfter/getStepRefreshAfter.js +24 -0
- package/build/domain/features/refreshAfter/getStepRefreshAfter.test.js +40 -0
- package/build/domain/features/schema-on-change/getDebouncedSchemaOnChange.js +50 -0
- package/build/domain/features/schema-on-change/getSchemaOnChange.js +34 -0
- package/build/domain/features/search/getPerformSearchFunction.js +75 -0
- package/build/domain/features/search/getPerformSearchFunction.test.js +301 -0
- package/build/domain/features/summary/summary-utils.js +40 -0
- package/build/domain/features/summary/summary-utils.test.js +125 -0
- package/build/domain/features/utils/http-utils.js +21 -0
- package/build/domain/features/utils/response-utils.js +9 -0
- package/build/domain/features/validation/spec-utils.js +19 -0
- package/build/domain/features/validation/validateStringPattern.js +24 -0
- package/build/domain/features/validation/validation-functions.js +6 -0
- package/build/domain/features/validation/validation-functions.test.js +108 -0
- package/build/domain/features/validation/value-checks.js +125 -0
- package/build/domain/features/validation/value-checks.test.js +262 -0
- package/build/domain/features/validationAsync/getComponentValidationAsync.js +53 -0
- package/build/domain/features/validationAsync/getComponentValidationAsync.test.js +67 -0
- package/build/domain/features/validationAsync/getInitialValidationAsyncState.js +5 -0
- package/build/domain/features/validationAsync/getPerformValidationAsync.js +45 -0
- package/build/domain/features/validationAsync/getPerformValidationAsync.test.js +143 -0
- package/build/domain/mappers/layout/alertLayoutToComponent.js +16 -0
- package/build/domain/mappers/layout/boxLayoutToComponent.js +13 -0
- package/build/domain/mappers/layout/buttonLayoutToComponent.js +77 -0
- package/build/domain/mappers/layout/columnsLayoutToComponent.js +13 -0
- package/build/domain/mappers/layout/decisionLayoutToComponent.js +22 -0
- package/build/domain/mappers/layout/deprecatedListLayoutToComponent.js +30 -0
- package/build/domain/mappers/layout/dividerLayoutToComponent.js +2 -0
- package/build/domain/mappers/layout/formLayoutToComponent.js +19 -0
- package/build/domain/mappers/layout/headingLayoutToComponent.js +12 -0
- package/build/domain/mappers/layout/imageLayoutToComponent.js +20 -0
- package/build/domain/mappers/layout/infoLayoutToComponent.js +12 -0
- package/build/domain/mappers/layout/instructionsLayoutToComponent.js +12 -0
- package/build/domain/mappers/layout/listLayoutToComponent.js +39 -0
- package/build/domain/mappers/layout/loadingIndicatorLayoutToComponent.js +9 -0
- package/build/domain/mappers/layout/markdownLayoutToComponent.js +12 -0
- package/build/domain/mappers/layout/mediaLayoutToComponent.js +12 -0
- package/build/domain/mappers/layout/modalLayoutToComponent.js +17 -0
- package/build/domain/mappers/layout/modalToComponent.js +8 -0
- package/build/domain/mappers/layout/paragraphLayoutToComponent.js +12 -0
- package/build/domain/mappers/layout/progressLayoutToComponent.js +15 -0
- package/build/domain/mappers/layout/reviewLayoutToComponent.js +48 -0
- package/build/domain/mappers/layout/searchLayoutToComponent.js +44 -0
- package/build/domain/mappers/layout/sectionLayoutToComponent.js +15 -0
- package/build/domain/mappers/layout/statusListLayoutToComponent.js +15 -0
- package/build/domain/mappers/layout/tabsLayoutToComponent.js +16 -0
- package/build/domain/mappers/layout/upsellLayoutToComponent.js +25 -0
- package/build/domain/mappers/mapLayoutToComponent.js +81 -0
- package/build/domain/mappers/mapSchemaToComponent.js +61 -0
- package/build/domain/mappers/mapSchemaToComponent.test.js +112 -0
- package/build/domain/mappers/mapStepSchemas.js +17 -0
- package/build/domain/mappers/mapStepToComponent.js +132 -0
- package/build/domain/mappers/mapStepToComponent.test.js +221 -0
- package/build/domain/mappers/mapToolbarToComponent.js +15 -0
- package/build/domain/mappers/schema/allOfSchemaToComponent.js +16 -0
- package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToComponent.js +26 -0
- package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToMultiSelectComponent.js +55 -0
- package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToMultiUploadComponent.js +67 -0
- package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToRepeatableComponent.js +57 -0
- package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToTupleComponent.js +20 -0
- package/build/domain/mappers/schema/blobSchemaToComponent.js +15 -0
- package/build/domain/mappers/schema/booleanSchemaToComponent.js +29 -0
- package/build/domain/mappers/schema/constSchemaToComponent.js +23 -0
- package/build/domain/mappers/schema/integerSchemaToComponent.js +28 -0
- package/build/domain/mappers/schema/numberSchemaToComponent.js +26 -0
- package/build/domain/mappers/schema/objectSchemaToComponent/assertDisplayOrder.js +23 -0
- package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToFormattedValueComponent.js +9 -0
- package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToMoneyInputComponent.js +119 -0
- package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToMoneyInputComponent.test.js +96 -0
- package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToObjectComponent.js +31 -0
- package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToObjectComponent.test.js +99 -0
- package/build/domain/mappers/schema/oneOfSchemaToComponent/oneOfSchemaToComponent.js +76 -0
- package/build/domain/mappers/schema/oneOfSchemaToComponent/oneOfSchemaToComponent.test.js +265 -0
- package/build/domain/mappers/schema/persistAsyncSchemaToComponent.js +29 -0
- package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToComponent.js +18 -0
- package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToComponent.test.js +133 -0
- package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToDateInputComponent.js +48 -0
- package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToTextInputComponent.js +37 -0
- package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToUploadInputComponent.js +28 -0
- package/build/domain/mappers/schema/tests/test-utils.js +16 -0
- package/build/domain/mappers/schema/types.js +1 -0
- package/build/domain/mappers/schema/utils/getPerformPersistAsyncFn.js +19 -0
- package/build/domain/mappers/schema/utils/getValidationAsyncInitialState.js +23 -0
- package/build/domain/mappers/schema/utils/mapCommonSchemaProps.js +16 -0
- package/build/domain/mappers/types.js +1 -0
- package/build/domain/mappers/utils/FeatureFlags.js +22 -0
- package/build/domain/mappers/utils/behavior-utils.js +44 -0
- package/build/domain/mappers/utils/call-to-action-utils.js +21 -0
- package/build/domain/mappers/utils/getAutocompleteString.js +76 -0
- package/build/domain/mappers/utils/getAutocompleteString.test.js +21 -0
- package/build/domain/mappers/utils/groupLayoutByPinned.js +38 -0
- package/build/domain/mappers/utils/groupLayoutByPinned.test.js +166 -0
- package/build/domain/mappers/utils/image.js +9 -0
- package/build/domain/mappers/utils/layout-utils.js +11 -0
- package/build/domain/mappers/utils/legacy-utils.js +49 -0
- package/build/domain/mappers/utils/media-utils.js +14 -0
- package/build/domain/mappers/utils/suggestions-utils.js +26 -0
- package/build/domain/mappers/utils/suggestions-utils.test.js +36 -0
- package/build/domain/mappers/utils/tags-utils.js +1 -0
- package/build/domain/mappers/utils/utils.js +35 -0
- package/build/domain/types.js +1 -0
- package/build/getSubflowCallbacks.js +38 -0
- package/build/i18n/index.js +40 -0
- package/build/index.js +8 -0
- package/build/main.js +112 -54
- package/build/main.mjs +111 -53
- package/build/renderers/CoreContainerRenderer.js +5 -0
- package/build/renderers/CoreRootRenderer.js +12 -0
- package/build/renderers/EmptyLoadingStateRenderer.js +5 -0
- package/build/renderers/getRenderFunction.js +24 -0
- package/build/renderers/getSchemaErrorMessageFunction.js +97 -0
- package/build/renderers/mappers/alertComponentToProps.js +2 -0
- package/build/renderers/mappers/allOfComponentToProps.js +6 -0
- package/build/renderers/mappers/booleanInputComponentToProps.js +5 -0
- package/build/renderers/mappers/boxComponentToProps.js +13 -0
- package/build/renderers/mappers/buttonComponentToProps.js +4 -0
- package/build/renderers/mappers/columnsComponentToProps.js +11 -0
- package/build/renderers/mappers/componentToRendererProps.js +164 -0
- package/build/renderers/mappers/constComponentToProps.js +5 -0
- package/build/renderers/mappers/containerComponentToProps.js +7 -0
- package/build/renderers/mappers/dateInputComponentToProps.js +2 -0
- package/build/renderers/mappers/decisionComponentToProps.js +16 -0
- package/build/renderers/mappers/dividerComponentToProps.js +2 -0
- package/build/renderers/mappers/externalComponentToProps.js +3 -0
- package/build/renderers/mappers/formComponentToProps.js +12 -0
- package/build/renderers/mappers/formattedValueComponentToProps.js +5 -0
- package/build/renderers/mappers/headingComponentToProps.js +2 -0
- package/build/renderers/mappers/hiddenComponentToProps.js +4 -0
- package/build/renderers/mappers/imageComponentToProps.js +2 -0
- package/build/renderers/mappers/instructionsComponentToProps.js +2 -0
- package/build/renderers/mappers/integerInputComponentToProps.js +2 -0
- package/build/renderers/mappers/listComponentToProps.js +2 -0
- package/build/renderers/mappers/loadingIndicatorComponentToProps.js +2 -0
- package/build/renderers/mappers/markdownComponentToProps.js +2 -0
- package/build/renderers/mappers/mediaComponentToProps.js +2 -0
- package/build/renderers/mappers/modalComponentToProps.js +11 -0
- package/build/renderers/mappers/modalLayoutComponentToProps.js +16 -0
- package/build/renderers/mappers/moneyInputComponentToProps.js +36 -0
- package/build/renderers/mappers/multiSelectComponentToProps.js +23 -0
- package/build/renderers/mappers/multiUploadInputComponentToProps.js +12 -0
- package/build/renderers/mappers/numberInputComponentToProps.js +2 -0
- package/build/renderers/mappers/objectComponentToProps.js +8 -0
- package/build/renderers/mappers/paragraphComponentToProps.js +2 -0
- package/build/renderers/mappers/persistAsyncComponentToProps.js +8 -0
- package/build/renderers/mappers/progressComponentToProps.js +2 -0
- package/build/renderers/mappers/repeatableComponentToProps.js +30 -0
- package/build/renderers/mappers/reviewComponentToProps.js +2 -0
- package/build/renderers/mappers/rootComponentToProps.js +21 -0
- package/build/renderers/mappers/searchComponentToProps.js +57 -0
- package/build/renderers/mappers/sectionComponentToProps.js +6 -0
- package/build/renderers/mappers/selectInputComponentToProps.js +26 -0
- package/build/renderers/mappers/statusListComponentToProps.js +2 -0
- package/build/renderers/mappers/subflowComponentToRendererProps.js +4 -0
- package/build/renderers/mappers/tabsComponentToProps.js +14 -0
- package/build/renderers/mappers/textInputComponentToProps.js +2 -0
- package/build/renderers/mappers/tupleComponentToProps.js +8 -0
- package/build/renderers/mappers/uploadInputComponentToProps.js +8 -0
- package/build/renderers/mappers/upsellComponentToProps.js +2 -0
- package/build/renderers/mappers/utils/getValidationState.js +12 -0
- package/build/renderers/mappers/utils/inputComponentToProps.js +26 -0
- package/build/renderers/mappers/utils/mapErrorsToValidationState.js +9 -0
- package/build/renderers/mappers/utils/pick.js +8 -0
- package/build/renderers/mappers/utils/selectInputOptionsToProps.js +11 -0
- package/build/renderers/stepComponentToProps.js +32 -0
- package/build/renderers/utils.js +69 -0
- package/build/renderers/utils.test.js +70 -0
- package/build/stories/dev-tools/ContainerQueries.story.js +66 -0
- package/build/stories/dev-tools/Debugger.story.js +38 -0
- package/build/stories/dev-tools/FixtureSelect.story.js +23 -0
- package/build/stories/dev-tools/TestServer.story.js +32 -0
- package/build/stories/examples/NativeFlow.story.js +80 -0
- package/build/stories/examples/Recipients.story.js +568 -0
- package/build/stories/spec/behavior/Copy.story.js +59 -0
- package/build/stories/spec/behavior/Modal.story.js +76 -0
- package/build/stories/spec/behavior/Subflow.story.js +267 -0
- package/build/stories/spec/layouts/Decision.story.js +241 -0
- package/build/stories/spec/layouts/Image.story.js +42 -0
- package/build/stories/spec/layouts/Modal.story.js +81 -0
- package/build/stories/spec/layouts/Search.story.js +325 -0
- package/build/stories/spec/layouts/Upsell.story.js +55 -0
- package/build/stories/spec/layouts/button/Button.story.js +100 -0
- package/build/stories/spec/layouts/button/PinnedButton.story.js +81 -0
- package/build/stories/spec/response/ActionResponse.story.js +66 -0
- package/build/stories/spec/schemas/MultiSelect.story.js +148 -0
- package/build/stories/spec/schemas/Upload.story.js +168 -0
- package/build/stories/spec/schemas/const/ConstLayout.story.js +159 -0
- package/build/stories/spec/schemas/const/ObjectConst.story.js +94 -0
- package/build/stories/spec/schemas/features/PersistAsync.story.js +176 -0
- package/build/stories/spec/schemas/features/ValidationAsync.story.js +103 -0
- package/build/stories/spec/schemas/object/FormattedValue.story.js +92 -0
- package/build/stories/spec/schemas/object/MoneyInput.story.js +240 -0
- package/build/stories/spec/schemas/oneOf/OneOfInitialisation.story.js +55 -0
- package/build/stories/spec/step/Controls.story.js +109 -0
- package/build/stories/spec/step/DFModal.story.js +58 -0
- package/build/stories/spec/step/Footer.story.js +70 -0
- package/build/stories/spec/step/Navigation.story.js +20 -0
- package/build/stories/spec/step/Tags.story.js +39 -0
- package/build/stories/spec/step/ToolBar.story.js +60 -0
- package/build/stories/spec/step/features/ErrorHandling.story.js +92 -0
- package/build/stories/spec/step/features/External.story.js +44 -0
- package/build/stories/spec/step/features/Polling.story.js +108 -0
- package/build/stories/spec/step/features/RefreshAfter.story.js +92 -0
- package/build/stories/spec/step/features/refresh/Refresh.story.js +258 -0
- package/build/stories/spec/step/features/refresh/RefreshWithPersistAsync.story.js +958 -0
- package/build/stories/types.js +1 -0
- package/build/stories/utils/fixtureHttpClient.js +70 -0
- package/build/stories/utils/getBasicStep.js +223 -0
- package/build/stories/utils/mockSearchHandler.js +71 -0
- package/build/stories/utils/render-utils.js +41 -0
- package/build/stories/visual-tests/layouts/NotUsingListItem.story.js +17 -0
- package/build/test-utils/DynamicFlowWise.js +32 -0
- package/build/test-utils/DynamicFlowWiseModal.js +34 -0
- package/build/test-utils/NeptuneProviders.js +11 -0
- package/build/test-utils/component-utils.js +5 -0
- package/build/test-utils/fetch-utils.js +45 -0
- package/build/test-utils/getMergedTestRenderers.js +34 -0
- package/build/test-utils/getRandomId.js +1 -0
- package/build/test-utils/index.js +3 -0
- package/build/test-utils/rtl-utils.js +7 -0
- package/build/test-utils/step-utils.js +6 -0
- package/build/test-utils/wait.js +3 -0
- package/build/tests/AlertLayout.test.js +78 -0
- package/build/tests/ArrayTuple.test.js +118 -0
- package/build/tests/ButtonLayout.test.js +308 -0
- package/build/tests/ConstLayout.test.js +95 -0
- package/build/tests/DateInput.test.js +163 -0
- package/build/tests/DecisionLayout.test.js +146 -0
- package/build/tests/DynamicFlow.test.js +147 -0
- package/build/tests/External.test.js +169 -0
- package/build/tests/Flow.test.js +328 -0
- package/build/tests/FormLayout.test.js +28 -0
- package/build/tests/FormattedValue.test.js +107 -0
- package/build/tests/ImageRenderer.test.js +78 -0
- package/build/tests/InitialAction.test.js +179 -0
- package/build/tests/InitialStep.test.js +168 -0
- package/build/tests/InstructionsLayout.test.js +45 -0
- package/build/tests/ListLayout.test.js +145 -0
- package/build/tests/Logging.test.js +53 -0
- package/build/tests/ModalBehavior.test.js +149 -0
- package/build/tests/MoneyInput.test.js +316 -0
- package/build/tests/MultiUpload.test.js +293 -0
- package/build/tests/NativeBack.test.js +267 -0
- package/build/tests/OneOfInitialisation.test.js +571 -0
- package/build/tests/PersistAsync.test.js +653 -0
- package/build/tests/Polling.test.js +617 -0
- package/build/tests/Prefetching.test.js +230 -0
- package/build/tests/RefreshAfter.test.js +63 -0
- package/build/tests/RefreshOnChange.ResponseHandling.test.js +205 -0
- package/build/tests/RefreshOnChange.test.js +233 -0
- package/build/tests/RefreshOnChange.with.Segmented.test.js +348 -0
- package/build/tests/RefreshOnChange.with.Tabs.test.js +358 -0
- package/build/tests/RefreshOnChangePreserve.test.js +224 -0
- package/build/tests/RendererProps.test.js +342 -0
- package/build/tests/Repeatable.test.js +107 -0
- package/build/tests/Rerendering.test.js +67 -0
- package/build/tests/ReviewLayout.test.js +274 -0
- package/build/tests/SchemaOnChange.test.js +133 -0
- package/build/tests/ScrollToError.test.js +217 -0
- package/build/tests/SegmentedControl.test.js +49 -0
- package/build/tests/SingleFileUpload.test.js +88 -0
- package/build/tests/StatusList.test.js +85 -0
- package/build/tests/Subflow.test.js +710 -0
- package/build/tests/Submission.ResponseHandling.test.js +557 -0
- package/build/tests/Submission.merging.test.js +202 -0
- package/build/tests/Submission.test.js +523 -0
- package/build/tests/Tags.test.js +475 -0
- package/build/tests/Upsell.test.js +126 -0
- package/build/tests/ValidationAsync.test.js +295 -0
- package/build/tests/legacy/Actions.test.js +158 -0
- package/build/tests/legacy/BackButton.test.js +114 -0
- package/build/tests/legacy/HiddenSchemas.test.js +246 -0
- package/build/tests/legacy/MultiSelect.test.js +497 -0
- package/build/tests/legacy/MultipleFileUploadSchema.test.js +341 -0
- package/build/tests/legacy/PersistAsync.blob-schema.test.js +224 -0
- package/build/tests/legacy/PersistAsync.string-schema.test.js +211 -0
- package/build/tests/legacy/RefreshStepOnChange.debouncing.test.js +209 -0
- package/build/tests/legacy/RefreshStepOnChange.test.js +424 -0
- package/build/tests/legacy/Search.test.js +437 -0
- package/build/tests/renderers/MultiSelectInputRendererProps.test.js +58 -0
- package/build/tests/renderers/SelectInputRendererProps.test.js +42 -0
- package/build/tests/renderers/TextInputRenderer.test.js +51 -0
- package/build/types/domain/components/UpsellComponent.d.ts +2 -3
- package/build/types/domain/components/UpsellComponent.d.ts.map +1 -1
- package/build/types/domain/components/searchComponent/SearchComponent.d.ts +5 -4
- package/build/types/domain/components/searchComponent/SearchComponent.d.ts.map +1 -1
- package/build/types/domain/features/search/getPerformSearchFunction.d.ts +1 -1
- package/build/types/domain/features/search/getPerformSearchFunction.d.ts.map +1 -1
- package/build/types/domain/mappers/layout/searchLayoutToComponent.d.ts +2 -1
- package/build/types/domain/mappers/layout/searchLayoutToComponent.d.ts.map +1 -1
- package/build/types/domain/mappers/layout/upsellLayoutToComponent.d.ts +1 -1
- package/build/types/domain/mappers/layout/upsellLayoutToComponent.d.ts.map +1 -1
- package/build/types/renderers/EmptyLoadingStateRenderer.d.ts +3 -0
- package/build/types/renderers/EmptyLoadingStateRenderer.d.ts.map +1 -0
- package/build/types/renderers/mappers/rootComponentToProps.d.ts.map +1 -1
- package/build/types/renderers/mappers/searchComponentToProps.d.ts +1 -1
- package/build/types/renderers/mappers/searchComponentToProps.d.ts.map +1 -1
- package/build/types/renderers/mappers/upsellComponentToProps.d.ts +2 -2
- package/build/types/renderers/mappers/upsellComponentToProps.d.ts.map +1 -1
- package/build/types/test-utils/getMergedTestRenderers.d.ts +1 -1
- package/build/types/test-utils/getMergedTestRenderers.d.ts.map +1 -1
- package/build/types/useDynamicFlow.d.ts.map +1 -1
- package/build/types.js +1 -0
- package/build/useDynamicFlow.js +104 -0
- package/build/useDynamicFlowModal.js +58 -0
- package/build/utils/analyse-step.js +14 -0
- package/build/utils/component-utils.js +8 -0
- package/build/utils/component-utils.test.js +113 -0
- package/build/utils/getScrollToTop.js +12 -0
- package/build/utils/normalise-flow-id.js +1 -0
- package/build/utils/normalise-flow-id.test.js +24 -0
- package/build/utils/openLinkInNewTab.js +10 -0
- package/build/utils/recursiveMerge.js +40 -0
- package/build/utils/recursiveMerge.test.js +93 -0
- package/build/utils/type-utils.js +21 -0
- package/build/utils/type-validators.js +11 -0
- package/build/utils/type-validators.test.js +180 -0
- package/build/utils/useStableCallback.js +15 -0
- package/package.json +4 -4
|
@@ -0,0 +1,341 @@
|
|
|
1
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
import { screen, waitFor } from '@testing-library/react';
|
|
3
|
+
import { userEvent } from '@testing-library/user-event';
|
|
4
|
+
import { vi } from 'vitest';
|
|
5
|
+
import { renderWithProviders } from '../../test-utils';
|
|
6
|
+
import DynamicFlowWise from '../../test-utils/DynamicFlowWise';
|
|
7
|
+
const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
|
|
8
|
+
// This file is a copy of the E2E test in legacy. It should remain as unchanged
|
|
9
|
+
// as possible, to ensure the new client functionality matches the old one.
|
|
10
|
+
const accepts = ['image/png', 'image/jpg', 'application/pdf'];
|
|
11
|
+
const maxSize = 5242000;
|
|
12
|
+
const description = 'PNG, JPG, or PDF, less than 5mb';
|
|
13
|
+
const idProperty = 'responseKey';
|
|
14
|
+
const param = 'bodyAttribute';
|
|
15
|
+
const uploadTitle = 'Invoice (optional)';
|
|
16
|
+
describe('Given a multiple file upload schema with persist-async', () => {
|
|
17
|
+
const getArraySchema = (overrides) => (Object.assign({ type: 'array', title: 'upload schema title', minItems: 1, maxItems: 2, addItemTitle: '', editItemTitle: '', items: {
|
|
18
|
+
type: 'string',
|
|
19
|
+
persistAsync: {
|
|
20
|
+
url: 'http://localhost:3001/v1/save-file/',
|
|
21
|
+
method: 'POST',
|
|
22
|
+
param,
|
|
23
|
+
idProperty,
|
|
24
|
+
schema: {
|
|
25
|
+
// Modification: we now read the analytics id from the inner schema
|
|
26
|
+
$id: 'upload-id',
|
|
27
|
+
type: 'string',
|
|
28
|
+
format: 'base64url',
|
|
29
|
+
title: uploadTitle,
|
|
30
|
+
description,
|
|
31
|
+
maxSize,
|
|
32
|
+
accepts,
|
|
33
|
+
validationMessages: {
|
|
34
|
+
maxSize: 'Your file is too large',
|
|
35
|
+
},
|
|
36
|
+
},
|
|
37
|
+
},
|
|
38
|
+
} }, overrides));
|
|
39
|
+
const getMultipleFileUploadSchemaObject = (arraySchema = getArraySchema({})) => ({
|
|
40
|
+
type: 'object',
|
|
41
|
+
$id: '#schema',
|
|
42
|
+
displayOrder: ['files'],
|
|
43
|
+
required: ['files'],
|
|
44
|
+
title: 'upload schema title',
|
|
45
|
+
properties: {
|
|
46
|
+
files: Object.assign({ $id: 'upload-id' }, arraySchema),
|
|
47
|
+
},
|
|
48
|
+
});
|
|
49
|
+
const getMultipleFileUploadSchemaObjectWithoutId = () => ({
|
|
50
|
+
type: 'object',
|
|
51
|
+
$id: '#schema',
|
|
52
|
+
displayOrder: ['files'],
|
|
53
|
+
required: ['files'],
|
|
54
|
+
title: 'upload schema title',
|
|
55
|
+
properties: {
|
|
56
|
+
files: Object.assign(Object.assign({}, getArraySchema({})), { minItems: 2, maxItems: 3, validationMessages: {
|
|
57
|
+
minItems: 'minimum items not met error (from schema)',
|
|
58
|
+
maxItems: 'maximum items uploaded already (from schema)',
|
|
59
|
+
} }),
|
|
60
|
+
},
|
|
61
|
+
validationMessages: {
|
|
62
|
+
minItems: 'minimum items not met error (from schema)',
|
|
63
|
+
maxItems: 'maximum items uploaded already (from schema)',
|
|
64
|
+
},
|
|
65
|
+
});
|
|
66
|
+
const testFile = new File([], 'test-file.pdf', { type: 'application/pdf' });
|
|
67
|
+
const mockHttpClient = vi.fn();
|
|
68
|
+
const initialStep = {
|
|
69
|
+
id: 'multiple-file-upload-schema-test-step-id',
|
|
70
|
+
title: 'OneOf Schema Step Title',
|
|
71
|
+
layout: [
|
|
72
|
+
{ type: 'form', schemaId: '#schema' },
|
|
73
|
+
{
|
|
74
|
+
type: 'button',
|
|
75
|
+
title: 'Submit',
|
|
76
|
+
action: { url: '/submit', method: 'POST' },
|
|
77
|
+
},
|
|
78
|
+
],
|
|
79
|
+
schemas: [getMultipleFileUploadSchemaObject()],
|
|
80
|
+
model: {},
|
|
81
|
+
errors: {},
|
|
82
|
+
};
|
|
83
|
+
const onEvent = vi.fn();
|
|
84
|
+
const getProps = (overrides = {}) => (Object.assign({ flowId: 'multiple-file-upload-schema-test-flow-id', initialStep, httpClient: mockHttpClient, onCompletion: vi.fn(), onError: vi.fn(), onEvent, onLog: vi.fn() }, overrides));
|
|
85
|
+
// We have some duplicate key warnings because we're mocking the random id generator
|
|
86
|
+
// But I want to keep this file as close to the original as possible, so I'm not going to fix it
|
|
87
|
+
const originalError = console.error;
|
|
88
|
+
beforeEach(() => {
|
|
89
|
+
mockHttpClient.mockResolvedValue({
|
|
90
|
+
ok: true,
|
|
91
|
+
status: 200,
|
|
92
|
+
json: vi.fn().mockResolvedValue({ [idProperty]: 'mockFileId' }),
|
|
93
|
+
});
|
|
94
|
+
console.error = vi.fn();
|
|
95
|
+
});
|
|
96
|
+
afterEach(() => {
|
|
97
|
+
vi.clearAllMocks();
|
|
98
|
+
onEvent.mockClear();
|
|
99
|
+
console.error = originalError;
|
|
100
|
+
});
|
|
101
|
+
describe('and the schema has an $id prop', () => {
|
|
102
|
+
// We use the inner id now (and it doesn't really matter anyway)
|
|
103
|
+
it.skip('should be the $id given to the input component', () => {
|
|
104
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, getProps())));
|
|
105
|
+
expect(screen.getByLabelText('upload schema title')).toBeInTheDocument();
|
|
106
|
+
expect(screen.getByLabelText('upload schema title')).toHaveAttribute('id', 'upload-id');
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
describe('and the schema has no $id prop', () => {
|
|
110
|
+
it.skip('should be a generated id given to the input component', () => {
|
|
111
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, getProps({
|
|
112
|
+
initialStep: Object.assign(Object.assign({}, initialStep), { schemas: [getMultipleFileUploadSchemaObjectWithoutId()] }),
|
|
113
|
+
}))));
|
|
114
|
+
expect(screen.getByLabelText('upload schema title')).toBeInTheDocument();
|
|
115
|
+
expect(screen.getByLabelText('upload schema title')).toHaveAttribute('id', 'DUMMY_GENERATED_UNIQUE_ID');
|
|
116
|
+
});
|
|
117
|
+
});
|
|
118
|
+
it('should render the file input', () => {
|
|
119
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, getProps())));
|
|
120
|
+
expect(screen.getByLabelText('upload schema title')).toHaveAttribute('type', 'file');
|
|
121
|
+
});
|
|
122
|
+
describe('when upload successful', () => {
|
|
123
|
+
it('the files should show as uploaded', async () => {
|
|
124
|
+
const props = getProps();
|
|
125
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
|
|
126
|
+
onEvent.mockClear();
|
|
127
|
+
await user.upload(screen.getByLabelText('upload schema title'), testFile);
|
|
128
|
+
await waitFor(() => expect(onEvent).toHaveBeenCalledTimes(2));
|
|
129
|
+
// Modification (we've changed the events):
|
|
130
|
+
expect(onEvent).toHaveBeenCalledWith('Dynamic Flow - PersistAsync Triggered', expect.objectContaining({
|
|
131
|
+
flowId: 'multiple-file-upload-schema-test-flow-id',
|
|
132
|
+
stepId: 'multiple-file-upload-schema-test-step-id',
|
|
133
|
+
schema: 'upload-id',
|
|
134
|
+
}));
|
|
135
|
+
expect(onEvent).toHaveBeenCalledWith('Dynamic Flow - PersistAsync Succeeded', expect.objectContaining({
|
|
136
|
+
flowId: 'multiple-file-upload-schema-test-flow-id',
|
|
137
|
+
stepId: 'multiple-file-upload-schema-test-step-id',
|
|
138
|
+
schema: 'upload-id',
|
|
139
|
+
}));
|
|
140
|
+
// Original:
|
|
141
|
+
// expect(onEvent).toHaveBeenCalledWith(
|
|
142
|
+
// 'Dynamic Flow - PersistAsync',
|
|
143
|
+
// expect.objectContaining({ status: 'pending' }),
|
|
144
|
+
// );
|
|
145
|
+
// expect(onEvent).toHaveBeenCalledWith(
|
|
146
|
+
// 'Dynamic Flow - PersistAsync',
|
|
147
|
+
// expect.objectContaining({ status: 'success' }),
|
|
148
|
+
// );
|
|
149
|
+
expect(screen.getByText('test-file.pdf')).toBeInTheDocument();
|
|
150
|
+
expect(screen.getByText('Uploaded')).toBeInTheDocument();
|
|
151
|
+
});
|
|
152
|
+
it('the file IDs should be in the payload on submission', async () => {
|
|
153
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, getProps())));
|
|
154
|
+
onEvent.mockClear();
|
|
155
|
+
await user.upload(screen.getByLabelText('upload schema title'), testFile);
|
|
156
|
+
await waitFor(() => expect(onEvent).toHaveBeenCalledTimes(2));
|
|
157
|
+
await user.click(screen.getByText('Submit'));
|
|
158
|
+
await waitFor(() => expect(mockHttpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({ body: JSON.stringify({ files: ['mockFileId'] }) })));
|
|
159
|
+
});
|
|
160
|
+
});
|
|
161
|
+
describe('when upload fails', () => {
|
|
162
|
+
it('shows error message to the user', async () => {
|
|
163
|
+
const errorMessage = 'Error uploading';
|
|
164
|
+
mockHttpClient.mockResolvedValueOnce({
|
|
165
|
+
ok: false,
|
|
166
|
+
status: 401,
|
|
167
|
+
json: vi.fn().mockResolvedValue({ validation: { [param]: errorMessage } }),
|
|
168
|
+
});
|
|
169
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, getProps())));
|
|
170
|
+
onEvent.mockClear();
|
|
171
|
+
await user.upload(screen.getByLabelText('upload schema title'), testFile);
|
|
172
|
+
await waitFor(() => expect(onEvent).toHaveBeenCalledTimes(2));
|
|
173
|
+
// Modification (we've changed the events):
|
|
174
|
+
expect(onEvent).toHaveBeenCalledWith('Dynamic Flow - PersistAsync Triggered', expect.objectContaining({
|
|
175
|
+
flowId: 'multiple-file-upload-schema-test-flow-id',
|
|
176
|
+
stepId: 'multiple-file-upload-schema-test-step-id',
|
|
177
|
+
schema: 'upload-id',
|
|
178
|
+
}));
|
|
179
|
+
expect(onEvent).toHaveBeenCalledWith('Dynamic Flow - PersistAsync Failed', expect.objectContaining({
|
|
180
|
+
flowId: 'multiple-file-upload-schema-test-flow-id',
|
|
181
|
+
stepId: 'multiple-file-upload-schema-test-step-id',
|
|
182
|
+
schema: 'upload-id',
|
|
183
|
+
}));
|
|
184
|
+
// Original
|
|
185
|
+
// expect(onEvent).toHaveBeenCalledWith(
|
|
186
|
+
// 'Dynamic Flow - PersistAsync',
|
|
187
|
+
// expect.objectContaining({ status: 'pending' }),
|
|
188
|
+
// );
|
|
189
|
+
// expect(onEvent).toHaveBeenCalledWith(
|
|
190
|
+
// 'Dynamic Flow - PersistAsync',
|
|
191
|
+
// expect.objectContaining({ status: 'failure' }),
|
|
192
|
+
// );
|
|
193
|
+
expect(screen.getByText('test-file.pdf')).toBeInTheDocument();
|
|
194
|
+
expect(screen.getByText('Error uploading')).toBeInTheDocument();
|
|
195
|
+
});
|
|
196
|
+
it('model is NOT updated', async () => {
|
|
197
|
+
const errorMessage = 'Error uploading';
|
|
198
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, getProps())));
|
|
199
|
+
onEvent.mockClear();
|
|
200
|
+
await user.upload(screen.getByLabelText('upload schema title'), testFile);
|
|
201
|
+
mockHttpClient.mockResolvedValueOnce({
|
|
202
|
+
ok: false,
|
|
203
|
+
status: 401,
|
|
204
|
+
json: vi.fn().mockResolvedValue({ validation: { [param]: errorMessage } }),
|
|
205
|
+
});
|
|
206
|
+
await user.upload(screen.getByLabelText('upload schema title'), new File([], 'test-file2.pdf', { type: 'application/pdf' }));
|
|
207
|
+
await waitFor(() => expect(onEvent).toHaveBeenCalledTimes(4));
|
|
208
|
+
await user.click(screen.getByText('Submit'));
|
|
209
|
+
// only valid file is added to model
|
|
210
|
+
await waitFor(() => expect(mockHttpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({ body: JSON.stringify({ files: ['mockFileId'] }) })));
|
|
211
|
+
});
|
|
212
|
+
});
|
|
213
|
+
describe('when uploading fewer files than required', () => {
|
|
214
|
+
it('should show default min items error when submitted with no. of files less than expected if no custom validation message is given', async () => {
|
|
215
|
+
const props = getProps({
|
|
216
|
+
initialStep: {
|
|
217
|
+
id: 'multiple-file-upload-schema-test-step-id',
|
|
218
|
+
title: 'OneOf Schema Step Title',
|
|
219
|
+
layout: [
|
|
220
|
+
{ type: 'form', schemaId: '#schema' },
|
|
221
|
+
{
|
|
222
|
+
type: 'button',
|
|
223
|
+
title: 'Submit',
|
|
224
|
+
action: { url: '/submit', method: 'POST' },
|
|
225
|
+
},
|
|
226
|
+
],
|
|
227
|
+
schemas: [
|
|
228
|
+
getMultipleFileUploadSchemaObject(getArraySchema({ minItems: 2, maxItems: 2 })),
|
|
229
|
+
],
|
|
230
|
+
model: {},
|
|
231
|
+
errors: {},
|
|
232
|
+
},
|
|
233
|
+
});
|
|
234
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
|
|
235
|
+
await user.upload(screen.getByLabelText('upload schema title'), testFile);
|
|
236
|
+
await user.click(screen.getByText('Submit'));
|
|
237
|
+
await waitFor(() => {
|
|
238
|
+
expect(screen.getByText('Please upload at least 2 file(s).')).toBeInTheDocument();
|
|
239
|
+
});
|
|
240
|
+
});
|
|
241
|
+
it('should show min items error from schema, when submitted with no. of files less than expected', async () => {
|
|
242
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, getProps({
|
|
243
|
+
initialStep: Object.assign(Object.assign({}, initialStep), { schemas: [getMultipleFileUploadSchemaObjectWithoutId()] }),
|
|
244
|
+
}))));
|
|
245
|
+
await user.upload(screen.getByLabelText('upload schema title'), testFile);
|
|
246
|
+
await user.click(screen.getByText('Submit'));
|
|
247
|
+
expect(screen.getByText('minimum items not met error (from schema)')).toBeInTheDocument();
|
|
248
|
+
});
|
|
249
|
+
});
|
|
250
|
+
// Currently Neptune's UploadInput handles all the max upload errors on drag-drop, bulk upload cases.
|
|
251
|
+
describe('when trying to upload more files than are allowed', () => {
|
|
252
|
+
it('should show the maximum allowed number of file uploads', async () => {
|
|
253
|
+
let it = 0;
|
|
254
|
+
const httpClient = vi.fn().mockImplementation(() => ({
|
|
255
|
+
ok: true,
|
|
256
|
+
status: 200,
|
|
257
|
+
// eslint-disable-next-line no-plusplus
|
|
258
|
+
json: vi.fn().mockResolvedValue({ [idProperty]: `mockFileId${it++}` }),
|
|
259
|
+
}));
|
|
260
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, getProps({
|
|
261
|
+
initialStep: Object.assign(Object.assign({}, initialStep), { schemas: [getMultipleFileUploadSchemaObjectWithoutId()] }),
|
|
262
|
+
httpClient,
|
|
263
|
+
}))));
|
|
264
|
+
onEvent.mockClear();
|
|
265
|
+
// upload 4 items
|
|
266
|
+
await user.upload(screen.getByLabelText('upload schema title'), new File([], 'test-file1.pdf', { type: 'application/pdf' }));
|
|
267
|
+
await user.upload(screen.getByLabelText('upload schema title'), new File([], 'test-file2.pdf', { type: 'application/pdf' }));
|
|
268
|
+
await user.upload(screen.getByLabelText('upload schema title'), new File([], 'test-file3.pdf', { type: 'application/pdf' }));
|
|
269
|
+
await user.upload(screen.getByLabelText('upload schema title'), new File([], 'test-file4.pdf', { type: 'application/pdf' }));
|
|
270
|
+
// only 3 shown
|
|
271
|
+
await expect(waitFor(async () => screen.findAllByText('Uploaded'))).resolves.toHaveLength(3);
|
|
272
|
+
await user.click(screen.getByText('Submit'));
|
|
273
|
+
await waitFor(() => expect(httpClient).toHaveBeenCalledWith('/submit', expect.objectContaining({
|
|
274
|
+
body: JSON.stringify({ files: ['mockFileId0', 'mockFileId1', 'mockFileId2'] }),
|
|
275
|
+
})));
|
|
276
|
+
});
|
|
277
|
+
});
|
|
278
|
+
});
|
|
279
|
+
describe('Given a multiple file upload schema without persist-async', () => {
|
|
280
|
+
const initialStep = {
|
|
281
|
+
id: 'multiple-file-upload-schema-test-step-id',
|
|
282
|
+
title: 'OneOf Schema Step Title',
|
|
283
|
+
layout: [
|
|
284
|
+
{ type: 'form', schemaId: '#schema' },
|
|
285
|
+
{
|
|
286
|
+
type: 'button',
|
|
287
|
+
title: 'Refresh',
|
|
288
|
+
behavior: { type: 'refresh' },
|
|
289
|
+
},
|
|
290
|
+
],
|
|
291
|
+
refreshUrl: '/refresh',
|
|
292
|
+
schemas: [
|
|
293
|
+
{
|
|
294
|
+
type: 'object',
|
|
295
|
+
$id: '#schema',
|
|
296
|
+
displayOrder: ['files'],
|
|
297
|
+
required: ['files'],
|
|
298
|
+
title: 'upload schema title',
|
|
299
|
+
properties: {
|
|
300
|
+
files: {
|
|
301
|
+
$id: 'upload-id',
|
|
302
|
+
type: 'array',
|
|
303
|
+
title: 'upload schema title',
|
|
304
|
+
minItems: 1,
|
|
305
|
+
maxItems: 2,
|
|
306
|
+
addItemTitle: '',
|
|
307
|
+
editItemTitle: '',
|
|
308
|
+
items: {
|
|
309
|
+
type: 'string',
|
|
310
|
+
$id: 'upload-id',
|
|
311
|
+
format: 'base64url',
|
|
312
|
+
title: uploadTitle,
|
|
313
|
+
description,
|
|
314
|
+
maxSize,
|
|
315
|
+
accepts,
|
|
316
|
+
validationMessages: {
|
|
317
|
+
maxSize: 'Your file is too large',
|
|
318
|
+
},
|
|
319
|
+
},
|
|
320
|
+
},
|
|
321
|
+
},
|
|
322
|
+
},
|
|
323
|
+
],
|
|
324
|
+
model: {},
|
|
325
|
+
errors: {},
|
|
326
|
+
};
|
|
327
|
+
const getProps = (overrides = {}) => (Object.assign({ flowId: 'multiple-file-upload-schema-test-flow-id', initialStep, httpClient: vi.fn(), onCompletion: vi.fn(), onError: vi.fn(), onEvent: vi.fn(), onLog: vi.fn() }, overrides));
|
|
328
|
+
it('should contain the base64uri of the selected files in the payload for a refresh request', async () => {
|
|
329
|
+
const props = getProps();
|
|
330
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
|
|
331
|
+
// first upload 2 files
|
|
332
|
+
await user.upload(screen.getByLabelText('upload schema title'), new File([], 'mockFile0.pdf', { type: 'application/pdf' }));
|
|
333
|
+
await user.upload(screen.getByLabelText('upload schema title'), new File([], 'mockFile1.pdf', { type: 'application/pdf' }));
|
|
334
|
+
await user.click(screen.getByText('Refresh'));
|
|
335
|
+
await waitFor(() => expect(props.httpClient).toHaveBeenCalledWith('/refresh', expect.objectContaining({
|
|
336
|
+
body: JSON.stringify({
|
|
337
|
+
files: ['data:application/pdf;base64,', 'data:application/pdf;base64,'],
|
|
338
|
+
}),
|
|
339
|
+
})));
|
|
340
|
+
});
|
|
341
|
+
});
|
|
@@ -0,0 +1,224 @@
|
|
|
1
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
import { screen, waitFor } from '@testing-library/react';
|
|
3
|
+
import { userEvent } from '@testing-library/user-event';
|
|
4
|
+
import { vi } from 'vitest';
|
|
5
|
+
import { renderWithProviders } from '../../test-utils';
|
|
6
|
+
import DynamicFlowWise from '../../test-utils/DynamicFlowWise';
|
|
7
|
+
const user = userEvent.setup({ applyAccept: false });
|
|
8
|
+
// This file is a copy of the E2E test in legacy. It should remain as unchanged
|
|
9
|
+
// as possible, to ensure the new client functionality matches the old one.
|
|
10
|
+
describe('PersistAsync with blob schema', () => {
|
|
11
|
+
const schema = {
|
|
12
|
+
$id: '#schema',
|
|
13
|
+
type: 'object',
|
|
14
|
+
title: 'File Upload',
|
|
15
|
+
displayOrder: ['file'],
|
|
16
|
+
properties: {
|
|
17
|
+
file: {
|
|
18
|
+
$id: '#file-upload-schema',
|
|
19
|
+
type: 'integer',
|
|
20
|
+
title: 'File Upload',
|
|
21
|
+
description: 'Upload a JPG or PNG file (max 150kb)',
|
|
22
|
+
persistAsync: {
|
|
23
|
+
method: 'POST',
|
|
24
|
+
url: '/upload-file-url',
|
|
25
|
+
param: 'file',
|
|
26
|
+
idProperty: 'token',
|
|
27
|
+
schema: {
|
|
28
|
+
$id: '#persist-async-schema',
|
|
29
|
+
type: 'blob',
|
|
30
|
+
maxSize: 150000,
|
|
31
|
+
accepts: ['image/png', 'image/jpg'],
|
|
32
|
+
},
|
|
33
|
+
},
|
|
34
|
+
},
|
|
35
|
+
},
|
|
36
|
+
};
|
|
37
|
+
const model = {};
|
|
38
|
+
const initialStep = {
|
|
39
|
+
id: 'persist-async-blob-step-id',
|
|
40
|
+
title: 'Persist Async Blob Test',
|
|
41
|
+
layout: [
|
|
42
|
+
{ type: 'form', schemaId: '#schema' },
|
|
43
|
+
{
|
|
44
|
+
type: 'button',
|
|
45
|
+
title: 'Submit',
|
|
46
|
+
action: { url: '/submit', method: 'POST' },
|
|
47
|
+
},
|
|
48
|
+
],
|
|
49
|
+
schemas: [schema],
|
|
50
|
+
model,
|
|
51
|
+
};
|
|
52
|
+
const getProps = (overrides = {}) => (Object.assign({ flowId: 'flow-id', initialStep, httpClient: vi.fn(), onCompletion: vi.fn(), onError: vi.fn(), onEvent: vi.fn(), onLog: vi.fn() }, overrides));
|
|
53
|
+
describe('when initialised', () => {
|
|
54
|
+
it('renders a file upload control', () => {
|
|
55
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, getProps())));
|
|
56
|
+
const input = screen.getByTestId('uploadInput');
|
|
57
|
+
expect(input).toBeInTheDocument();
|
|
58
|
+
expect(input.tagName).toBe('INPUT');
|
|
59
|
+
expect(input).toHaveAttribute('type', 'file');
|
|
60
|
+
});
|
|
61
|
+
describe('when the schema has an $id property', () => {
|
|
62
|
+
// We don't care about this - we're only going to use the inner ID
|
|
63
|
+
it.skip('renders the UploadInput with an id prop based on the schema $id', () => {
|
|
64
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, getProps())));
|
|
65
|
+
expect(screen.getByTestId('uploadInput')).toHaveAttribute('id', '#file-upload-schema');
|
|
66
|
+
});
|
|
67
|
+
});
|
|
68
|
+
describe('when the main schema has no $id, but the persist async schema has an $id property', () => {
|
|
69
|
+
it('renders the UploadInput with an id prop based on the persist async schema $id', () => {
|
|
70
|
+
const schemaWithNoId = {
|
|
71
|
+
$id: '#schema',
|
|
72
|
+
type: 'object',
|
|
73
|
+
title: 'File Upload',
|
|
74
|
+
displayOrder: ['file'],
|
|
75
|
+
properties: {
|
|
76
|
+
file: {
|
|
77
|
+
type: 'integer',
|
|
78
|
+
title: 'File Upload',
|
|
79
|
+
description: 'Upload a JPG or PNG file (max 150kb)',
|
|
80
|
+
persistAsync: {
|
|
81
|
+
method: 'POST',
|
|
82
|
+
url: '/upload-file-url',
|
|
83
|
+
param: 'file',
|
|
84
|
+
idProperty: 'token',
|
|
85
|
+
schema: {
|
|
86
|
+
$id: '#persist-async-schema',
|
|
87
|
+
type: 'blob',
|
|
88
|
+
maxSize: 150000,
|
|
89
|
+
accepts: ['image/png', 'image/jpg'],
|
|
90
|
+
},
|
|
91
|
+
},
|
|
92
|
+
},
|
|
93
|
+
},
|
|
94
|
+
};
|
|
95
|
+
const props = getProps({ initialStep: Object.assign(Object.assign({}, initialStep), { schemas: [schemaWithNoId] }) });
|
|
96
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
|
|
97
|
+
expect(screen.getByTestId('uploadInput')).toHaveAttribute('id', '#persist-async-schema');
|
|
98
|
+
});
|
|
99
|
+
});
|
|
100
|
+
describe('when none of the schemas have an $id', () => {
|
|
101
|
+
// This is strange behaviour which we're not going to replicate
|
|
102
|
+
it.skip('renders the UploadInput with an id prop based on the idProperty', () => {
|
|
103
|
+
const schemaWithNoId = {
|
|
104
|
+
$id: '#schema',
|
|
105
|
+
type: 'object',
|
|
106
|
+
title: 'File Upload',
|
|
107
|
+
displayOrder: ['file'],
|
|
108
|
+
properties: {
|
|
109
|
+
file: {
|
|
110
|
+
type: 'integer',
|
|
111
|
+
title: 'File Upload',
|
|
112
|
+
description: 'Upload a JPG or PNG file (max 150kb)',
|
|
113
|
+
persistAsync: {
|
|
114
|
+
method: 'POST',
|
|
115
|
+
url: '/upload-file-url',
|
|
116
|
+
param: 'file',
|
|
117
|
+
idProperty: 'token',
|
|
118
|
+
schema: {
|
|
119
|
+
type: 'blob',
|
|
120
|
+
maxSize: 150000,
|
|
121
|
+
accepts: ['image/png', 'image/jpg'],
|
|
122
|
+
},
|
|
123
|
+
},
|
|
124
|
+
},
|
|
125
|
+
},
|
|
126
|
+
};
|
|
127
|
+
const props = getProps({ initialStep: Object.assign(Object.assign({}, initialStep), { schemas: [schemaWithNoId] }) });
|
|
128
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
|
|
129
|
+
expect(screen.getByTestId('uploadInput')).toHaveAttribute('id', 'token');
|
|
130
|
+
});
|
|
131
|
+
});
|
|
132
|
+
});
|
|
133
|
+
describe('when a file is selected', () => {
|
|
134
|
+
it('calls the httpClient to upload the file with a FormData body containing the file', async () => {
|
|
135
|
+
var _a, _b;
|
|
136
|
+
const httpClient = vi.fn(async () => Promise.resolve(new Response(JSON.stringify({ token: 123 }))));
|
|
137
|
+
const props = getProps({ httpClient });
|
|
138
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
|
|
139
|
+
const file = new File(['foo'], 'foo.png', { type: 'image/png' });
|
|
140
|
+
await user.upload(screen.getByTestId('uploadInput'), file);
|
|
141
|
+
await waitFor(() => expect(httpClient).toHaveBeenCalledTimes(1));
|
|
142
|
+
// Modification (call now has headers and signal):
|
|
143
|
+
expect(httpClient).toHaveBeenCalledWith('/upload-file-url', expect.objectContaining({ method: 'POST', body: expect.any(FormData) }));
|
|
144
|
+
// Original:
|
|
145
|
+
// expect(httpClient).toHaveBeenCalledWith('/upload-file-url', {
|
|
146
|
+
// method: 'POST',
|
|
147
|
+
// body: expect.any(FormData) as FormData,
|
|
148
|
+
// });
|
|
149
|
+
const formData = (_b = (_a = httpClient.mock.calls[0]) === null || _a === void 0 ? void 0 : _a[1]) === null || _b === void 0 ? void 0 : _b.body;
|
|
150
|
+
expect(formData.get('file')).toBe(file);
|
|
151
|
+
});
|
|
152
|
+
describe('when the file is too large', () => {
|
|
153
|
+
it('does not call the httpClient and displays an error message', async () => {
|
|
154
|
+
const httpClient = vi.fn();
|
|
155
|
+
const props = getProps({ httpClient });
|
|
156
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
|
|
157
|
+
const file = new File(['foo'], 'foo.png', { type: 'image/png' });
|
|
158
|
+
// eslint-disable-next-line functional/immutable-data
|
|
159
|
+
Object.defineProperty(file, 'size', { value: 150001 });
|
|
160
|
+
await user.upload(screen.getByTestId('uploadInput'), file);
|
|
161
|
+
await waitFor(() => {
|
|
162
|
+
expect(screen.getByText('File is too large')).toBeInTheDocument();
|
|
163
|
+
});
|
|
164
|
+
expect(httpClient).not.toHaveBeenCalled();
|
|
165
|
+
});
|
|
166
|
+
});
|
|
167
|
+
describe('when the file is not of the correct type', () => {
|
|
168
|
+
it('does not call the httpClient and displays an error message', async () => {
|
|
169
|
+
const httpClient = vi.fn();
|
|
170
|
+
const props = getProps({ httpClient });
|
|
171
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
|
|
172
|
+
const file = new File(['foo'], 'foo.gif', { type: 'image/gif' });
|
|
173
|
+
await user.upload(screen.getByTestId('uploadInput'), file);
|
|
174
|
+
await waitFor(() => {
|
|
175
|
+
expect(screen.getByText('File type not supported')).toBeInTheDocument();
|
|
176
|
+
});
|
|
177
|
+
expect(httpClient).not.toHaveBeenCalled();
|
|
178
|
+
});
|
|
179
|
+
});
|
|
180
|
+
describe('when a file is successfully uploaded', () => {
|
|
181
|
+
it('displays the file name on screen with the "Uploaded" tag', async () => {
|
|
182
|
+
const httpClient = vi.fn(async () => Promise.resolve(new Response(JSON.stringify({ token: 123 }))));
|
|
183
|
+
const props = getProps({ httpClient });
|
|
184
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
|
|
185
|
+
await user.upload(screen.getByTestId('uploadInput'), new File(['foo'], 'foo.png', { type: 'image/png' }));
|
|
186
|
+
await waitFor(() => {
|
|
187
|
+
expect(screen.getByText('Uploaded')).toBeInTheDocument();
|
|
188
|
+
});
|
|
189
|
+
expect(screen.getByText('foo.png')).toBeInTheDocument();
|
|
190
|
+
});
|
|
191
|
+
});
|
|
192
|
+
describe('when a file fails to upload', () => {
|
|
193
|
+
describe('when there is a general failure', () => {
|
|
194
|
+
it('displays an error message', async () => {
|
|
195
|
+
const httpClient = vi.fn(async () => Promise.resolve(new Response(null, { status: 500 })));
|
|
196
|
+
const props = getProps({ httpClient });
|
|
197
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
|
|
198
|
+
await user.upload(screen.getByTestId('uploadInput'), new File(['foo'], 'foo.png', { type: 'image/png' }));
|
|
199
|
+
// Modification (we now use our general error message, not Neptune's):
|
|
200
|
+
await waitFor(() => {
|
|
201
|
+
expect(screen.getByText('Something went wrong.')).toBeInTheDocument();
|
|
202
|
+
});
|
|
203
|
+
// Original:
|
|
204
|
+
// await waitFor(() => {
|
|
205
|
+
// expect(screen.getByText('Uploading failed')).toBeInTheDocument();
|
|
206
|
+
// });
|
|
207
|
+
});
|
|
208
|
+
});
|
|
209
|
+
describe('when there is a validation error', () => {
|
|
210
|
+
it('displays the validation error message provided in the response', async () => {
|
|
211
|
+
const httpClient = vi.fn(async () => Promise.resolve(new Response(JSON.stringify({
|
|
212
|
+
validation: { file: 'The server overlords dislike the uploaded file.' },
|
|
213
|
+
}), { status: 401 })));
|
|
214
|
+
const props = getProps({ httpClient });
|
|
215
|
+
renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
|
|
216
|
+
await user.upload(screen.getByTestId('uploadInput'), new File(['foo'], 'foo.png', { type: 'image/png' }));
|
|
217
|
+
await waitFor(() => {
|
|
218
|
+
expect(screen.getByText('The server overlords dislike the uploaded file.')).toBeInTheDocument();
|
|
219
|
+
});
|
|
220
|
+
});
|
|
221
|
+
});
|
|
222
|
+
});
|
|
223
|
+
});
|
|
224
|
+
});
|