@wise/dynamic-flow-client 5.12.0 → 5.12.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (405) hide show
  1. package/build/DynamicFlowCore.js +5 -0
  2. package/build/common/errorBoundary/ErrorBoundary.js +26 -0
  3. package/build/common/errorBoundary/ErrorBoundary.test.js +29 -0
  4. package/build/common/errorBoundary/ErrorBoundaryAlert.js +11 -0
  5. package/build/common/makeHttpClient/api-utils.js +3 -0
  6. package/build/common/makeHttpClient/index.js +1 -0
  7. package/build/common/makeHttpClient/makeHttpClient.js +10 -0
  8. package/build/common/makeHttpClient/makeHttpClient.test.js +186 -0
  9. package/build/common/messages/external-confirmation.messages.js +23 -0
  10. package/build/common/messages/file-upload.messages.js +13 -0
  11. package/build/common/messages/generic-error.messages.js +18 -0
  12. package/build/common/messages/help.messages.js +8 -0
  13. package/build/common/messages/multi-file-upload.messages.js +18 -0
  14. package/build/common/messages/multi-select.messages.js +8 -0
  15. package/build/common/messages/paragraph.messages.js +13 -0
  16. package/build/common/messages/repeatable.messages.js +23 -0
  17. package/build/common/messages/search.messages.js +8 -0
  18. package/build/common/messages/validation.array.messages.js +13 -0
  19. package/build/common/messages/validation.messages.js +53 -0
  20. package/build/controller/FlowController.js +373 -0
  21. package/build/controller/executePoll.js +64 -0
  22. package/build/controller/executeRefresh.js +45 -0
  23. package/build/controller/executeRequest.js +80 -0
  24. package/build/controller/executeSubmission.js +70 -0
  25. package/build/controller/getErrorMessage.js +7 -0
  26. package/build/controller/getRequestAbortController.js +13 -0
  27. package/build/controller/getResponseType.js +35 -0
  28. package/build/controller/getSafeHttpClient.js +8 -0
  29. package/build/controller/getStepCounter.js +16 -0
  30. package/build/controller/handleErrorResponse.js +26 -0
  31. package/build/controller/makeSafeHttpClient.js +8 -0
  32. package/build/controller/response-utils.js +72 -0
  33. package/build/domain/components/AlertComponent.js +1 -0
  34. package/build/domain/components/AllOfComponent.js +40 -0
  35. package/build/domain/components/BooleanInputComponent.js +50 -0
  36. package/build/domain/components/BoxComponent.js +3 -0
  37. package/build/domain/components/ButtonComponent.js +1 -0
  38. package/build/domain/components/ColumnsComponent.js +3 -0
  39. package/build/domain/components/ConstComponent.js +18 -0
  40. package/build/domain/components/ContainerComponent.js +3 -0
  41. package/build/domain/components/DateInputComponent.js +75 -0
  42. package/build/domain/components/DecisionComponent.js +1 -0
  43. package/build/domain/components/DividerComponent.js +1 -0
  44. package/build/domain/components/FormComponent.js +3 -0
  45. package/build/domain/components/FormattedValueComponent.js +44 -0
  46. package/build/domain/components/HeadingComponent.js +1 -0
  47. package/build/domain/components/ImageComponent.js +1 -0
  48. package/build/domain/components/InstructionsComponent.js +1 -0
  49. package/build/domain/components/IntegerInputComponent.js +74 -0
  50. package/build/domain/components/ListComponent.js +1 -0
  51. package/build/domain/components/LoadingIndicatorComponent.js +1 -0
  52. package/build/domain/components/MarkdownComponent.js +1 -0
  53. package/build/domain/components/MediaComponent.js +1 -0
  54. package/build/domain/components/ModalComponent.js +16 -0
  55. package/build/domain/components/ModalLayoutComponent.js +3 -0
  56. package/build/domain/components/MoneyInputComponent.js +57 -0
  57. package/build/domain/components/MultiSelectInputComponent.js +81 -0
  58. package/build/domain/components/MultiUploadInputComponent.js +88 -0
  59. package/build/domain/components/NumberInputComponent.js +73 -0
  60. package/build/domain/components/ObjectComponent.js +45 -0
  61. package/build/domain/components/ParagraphComponent.js +1 -0
  62. package/build/domain/components/PersistAsyncComponent.js +92 -0
  63. package/build/domain/components/ProgressComponent.js +1 -0
  64. package/build/domain/components/RepeatableComponent.js +103 -0
  65. package/build/domain/components/ReviewComponent.js +1 -0
  66. package/build/domain/components/RootDomainComponent.js +173 -0
  67. package/build/domain/components/SectionComponent.js +5 -0
  68. package/build/domain/components/SelectInputComponent.js +88 -0
  69. package/build/domain/components/StatusListComponent.js +1 -0
  70. package/build/domain/components/SubflowDomainComponent.js +9 -0
  71. package/build/domain/components/TabsComponent.js +1 -0
  72. package/build/domain/components/TextInputComponent.js +76 -0
  73. package/build/domain/components/TupleComponent.js +41 -0
  74. package/build/domain/components/UploadInputComponent.js +83 -0
  75. package/build/domain/components/UpsellComponent.js +25 -0
  76. package/build/domain/components/searchComponent/SearchComponent.js +92 -0
  77. package/build/domain/components/searchComponent/SearchComponent.test.js +190 -0
  78. package/build/domain/components/step/ExternalConfirmationComponent.js +24 -0
  79. package/build/domain/components/step/StepDomainComponent.js +78 -0
  80. package/build/domain/components/step/ToolbarComponent.js +1 -0
  81. package/build/domain/components/utils/WithUpdate.js +1 -0
  82. package/build/domain/components/utils/component-utils.js +12 -0
  83. package/build/domain/components/utils/debounce.js +34 -0
  84. package/build/domain/components/utils/debounce.test.js +67 -0
  85. package/build/domain/components/utils/file-utils.js +21 -0
  86. package/build/domain/components/utils/file-utils.test.js +27 -0
  87. package/build/domain/components/utils/getRandomId.js +1 -0
  88. package/build/domain/components/utils/isExactLocalValueMatch.js +14 -0
  89. package/build/domain/components/utils/isOrWasValid.js +5 -0
  90. package/build/domain/components/utils/isPartialModelMatch.js +18 -0
  91. package/build/domain/components/utils/isPartialModelMatch.test.js +74 -0
  92. package/build/domain/features/eventNames.js +24 -0
  93. package/build/domain/features/events.js +1 -0
  94. package/build/domain/features/persistAsync/getComponentMultiPersistAsync.js +50 -0
  95. package/build/domain/features/persistAsync/getInitialPersistedState.js +7 -0
  96. package/build/domain/features/persistAsync/getPerformPersistAsync.js +43 -0
  97. package/build/domain/features/persistAsync/getPerformPersistAsync.test.js +139 -0
  98. package/build/domain/features/polling/getStepPolling.js +43 -0
  99. package/build/domain/features/polling/getStepPolling.test.js +90 -0
  100. package/build/domain/features/prefetch/getStepPrefetch.js +43 -0
  101. package/build/domain/features/prefetch/request-cache.js +49 -0
  102. package/build/domain/features/prefetch/request-cache.test.js +70 -0
  103. package/build/domain/features/refreshAfter/getStepRefreshAfter.js +24 -0
  104. package/build/domain/features/refreshAfter/getStepRefreshAfter.test.js +40 -0
  105. package/build/domain/features/schema-on-change/getDebouncedSchemaOnChange.js +50 -0
  106. package/build/domain/features/schema-on-change/getSchemaOnChange.js +34 -0
  107. package/build/domain/features/search/getPerformSearchFunction.js +75 -0
  108. package/build/domain/features/search/getPerformSearchFunction.test.js +301 -0
  109. package/build/domain/features/summary/summary-utils.js +40 -0
  110. package/build/domain/features/summary/summary-utils.test.js +125 -0
  111. package/build/domain/features/utils/http-utils.js +21 -0
  112. package/build/domain/features/utils/response-utils.js +9 -0
  113. package/build/domain/features/validation/spec-utils.js +19 -0
  114. package/build/domain/features/validation/validateStringPattern.js +24 -0
  115. package/build/domain/features/validation/validation-functions.js +6 -0
  116. package/build/domain/features/validation/validation-functions.test.js +108 -0
  117. package/build/domain/features/validation/value-checks.js +125 -0
  118. package/build/domain/features/validation/value-checks.test.js +262 -0
  119. package/build/domain/features/validationAsync/getComponentValidationAsync.js +53 -0
  120. package/build/domain/features/validationAsync/getComponentValidationAsync.test.js +67 -0
  121. package/build/domain/features/validationAsync/getInitialValidationAsyncState.js +5 -0
  122. package/build/domain/features/validationAsync/getPerformValidationAsync.js +45 -0
  123. package/build/domain/features/validationAsync/getPerformValidationAsync.test.js +143 -0
  124. package/build/domain/mappers/layout/alertLayoutToComponent.js +16 -0
  125. package/build/domain/mappers/layout/boxLayoutToComponent.js +13 -0
  126. package/build/domain/mappers/layout/buttonLayoutToComponent.js +77 -0
  127. package/build/domain/mappers/layout/columnsLayoutToComponent.js +13 -0
  128. package/build/domain/mappers/layout/decisionLayoutToComponent.js +22 -0
  129. package/build/domain/mappers/layout/deprecatedListLayoutToComponent.js +30 -0
  130. package/build/domain/mappers/layout/dividerLayoutToComponent.js +2 -0
  131. package/build/domain/mappers/layout/formLayoutToComponent.js +19 -0
  132. package/build/domain/mappers/layout/headingLayoutToComponent.js +12 -0
  133. package/build/domain/mappers/layout/imageLayoutToComponent.js +20 -0
  134. package/build/domain/mappers/layout/infoLayoutToComponent.js +12 -0
  135. package/build/domain/mappers/layout/instructionsLayoutToComponent.js +12 -0
  136. package/build/domain/mappers/layout/listLayoutToComponent.js +39 -0
  137. package/build/domain/mappers/layout/loadingIndicatorLayoutToComponent.js +9 -0
  138. package/build/domain/mappers/layout/markdownLayoutToComponent.js +12 -0
  139. package/build/domain/mappers/layout/mediaLayoutToComponent.js +12 -0
  140. package/build/domain/mappers/layout/modalLayoutToComponent.js +17 -0
  141. package/build/domain/mappers/layout/modalToComponent.js +8 -0
  142. package/build/domain/mappers/layout/paragraphLayoutToComponent.js +12 -0
  143. package/build/domain/mappers/layout/progressLayoutToComponent.js +15 -0
  144. package/build/domain/mappers/layout/reviewLayoutToComponent.js +48 -0
  145. package/build/domain/mappers/layout/searchLayoutToComponent.js +44 -0
  146. package/build/domain/mappers/layout/sectionLayoutToComponent.js +15 -0
  147. package/build/domain/mappers/layout/statusListLayoutToComponent.js +15 -0
  148. package/build/domain/mappers/layout/tabsLayoutToComponent.js +16 -0
  149. package/build/domain/mappers/layout/upsellLayoutToComponent.js +25 -0
  150. package/build/domain/mappers/mapLayoutToComponent.js +81 -0
  151. package/build/domain/mappers/mapSchemaToComponent.js +61 -0
  152. package/build/domain/mappers/mapSchemaToComponent.test.js +112 -0
  153. package/build/domain/mappers/mapStepSchemas.js +15 -0
  154. package/build/domain/mappers/mapStepToComponent.js +133 -0
  155. package/build/domain/mappers/mapStepToComponent.test.js +221 -0
  156. package/build/domain/mappers/mapToolbarToComponent.js +15 -0
  157. package/build/domain/mappers/schema/allOfSchemaToComponent.js +16 -0
  158. package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToComponent.js +26 -0
  159. package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToMultiSelectComponent.js +55 -0
  160. package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToMultiUploadComponent.js +67 -0
  161. package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToRepeatableComponent.js +57 -0
  162. package/build/domain/mappers/schema/arraySchemaToComponent/arraySchemaToTupleComponent.js +20 -0
  163. package/build/domain/mappers/schema/blobSchemaToComponent.js +15 -0
  164. package/build/domain/mappers/schema/booleanSchemaToComponent.js +29 -0
  165. package/build/domain/mappers/schema/constSchemaToComponent.js +23 -0
  166. package/build/domain/mappers/schema/integerSchemaToComponent.js +28 -0
  167. package/build/domain/mappers/schema/numberSchemaToComponent.js +26 -0
  168. package/build/domain/mappers/schema/objectSchemaToComponent/assertDisplayOrder.js +23 -0
  169. package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToFormattedValueComponent.js +9 -0
  170. package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToMoneyInputComponent.js +119 -0
  171. package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToMoneyInputComponent.test.js +96 -0
  172. package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToObjectComponent.js +31 -0
  173. package/build/domain/mappers/schema/objectSchemaToComponent/objectSchemaToObjectComponent.test.js +99 -0
  174. package/build/domain/mappers/schema/oneOfSchemaToComponent/oneOfSchemaToComponent.js +66 -0
  175. package/build/domain/mappers/schema/oneOfSchemaToComponent/oneOfSchemaToComponent.test.js +225 -0
  176. package/build/domain/mappers/schema/persistAsyncSchemaToComponent.js +29 -0
  177. package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToComponent.js +18 -0
  178. package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToComponent.test.js +133 -0
  179. package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToDateInputComponent.js +48 -0
  180. package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToTextInputComponent.js +37 -0
  181. package/build/domain/mappers/schema/stringSchemaToComponent/stringSchemaToUploadInputComponent.js +28 -0
  182. package/build/domain/mappers/schema/tests/test-utils.js +16 -0
  183. package/build/domain/mappers/schema/types.js +1 -0
  184. package/build/domain/mappers/schema/utils/getPerformPersistAsyncFn.js +19 -0
  185. package/build/domain/mappers/schema/utils/getValidationAsyncInitialState.js +23 -0
  186. package/build/domain/mappers/schema/utils/mapCommonSchemaProps.js +16 -0
  187. package/build/domain/mappers/types.js +1 -0
  188. package/build/domain/mappers/utils/FeatureFlags.js +22 -0
  189. package/build/domain/mappers/utils/behavior-utils.js +44 -0
  190. package/build/domain/mappers/utils/call-to-action-utils.js +21 -0
  191. package/build/domain/mappers/utils/getAutocompleteString.js +76 -0
  192. package/build/domain/mappers/utils/getAutocompleteString.test.js +21 -0
  193. package/build/domain/mappers/utils/groupLayoutByPinned.js +38 -0
  194. package/build/domain/mappers/utils/groupLayoutByPinned.test.js +166 -0
  195. package/build/domain/mappers/utils/image.js +9 -0
  196. package/build/domain/mappers/utils/layout-utils.js +11 -0
  197. package/build/domain/mappers/utils/legacy-utils.js +49 -0
  198. package/build/domain/mappers/utils/media-utils.js +14 -0
  199. package/build/domain/mappers/utils/suggestions-utils.js +26 -0
  200. package/build/domain/mappers/utils/suggestions-utils.test.js +36 -0
  201. package/build/domain/mappers/utils/tags-utils.js +1 -0
  202. package/build/domain/mappers/utils/utils.js +35 -0
  203. package/build/domain/types.js +1 -0
  204. package/build/getSubflowCallbacks.js +38 -0
  205. package/build/i18n/index.js +40 -0
  206. package/build/index.js +8 -0
  207. package/build/renderers/CoreContainerRenderer.js +5 -0
  208. package/build/renderers/CoreRootRenderer.js +12 -0
  209. package/build/renderers/EmptyLoadingStateRenderer.js +5 -0
  210. package/build/renderers/getRenderFunction.js +24 -0
  211. package/build/renderers/getSchemaErrorMessageFunction.js +97 -0
  212. package/build/renderers/mappers/alertComponentToProps.js +2 -0
  213. package/build/renderers/mappers/allOfComponentToProps.js +6 -0
  214. package/build/renderers/mappers/booleanInputComponentToProps.js +5 -0
  215. package/build/renderers/mappers/boxComponentToProps.js +13 -0
  216. package/build/renderers/mappers/buttonComponentToProps.js +4 -0
  217. package/build/renderers/mappers/columnsComponentToProps.js +11 -0
  218. package/build/renderers/mappers/componentToRendererProps.js +164 -0
  219. package/build/renderers/mappers/constComponentToProps.js +5 -0
  220. package/build/renderers/mappers/containerComponentToProps.js +7 -0
  221. package/build/renderers/mappers/dateInputComponentToProps.js +2 -0
  222. package/build/renderers/mappers/decisionComponentToProps.js +16 -0
  223. package/build/renderers/mappers/dividerComponentToProps.js +2 -0
  224. package/build/renderers/mappers/externalComponentToProps.js +3 -0
  225. package/build/renderers/mappers/formComponentToProps.js +12 -0
  226. package/build/renderers/mappers/formattedValueComponentToProps.js +5 -0
  227. package/build/renderers/mappers/headingComponentToProps.js +2 -0
  228. package/build/renderers/mappers/hiddenComponentToProps.js +4 -0
  229. package/build/renderers/mappers/imageComponentToProps.js +2 -0
  230. package/build/renderers/mappers/instructionsComponentToProps.js +2 -0
  231. package/build/renderers/mappers/integerInputComponentToProps.js +2 -0
  232. package/build/renderers/mappers/listComponentToProps.js +2 -0
  233. package/build/renderers/mappers/loadingIndicatorComponentToProps.js +2 -0
  234. package/build/renderers/mappers/markdownComponentToProps.js +2 -0
  235. package/build/renderers/mappers/mediaComponentToProps.js +2 -0
  236. package/build/renderers/mappers/modalComponentToProps.js +11 -0
  237. package/build/renderers/mappers/modalLayoutComponentToProps.js +16 -0
  238. package/build/renderers/mappers/moneyInputComponentToProps.js +36 -0
  239. package/build/renderers/mappers/multiSelectComponentToProps.js +23 -0
  240. package/build/renderers/mappers/multiUploadInputComponentToProps.js +12 -0
  241. package/build/renderers/mappers/numberInputComponentToProps.js +2 -0
  242. package/build/renderers/mappers/objectComponentToProps.js +8 -0
  243. package/build/renderers/mappers/paragraphComponentToProps.js +2 -0
  244. package/build/renderers/mappers/persistAsyncComponentToProps.js +8 -0
  245. package/build/renderers/mappers/progressComponentToProps.js +2 -0
  246. package/build/renderers/mappers/repeatableComponentToProps.js +30 -0
  247. package/build/renderers/mappers/reviewComponentToProps.js +2 -0
  248. package/build/renderers/mappers/rootComponentToProps.js +21 -0
  249. package/build/renderers/mappers/searchComponentToProps.js +57 -0
  250. package/build/renderers/mappers/sectionComponentToProps.js +6 -0
  251. package/build/renderers/mappers/selectInputComponentToProps.js +34 -0
  252. package/build/renderers/mappers/statusListComponentToProps.js +2 -0
  253. package/build/renderers/mappers/subflowComponentToRendererProps.js +4 -0
  254. package/build/renderers/mappers/tabsComponentToProps.js +14 -0
  255. package/build/renderers/mappers/textInputComponentToProps.js +2 -0
  256. package/build/renderers/mappers/tupleComponentToProps.js +8 -0
  257. package/build/renderers/mappers/uploadInputComponentToProps.js +8 -0
  258. package/build/renderers/mappers/upsellComponentToProps.js +2 -0
  259. package/build/renderers/mappers/utils/getValidationState.js +12 -0
  260. package/build/renderers/mappers/utils/inputComponentToProps.js +26 -0
  261. package/build/renderers/mappers/utils/mapErrorsToValidationState.js +9 -0
  262. package/build/renderers/mappers/utils/pick.js +8 -0
  263. package/build/renderers/mappers/utils/selectInputOptionsToProps.js +11 -0
  264. package/build/renderers/stepComponentToProps.js +32 -0
  265. package/build/renderers/utils.js +69 -0
  266. package/build/renderers/utils.test.js +70 -0
  267. package/build/stories/dev-tools/ContainerQueries.story.js +66 -0
  268. package/build/stories/dev-tools/Debugger.story.js +38 -0
  269. package/build/stories/dev-tools/FixtureSelect.story.js +23 -0
  270. package/build/stories/dev-tools/TestServer.story.js +32 -0
  271. package/build/stories/examples/NativeFlow.story.js +80 -0
  272. package/build/stories/examples/Recipients.story.js +568 -0
  273. package/build/stories/spec/behavior/Copy.story.js +71 -0
  274. package/build/stories/spec/behavior/Link.story.js +40 -0
  275. package/build/stories/spec/behavior/Modal.story.js +79 -0
  276. package/build/stories/spec/behavior/Subflow.story.js +267 -0
  277. package/build/stories/spec/layouts/Decision.story.js +241 -0
  278. package/build/stories/spec/layouts/Image.story.js +42 -0
  279. package/build/stories/spec/layouts/Modal.story.js +81 -0
  280. package/build/stories/spec/layouts/Search.story.js +325 -0
  281. package/build/stories/spec/layouts/Upsell.story.js +55 -0
  282. package/build/stories/spec/layouts/button/Button.story.js +100 -0
  283. package/build/stories/spec/layouts/button/PinnedButton.story.js +81 -0
  284. package/build/stories/spec/response/ActionResponse.story.js +66 -0
  285. package/build/stories/spec/schemas/MultiSelect.story.js +148 -0
  286. package/build/stories/spec/schemas/Upload.story.js +168 -0
  287. package/build/stories/spec/schemas/const/ConstLayout.story.js +159 -0
  288. package/build/stories/spec/schemas/const/ObjectConst.story.js +94 -0
  289. package/build/stories/spec/schemas/features/PersistAsync.story.js +211 -0
  290. package/build/stories/spec/schemas/features/ValidationAsync.story.js +103 -0
  291. package/build/stories/spec/schemas/object/FormattedValue.story.js +92 -0
  292. package/build/stories/spec/schemas/object/MoneyInput.story.js +240 -0
  293. package/build/stories/spec/schemas/oneOf/OneOfInitialisation.story.js +55 -0
  294. package/build/stories/spec/schemas/oneOf/OneOfWithSingleOption.story.js +114 -0
  295. package/build/stories/spec/step/Controls.story.js +109 -0
  296. package/build/stories/spec/step/DFModal.story.js +58 -0
  297. package/build/stories/spec/step/Footer.story.js +70 -0
  298. package/build/stories/spec/step/Navigation.story.js +20 -0
  299. package/build/stories/spec/step/ScrollToBottom.story.js +103 -0
  300. package/build/stories/spec/step/Tags.story.js +39 -0
  301. package/build/stories/spec/step/ToolBar.story.js +60 -0
  302. package/build/stories/spec/step/features/ErrorHandling.story.js +92 -0
  303. package/build/stories/spec/step/features/External.story.js +91 -0
  304. package/build/stories/spec/step/features/Polling.story.js +108 -0
  305. package/build/stories/spec/step/features/RefreshAfter.story.js +92 -0
  306. package/build/stories/spec/step/features/refresh/Refresh.story.js +258 -0
  307. package/build/stories/spec/step/features/refresh/RefreshWithPersistAsync.story.js +958 -0
  308. package/build/stories/types.js +1 -0
  309. package/build/stories/utils/fixtureHttpClient.js +70 -0
  310. package/build/stories/utils/getBasicStep.js +223 -0
  311. package/build/stories/utils/mockSearchHandler.js +71 -0
  312. package/build/stories/utils/render-utils.js +41 -0
  313. package/build/stories/visual-tests/layouts/NotUsingListItem.story.js +17 -0
  314. package/build/test-utils/DynamicFlowWise.js +32 -0
  315. package/build/test-utils/DynamicFlowWiseModal.js +34 -0
  316. package/build/test-utils/NeptuneProviders.js +11 -0
  317. package/build/test-utils/component-utils.js +5 -0
  318. package/build/test-utils/fetch-utils.js +45 -0
  319. package/build/test-utils/getMergedTestRenderers.js +34 -0
  320. package/build/test-utils/getRandomId.js +1 -0
  321. package/build/test-utils/index.js +3 -0
  322. package/build/test-utils/openLinkInNewTab.js +15 -0
  323. package/build/test-utils/rtl-utils.js +7 -0
  324. package/build/test-utils/step-utils.js +6 -0
  325. package/build/test-utils/wait.js +3 -0
  326. package/build/tests/AlertLayout.test.js +78 -0
  327. package/build/tests/ArrayTuple.test.js +118 -0
  328. package/build/tests/ButtonLayout.test.js +308 -0
  329. package/build/tests/ConstLayout.test.js +95 -0
  330. package/build/tests/DateInput.test.js +163 -0
  331. package/build/tests/DecisionLayout.test.js +146 -0
  332. package/build/tests/DynamicFlow.test.js +147 -0
  333. package/build/tests/External.test.js +169 -0
  334. package/build/tests/Flow.test.js +328 -0
  335. package/build/tests/FormLayout.test.js +28 -0
  336. package/build/tests/FormattedValue.test.js +107 -0
  337. package/build/tests/ImageRenderer.test.js +78 -0
  338. package/build/tests/InitialAction.test.js +179 -0
  339. package/build/tests/InitialStep.test.js +168 -0
  340. package/build/tests/InstructionsLayout.test.js +45 -0
  341. package/build/tests/ListLayout.test.js +168 -0
  342. package/build/tests/Logging.test.js +53 -0
  343. package/build/tests/ModalBehavior.test.js +149 -0
  344. package/build/tests/MoneyInput.test.js +316 -0
  345. package/build/tests/MultiUpload.test.js +293 -0
  346. package/build/tests/NativeBack.test.js +267 -0
  347. package/build/tests/NoOp.test.js +194 -0
  348. package/build/tests/OneOfInitialisation.test.js +571 -0
  349. package/build/tests/PersistAsync.test.js +687 -0
  350. package/build/tests/Polling.test.js +702 -0
  351. package/build/tests/Prefetching.test.js +230 -0
  352. package/build/tests/RefreshAfter.test.js +63 -0
  353. package/build/tests/RefreshOnChange.ResponseHandling.test.js +205 -0
  354. package/build/tests/RefreshOnChange.test.js +233 -0
  355. package/build/tests/RefreshOnChange.with.Segmented.test.js +348 -0
  356. package/build/tests/RefreshOnChange.with.Tabs.test.js +358 -0
  357. package/build/tests/RefreshOnChangePreserve.test.js +224 -0
  358. package/build/tests/RendererProps.test.js +342 -0
  359. package/build/tests/Repeatable.test.js +107 -0
  360. package/build/tests/Rerendering.test.js +67 -0
  361. package/build/tests/ReviewLayout.test.js +274 -0
  362. package/build/tests/SchemaOnChange.test.js +133 -0
  363. package/build/tests/SchemaReferences.test.js +88 -0
  364. package/build/tests/ScrollToBottom.test.js +122 -0
  365. package/build/tests/ScrollToError.test.js +217 -0
  366. package/build/tests/SegmentedControl.test.js +49 -0
  367. package/build/tests/SingleFileUpload.test.js +168 -0
  368. package/build/tests/StatusList.test.js +85 -0
  369. package/build/tests/Subflow.test.js +710 -0
  370. package/build/tests/Submission.ResponseHandling.test.js +557 -0
  371. package/build/tests/Submission.merging.test.js +202 -0
  372. package/build/tests/Submission.test.js +748 -0
  373. package/build/tests/Tags.test.js +475 -0
  374. package/build/tests/Upsell.test.js +154 -0
  375. package/build/tests/ValidationAsync.test.js +295 -0
  376. package/build/tests/legacy/Actions.test.js +158 -0
  377. package/build/tests/legacy/BackButton.test.js +114 -0
  378. package/build/tests/legacy/HiddenSchemas.test.js +246 -0
  379. package/build/tests/legacy/MultiSelect.test.js +501 -0
  380. package/build/tests/legacy/MultipleFileUploadSchema.test.js +341 -0
  381. package/build/tests/legacy/PersistAsync.blob-schema.test.js +224 -0
  382. package/build/tests/legacy/PersistAsync.string-schema.test.js +211 -0
  383. package/build/tests/legacy/RefreshStepOnChange.debouncing.test.js +209 -0
  384. package/build/tests/legacy/RefreshStepOnChange.test.js +424 -0
  385. package/build/tests/legacy/Search.test.js +437 -0
  386. package/build/tests/renderers/MultiSelectInputRendererProps.test.js +58 -0
  387. package/build/tests/renderers/SelectInputRendererProps.test.js +43 -0
  388. package/build/tests/renderers/TextInputRenderer.test.js +51 -0
  389. package/build/tsconfig.types.tsbuildinfo +1 -0
  390. package/build/types.js +1 -0
  391. package/build/useDynamicFlow.js +104 -0
  392. package/build/useDynamicFlowModal.js +58 -0
  393. package/build/utils/analyse-step.js +14 -0
  394. package/build/utils/component-utils.js +8 -0
  395. package/build/utils/component-utils.test.js +113 -0
  396. package/build/utils/getScrollToTop.js +12 -0
  397. package/build/utils/normalise-flow-id.js +1 -0
  398. package/build/utils/normalise-flow-id.test.js +24 -0
  399. package/build/utils/recursiveMerge.js +40 -0
  400. package/build/utils/recursiveMerge.test.js +93 -0
  401. package/build/utils/type-utils.js +21 -0
  402. package/build/utils/type-validators.js +11 -0
  403. package/build/utils/type-validators.test.js +180 -0
  404. package/build/utils/useStableCallback.js +15 -0
  405. package/package.json +20 -21
@@ -0,0 +1,230 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ import { screen } from '@testing-library/react';
3
+ import { userEvent } from '@testing-library/user-event';
4
+ import { vi } from 'vitest';
5
+ import { createFlowController } from '../controller/FlowController';
6
+ import { FeatureFlags } from '../domain/mappers/utils/FeatureFlags';
7
+ import { renderWithProviders } from '../test-utils';
8
+ import DynamicFlowWise from '../test-utils/DynamicFlowWise';
9
+ import { respondWith } from '../test-utils/fetch-utils';
10
+ const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
11
+ describe('Prefetching actions in a step', () => {
12
+ const initialStep = () => ({
13
+ id: 'step',
14
+ title: 'Step Title',
15
+ schemas: [
16
+ {
17
+ $id: 'schema-1',
18
+ type: 'string',
19
+ title: 'A string schema',
20
+ default: 'a string value',
21
+ },
22
+ ],
23
+ layout: [
24
+ {
25
+ type: 'form',
26
+ schemaId: 'schema-1',
27
+ },
28
+ {
29
+ type: 'review',
30
+ callToAction: {
31
+ title: 'Edit',
32
+ behavior: { type: 'action', action: { url: '/edit', prefetch: true } },
33
+ },
34
+ fields: [
35
+ {
36
+ label: 'Field A',
37
+ value: 'A value',
38
+ callToAction: {
39
+ title: 'Edit Field A',
40
+ behavior: { type: 'action', action: { url: '/edit-field-a', prefetch: true } },
41
+ },
42
+ },
43
+ ],
44
+ },
45
+ {
46
+ type: 'button',
47
+ title: 'Button 1',
48
+ behavior: {
49
+ type: 'action',
50
+ action: { url: '/button-1', prefetch: true },
51
+ },
52
+ },
53
+ {
54
+ type: 'button',
55
+ title: 'Button 2',
56
+ behavior: {
57
+ type: 'action',
58
+ action: { url: '/button-2' },
59
+ },
60
+ },
61
+ ],
62
+ });
63
+ const secondStep = () => ({
64
+ id: 'second-step',
65
+ title: 'Second Step',
66
+ schemas: [],
67
+ layout: [],
68
+ });
69
+ it('prefetches actions marked for prefetching', async () => {
70
+ const mockHttpClient = vi.fn(async () => {
71
+ return respondWith(secondStep());
72
+ });
73
+ const eventCountByName = {};
74
+ const onEvent = vi.fn((eventName) => {
75
+ var _a;
76
+ eventCountByName[eventName] = ((_a = eventCountByName[eventName]) !== null && _a !== void 0 ? _a : 0) + 1;
77
+ });
78
+ const onError = vi.fn();
79
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "prefetching-flow", httpClient: mockHttpClient, initialStep: initialStep(), onEvent: onEvent, onCompletion: () => { }, onError: onError }));
80
+ expect(eventCountByName['Dynamic Flow - Step Shown']).toBe(1); // the initial step shown
81
+ expect(mockHttpClient).toHaveBeenCalledTimes(3); // three prefetches
82
+ await user.click(screen.getByRole('button', { name: 'Edit' })); // trigger prefetched action
83
+ expect(mockHttpClient).toHaveBeenCalledTimes(3); // no extra requests
84
+ expect(eventCountByName['Dynamic Flow - Step Shown']).toBe(2); // the second step shown
85
+ expect(onError).not.toHaveBeenCalled();
86
+ });
87
+ it('does not prefetch actions not marked for prefetching', async () => {
88
+ const mockHttpClient = vi.fn(async () => {
89
+ return respondWith(secondStep());
90
+ });
91
+ const eventCountByName = {};
92
+ const onEvent = vi.fn((eventName) => {
93
+ var _a;
94
+ eventCountByName[eventName] = ((_a = eventCountByName[eventName]) !== null && _a !== void 0 ? _a : 0) + 1;
95
+ });
96
+ const onError = vi.fn();
97
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "prefetching-flow", httpClient: mockHttpClient, initialStep: initialStep(), onEvent: onEvent, onCompletion: () => { }, onError: onError }));
98
+ expect(eventCountByName['Dynamic Flow - Step Shown']).toBe(1); // the initial step shown
99
+ expect(mockHttpClient).toHaveBeenCalledTimes(3); // three prefetches
100
+ await user.click(screen.getByRole('button', { name: 'Button 2' })); // trigger non-prefetched action
101
+ expect(mockHttpClient).toHaveBeenCalledTimes(4); // one extra request
102
+ expect(eventCountByName['Dynamic Flow - Step Shown']).toBe(2); // the second step shown
103
+ expect(onError).not.toHaveBeenCalled();
104
+ });
105
+ it('includes schema values and headers in prefetching requests', async () => {
106
+ const mockHttpClient = vi.fn(async () => {
107
+ return respondWith(secondStep());
108
+ });
109
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "prefetching-flow", httpClient: mockHttpClient, initialStep: initialStep(), onCompletion: () => { }, onError: () => { } }));
110
+ const expectedRequestInit = {
111
+ method: 'POST',
112
+ body: '"a string value"',
113
+ headers: { 'Content-Type': 'application/json' },
114
+ };
115
+ expect(mockHttpClient).toHaveBeenCalledWith('/edit', expectedRequestInit);
116
+ expect(mockHttpClient).toHaveBeenCalledWith('/edit-field-a', expectedRequestInit);
117
+ expect(mockHttpClient).toHaveBeenCalledWith('/button-1', expectedRequestInit);
118
+ });
119
+ it('does not use cached responses when their status code is not OK', async () => {
120
+ const step = {
121
+ id: 'initial-step',
122
+ title: 'Initial Step Title',
123
+ schemas: [],
124
+ layout: [
125
+ {
126
+ type: 'button',
127
+ title: 'Button 1',
128
+ behavior: {
129
+ type: 'action',
130
+ action: { url: '/button-1', prefetch: true },
131
+ },
132
+ },
133
+ ],
134
+ };
135
+ const mockHttpClient = vi
136
+ .fn()
137
+ .mockResolvedValueOnce(await respondWith(null, { status: 500 }))
138
+ .mockResolvedValueOnce(await respondWith(secondStep()));
139
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "prefetching-flow", httpClient: mockHttpClient, initialStep: step, onCompletion: () => { }, onError: () => { } }));
140
+ expect(mockHttpClient).toHaveBeenCalledTimes(1); // one prefetch attempt
141
+ await user.click(screen.getByRole('button', { name: 'Button 1' })); // trigger action from initial step
142
+ expect(mockHttpClient).toHaveBeenCalledTimes(2); // one extra request due to non-OK cached response
143
+ });
144
+ it('only performs one request for identical actions', () => {
145
+ const step = Object.assign(Object.assign({}, initialStep()), { layout: [
146
+ ...initialStep().layout,
147
+ {
148
+ type: 'button',
149
+ title: 'Button 1 again',
150
+ behavior: {
151
+ type: 'action',
152
+ action: { url: '/button-1', prefetch: true },
153
+ },
154
+ },
155
+ ] });
156
+ const mockHttpClient = vi.fn().mockImplementation(() => respondWith((Object.assign(Object.assign({}, initialStep()), { title: 'This is a new step' }))));
157
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "prefetching-flow", httpClient: mockHttpClient, initialStep: step, onCompletion: () => { }, onError: () => { } }));
158
+ expect(mockHttpClient).toHaveBeenCalledTimes(3);
159
+ });
160
+ describe('Prefetching scope is constrained to the step', () => {
161
+ it('does not use prefetched actions from previous steps', async () => {
162
+ const secondStepLocal = () => ({
163
+ id: 'step',
164
+ title: 'Second Step Title',
165
+ schemas: [
166
+ {
167
+ $id: 'schema-1',
168
+ type: 'string',
169
+ title: 'A string schema',
170
+ default: 'a string value',
171
+ },
172
+ ],
173
+ layout: [
174
+ {
175
+ type: 'form',
176
+ schemaId: 'schema-1',
177
+ },
178
+ {
179
+ type: 'button',
180
+ title: 'Button 1',
181
+ behavior: {
182
+ type: 'action',
183
+ action: { url: '/button-1' },
184
+ },
185
+ },
186
+ ],
187
+ });
188
+ const mockHttpClient = vi.fn(async () => {
189
+ return respondWith(secondStepLocal());
190
+ });
191
+ const eventCountByName = {};
192
+ const onEvent = vi.fn((eventName) => {
193
+ var _a;
194
+ eventCountByName[eventName] = ((_a = eventCountByName[eventName]) !== null && _a !== void 0 ? _a : 0) + 1;
195
+ });
196
+ const onError = vi.fn();
197
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "prefetching-flow", httpClient: mockHttpClient, initialStep: initialStep(), onEvent: onEvent, onCompletion: () => { }, onError: onError }));
198
+ expect(eventCountByName['Dynamic Flow - Step Shown']).toBe(1); // the initial step shown
199
+ expect(mockHttpClient).toHaveBeenCalledTimes(3); // three prefetches
200
+ await user.click(screen.getByRole('button', { name: 'Button 1' })); // trigger action from initial step
201
+ expect(mockHttpClient).toHaveBeenCalledTimes(3); // no extra requests
202
+ expect(eventCountByName['Dynamic Flow - Step Shown']).toBe(2); // the second step shown
203
+ await user.click(screen.getByRole('button', { name: 'Button 1' })); // trigger action from second step
204
+ expect(mockHttpClient).toHaveBeenCalledTimes(4); // one extra request -- no prefetching from previous step
205
+ expect(onError).not.toHaveBeenCalled(); // no errors
206
+ });
207
+ });
208
+ describe('Initialisation', () => {
209
+ it('does not prefetch before the DF controller is started', async () => {
210
+ const mockHttpClient = vi.fn();
211
+ const df = createFlowController({
212
+ flowId: 'prefetching-flow',
213
+ backConfig: { isNativeBackEnabled: true, isFlowCancellable: true },
214
+ features: new FeatureFlags({}),
215
+ initialStep: initialStep(),
216
+ httpClient: mockHttpClient,
217
+ scrollToTop: vi.fn(),
218
+ onCompletion: vi.fn(),
219
+ onError: vi.fn(),
220
+ onLink: vi.fn(),
221
+ getErrorMessageFunctions: vi.fn(),
222
+ onChange: vi.fn(),
223
+ onValueChange: vi.fn(),
224
+ });
225
+ expect(mockHttpClient).not.toHaveBeenCalled();
226
+ df.start();
227
+ expect(mockHttpClient).toHaveBeenCalledTimes(3); // prefetches after starting
228
+ });
229
+ });
230
+ });
@@ -0,0 +1,63 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ import { act, waitFor } from '@testing-library/react';
3
+ import { renderWithProviders } from '../test-utils';
4
+ import DynamicFlowWise from '../test-utils/DynamicFlowWise';
5
+ import { vi } from 'vitest';
6
+ describe('given a step with a refreshAfter value', () => {
7
+ const getStep = (refreshAfter) => ({
8
+ id: 'step',
9
+ title: 'Step',
10
+ layout: [
11
+ {
12
+ type: 'paragraph',
13
+ text: refreshAfter
14
+ ? `This step will refresh at ${refreshAfter}`
15
+ : 'This step will not refresh',
16
+ },
17
+ ],
18
+ schemas: [],
19
+ refreshUrl: '/refresh',
20
+ refreshAfter,
21
+ });
22
+ describe('when refreshAfter is in the future', () => {
23
+ it('triggers a refresh when the time is reached', async () => {
24
+ const httpClient = vi.fn();
25
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "flow-id", initialStep: getStep(new Date(Date.now() + 3000).toISOString()), httpClient: httpClient, onCompletion: () => { }, onError: () => { } }));
26
+ await act(async () => {
27
+ vi.advanceTimersByTime(2900);
28
+ });
29
+ await waitFor(() => {
30
+ expect(httpClient).not.toHaveBeenCalled();
31
+ });
32
+ await act(async () => {
33
+ vi.advanceTimersByTime(100); // 2900ms + 100ms = 3000ms
34
+ });
35
+ await waitFor(() => {
36
+ expect(httpClient).toHaveBeenCalledWith('/refresh', expect.anything());
37
+ });
38
+ });
39
+ });
40
+ describe('when refreshAfter is in the past', () => {
41
+ it('triggers a refresh after one second, not immediately', async () => {
42
+ const httpClient = vi.fn();
43
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "flow-id", initialStep: getStep(new Date(Date.now() - 2000).toISOString()), httpClient: httpClient, onCompletion: () => { }, onError: () => { } }));
44
+ await act(async () => {
45
+ vi.advanceTimersByTime(900);
46
+ });
47
+ expect(httpClient).not.toHaveBeenCalled();
48
+ await act(async () => {
49
+ vi.advanceTimersByTime(100); // 900ms + 100ms = 1 second
50
+ });
51
+ expect(httpClient).toHaveBeenCalledWith('/refresh', expect.anything());
52
+ });
53
+ });
54
+ describe('when refreshAfter is invalid', () => {
55
+ it('fails the flow, triggering onError', async () => {
56
+ const onError = vi.fn();
57
+ renderWithProviders(_jsx(DynamicFlowWise, { flowId: "flow-id", initialStep: getStep('banana'), httpClient: vi.fn(), onCompletion: () => { }, onError: onError }));
58
+ await waitFor(() => {
59
+ expect(onError).toHaveBeenCalledWith(new Error('Invalid refreshAfter value: banana'), undefined);
60
+ });
61
+ });
62
+ });
63
+ });
@@ -0,0 +1,205 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ /* eslint-disable testing-library/render-result-naming-convention */
3
+ import { act, screen, waitFor } from '@testing-library/react';
4
+ import { userEvent } from '@testing-library/user-event';
5
+ import { vi } from 'vitest';
6
+ import { renderWithProviders, respondWith } from '../test-utils';
7
+ import DynamicFlowWise from '../test-utils/DynamicFlowWise';
8
+ import { abortableDelay } from '../test-utils/fetch-utils';
9
+ const user = userEvent.setup({ advanceTimers: vi.advanceTimersByTime });
10
+ const getProps = () => ({
11
+ flowId: 'flow-id',
12
+ onCompletion: vi.fn(),
13
+ onError: vi.fn(),
14
+ onEvent: vi.fn(),
15
+ onLog: vi.fn(),
16
+ });
17
+ const initialStep = {
18
+ id: 'step-id',
19
+ title: 'Submission test',
20
+ schemas: [
21
+ {
22
+ $id: '#form',
23
+ type: 'object',
24
+ displayOrder: ['color'],
25
+ properties: {
26
+ color: {
27
+ title: 'Color',
28
+ control: 'radio',
29
+ analyticsId: '#color-schema',
30
+ oneOf: [
31
+ { const: 'red', title: 'Red' },
32
+ { const: 'blue', title: 'Blue' },
33
+ { const: 'green', title: 'Green' },
34
+ ],
35
+ refreshStepOnChange: true,
36
+ },
37
+ },
38
+ },
39
+ ],
40
+ layout: [
41
+ {
42
+ type: 'form',
43
+ schemaId: '#form',
44
+ },
45
+ {
46
+ type: 'button',
47
+ title: 'Submit',
48
+ action: { id: 'action-id', url: '/submit', method: 'POST' },
49
+ },
50
+ ],
51
+ refreshUrl: '/refresh',
52
+ };
53
+ const renderAndRefresh = async (httpClient) => {
54
+ const props = Object.assign(Object.assign({}, getProps()), { initialStep, httpClient });
55
+ renderWithProviders(_jsx(DynamicFlowWise, Object.assign({}, props)));
56
+ await waitFor(() => {
57
+ expect(screen.getByText(initialStep.title)).toBeInTheDocument();
58
+ });
59
+ await user.click(screen.getByText('Red'));
60
+ return props;
61
+ };
62
+ describe('RefreshOnChange response handling', () => {
63
+ describe('When the fetch call throws', () => {
64
+ it('should trigger onError and the "Action Failed" and "Failed" events', async () => {
65
+ const httpClient = vi.fn().mockRejectedValue(new Error('Network error'));
66
+ const props = await renderAndRefresh(httpClient);
67
+ await waitFor(() => {
68
+ expect(props.onError).toHaveBeenCalledWith(expect.any(Error), undefined);
69
+ });
70
+ expect(props.onEvent).toHaveBeenCalledWith('Dynamic Flow - Refresh Failed', expect.objectContaining({
71
+ flowId: 'flow-id',
72
+ stepId: 'step-id',
73
+ stepCount: 0,
74
+ schema: '#color-schema',
75
+ }));
76
+ expect(props.onEvent).toHaveBeenCalledWith('Dynamic Flow - Failed', expect.objectContaining({
77
+ flowId: 'flow-id',
78
+ stepId: 'step-id',
79
+ stepCount: 0,
80
+ }));
81
+ });
82
+ });
83
+ describe('Status 2xx (response.ok)', () => {
84
+ describe('when the response is a valid step', () => {
85
+ it('should render the new step, and trigger the "Refresh Succeeded" event', async () => {
86
+ const httpClient = vi
87
+ .fn()
88
+ .mockResolvedValue(respondWith(Object.assign(Object.assign({}, initialStep), { title: 'Refreshed', model: { color: 'red' } })));
89
+ const props = await renderAndRefresh(httpClient);
90
+ await waitFor(() => {
91
+ expect(screen.getByText('Refreshed')).toBeInTheDocument();
92
+ });
93
+ expect(screen.queryByText(props.initialStep.title)).not.toBeInTheDocument();
94
+ expect(props.onEvent).toHaveBeenCalledWith('Dynamic Flow - Refresh Succeeded', expect.objectContaining({
95
+ flowId: 'flow-id',
96
+ stepId: 'step-id',
97
+ stepCount: 0,
98
+ schema: '#color-schema',
99
+ }));
100
+ });
101
+ });
102
+ describe('when the response is an invalid step', () => {
103
+ it('should trigger onError and the "Action Failed" and "Failed" events', async () => {
104
+ const httpClient = vi.fn().mockResolvedValue(respondWith('not a step, just a string'));
105
+ const props = await renderAndRefresh(httpClient);
106
+ await waitFor(() => {
107
+ expect(props.onError).toHaveBeenCalledWith(expect.any(Error), undefined);
108
+ });
109
+ expect(props.onEvent).toHaveBeenCalledWith('Dynamic Flow - Refresh Failed', expect.objectContaining({
110
+ flowId: 'flow-id',
111
+ stepId: 'step-id',
112
+ stepCount: 0,
113
+ schema: '#color-schema',
114
+ }));
115
+ expect(props.onEvent).toHaveBeenCalledWith('Dynamic Flow - Failed', expect.objectContaining({
116
+ flowId: 'flow-id',
117
+ stepId: 'step-id',
118
+ stepCount: 0,
119
+ }));
120
+ });
121
+ });
122
+ });
123
+ describe('Status 304', () => {
124
+ it('should trigger the "Refresh Succeeded" event and not change the step', async () => {
125
+ const httpClient = vi.fn().mockResolvedValue(respondWith(null, { status: 304 }));
126
+ const props = await renderAndRefresh(httpClient);
127
+ await waitFor(() => {
128
+ expect(screen.getByText(initialStep.title)).toBeInTheDocument();
129
+ });
130
+ expect(props.onEvent).toHaveBeenCalledWith('Dynamic Flow - Refresh Succeeded', expect.objectContaining({
131
+ flowId: 'flow-id',
132
+ stepId: 'step-id',
133
+ stepCount: 0,
134
+ schema: '#color-schema',
135
+ }));
136
+ });
137
+ });
138
+ describe('Status 4xx or 5xx', () => {
139
+ describe('When the response body can be parsed as an ErrorResponseBody', () => {
140
+ const getHttpClient = () => vi.fn().mockResolvedValue(respondWith({
141
+ analytics: { shopping: 'list' },
142
+ }, { status: 422 }));
143
+ it('should trigger a "Refresh failed" and "Failed" events with analytics', async () => {
144
+ const props = await renderAndRefresh(getHttpClient());
145
+ expect(props.onEvent).toHaveBeenCalledWith('Dynamic Flow - Refresh Failed', expect.objectContaining({
146
+ flowId: 'flow-id',
147
+ stepId: 'step-id',
148
+ schema: '#color-schema',
149
+ shopping: 'list',
150
+ statusCode: 422,
151
+ }));
152
+ expect(props.onEvent).toHaveBeenLastCalledWith('Dynamic Flow - Failed', expect.objectContaining({
153
+ flowId: 'flow-id',
154
+ stepId: 'step-id',
155
+ shopping: 'list',
156
+ }));
157
+ });
158
+ });
159
+ describe('When the response body can NOT be parsed as an ErrorResponseBody', () => {
160
+ const getHttpClient = () => vi.fn().mockResolvedValue(respondWith('', { status: 422 }));
161
+ it('should trigger a "Refresh failed" and "Failed" events', async () => {
162
+ const props = await renderAndRefresh(getHttpClient());
163
+ expect(props.onEvent).toHaveBeenCalledWith('Dynamic Flow - Refresh Failed', expect.objectContaining({
164
+ flowId: 'flow-id',
165
+ stepId: 'step-id',
166
+ schema: '#color-schema',
167
+ statusCode: 422,
168
+ }));
169
+ expect(props.onEvent).toHaveBeenLastCalledWith('Dynamic Flow - Failed', expect.objectContaining({
170
+ flowId: 'flow-id',
171
+ stepId: 'step-id',
172
+ }));
173
+ });
174
+ });
175
+ });
176
+ describe('Aborting a refresh request', () => {
177
+ describe('When a second refresh request is made', () => {
178
+ it('should abort the first request and only use the response from the second request', async () => {
179
+ // The first request takes 100ms to respond
180
+ // The second request takes only 10ms to respond
181
+ const httpClient = vi
182
+ .fn()
183
+ .mockImplementationOnce(async (input, init) => {
184
+ var _a;
185
+ return abortableDelay(200, (_a = init === null || init === void 0 ? void 0 : init.signal) !== null && _a !== void 0 ? _a : null).then(async () => respondWith(Object.assign(Object.assign({}, initialStep), { title: 'Refreshed One', model: { color: 'red' } })));
186
+ })
187
+ .mockImplementationOnce(async (input, init) => {
188
+ var _a;
189
+ return abortableDelay(100, (_a = init === null || init === void 0 ? void 0 : init.signal) !== null && _a !== void 0 ? _a : null).then(async () => respondWith(Object.assign(Object.assign({}, initialStep), { title: 'Refreshed Two', model: { color: 'blue' } })));
190
+ });
191
+ await renderAndRefresh(httpClient);
192
+ await user.click(screen.getByText('Blue'));
193
+ await waitFor(() => {
194
+ expect(screen.getByText('Refreshed Two')).toBeInTheDocument();
195
+ });
196
+ // wait enough time for the first request to finish
197
+ // had it not been cancelled, it would have resolved after 100ms
198
+ await act(async () => vi.advanceTimersByTimeAsync(100));
199
+ // We expect the first request to be cancelled and not to have updated the UI
200
+ expect(screen.queryByText('Refreshed One')).not.toBeInTheDocument();
201
+ expect(screen.getByText('Refreshed Two')).toBeInTheDocument();
202
+ });
203
+ });
204
+ });
205
+ });