@defra/forms-engine-plugin 0.0.3 → 0.0.4

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 (328) hide show
  1. package/.public/assets/fonts/bold-affa96571d-v2.woff +0 -0
  2. package/.public/assets/fonts/bold-b542beb274-v2.woff2 +0 -0
  3. package/.public/assets/fonts/light-94a07e06a1-v2.woff2 +0 -0
  4. package/.public/assets/fonts/light-f591b13f7d-v2.woff +0 -0
  5. package/.public/assets/images/favicon.ico +0 -0
  6. package/.public/assets/images/favicon.svg +1 -0
  7. package/.public/assets/images/govuk-crest.svg +1 -0
  8. package/.public/assets/images/govuk-icon-180.png +0 -0
  9. package/.public/assets/images/govuk-icon-192.png +0 -0
  10. package/.public/assets/images/govuk-icon-512.png +0 -0
  11. package/.public/assets/images/govuk-icon-mask.svg +1 -0
  12. package/.public/assets/images/govuk-opengraph-image.png +0 -0
  13. package/.public/assets/manifest.json +39 -0
  14. package/.public/assets-manifest.json +24 -0
  15. package/.public/javascripts/application.0fd8c18.min.js +3 -0
  16. package/.public/javascripts/application.0fd8c18.min.js.LICENSE.txt +58 -0
  17. package/.public/javascripts/application.0fd8c18.min.js.map +1 -0
  18. package/.public/javascripts/file-upload.b2f18f5.min.js +2 -0
  19. package/.public/javascripts/file-upload.b2f18f5.min.js.map +1 -0
  20. package/.public/javascripts/vendor/accessible-autocomplete.275d332.min.js +2 -0
  21. package/.public/javascripts/vendor/accessible-autocomplete.275d332.min.js.map +1 -0
  22. package/.public/stylesheets/application.e340021.min.css +14 -0
  23. package/.public/stylesheets/application.e340021.min.css.map +1 -0
  24. package/package.json +6 -2
  25. package/.browserslistrc +0 -16
  26. package/.editorconfig +0 -9
  27. package/.eslintrc.cjs +0 -266
  28. package/.github/dependabot.yml +0 -85
  29. package/.github/workflows/check-pull-request.yml +0 -209
  30. package/.github/workflows/pr-notifier.yml +0 -98
  31. package/.github/workflows/publish.yml +0 -111
  32. package/.husky/pre-commit +0 -1
  33. package/.lintstagedrc.js +0 -4
  34. package/.nvmrc +0 -1
  35. package/.prettierignore +0 -8
  36. package/.prettierrc.cjs +0 -26
  37. package/Dockerfile +0 -61
  38. package/Procfile +0 -1
  39. package/babel.config.cjs +0 -55
  40. package/compose/aws.env +0 -4
  41. package/compose/start-localstack.sh +0 -26
  42. package/docker-compose.yaml +0 -86
  43. package/globals.d.ts +0 -1
  44. package/jest.config.cjs +0 -54
  45. package/jest.environment.js +0 -4
  46. package/jest.setup.cjs +0 -14
  47. package/postcss.config.js +0 -26
  48. package/sonar-project.properties +0 -17
  49. package/src/client/javascripts/application.js +0 -87
  50. package/src/client/javascripts/file-upload.js +0 -386
  51. package/src/client/stylesheets/_code.scss +0 -33
  52. package/src/client/stylesheets/_govuk-frontend.scss +0 -4
  53. package/src/client/stylesheets/_prose.scss +0 -56
  54. package/src/client/stylesheets/_service-banner.scss +0 -24
  55. package/src/client/stylesheets/_summary-list.scss +0 -28
  56. package/src/client/stylesheets/_tag-env.scss +0 -24
  57. package/src/client/stylesheets/application.scss +0 -14
  58. package/src/common/cookies.js +0 -58
  59. package/src/common/cookies.test.js +0 -23
  60. package/src/common/types.js +0 -5
  61. package/src/config/index.ts +0 -271
  62. package/src/index.ts +0 -31
  63. package/src/server/common/helpers/logging/logger-options.test.ts +0 -50
  64. package/src/server/common/helpers/logging/logger-options.ts +0 -46
  65. package/src/server/common/helpers/logging/logger.ts +0 -7
  66. package/src/server/common/helpers/logging/request-logger.ts +0 -9
  67. package/src/server/common/helpers/logging/request-tracing.js +0 -10
  68. package/src/server/common/helpers/redis-client.js +0 -70
  69. package/src/server/constants.js +0 -1
  70. package/src/server/forms/README.md +0 -10
  71. package/src/server/forms/components.json +0 -1015
  72. package/src/server/forms/report-a-terrorist.json +0 -270
  73. package/src/server/forms/runner-components-test.json +0 -365
  74. package/src/server/forms/test.json +0 -581
  75. package/src/server/index.test.ts +0 -582
  76. package/src/server/index.ts +0 -140
  77. package/src/server/plugins/blankie.test.ts +0 -73
  78. package/src/server/plugins/blankie.ts +0 -48
  79. package/src/server/plugins/crumb.ts +0 -20
  80. package/src/server/plugins/engine/README.md +0 -87
  81. package/src/server/plugins/engine/components/AutocompleteField.test.ts +0 -294
  82. package/src/server/plugins/engine/components/AutocompleteField.ts +0 -49
  83. package/src/server/plugins/engine/components/CheckboxesField.test.ts +0 -379
  84. package/src/server/plugins/engine/components/CheckboxesField.ts +0 -106
  85. package/src/server/plugins/engine/components/ComponentBase.ts +0 -97
  86. package/src/server/plugins/engine/components/ComponentCollection.ts +0 -278
  87. package/src/server/plugins/engine/components/DatePartsField.test.ts +0 -822
  88. package/src/server/plugins/engine/components/DatePartsField.ts +0 -264
  89. package/src/server/plugins/engine/components/Details.test.ts +0 -49
  90. package/src/server/plugins/engine/components/Details.ts +0 -30
  91. package/src/server/plugins/engine/components/EmailAddressField.test.ts +0 -395
  92. package/src/server/plugins/engine/components/EmailAddressField.ts +0 -55
  93. package/src/server/plugins/engine/components/FileUploadField.test.ts +0 -778
  94. package/src/server/plugins/engine/components/FileUploadField.ts +0 -262
  95. package/src/server/plugins/engine/components/FormComponent.ts +0 -249
  96. package/src/server/plugins/engine/components/Html.test.ts +0 -48
  97. package/src/server/plugins/engine/components/Html.ts +0 -29
  98. package/src/server/plugins/engine/components/InsetText.test.ts +0 -48
  99. package/src/server/plugins/engine/components/InsetText.ts +0 -27
  100. package/src/server/plugins/engine/components/List.test.ts +0 -76
  101. package/src/server/plugins/engine/components/List.ts +0 -72
  102. package/src/server/plugins/engine/components/ListFormComponent.ts +0 -140
  103. package/src/server/plugins/engine/components/MonthYearField.test.ts +0 -567
  104. package/src/server/plugins/engine/components/MonthYearField.ts +0 -222
  105. package/src/server/plugins/engine/components/MultilineTextField.test.ts +0 -558
  106. package/src/server/plugins/engine/components/MultilineTextField.ts +0 -138
  107. package/src/server/plugins/engine/components/NumberField.test.ts +0 -701
  108. package/src/server/plugins/engine/components/NumberField.ts +0 -163
  109. package/src/server/plugins/engine/components/RadiosField.test.ts +0 -288
  110. package/src/server/plugins/engine/components/RadiosField.ts +0 -24
  111. package/src/server/plugins/engine/components/SelectField.test.ts +0 -288
  112. package/src/server/plugins/engine/components/SelectField.ts +0 -47
  113. package/src/server/plugins/engine/components/SelectionControlField.ts +0 -43
  114. package/src/server/plugins/engine/components/TelephoneNumberField.test.ts +0 -356
  115. package/src/server/plugins/engine/components/TelephoneNumberField.ts +0 -67
  116. package/src/server/plugins/engine/components/TextField.test.ts +0 -489
  117. package/src/server/plugins/engine/components/TextField.ts +0 -96
  118. package/src/server/plugins/engine/components/UkAddressField.test.ts +0 -623
  119. package/src/server/plugins/engine/components/UkAddressField.ts +0 -172
  120. package/src/server/plugins/engine/components/YesNoField.test.ts +0 -248
  121. package/src/server/plugins/engine/components/YesNoField.ts +0 -31
  122. package/src/server/plugins/engine/components/constants.ts +0 -1
  123. package/src/server/plugins/engine/components/helpers.ts +0 -330
  124. package/src/server/plugins/engine/components/index.ts +0 -24
  125. package/src/server/plugins/engine/components/types.ts +0 -117
  126. package/src/server/plugins/engine/configureEnginePlugin.ts +0 -47
  127. package/src/server/plugins/engine/helpers.test.ts +0 -791
  128. package/src/server/plugins/engine/helpers.ts +0 -379
  129. package/src/server/plugins/engine/index.ts +0 -7
  130. package/src/server/plugins/engine/models/FormModel.test.ts +0 -42
  131. package/src/server/plugins/engine/models/FormModel.ts +0 -443
  132. package/src/server/plugins/engine/models/RepeatingSummaryViewModel.ts +0 -0
  133. package/src/server/plugins/engine/models/Section.ts +0 -0
  134. package/src/server/plugins/engine/models/SummaryViewModel.test.ts +0 -209
  135. package/src/server/plugins/engine/models/SummaryViewModel.ts +0 -220
  136. package/src/server/plugins/engine/models/index.ts +0 -2
  137. package/src/server/plugins/engine/models/types.ts +0 -114
  138. package/src/server/plugins/engine/outputFormatters/human/v1.test.ts +0 -143
  139. package/src/server/plugins/engine/outputFormatters/human/v1.ts +0 -73
  140. package/src/server/plugins/engine/outputFormatters/index.test.ts +0 -17
  141. package/src/server/plugins/engine/outputFormatters/index.ts +0 -44
  142. package/src/server/plugins/engine/outputFormatters/machine/v1.test.ts +0 -229
  143. package/src/server/plugins/engine/outputFormatters/machine/v1.ts +0 -140
  144. package/src/server/plugins/engine/outputFormatters/machine/v2.test.ts +0 -229
  145. package/src/server/plugins/engine/outputFormatters/machine/v2.ts +0 -153
  146. package/src/server/plugins/engine/pageControllers/FileUploadPageController.test.ts +0 -1108
  147. package/src/server/plugins/engine/pageControllers/FileUploadPageController.ts +0 -446
  148. package/src/server/plugins/engine/pageControllers/PageController.test.ts +0 -205
  149. package/src/server/plugins/engine/pageControllers/PageController.ts +0 -176
  150. package/src/server/plugins/engine/pageControllers/QuestionPageController.test.ts +0 -1264
  151. package/src/server/plugins/engine/pageControllers/QuestionPageController.ts +0 -561
  152. package/src/server/plugins/engine/pageControllers/README.md +0 -28
  153. package/src/server/plugins/engine/pageControllers/RepeatPageController.test.ts +0 -264
  154. package/src/server/plugins/engine/pageControllers/RepeatPageController.ts +0 -458
  155. package/src/server/plugins/engine/pageControllers/StartPageController.ts +0 -18
  156. package/src/server/plugins/engine/pageControllers/StatusPageController.ts +0 -50
  157. package/src/server/plugins/engine/pageControllers/SummaryPageController.ts +0 -261
  158. package/src/server/plugins/engine/pageControllers/TerminalController.test.ts +0 -28
  159. package/src/server/plugins/engine/pageControllers/TerminalPageController.ts +0 -19
  160. package/src/server/plugins/engine/pageControllers/helpers.test.ts +0 -198
  161. package/src/server/plugins/engine/pageControllers/helpers.ts +0 -101
  162. package/src/server/plugins/engine/pageControllers/index.ts +0 -10
  163. package/src/server/plugins/engine/pageControllers/validationOptions.ts +0 -89
  164. package/src/server/plugins/engine/plugin.ts +0 -673
  165. package/src/server/plugins/engine/services/formSubmissionService.js +0 -46
  166. package/src/server/plugins/engine/services/formsService.js +0 -46
  167. package/src/server/plugins/engine/services/formsService.test.js +0 -90
  168. package/src/server/plugins/engine/services/index.js +0 -3
  169. package/src/server/plugins/engine/services/notifyService.test.ts +0 -132
  170. package/src/server/plugins/engine/services/notifyService.ts +0 -64
  171. package/src/server/plugins/engine/services/uploadService.js +0 -60
  172. package/src/server/plugins/engine/types.ts +0 -315
  173. package/src/server/plugins/engine/views/components/autocompletefield.html +0 -5
  174. package/src/server/plugins/engine/views/components/checkboxesfield.html +0 -5
  175. package/src/server/plugins/engine/views/components/datepartsfield.html +0 -5
  176. package/src/server/plugins/engine/views/components/details.html +0 -6
  177. package/src/server/plugins/engine/views/components/emailaddressfield.html +0 -5
  178. package/src/server/plugins/engine/views/components/fileuploadfield-key.html +0 -8
  179. package/src/server/plugins/engine/views/components/fileuploadfield-value.html +0 -3
  180. package/src/server/plugins/engine/views/components/fileuploadfield.html +0 -24
  181. package/src/server/plugins/engine/views/components/html.html +0 -3
  182. package/src/server/plugins/engine/views/components/insettext.html +0 -7
  183. package/src/server/plugins/engine/views/components/list.html +0 -36
  184. package/src/server/plugins/engine/views/components/monthyearfield.html +0 -5
  185. package/src/server/plugins/engine/views/components/multilinetextfield.html +0 -10
  186. package/src/server/plugins/engine/views/components/numberfield.html +0 -5
  187. package/src/server/plugins/engine/views/components/radiosfield.html +0 -5
  188. package/src/server/plugins/engine/views/components/selectfield.html +0 -5
  189. package/src/server/plugins/engine/views/components/telephonenumberfield.html +0 -5
  190. package/src/server/plugins/engine/views/components/textfield.html +0 -5
  191. package/src/server/plugins/engine/views/components/ukaddressfield.html +0 -25
  192. package/src/server/plugins/engine/views/components/yesnofield.html +0 -5
  193. package/src/server/plugins/engine/views/file-upload.html +0 -45
  194. package/src/server/plugins/engine/views/index.html +0 -39
  195. package/src/server/plugins/engine/views/item-delete.html +0 -56
  196. package/src/server/plugins/engine/views/partials/components.html +0 -6
  197. package/src/server/plugins/engine/views/partials/conditional-components.html +0 -3
  198. package/src/server/plugins/engine/views/partials/debug.html +0 -44
  199. package/src/server/plugins/engine/views/partials/form.html +0 -15
  200. package/src/server/plugins/engine/views/partials/heading.html +0 -16
  201. package/src/server/plugins/engine/views/partials/preview-banner.html +0 -32
  202. package/src/server/plugins/engine/views/partials/preview-banner.test.js +0 -122
  203. package/src/server/plugins/engine/views/partials/warn-missing-notification-email.html +0 -10
  204. package/src/server/plugins/engine/views/repeat-list-summary.html +0 -53
  205. package/src/server/plugins/errorPages.ts +0 -58
  206. package/src/server/plugins/nunjucks/context.js +0 -88
  207. package/src/server/plugins/nunjucks/context.test.js +0 -142
  208. package/src/server/plugins/nunjucks/enviroment.test.js +0 -201
  209. package/src/server/plugins/nunjucks/environment.js +0 -116
  210. package/src/server/plugins/nunjucks/filters/answer.js +0 -27
  211. package/src/server/plugins/nunjucks/filters/answer.test.js +0 -89
  212. package/src/server/plugins/nunjucks/filters/evaluate.js +0 -21
  213. package/src/server/plugins/nunjucks/filters/field.js +0 -28
  214. package/src/server/plugins/nunjucks/filters/field.test.js +0 -75
  215. package/src/server/plugins/nunjucks/filters/highlight.js +0 -11
  216. package/src/server/plugins/nunjucks/filters/href.js +0 -30
  217. package/src/server/plugins/nunjucks/filters/href.test.js +0 -80
  218. package/src/server/plugins/nunjucks/filters/index.js +0 -8
  219. package/src/server/plugins/nunjucks/filters/inspect.js +0 -15
  220. package/src/server/plugins/nunjucks/filters/page.js +0 -24
  221. package/src/server/plugins/nunjucks/filters/page.test.js +0 -65
  222. package/src/server/plugins/nunjucks/index.js +0 -3
  223. package/src/server/plugins/nunjucks/plugin.js +0 -40
  224. package/src/server/plugins/nunjucks/render.js +0 -42
  225. package/src/server/plugins/nunjucks/types.js +0 -40
  226. package/src/server/plugins/pulse.ts +0 -11
  227. package/src/server/plugins/router.ts +0 -201
  228. package/src/server/plugins/session.ts +0 -28
  229. package/src/server/routes/health.js +0 -13
  230. package/src/server/routes/health.test.js +0 -35
  231. package/src/server/routes/index.test.ts +0 -125
  232. package/src/server/routes/index.ts +0 -2
  233. package/src/server/routes/public.ts +0 -47
  234. package/src/server/routes/types.ts +0 -48
  235. package/src/server/schemas/index.ts +0 -34
  236. package/src/server/secure-context.js +0 -43
  237. package/src/server/services/cacheService.test.ts +0 -276
  238. package/src/server/services/cacheService.ts +0 -131
  239. package/src/server/services/httpService.test.js +0 -491
  240. package/src/server/services/httpService.ts +0 -50
  241. package/src/server/services/index.ts +0 -1
  242. package/src/server/types.ts +0 -54
  243. package/src/server/utils/notify.test.ts +0 -37
  244. package/src/server/utils/notify.ts +0 -50
  245. package/src/server/utils/secure-context/get-trust-store-certs.js +0 -11
  246. package/src/server/utils/secure-context/get-trust-store-certs.test.js +0 -19
  247. package/src/server/utils/utils.js +0 -24
  248. package/src/server/utils/utils.test.js +0 -54
  249. package/src/server/views/404.html +0 -16
  250. package/src/server/views/500.html +0 -19
  251. package/src/server/views/components/debug/macro.njk +0 -3
  252. package/src/server/views/components/debug/template.njk +0 -13
  253. package/src/server/views/components/service-banner/macro.njk +0 -3
  254. package/src/server/views/components/service-banner/template.njk +0 -20
  255. package/src/server/views/components/service-banner/template.test.js +0 -43
  256. package/src/server/views/components/tag-env/macro.njk +0 -3
  257. package/src/server/views/components/tag-env/template.njk +0 -30
  258. package/src/server/views/components/tag-env/template.test.js +0 -66
  259. package/src/server/views/confirmation.html +0 -19
  260. package/src/server/views/help/accessibility-statement.html +0 -58
  261. package/src/server/views/help/cookie-preferences.html +0 -57
  262. package/src/server/views/help/cookies.html +0 -71
  263. package/src/server/views/help/get-support.html +0 -37
  264. package/src/server/views/help/privacy-notice.html +0 -68
  265. package/src/server/views/help/terms-and-conditions.html +0 -83
  266. package/src/server/views/layout.html +0 -199
  267. package/src/server/views/summary.html +0 -50
  268. package/src/typings/hapi/index.d.ts +0 -95
  269. package/src/typings/hapi-tracing/index.d.ts +0 -6
  270. package/src/typings/index.d.ts +0 -3
  271. package/src/typings/joi/index.d.ts +0 -22
  272. package/stylelint.config.js +0 -10
  273. package/test/client/javascripts/file-upload.test.js +0 -1197
  274. package/test/condition/checkboxes.test.js +0 -112
  275. package/test/condition/radios.test.js +0 -112
  276. package/test/condition/text.test.js +0 -103
  277. package/test/fixtures/assets-manifest.json +0 -4
  278. package/test/fixtures/form.js +0 -86
  279. package/test/fixtures/index.js +0 -2
  280. package/test/fixtures/list.js +0 -92
  281. package/test/form/cookies.test.js +0 -338
  282. package/test/form/csrf.test.js +0 -87
  283. package/test/form/definitions/basic.js +0 -101
  284. package/test/form/definitions/blank.js +0 -10
  285. package/test/form/definitions/checkboxes.json +0 -88
  286. package/test/form/definitions/components.json +0 -452
  287. package/test/form/definitions/conditional-reveal.js +0 -140
  288. package/test/form/definitions/conditions-basic.js +0 -187
  289. package/test/form/definitions/conditions-complex.js +0 -338
  290. package/test/form/definitions/conditions-dates.js +0 -78
  291. package/test/form/definitions/conditions-escaping.js +0 -143
  292. package/test/form/definitions/demo-cph-number.js +0 -3099
  293. package/test/form/definitions/feedback.json +0 -45
  294. package/test/form/definitions/fields-optional.js +0 -402
  295. package/test/form/definitions/fields-required.js +0 -402
  296. package/test/form/definitions/file-upload-basic.js +0 -44
  297. package/test/form/definitions/file-upload.js +0 -66
  298. package/test/form/definitions/minimal.js +0 -39
  299. package/test/form/definitions/phase-alpha.json +0 -33
  300. package/test/form/definitions/phase-default.json +0 -26
  301. package/test/form/definitions/radios.json +0 -88
  302. package/test/form/definitions/repeat-mixed.js +0 -54
  303. package/test/form/definitions/repeat.js +0 -70
  304. package/test/form/definitions/status.json +0 -126
  305. package/test/form/definitions/templates.js +0 -183
  306. package/test/form/definitions/test.json +0 -581
  307. package/test/form/definitions/text.json +0 -75
  308. package/test/form/definitions/titles.json +0 -170
  309. package/test/form/definitions.test.js +0 -47
  310. package/test/form/exit-page.test.js +0 -210
  311. package/test/form/feedback.test.js +0 -68
  312. package/test/form/fields-optional.test.js +0 -237
  313. package/test/form/fields-required.test.js +0 -294
  314. package/test/form/file-upload.test.js +0 -313
  315. package/test/form/govuk-notify.test.js +0 -449
  316. package/test/form/journey-basic.test.js +0 -444
  317. package/test/form/persist-files.test.js +0 -227
  318. package/test/form/phase-banner.test.js +0 -71
  319. package/test/form/repeat.test.js +0 -628
  320. package/test/form/summary-submission-email.test.js +0 -95
  321. package/test/form/template.test.js +0 -288
  322. package/test/form/titles.test.js +0 -204
  323. package/test/helpers/component-helpers.js +0 -74
  324. package/test/utils/get-cookie.js +0 -42
  325. package/test/utils/get-form-definitions.js +0 -18
  326. package/tmp.pdf +0 -1
  327. package/tsconfig.json +0 -28
  328. package/webpack.config.js +0 -208
@@ -1,623 +0,0 @@
1
- import { ComponentType, type UkAddressFieldComponent } from '@defra/forms-model'
2
-
3
- import { ComponentCollection } from '~/src/server/plugins/engine/components/ComponentCollection.js'
4
- import {
5
- getAnswer,
6
- type Field
7
- } from '~/src/server/plugins/engine/components/helpers.js'
8
- import { type ViewModel } from '~/src/server/plugins/engine/components/types.js'
9
- import { FormModel } from '~/src/server/plugins/engine/models/FormModel.js'
10
- import {
11
- type FormPayload,
12
- type FormState
13
- } from '~/src/server/plugins/engine/types.js'
14
- import definition from '~/test/form/definitions/blank.js'
15
-
16
- describe('UkAddressField', () => {
17
- let model: FormModel
18
-
19
- beforeEach(() => {
20
- model = new FormModel(definition, {
21
- basePath: 'test'
22
- })
23
- })
24
-
25
- describe('Defaults', () => {
26
- let def: UkAddressFieldComponent
27
- let collection: ComponentCollection
28
- let field: Field
29
-
30
- beforeEach(() => {
31
- def = {
32
- title: 'Example UK address',
33
- name: 'myComponent',
34
- type: ComponentType.UkAddressField,
35
- options: {}
36
- } satisfies UkAddressFieldComponent
37
-
38
- collection = new ComponentCollection([def], { model })
39
- field = collection.fields[0]
40
- })
41
-
42
- describe('Schema', () => {
43
- it('uses collection titles as labels', () => {
44
- const { formSchema } = collection
45
- const { keys } = formSchema.describe()
46
-
47
- expect(keys).toHaveProperty(
48
- 'myComponent__addressLine1',
49
- expect.objectContaining({
50
- flags: expect.objectContaining({ label: 'Address line 1' })
51
- })
52
- )
53
-
54
- expect(keys).toHaveProperty(
55
- 'myComponent__addressLine2',
56
- expect.objectContaining({
57
- flags: expect.objectContaining({ label: 'Address line 2' })
58
- })
59
- )
60
-
61
- expect(keys).toHaveProperty(
62
- 'myComponent__town',
63
- expect.objectContaining({
64
- flags: expect.objectContaining({ label: 'Town or city' })
65
- })
66
- )
67
-
68
- expect(keys).toHaveProperty(
69
- `myComponent__postcode`,
70
- expect.objectContaining({
71
- flags: expect.objectContaining({ label: 'Postcode' })
72
- })
73
- )
74
- })
75
-
76
- it('uses collection names as keys', () => {
77
- const { formSchema } = collection
78
- const { keys } = formSchema.describe()
79
-
80
- expect(field.keys).toEqual([
81
- 'myComponent',
82
- 'myComponent__addressLine1',
83
- 'myComponent__addressLine2',
84
- 'myComponent__town',
85
- 'myComponent__postcode'
86
- ])
87
-
88
- expect(field.collection?.keys).not.toHaveProperty('myComponent')
89
-
90
- for (const key of field.collection?.keys ?? []) {
91
- expect(keys).toHaveProperty(key)
92
- }
93
- })
94
-
95
- it('is required by default', () => {
96
- const { formSchema } = collection
97
- const { keys } = formSchema.describe()
98
-
99
- expect(keys).toHaveProperty(
100
- 'myComponent__addressLine1',
101
- expect.objectContaining({
102
- flags: expect.objectContaining({ presence: 'required' })
103
- })
104
- )
105
-
106
- expect(keys).toHaveProperty(
107
- 'myComponent__addressLine2',
108
- expect.objectContaining({
109
- allow: [''], // Required but empty string is allowed
110
- flags: expect.objectContaining({ presence: 'required' })
111
- })
112
- )
113
-
114
- expect(keys).toHaveProperty(
115
- 'myComponent__town',
116
- expect.objectContaining({
117
- flags: expect.objectContaining({ presence: 'required' })
118
- })
119
- )
120
-
121
- expect(keys).toHaveProperty(
122
- `myComponent__postcode`,
123
- expect.objectContaining({
124
- flags: expect.objectContaining({ presence: 'required' })
125
- })
126
- )
127
- })
128
-
129
- it('is optional when configured', () => {
130
- const collectionOptional = new ComponentCollection(
131
- [
132
- {
133
- title: 'Example UK address',
134
- name: 'myComponent',
135
- type: ComponentType.UkAddressField,
136
- options: { required: false }
137
- }
138
- ],
139
- { model }
140
- )
141
-
142
- const { formSchema } = collectionOptional
143
- const { keys } = formSchema.describe()
144
-
145
- expect(keys).toHaveProperty(
146
- 'myComponent__addressLine1',
147
- expect.objectContaining({ allow: [''] })
148
- )
149
-
150
- expect(keys).toHaveProperty(
151
- 'myComponent__addressLine2',
152
- expect.objectContaining({ allow: [''] })
153
- )
154
-
155
- expect(keys).toHaveProperty(
156
- 'myComponent__town',
157
- expect.objectContaining({ allow: [''] })
158
- )
159
-
160
- expect(keys).toHaveProperty(
161
- `myComponent__postcode`,
162
- expect.objectContaining({ allow: [''] })
163
- )
164
-
165
- const result = collectionOptional.validate(
166
- getFormData({
167
- addressLine1: '',
168
- addressLine2: '',
169
- town: '',
170
- postcode: ''
171
- })
172
- )
173
-
174
- expect(result.errors).toBeUndefined()
175
- })
176
-
177
- it('accepts valid values', () => {
178
- const result1 = collection.validate(
179
- getFormData({
180
- addressLine1: 'Richard Fairclough House',
181
- addressLine2: 'Knutsford Road',
182
- town: 'Warrington',
183
- postcode: 'WA4 1HT'
184
- })
185
- )
186
-
187
- const result2 = collection.validate(
188
- getFormData({
189
- addressLine1: 'Richard Fairclough House',
190
- addressLine2: '', // Optional field
191
- town: 'Warrington',
192
- postcode: 'WA4 1HT'
193
- })
194
- )
195
-
196
- expect(result1.errors).toBeUndefined()
197
- expect(result2.errors).toBeUndefined()
198
- })
199
-
200
- it('adds errors for empty value', () => {
201
- const result = collection.validate(
202
- getFormData({
203
- addressLine1: '',
204
- addressLine2: '',
205
- town: '',
206
- postcode: ''
207
- })
208
- )
209
-
210
- expect(result.errors).toEqual([
211
- expect.objectContaining({
212
- text: 'Enter address line 1'
213
- }),
214
- expect.objectContaining({
215
- text: 'Enter town or city'
216
- }),
217
- expect.objectContaining({
218
- text: 'Enter postcode'
219
- })
220
- ])
221
- })
222
-
223
- it('adds errors for invalid values', () => {
224
- const result1 = collection.validate(getFormData({ unknown: 'invalid' }))
225
-
226
- const result2 = collection.validate(
227
- getFormData({
228
- addressLine1: ['invalid'],
229
- addressLine2: ['invalid'],
230
- town: ['invalid'],
231
- postcode: ['invalid']
232
- })
233
- )
234
-
235
- const result3 = collection.validate(
236
- getFormData({
237
- addressLine1: 'invalid',
238
- addressLine2: 'invalid',
239
- town: 'invalid',
240
- postcode: 'invalid'
241
- })
242
- )
243
-
244
- expect(result1.errors).toBeTruthy()
245
- expect(result2.errors).toBeTruthy()
246
- expect(result3.errors).toBeTruthy()
247
- })
248
- })
249
-
250
- describe('State', () => {
251
- const address: FormPayload = {
252
- addressLine1: 'Richard Fairclough House',
253
- addressLine2: 'Knutsford Road',
254
- town: 'Warrington',
255
- postcode: 'WA4 1HT'
256
- }
257
-
258
- it('returns text from state', () => {
259
- const state1 = getFormState(address)
260
- const state2 = getFormState({})
261
-
262
- const answer1 = getAnswer(field, state1)
263
- const answer2 = getAnswer(field, state2)
264
-
265
- expect(answer1).toBe(
266
- 'Richard Fairclough House<br>Knutsford Road<br>Warrington<br>WA4 1HT<br>'
267
- )
268
-
269
- expect(answer2).toBe('')
270
- })
271
-
272
- it('returns payload from state', () => {
273
- const state1 = getFormState(address)
274
- const state2 = getFormState({})
275
-
276
- const payload1 = field.getFormDataFromState(state1)
277
- const payload2 = field.getFormDataFromState(state2)
278
-
279
- expect(payload1).toEqual(getFormData(address))
280
- expect(payload2).toEqual(getFormData({}))
281
- })
282
-
283
- it('returns value from state', () => {
284
- const state1 = getFormState(address)
285
- const state2 = getFormState({})
286
-
287
- const value1 = field.getFormValueFromState(state1)
288
- const value2 = field.getFormValueFromState(state2)
289
-
290
- expect(value1).toEqual(address)
291
- expect(value2).toBeUndefined()
292
- })
293
-
294
- it('returns context for conditions and form submission', () => {
295
- const state1 = getFormState(address)
296
- const state2 = getFormState({})
297
-
298
- const value1 = field.getContextValueFromState(state1)
299
- const value2 = field.getContextValueFromState(state2)
300
-
301
- expect(value1).toEqual([
302
- 'Richard Fairclough House',
303
- 'Knutsford Road',
304
- 'Warrington',
305
- 'WA4 1HT'
306
- ])
307
-
308
- expect(value2).toBeNull()
309
- })
310
-
311
- it('returns state from payload', () => {
312
- const payload1 = getFormData(address)
313
- const payload2 = getFormData({})
314
-
315
- const value1 = field.getStateFromValidForm(payload1)
316
- const value2 = field.getStateFromValidForm(payload2)
317
-
318
- expect(value1).toEqual(getFormState(address))
319
- expect(value2).toEqual(getFormState({}))
320
- })
321
- })
322
-
323
- describe('View model', () => {
324
- const address: FormPayload = {
325
- addressLine1: 'Richard Fairclough House',
326
- addressLine2: 'Knutsford Road',
327
- town: 'Warrington',
328
- postcode: 'WA4 1HT'
329
- }
330
-
331
- it('sets Nunjucks component defaults', () => {
332
- const payload = getFormData(address)
333
- const viewModel = field.getViewModel(payload)
334
-
335
- expect(viewModel).toEqual(
336
- expect.objectContaining({
337
- label: { text: def.title },
338
- name: 'myComponent',
339
- id: 'myComponent',
340
- value: undefined,
341
- components: expect.arrayContaining([
342
- expect.objectContaining({
343
- model: getViewModel(address, 'addressLine1', {
344
- label: { text: 'Address line 1' },
345
- attributes: { autocomplete: 'address-line1' }
346
- })
347
- }),
348
-
349
- expect.objectContaining({
350
- model: getViewModel(address, 'addressLine2', {
351
- label: { text: 'Address line 2 (optional)' },
352
- attributes: { autocomplete: 'address-line2' },
353
- value: address.addressLine2
354
- })
355
- }),
356
-
357
- expect.objectContaining({
358
- model: getViewModel(address, 'town', {
359
- label: { text: 'Town or city' },
360
- classes: 'govuk-!-width-two-thirds',
361
- attributes: { autocomplete: 'address-level2' },
362
- value: address.town
363
- })
364
- }),
365
-
366
- expect.objectContaining({
367
- model: getViewModel(address, 'postcode', {
368
- label: { text: 'Postcode' },
369
- classes: 'govuk-input--width-10',
370
- attributes: { autocomplete: 'postal-code' },
371
- value: address.postcode
372
- })
373
- })
374
- ])
375
- })
376
- )
377
- })
378
-
379
- it('sets Nunjucks component fieldset', () => {
380
- const payload = getFormData(address)
381
- const viewModel = field.getViewModel(payload)
382
-
383
- expect(viewModel.fieldset).toEqual({
384
- legend: {
385
- text: def.title,
386
- classes: 'govuk-fieldset__legend--m'
387
- }
388
- })
389
- })
390
- })
391
- })
392
-
393
- describe('Validation', () => {
394
- const address: FormPayload = {
395
- addressLine1: 'Richard Fairclough House',
396
- addressLine2: 'Knutsford Road',
397
- town: 'Warrington',
398
- postcode: 'WA4 1HT'
399
- }
400
-
401
- const addressLine1Invalid =
402
- 'Address line 1000000000000000000000000000000000000000000000000000000000000000000000000000000000000000'
403
-
404
- const addressLine2Invalid =
405
- 'Address line 2000000000000000000000000000000000000000000000000000000000000000000000000000000000000000'
406
-
407
- const townInvalid =
408
- 'Town 000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000'
409
-
410
- const postcodeInvalid = '111 XX2'
411
-
412
- describe.each([
413
- {
414
- description: 'Trim empty spaces',
415
- component: {
416
- title: 'Example UK address',
417
- name: 'myComponent',
418
- type: ComponentType.UkAddressField,
419
- options: {}
420
- } satisfies UkAddressFieldComponent,
421
- assertions: [
422
- {
423
- input: getFormData({
424
- addressLine1: ' Richard Fairclough House',
425
- addressLine2: ' Knutsford Road',
426
- town: ' Warrington',
427
- postcode: ' WA4 1HT'
428
- }),
429
- output: {
430
- value: getFormData(address)
431
- }
432
- },
433
- {
434
- input: getFormData({
435
- addressLine1: 'Richard Fairclough House ',
436
- addressLine2: 'Knutsford Road ',
437
- town: 'Warrington ',
438
- postcode: 'WA4 1HT '
439
- }),
440
- output: {
441
- value: getFormData(address)
442
- }
443
- },
444
- {
445
- input: getFormData({
446
- addressLine1: ' Richard Fairclough House \n\n',
447
- addressLine2: ' Knutsford Road \n\n',
448
- town: ' Warrington \n\n',
449
- postcode: ' WA4 1HT \n\n'
450
- }),
451
- output: {
452
- value: getFormData(address)
453
- }
454
- }
455
- ]
456
- },
457
- {
458
- description: 'Out of range values',
459
- component: {
460
- title: 'Example UK address',
461
- name: 'myComponent',
462
- type: ComponentType.UkAddressField,
463
- options: {}
464
- } satisfies UkAddressFieldComponent,
465
- assertions: [
466
- {
467
- input: getFormData({
468
- addressLine1: addressLine1Invalid,
469
- addressLine2: 'Knutsford Road',
470
- town: 'Warrington',
471
- postcode: 'WA4 1HT'
472
- }),
473
- output: {
474
- value: getFormData({
475
- addressLine1: addressLine1Invalid,
476
- addressLine2: 'Knutsford Road',
477
- town: 'Warrington',
478
- postcode: 'WA4 1HT'
479
- }),
480
- errors: [
481
- expect.objectContaining({
482
- text: 'Address line 1 must be 100 characters or less'
483
- })
484
- ]
485
- }
486
- },
487
- {
488
- input: getFormData({
489
- addressLine1: 'Richard Fairclough House',
490
- addressLine2: addressLine2Invalid,
491
- town: 'Warrington',
492
- postcode: 'WA4 1HT'
493
- }),
494
- output: {
495
- value: getFormData({
496
- addressLine1: 'Richard Fairclough House',
497
- addressLine2: addressLine2Invalid,
498
- town: 'Warrington',
499
- postcode: 'WA4 1HT'
500
- }),
501
- errors: [
502
- expect.objectContaining({
503
- text: 'Address line 2 must be 100 characters or less'
504
- })
505
- ]
506
- }
507
- },
508
- {
509
- input: getFormData({
510
- addressLine1: 'Richard Fairclough House',
511
- addressLine2: 'Knutsford Road',
512
- town: townInvalid,
513
- postcode: 'WA4 1HT'
514
- }),
515
- output: {
516
- value: getFormData({
517
- addressLine1: 'Richard Fairclough House',
518
- addressLine2: 'Knutsford Road',
519
- town: townInvalid,
520
- postcode: 'WA4 1HT'
521
- }),
522
- errors: [
523
- expect.objectContaining({
524
- text: 'Town or city must be 100 characters or less'
525
- })
526
- ]
527
- }
528
- },
529
- {
530
- input: getFormData({
531
- addressLine1: 'Richard Fairclough House',
532
- addressLine2: 'Knutsford Road',
533
- town: 'Warrington',
534
- postcode: postcodeInvalid
535
- }),
536
- output: {
537
- value: getFormData({
538
- addressLine1: 'Richard Fairclough House',
539
- addressLine2: 'Knutsford Road',
540
- town: 'Warrington',
541
- postcode: postcodeInvalid
542
- }),
543
- errors: [
544
- expect.objectContaining({
545
- text: 'Enter a valid postcode'
546
- })
547
- ]
548
- }
549
- }
550
- ]
551
- }
552
- ])('$description', ({ component: def, assertions }) => {
553
- let collection: ComponentCollection
554
-
555
- beforeEach(() => {
556
- collection = new ComponentCollection([def], { model })
557
- })
558
-
559
- it.each([...assertions])(
560
- 'validates custom example',
561
- ({ input, output }) => {
562
- const result = collection.validate(input)
563
- expect(result).toEqual(output)
564
- }
565
- )
566
- })
567
- })
568
- })
569
-
570
- /**
571
- * UK address field view model
572
- */
573
- function getViewModel(
574
- address: FormPayload,
575
- name: string,
576
- overrides?: Partial<ViewModel>
577
- ): Partial<ViewModel> {
578
- const payload = getFormData(address)
579
- const fieldName = `myComponent__${name}`
580
- const fieldClasses = overrides?.classes ?? undefined
581
- const fieldAttributes = overrides?.attributes ?? expect.any(Object)
582
-
583
- return {
584
- label: expect.objectContaining(
585
- overrides?.label ?? {
586
- text: expect.any(String)
587
- }
588
- ),
589
- name: fieldName,
590
- id: fieldName,
591
- value: payload[fieldName],
592
- classes: fieldClasses,
593
- attributes: fieldAttributes
594
- }
595
- }
596
-
597
- /**
598
- * UK address form data
599
- */
600
- function getFormData(address: FormPayload): FormPayload {
601
- return {
602
- myComponent__addressLine1: address.addressLine1,
603
- myComponent__addressLine2: address.addressLine2,
604
- myComponent__town: address.town,
605
- myComponent__postcode: address.postcode
606
- }
607
- }
608
-
609
- /**
610
- * UK address session state
611
- */
612
- function getFormState(address: FormPayload): FormState {
613
- const [addressLine1, addressLine2, town, postcode] = Object.values(
614
- getFormData(address)
615
- )
616
-
617
- return {
618
- myComponent__addressLine1: addressLine1 ?? null,
619
- myComponent__addressLine2: addressLine2 ?? null,
620
- myComponent__town: town ?? null,
621
- myComponent__postcode: postcode ?? null
622
- }
623
- }