@defra/forms-engine-plugin 0.0.4 → 0.0.6

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 (260) hide show
  1. package/.server/server/index.js +0 -4
  2. package/.server/server/index.js.map +1 -1
  3. package/.server/server/plugins/engine/helpers.js +3 -0
  4. package/.server/server/plugins/engine/helpers.js.map +1 -1
  5. package/.server/server/plugins/engine/index.js +27 -1
  6. package/.server/server/plugins/engine/index.js.map +1 -1
  7. package/.server/server/plugins/engine/pageControllers/FileUploadPageController.js +2 -4
  8. package/.server/server/plugins/engine/pageControllers/FileUploadPageController.js.map +1 -1
  9. package/.server/server/plugins/engine/pageControllers/QuestionPageController.js +4 -10
  10. package/.server/server/plugins/engine/pageControllers/QuestionPageController.js.map +1 -1
  11. package/.server/server/plugins/engine/pageControllers/StatusPageController.js +2 -3
  12. package/.server/server/plugins/engine/pageControllers/StatusPageController.js.map +1 -1
  13. package/.server/server/plugins/engine/pageControllers/SummaryPageController.js +2 -4
  14. package/.server/server/plugins/engine/pageControllers/SummaryPageController.js.map +1 -1
  15. package/.server/server/plugins/engine/plugin.js +65 -6
  16. package/.server/server/plugins/engine/plugin.js.map +1 -1
  17. package/.server/server/plugins/engine/types.js.map +1 -1
  18. package/.server/server/{views → plugins/engine/views}/components/service-banner/template.test.js +1 -1
  19. package/.server/server/plugins/engine/views/components/service-banner/template.test.js.map +1 -0
  20. package/.server/server/{views → plugins/engine/views}/components/tag-env/template.test.js +1 -1
  21. package/.server/server/plugins/engine/views/components/tag-env/template.test.js.map +1 -0
  22. package/.server/server/services/cacheService.js +5 -2
  23. package/.server/server/services/cacheService.js.map +1 -1
  24. package/.server/typings/hapi/index.d.js.map +1 -1
  25. package/README.md +215 -4
  26. package/package.json +3 -3
  27. package/src/client/javascripts/application.js +87 -0
  28. package/src/client/javascripts/file-upload.js +386 -0
  29. package/src/client/stylesheets/_code.scss +33 -0
  30. package/src/client/stylesheets/_govuk-frontend.scss +4 -0
  31. package/src/client/stylesheets/_prose.scss +56 -0
  32. package/src/client/stylesheets/_service-banner.scss +24 -0
  33. package/src/client/stylesheets/_summary-list.scss +28 -0
  34. package/src/client/stylesheets/_tag-env.scss +24 -0
  35. package/src/client/stylesheets/application.scss +14 -0
  36. package/src/common/cookies.js +58 -0
  37. package/src/common/cookies.test.js +23 -0
  38. package/src/common/types.js +5 -0
  39. package/src/config/index.ts +271 -0
  40. package/src/index.ts +31 -0
  41. package/src/server/common/helpers/logging/logger-options.test.ts +50 -0
  42. package/src/server/common/helpers/logging/logger-options.ts +46 -0
  43. package/src/server/common/helpers/logging/logger.ts +7 -0
  44. package/src/server/common/helpers/logging/request-logger.ts +9 -0
  45. package/src/server/common/helpers/logging/request-tracing.js +10 -0
  46. package/src/server/common/helpers/redis-client.js +70 -0
  47. package/src/server/constants.js +1 -0
  48. package/src/server/forms/README.md +10 -0
  49. package/src/server/forms/components.json +1015 -0
  50. package/src/server/forms/report-a-terrorist.json +270 -0
  51. package/src/server/forms/runner-components-test.json +365 -0
  52. package/src/server/forms/test.json +581 -0
  53. package/src/server/index.test.ts +582 -0
  54. package/src/server/index.ts +135 -0
  55. package/src/server/plugins/blankie.test.ts +73 -0
  56. package/src/server/plugins/blankie.ts +48 -0
  57. package/src/server/plugins/crumb.ts +20 -0
  58. package/src/server/plugins/engine/README.md +87 -0
  59. package/src/server/plugins/engine/components/AutocompleteField.test.ts +294 -0
  60. package/src/server/plugins/engine/components/AutocompleteField.ts +49 -0
  61. package/src/server/plugins/engine/components/CheckboxesField.test.ts +379 -0
  62. package/src/server/plugins/engine/components/CheckboxesField.ts +106 -0
  63. package/src/server/plugins/engine/components/ComponentBase.ts +97 -0
  64. package/src/server/plugins/engine/components/ComponentCollection.ts +278 -0
  65. package/src/server/plugins/engine/components/DatePartsField.test.ts +822 -0
  66. package/src/server/plugins/engine/components/DatePartsField.ts +264 -0
  67. package/src/server/plugins/engine/components/Details.test.ts +49 -0
  68. package/src/server/plugins/engine/components/Details.ts +30 -0
  69. package/src/server/plugins/engine/components/EmailAddressField.test.ts +395 -0
  70. package/src/server/plugins/engine/components/EmailAddressField.ts +55 -0
  71. package/src/server/plugins/engine/components/FileUploadField.test.ts +778 -0
  72. package/src/server/plugins/engine/components/FileUploadField.ts +262 -0
  73. package/src/server/plugins/engine/components/FormComponent.ts +249 -0
  74. package/src/server/plugins/engine/components/Html.test.ts +48 -0
  75. package/src/server/plugins/engine/components/Html.ts +29 -0
  76. package/src/server/plugins/engine/components/InsetText.test.ts +48 -0
  77. package/src/server/plugins/engine/components/InsetText.ts +27 -0
  78. package/src/server/plugins/engine/components/List.test.ts +76 -0
  79. package/src/server/plugins/engine/components/List.ts +72 -0
  80. package/src/server/plugins/engine/components/ListFormComponent.ts +140 -0
  81. package/src/server/plugins/engine/components/MonthYearField.test.ts +567 -0
  82. package/src/server/plugins/engine/components/MonthYearField.ts +222 -0
  83. package/src/server/plugins/engine/components/MultilineTextField.test.ts +558 -0
  84. package/src/server/plugins/engine/components/MultilineTextField.ts +138 -0
  85. package/src/server/plugins/engine/components/NumberField.test.ts +701 -0
  86. package/src/server/plugins/engine/components/NumberField.ts +163 -0
  87. package/src/server/plugins/engine/components/RadiosField.test.ts +288 -0
  88. package/src/server/plugins/engine/components/RadiosField.ts +24 -0
  89. package/src/server/plugins/engine/components/SelectField.test.ts +288 -0
  90. package/src/server/plugins/engine/components/SelectField.ts +47 -0
  91. package/src/server/plugins/engine/components/SelectionControlField.ts +43 -0
  92. package/src/server/plugins/engine/components/TelephoneNumberField.test.ts +356 -0
  93. package/src/server/plugins/engine/components/TelephoneNumberField.ts +67 -0
  94. package/src/server/plugins/engine/components/TextField.test.ts +489 -0
  95. package/src/server/plugins/engine/components/TextField.ts +96 -0
  96. package/src/server/plugins/engine/components/UkAddressField.test.ts +623 -0
  97. package/src/server/plugins/engine/components/UkAddressField.ts +172 -0
  98. package/src/server/plugins/engine/components/YesNoField.test.ts +248 -0
  99. package/src/server/plugins/engine/components/YesNoField.ts +31 -0
  100. package/src/server/plugins/engine/components/constants.ts +1 -0
  101. package/src/server/plugins/engine/components/helpers.ts +330 -0
  102. package/src/server/plugins/engine/components/index.ts +24 -0
  103. package/src/server/plugins/engine/components/types.ts +117 -0
  104. package/src/server/plugins/engine/configureEnginePlugin.ts +47 -0
  105. package/src/server/plugins/engine/helpers.test.ts +791 -0
  106. package/src/server/plugins/engine/helpers.ts +384 -0
  107. package/src/server/plugins/engine/index.ts +47 -0
  108. package/src/server/plugins/engine/models/FormModel.test.ts +42 -0
  109. package/src/server/plugins/engine/models/FormModel.ts +443 -0
  110. package/src/server/plugins/engine/models/RepeatingSummaryViewModel.ts +0 -0
  111. package/src/server/plugins/engine/models/Section.ts +0 -0
  112. package/src/server/plugins/engine/models/SummaryViewModel.test.ts +209 -0
  113. package/src/server/plugins/engine/models/SummaryViewModel.ts +220 -0
  114. package/src/server/plugins/engine/models/index.ts +2 -0
  115. package/src/server/plugins/engine/models/types.ts +114 -0
  116. package/src/server/plugins/engine/outputFormatters/human/v1.test.ts +143 -0
  117. package/src/server/plugins/engine/outputFormatters/human/v1.ts +73 -0
  118. package/src/server/plugins/engine/outputFormatters/index.test.ts +17 -0
  119. package/src/server/plugins/engine/outputFormatters/index.ts +44 -0
  120. package/src/server/plugins/engine/outputFormatters/machine/v1.test.ts +229 -0
  121. package/src/server/plugins/engine/outputFormatters/machine/v1.ts +140 -0
  122. package/src/server/plugins/engine/outputFormatters/machine/v2.test.ts +229 -0
  123. package/src/server/plugins/engine/outputFormatters/machine/v2.ts +153 -0
  124. package/src/server/plugins/engine/pageControllers/FileUploadPageController.test.ts +1116 -0
  125. package/src/server/plugins/engine/pageControllers/FileUploadPageController.ts +447 -0
  126. package/src/server/plugins/engine/pageControllers/PageController.test.ts +205 -0
  127. package/src/server/plugins/engine/pageControllers/PageController.ts +176 -0
  128. package/src/server/plugins/engine/pageControllers/QuestionPageController.test.ts +1264 -0
  129. package/src/server/plugins/engine/pageControllers/QuestionPageController.ts +565 -0
  130. package/src/server/plugins/engine/pageControllers/README.md +28 -0
  131. package/src/server/plugins/engine/pageControllers/RepeatPageController.test.ts +264 -0
  132. package/src/server/plugins/engine/pageControllers/RepeatPageController.ts +458 -0
  133. package/src/server/plugins/engine/pageControllers/StartPageController.ts +18 -0
  134. package/src/server/plugins/engine/pageControllers/StatusPageController.ts +51 -0
  135. package/src/server/plugins/engine/pageControllers/SummaryPageController.ts +262 -0
  136. package/src/server/plugins/engine/pageControllers/TerminalController.test.ts +28 -0
  137. package/src/server/plugins/engine/pageControllers/TerminalPageController.ts +19 -0
  138. package/src/server/plugins/engine/pageControllers/helpers.test.ts +198 -0
  139. package/src/server/plugins/engine/pageControllers/helpers.ts +101 -0
  140. package/src/server/plugins/engine/pageControllers/index.ts +10 -0
  141. package/src/server/plugins/engine/pageControllers/validationOptions.ts +89 -0
  142. package/src/server/plugins/engine/plugin.ts +753 -0
  143. package/src/server/plugins/engine/services/formSubmissionService.js +46 -0
  144. package/src/server/plugins/engine/services/formsService.js +46 -0
  145. package/src/server/plugins/engine/services/formsService.test.js +90 -0
  146. package/src/server/plugins/engine/services/index.js +3 -0
  147. package/src/server/plugins/engine/services/notifyService.test.ts +132 -0
  148. package/src/server/plugins/engine/services/notifyService.ts +64 -0
  149. package/src/server/plugins/engine/services/uploadService.js +60 -0
  150. package/src/server/plugins/engine/types.ts +317 -0
  151. package/src/server/plugins/engine/views/components/autocompletefield.html +5 -0
  152. package/src/server/plugins/engine/views/components/checkboxesfield.html +5 -0
  153. package/src/server/plugins/engine/views/components/datepartsfield.html +5 -0
  154. package/src/server/plugins/engine/views/components/debug/macro.njk +3 -0
  155. package/src/server/plugins/engine/views/components/debug/template.njk +13 -0
  156. package/src/server/plugins/engine/views/components/details.html +6 -0
  157. package/src/server/plugins/engine/views/components/emailaddressfield.html +5 -0
  158. package/src/server/plugins/engine/views/components/fileuploadfield-key.html +8 -0
  159. package/src/server/plugins/engine/views/components/fileuploadfield-value.html +3 -0
  160. package/src/server/plugins/engine/views/components/fileuploadfield.html +24 -0
  161. package/src/server/plugins/engine/views/components/html.html +3 -0
  162. package/src/server/plugins/engine/views/components/insettext.html +7 -0
  163. package/src/server/plugins/engine/views/components/list.html +36 -0
  164. package/src/server/plugins/engine/views/components/monthyearfield.html +5 -0
  165. package/src/server/plugins/engine/views/components/multilinetextfield.html +10 -0
  166. package/src/server/plugins/engine/views/components/numberfield.html +5 -0
  167. package/src/server/plugins/engine/views/components/radiosfield.html +5 -0
  168. package/src/server/plugins/engine/views/components/selectfield.html +5 -0
  169. package/src/server/plugins/engine/views/components/service-banner/macro.njk +3 -0
  170. package/src/server/plugins/engine/views/components/service-banner/template.njk +20 -0
  171. package/src/server/plugins/engine/views/components/service-banner/template.test.js +43 -0
  172. package/src/server/plugins/engine/views/components/tag-env/macro.njk +3 -0
  173. package/src/server/plugins/engine/views/components/tag-env/template.njk +30 -0
  174. package/src/server/plugins/engine/views/components/tag-env/template.test.js +66 -0
  175. package/src/server/plugins/engine/views/components/telephonenumberfield.html +5 -0
  176. package/src/server/plugins/engine/views/components/textfield.html +5 -0
  177. package/src/server/plugins/engine/views/components/ukaddressfield.html +25 -0
  178. package/src/server/plugins/engine/views/components/yesnofield.html +5 -0
  179. package/src/server/plugins/engine/views/confirmation.html +19 -0
  180. package/src/server/plugins/engine/views/file-upload.html +45 -0
  181. package/src/server/plugins/engine/views/index.html +39 -0
  182. package/src/server/plugins/engine/views/item-delete.html +56 -0
  183. package/src/server/plugins/engine/views/layout.html +199 -0
  184. package/src/server/plugins/engine/views/partials/components.html +6 -0
  185. package/src/server/plugins/engine/views/partials/conditional-components.html +3 -0
  186. package/src/server/plugins/engine/views/partials/debug.html +44 -0
  187. package/src/server/plugins/engine/views/partials/form.html +15 -0
  188. package/src/server/plugins/engine/views/partials/heading.html +16 -0
  189. package/src/server/plugins/engine/views/partials/preview-banner.html +32 -0
  190. package/src/server/plugins/engine/views/partials/preview-banner.test.js +122 -0
  191. package/src/server/plugins/engine/views/partials/warn-missing-notification-email.html +10 -0
  192. package/src/server/plugins/engine/views/repeat-list-summary.html +53 -0
  193. package/src/server/plugins/engine/views/summary.html +50 -0
  194. package/src/server/plugins/errorPages.ts +58 -0
  195. package/src/server/plugins/nunjucks/context.js +88 -0
  196. package/src/server/plugins/nunjucks/context.test.js +142 -0
  197. package/src/server/plugins/nunjucks/enviroment.test.js +201 -0
  198. package/src/server/plugins/nunjucks/environment.js +116 -0
  199. package/src/server/plugins/nunjucks/filters/answer.js +27 -0
  200. package/src/server/plugins/nunjucks/filters/answer.test.js +89 -0
  201. package/src/server/plugins/nunjucks/filters/evaluate.js +21 -0
  202. package/src/server/plugins/nunjucks/filters/field.js +28 -0
  203. package/src/server/plugins/nunjucks/filters/field.test.js +75 -0
  204. package/src/server/plugins/nunjucks/filters/highlight.js +11 -0
  205. package/src/server/plugins/nunjucks/filters/href.js +30 -0
  206. package/src/server/plugins/nunjucks/filters/href.test.js +80 -0
  207. package/src/server/plugins/nunjucks/filters/index.js +8 -0
  208. package/src/server/plugins/nunjucks/filters/inspect.js +15 -0
  209. package/src/server/plugins/nunjucks/filters/page.js +24 -0
  210. package/src/server/plugins/nunjucks/filters/page.test.js +65 -0
  211. package/src/server/plugins/nunjucks/index.js +3 -0
  212. package/src/server/plugins/nunjucks/plugin.js +40 -0
  213. package/src/server/plugins/nunjucks/render.js +42 -0
  214. package/src/server/plugins/nunjucks/types.js +40 -0
  215. package/src/server/plugins/pulse.ts +11 -0
  216. package/src/server/plugins/router.ts +201 -0
  217. package/src/server/plugins/session.ts +28 -0
  218. package/src/server/routes/health.js +13 -0
  219. package/src/server/routes/health.test.js +35 -0
  220. package/src/server/routes/index.test.ts +125 -0
  221. package/src/server/routes/index.ts +2 -0
  222. package/src/server/routes/public.ts +47 -0
  223. package/src/server/routes/types.ts +48 -0
  224. package/src/server/schemas/index.ts +34 -0
  225. package/src/server/secure-context.js +43 -0
  226. package/src/server/services/cacheService.test.ts +277 -0
  227. package/src/server/services/cacheService.ts +138 -0
  228. package/src/server/services/httpService.test.js +491 -0
  229. package/src/server/services/httpService.ts +50 -0
  230. package/src/server/services/index.ts +1 -0
  231. package/src/server/types.ts +54 -0
  232. package/src/server/utils/notify.test.ts +37 -0
  233. package/src/server/utils/notify.ts +50 -0
  234. package/src/server/utils/secure-context/get-trust-store-certs.js +11 -0
  235. package/src/server/utils/secure-context/get-trust-store-certs.test.js +19 -0
  236. package/src/server/utils/utils.js +24 -0
  237. package/src/server/utils/utils.test.js +54 -0
  238. package/src/server/views/404.html +16 -0
  239. package/src/server/views/500.html +19 -0
  240. package/src/server/views/help/accessibility-statement.html +58 -0
  241. package/src/server/views/help/cookie-preferences.html +57 -0
  242. package/src/server/views/help/cookies.html +71 -0
  243. package/src/server/views/help/get-support.html +37 -0
  244. package/src/server/views/help/privacy-notice.html +68 -0
  245. package/src/server/views/help/terms-and-conditions.html +83 -0
  246. package/src/typings/hapi/index.d.ts +87 -0
  247. package/src/typings/hapi-tracing/index.d.ts +6 -0
  248. package/src/typings/index.d.ts +3 -0
  249. package/src/typings/joi/index.d.ts +22 -0
  250. package/.server/server/views/components/service-banner/template.test.js.map +0 -1
  251. package/.server/server/views/components/tag-env/template.test.js.map +0 -1
  252. /package/.server/server/{views → plugins/engine/views}/components/debug/macro.njk +0 -0
  253. /package/.server/server/{views → plugins/engine/views}/components/debug/template.njk +0 -0
  254. /package/.server/server/{views → plugins/engine/views}/components/service-banner/macro.njk +0 -0
  255. /package/.server/server/{views → plugins/engine/views}/components/service-banner/template.njk +0 -0
  256. /package/.server/server/{views → plugins/engine/views}/components/tag-env/macro.njk +0 -0
  257. /package/.server/server/{views → plugins/engine/views}/components/tag-env/template.njk +0 -0
  258. /package/.server/server/{views → plugins/engine/views}/confirmation.html +0 -0
  259. /package/.server/server/{views → plugins/engine/views}/layout.html +0 -0
  260. /package/.server/server/{views → plugins/engine/views}/summary.html +0 -0
@@ -0,0 +1,822 @@
1
+ import { ComponentType, type DatePartsFieldComponent } from '@defra/forms-model'
2
+ import { addDays, format, startOfDay } from 'date-fns'
3
+
4
+ import { ComponentCollection } from '~/src/server/plugins/engine/components/ComponentCollection.js'
5
+ import {
6
+ getAnswer,
7
+ type Field
8
+ } from '~/src/server/plugins/engine/components/helpers.js'
9
+ import { type DateInputItem } from '~/src/server/plugins/engine/components/types.js'
10
+ import { FormModel } from '~/src/server/plugins/engine/models/FormModel.js'
11
+ import {
12
+ type FormPayload,
13
+ type FormState
14
+ } from '~/src/server/plugins/engine/types.js'
15
+ import definition from '~/test/form/definitions/blank.js'
16
+
17
+ describe('DatePartsField', () => {
18
+ let model: FormModel
19
+
20
+ beforeEach(() => {
21
+ model = new FormModel(definition, {
22
+ basePath: 'test'
23
+ })
24
+ })
25
+
26
+ describe('Defaults', () => {
27
+ let def: DatePartsFieldComponent
28
+ let collection: ComponentCollection
29
+ let field: Field
30
+
31
+ beforeEach(() => {
32
+ def = {
33
+ title: 'Example date parts field',
34
+ name: 'myComponent',
35
+ type: ComponentType.DatePartsField,
36
+ options: {}
37
+ } satisfies DatePartsFieldComponent
38
+
39
+ collection = new ComponentCollection([def], { model })
40
+ field = collection.fields[0]
41
+ })
42
+
43
+ describe('Schema', () => {
44
+ it('uses collection titles as labels', () => {
45
+ const { formSchema } = collection
46
+ const { keys } = formSchema.describe()
47
+
48
+ expect(keys).toHaveProperty(
49
+ 'myComponent__day',
50
+ expect.objectContaining({
51
+ flags: expect.objectContaining({ label: 'Day' })
52
+ })
53
+ )
54
+
55
+ expect(keys).toHaveProperty(
56
+ 'myComponent__month',
57
+ expect.objectContaining({
58
+ flags: expect.objectContaining({ label: 'Month' })
59
+ })
60
+ )
61
+
62
+ expect(keys).toHaveProperty(
63
+ 'myComponent__year',
64
+ expect.objectContaining({
65
+ flags: expect.objectContaining({ label: 'Year' })
66
+ })
67
+ )
68
+ })
69
+
70
+ it('uses collection names as keys', () => {
71
+ const { formSchema } = collection
72
+ const { keys } = formSchema.describe()
73
+
74
+ expect(field.keys).toEqual([
75
+ 'myComponent',
76
+ 'myComponent__day',
77
+ 'myComponent__month',
78
+ 'myComponent__year'
79
+ ])
80
+
81
+ expect(field.collection?.keys).not.toHaveProperty('myComponent')
82
+
83
+ for (const key of field.collection?.keys ?? []) {
84
+ expect(keys).toHaveProperty(key)
85
+ }
86
+ })
87
+
88
+ it('is required by default', () => {
89
+ const { formSchema } = collection
90
+ const { keys } = formSchema.describe()
91
+
92
+ expect(keys).toHaveProperty(
93
+ 'myComponent__day',
94
+ expect.objectContaining({
95
+ flags: expect.objectContaining({ presence: 'required' })
96
+ })
97
+ )
98
+
99
+ expect(keys).toHaveProperty(
100
+ 'myComponent__month',
101
+ expect.objectContaining({
102
+ flags: expect.objectContaining({ presence: 'required' })
103
+ })
104
+ )
105
+
106
+ expect(keys).toHaveProperty(
107
+ 'myComponent__year',
108
+ expect.objectContaining({
109
+ flags: expect.objectContaining({ presence: 'required' })
110
+ })
111
+ )
112
+ })
113
+
114
+ it('is optional when configured', () => {
115
+ const collectionOptional = new ComponentCollection(
116
+ [
117
+ {
118
+ title: 'Example date parts field',
119
+ name: 'myComponent',
120
+ type: ComponentType.DatePartsField,
121
+ options: { required: false }
122
+ }
123
+ ],
124
+ { model }
125
+ )
126
+
127
+ const { formSchema } = collectionOptional
128
+ const { keys } = formSchema.describe()
129
+
130
+ expect(keys).toHaveProperty(
131
+ 'myComponent__day',
132
+ expect.objectContaining({ allow: [''] })
133
+ )
134
+
135
+ expect(keys).toHaveProperty(
136
+ 'myComponent__month',
137
+ expect.objectContaining({ allow: [''] })
138
+ )
139
+
140
+ expect(keys).toHaveProperty(
141
+ 'myComponent__year',
142
+ expect.objectContaining({ allow: [''] })
143
+ )
144
+
145
+ // Empty optional payload (valid)
146
+ const result1 = collectionOptional.validate(
147
+ getFormData({
148
+ day: '',
149
+ month: '',
150
+ year: ''
151
+ })
152
+ )
153
+
154
+ // Partial optional payload (invalid)
155
+ const result2 = collectionOptional.validate(
156
+ getFormData({
157
+ day: '31',
158
+ month: '',
159
+ year: ''
160
+ })
161
+ )
162
+
163
+ expect(result1.errors).toBeUndefined()
164
+ expect(result2.errors).toEqual([
165
+ expect.objectContaining({
166
+ text: 'Example date parts field must include a month'
167
+ })
168
+ ])
169
+ })
170
+
171
+ it('accepts valid values', () => {
172
+ const result1 = collection.validate(
173
+ getFormData({
174
+ day: '31',
175
+ month: '12',
176
+ year: '2024'
177
+ })
178
+ )
179
+
180
+ const result2 = collection.validate(
181
+ getFormData({
182
+ day: '1',
183
+ month: '2',
184
+ year: '2024'
185
+ })
186
+ )
187
+
188
+ // Leap year in 2024
189
+ const result3 = collection.validate(
190
+ getFormData({
191
+ day: '29',
192
+ month: '2',
193
+ year: '2024'
194
+ })
195
+ )
196
+
197
+ expect(result1.errors).toBeUndefined()
198
+ expect(result2.errors).toBeUndefined()
199
+ expect(result3.errors).toBeUndefined()
200
+ })
201
+
202
+ it('adds errors for empty value', () => {
203
+ const result = collection.validate(
204
+ getFormData({
205
+ day: '',
206
+ month: '',
207
+ year: ''
208
+ })
209
+ )
210
+
211
+ expect(result.errors).toEqual([
212
+ expect.objectContaining({
213
+ text: 'Example date parts field must include a day'
214
+ }),
215
+ expect.objectContaining({
216
+ text: 'Example date parts field must include a month'
217
+ }),
218
+ expect.objectContaining({
219
+ text: 'Example date parts field must include a year'
220
+ })
221
+ ])
222
+ })
223
+
224
+ it('adds errors for invalid values', () => {
225
+ const result1 = collection.validate(getFormData({ unknown: 'invalid' }))
226
+
227
+ const result2 = collection.validate(
228
+ getFormData({
229
+ day: ['invalid'],
230
+ month: ['invalid'],
231
+ year: ['invalid']
232
+ })
233
+ )
234
+
235
+ const result3 = collection.validate(
236
+ getFormData({
237
+ day: 'invalid',
238
+ month: 'invalid',
239
+ year: 'invalid'
240
+ })
241
+ )
242
+
243
+ expect(result1.errors).toBeTruthy()
244
+ expect(result2.errors).toBeTruthy()
245
+ expect(result3.errors).toBeTruthy()
246
+ })
247
+ })
248
+
249
+ describe('State', () => {
250
+ const date = new Date('2024-12-31')
251
+
252
+ it('returns text from state', () => {
253
+ const state1 = getFormState(date)
254
+ const state2 = getFormState({})
255
+
256
+ const answer1 = getAnswer(field, state1)
257
+ const answer2 = getAnswer(field, state2)
258
+
259
+ expect(answer1).toBe('31 December 2024')
260
+ expect(answer2).toBe('')
261
+ })
262
+
263
+ it('returns payload from state', () => {
264
+ const state1 = getFormState(startOfDay(date))
265
+ const state2 = getFormState({})
266
+
267
+ const payload1 = field.getFormDataFromState(state1)
268
+ const payload2 = field.getFormDataFromState(state2)
269
+
270
+ expect(payload1).toEqual(getFormData(date))
271
+ expect(payload2).toEqual(getFormData({}))
272
+ })
273
+
274
+ it('returns value from state', () => {
275
+ const state1 = getFormState(startOfDay(date))
276
+ const state2 = getFormState({})
277
+
278
+ const value1 = field.getFormValueFromState(state1)
279
+ const value2 = field.getFormValueFromState(state2)
280
+
281
+ expect(value1).toEqual({
282
+ day: 31,
283
+ month: 12,
284
+ year: 2024
285
+ })
286
+
287
+ expect(value2).toBeUndefined()
288
+ })
289
+
290
+ it('returns context for conditions and form submission', () => {
291
+ const state1 = getFormState(startOfDay(date))
292
+ const state2 = getFormState({})
293
+
294
+ const value1 = field.getContextValueFromState(state1)
295
+ const value2 = field.getContextValueFromState(state2)
296
+
297
+ expect(value1).toBe('2024-12-31')
298
+ expect(value2).toBeNull()
299
+ })
300
+
301
+ it('returns null context when date is invalid', () => {
302
+ const state1 = getFormState({ day: 1, month: 0, year: 2025 })
303
+ const state2 = getFormState({})
304
+ const state3 = getFormState({ day: 1, month: 13, year: 2025 })
305
+ const state4 = getFormState({ day: 32, month: 12, year: 2025 })
306
+
307
+ const value1 = field.getContextValueFromState(state1)
308
+ const value2 = field.getContextValueFromState(state2)
309
+ const value3 = field.getContextValueFromState(state3)
310
+ const value4 = field.getContextValueFromState(state4)
311
+
312
+ expect(value1).toBeNull()
313
+ expect(value2).toBeNull()
314
+ expect(value3).toBeNull()
315
+ expect(value4).toBeNull()
316
+ })
317
+
318
+ it('returns state from payload', () => {
319
+ const payload1 = getFormData(date)
320
+ const payload2 = getFormData({})
321
+
322
+ const value1 = field.getStateFromValidForm(payload1)
323
+ const value2 = field.getStateFromValidForm(payload2)
324
+
325
+ expect(value1).toEqual(getFormState(date))
326
+ expect(value2).toEqual(getFormState({}))
327
+ })
328
+ })
329
+
330
+ describe('View model', () => {
331
+ const date = new Date('2024-12-31')
332
+
333
+ it('sets Nunjucks component defaults', () => {
334
+ const payload = getFormData(date)
335
+ const viewModel = field.getViewModel(payload)
336
+
337
+ expect(viewModel).toEqual(
338
+ expect.objectContaining({
339
+ label: { text: def.title },
340
+ name: 'myComponent',
341
+ id: 'myComponent',
342
+ value: undefined,
343
+ items: [
344
+ expect.objectContaining(
345
+ getViewModel(date, 'day', {
346
+ label: { text: 'Day' },
347
+ classes: 'govuk-input--width-2',
348
+ value: 31
349
+ })
350
+ ),
351
+
352
+ expect.objectContaining(
353
+ getViewModel(date, 'month', {
354
+ label: { text: 'Month' },
355
+ classes: 'govuk-input--width-2',
356
+ value: 12
357
+ })
358
+ ),
359
+
360
+ expect.objectContaining(
361
+ getViewModel(date, 'year', {
362
+ label: { text: 'Year' },
363
+ classes: 'govuk-input--width-4',
364
+ value: 2024
365
+ })
366
+ )
367
+ ]
368
+ })
369
+ )
370
+ })
371
+
372
+ it('sets Nunjucks component value when invalid', () => {
373
+ const payload = getFormData({
374
+ day: 'DD',
375
+ month: 'MM',
376
+ year: 'YYYY'
377
+ })
378
+
379
+ const viewModel = field.getViewModel(payload)
380
+
381
+ expect(viewModel).toEqual(
382
+ expect.objectContaining({
383
+ items: [
384
+ expect.objectContaining(
385
+ getViewModel(date, 'day', { value: 'DD' })
386
+ ),
387
+
388
+ expect.objectContaining(
389
+ getViewModel(date, 'month', { value: 'MM' })
390
+ ),
391
+
392
+ expect.objectContaining(
393
+ getViewModel(date, 'year', { value: 'YYYY' })
394
+ )
395
+ ]
396
+ })
397
+ )
398
+ })
399
+
400
+ it('sets Nunjucks component fieldset', () => {
401
+ const payload = getFormData(date)
402
+ const viewModel = field.getViewModel(payload)
403
+
404
+ expect(viewModel.fieldset).toEqual({
405
+ legend: {
406
+ text: def.title,
407
+ classes: 'govuk-fieldset__legend--m'
408
+ }
409
+ })
410
+ })
411
+ })
412
+ })
413
+
414
+ describe('Validation', () => {
415
+ const today = startOfDay(new Date())
416
+ const date = new Date('2001-01-01')
417
+
418
+ const OneDayInPast = addDays(today, -1)
419
+ const TwoDaysInPast = addDays(today, -2)
420
+ const OneDayInFuture = addDays(today, 1)
421
+ const TwoDaysInFuture = addDays(today, 2)
422
+
423
+ describe.each([
424
+ {
425
+ description: 'Trim empty spaces',
426
+ component: {
427
+ title: 'Example date parts field',
428
+ name: 'myComponent',
429
+ type: ComponentType.DatePartsField,
430
+ options: {}
431
+ } satisfies DatePartsFieldComponent,
432
+ assertions: [
433
+ {
434
+ input: getFormData({
435
+ day: ' 01',
436
+ month: ' 01',
437
+ year: ' 2001'
438
+ }),
439
+ output: {
440
+ value: getFormData(date)
441
+ }
442
+ },
443
+ {
444
+ input: getFormData({
445
+ day: '01 ',
446
+ month: '01 ',
447
+ year: '2001 '
448
+ }),
449
+ output: {
450
+ value: getFormData(date)
451
+ }
452
+ },
453
+ {
454
+ input: getFormData({
455
+ day: ' 01 \n\n',
456
+ month: ' 01 \n\n',
457
+ year: ' 2001 \n\n'
458
+ }),
459
+ output: {
460
+ value: getFormData(date)
461
+ }
462
+ }
463
+ ]
464
+ },
465
+ {
466
+ description: 'Decimals',
467
+ component: {
468
+ title: 'Example date parts field',
469
+ name: 'myComponent',
470
+ type: ComponentType.DatePartsField,
471
+ options: {}
472
+ } satisfies DatePartsFieldComponent,
473
+ assertions: [
474
+ {
475
+ input: getFormData({
476
+ day: '1.1',
477
+ month: '1.2',
478
+ year: '2001.3'
479
+ }),
480
+ output: {
481
+ value: getFormData({
482
+ day: 1.1,
483
+ month: 1.2,
484
+ year: 2001.3
485
+ }),
486
+ errors: [
487
+ expect.objectContaining({
488
+ text: 'Example date parts field must be a real date'
489
+ }),
490
+ expect.objectContaining({
491
+ text: 'Example date parts field must be a real date'
492
+ }),
493
+ expect.objectContaining({
494
+ text: 'Example date parts field must be a real date'
495
+ })
496
+ ]
497
+ }
498
+ }
499
+ ]
500
+ },
501
+ {
502
+ description: 'Leap years',
503
+ component: {
504
+ title: 'Example date parts field',
505
+ name: 'myComponent',
506
+ type: ComponentType.DatePartsField,
507
+ options: {}
508
+ } satisfies DatePartsFieldComponent,
509
+ assertions: [
510
+ {
511
+ // Leap year in 2024
512
+ input: getFormData({
513
+ day: '29',
514
+ month: '2',
515
+ year: '2024'
516
+ }),
517
+ output: {
518
+ value: getFormData({
519
+ day: 29,
520
+ month: 2,
521
+ year: 2024
522
+ })
523
+ }
524
+ },
525
+ {
526
+ // Not a leap year in 2023
527
+ input: getFormData({
528
+ day: '29',
529
+ month: '2',
530
+ year: '2023'
531
+ }),
532
+ output: {
533
+ value: getFormData({
534
+ day: 29,
535
+ month: 2,
536
+ year: 2023
537
+ }),
538
+ errors: [
539
+ expect.objectContaining({
540
+ text: 'Example date parts field must be a real date'
541
+ })
542
+ ]
543
+ }
544
+ }
545
+ ]
546
+ },
547
+ {
548
+ description: 'Out of range values',
549
+ component: {
550
+ title: 'Example date parts field',
551
+ name: 'myComponent',
552
+ type: ComponentType.DatePartsField,
553
+ options: {}
554
+ } satisfies DatePartsFieldComponent,
555
+ assertions: [
556
+ {
557
+ input: getFormData({
558
+ day: '32',
559
+ month: '1',
560
+ year: '2024'
561
+ }),
562
+ output: {
563
+ value: getFormData({
564
+ day: 32,
565
+ month: 1,
566
+ year: 2024
567
+ }),
568
+ errors: [
569
+ expect.objectContaining({
570
+ text: 'Example date parts field must be a real date'
571
+ })
572
+ ]
573
+ }
574
+ },
575
+ {
576
+ input: getFormData({
577
+ day: '1',
578
+ month: '13',
579
+ year: '2024'
580
+ }),
581
+ output: {
582
+ value: getFormData({
583
+ day: 1,
584
+ month: 13,
585
+ year: 2024
586
+ }),
587
+ errors: [
588
+ expect.objectContaining({
589
+ text: 'Example date parts field must be a real date'
590
+ })
591
+ ]
592
+ }
593
+ },
594
+ {
595
+ input: getFormData({
596
+ day: '1',
597
+ month: '1',
598
+ year: '999'
599
+ }),
600
+ output: {
601
+ value: getFormData({
602
+ day: 1,
603
+ month: 1,
604
+ year: 999
605
+ }),
606
+ errors: [
607
+ expect.objectContaining({
608
+ text: 'Example date parts field must be a real date'
609
+ })
610
+ ]
611
+ }
612
+ }
613
+ ]
614
+ },
615
+ {
616
+ description: 'Impossible dates',
617
+ component: {
618
+ title: 'Example date parts field',
619
+ name: 'myComponent',
620
+ type: ComponentType.DatePartsField,
621
+ options: {}
622
+ } satisfies DatePartsFieldComponent,
623
+ assertions: [
624
+ {
625
+ input: getFormData({
626
+ day: '31',
627
+ month: '4',
628
+ year: '2024'
629
+ }),
630
+ output: {
631
+ value: getFormData({
632
+ day: 31,
633
+ month: 4,
634
+ year: 2024
635
+ }),
636
+ errors: [
637
+ expect.objectContaining({
638
+ text: 'Example date parts field must be a real date'
639
+ })
640
+ ]
641
+ }
642
+ },
643
+ {
644
+ input: getFormData({
645
+ day: '31',
646
+ month: '6',
647
+ year: '2024'
648
+ }),
649
+ output: {
650
+ value: getFormData({
651
+ day: 31,
652
+ month: 6,
653
+ year: 2024
654
+ }),
655
+ errors: [
656
+ expect.objectContaining({
657
+ text: 'Example date parts field must be a real date'
658
+ })
659
+ ]
660
+ }
661
+ }
662
+ ]
663
+ },
664
+ {
665
+ description: 'Max days in the past option',
666
+ component: {
667
+ title: 'Example date parts field',
668
+ name: 'myComponent',
669
+ type: ComponentType.DatePartsField,
670
+ options: {
671
+ maxDaysInPast: 1
672
+ }
673
+ } satisfies DatePartsFieldComponent,
674
+ assertions: [
675
+ {
676
+ input: getFormData(TwoDaysInPast),
677
+ output: {
678
+ value: getFormData(TwoDaysInPast),
679
+ errors: [
680
+ expect.objectContaining({
681
+ text: `Example date parts field must be the same as or after ${format(OneDayInPast, 'd MMMM yyyy')}`
682
+ })
683
+ ]
684
+ }
685
+ },
686
+ {
687
+ input: getFormData(today),
688
+ output: { value: getFormData(today) }
689
+ }
690
+ ]
691
+ },
692
+ {
693
+ description: 'Max days in the future option',
694
+ component: {
695
+ title: 'Example date parts field',
696
+ name: 'myComponent',
697
+ type: ComponentType.DatePartsField,
698
+ options: {
699
+ maxDaysInFuture: 1
700
+ }
701
+ } satisfies DatePartsFieldComponent,
702
+ assertions: [
703
+ {
704
+ input: getFormData(TwoDaysInFuture),
705
+ output: {
706
+ value: getFormData(TwoDaysInFuture),
707
+ errors: [
708
+ expect.objectContaining({
709
+ text: `Example date parts field must be the same as or before ${format(OneDayInFuture, 'd MMMM yyyy')}`
710
+ })
711
+ ]
712
+ }
713
+ },
714
+ {
715
+ input: getFormData(today),
716
+ output: { value: getFormData(today) }
717
+ }
718
+ ]
719
+ },
720
+ {
721
+ description: 'Optional fields',
722
+ component: {
723
+ title: 'Example date parts field',
724
+ name: 'myComponent',
725
+ type: ComponentType.DatePartsField,
726
+ options: {
727
+ required: false
728
+ }
729
+ } satisfies DatePartsFieldComponent,
730
+ assertions: [
731
+ {
732
+ input: getFormData({
733
+ day: '',
734
+ month: '',
735
+ year: ''
736
+ }),
737
+ output: {
738
+ value: getFormData({
739
+ day: '',
740
+ month: '',
741
+ year: ''
742
+ })
743
+ }
744
+ }
745
+ ]
746
+ }
747
+ ])('$description', ({ component: def, assertions }) => {
748
+ let collection: ComponentCollection
749
+
750
+ beforeEach(() => {
751
+ collection = new ComponentCollection([def], { model })
752
+ })
753
+
754
+ it.each([...assertions])(
755
+ 'validates custom example',
756
+ ({ input, output }) => {
757
+ const result = collection.validate(input)
758
+ expect(result).toEqual(output)
759
+ }
760
+ )
761
+ })
762
+ })
763
+ })
764
+
765
+ /**
766
+ * Date field view model
767
+ */
768
+ function getViewModel(
769
+ date: Date,
770
+ name: string,
771
+ overrides?: Partial<DateInputItem>
772
+ ): DateInputItem {
773
+ const payload = getFormData(date)
774
+
775
+ const fieldName = `myComponent__${name}`
776
+ const fieldValue = overrides?.value ?? payload[fieldName]
777
+ const fieldClasses = overrides?.classes ?? expect.any(String)
778
+
779
+ return {
780
+ label: expect.objectContaining(
781
+ overrides?.label ?? {
782
+ text: expect.any(String)
783
+ }
784
+ ),
785
+ name: fieldName,
786
+ id: fieldName,
787
+ value: fieldValue as DateInputItem['value'],
788
+ classes: fieldClasses
789
+ }
790
+ }
791
+
792
+ /**
793
+ * Date form data
794
+ */
795
+ function getFormData(date: Date | FormPayload): FormPayload {
796
+ if (date instanceof Date) {
797
+ date = {
798
+ day: date.getDate(),
799
+ month: date.getMonth() + 1,
800
+ year: date.getFullYear()
801
+ }
802
+ }
803
+
804
+ return {
805
+ myComponent__day: date.day,
806
+ myComponent__month: date.month,
807
+ myComponent__year: date.year
808
+ }
809
+ }
810
+
811
+ /**
812
+ * Date session state
813
+ */
814
+ function getFormState(date: Date | FormPayload): FormState {
815
+ const [day, month, year] = Object.values(getFormData(date))
816
+
817
+ return {
818
+ myComponent__day: day ?? null,
819
+ myComponent__month: month ?? null,
820
+ myComponent__year: year ?? null
821
+ }
822
+ }