@defra/forms-engine-plugin 0.0.4 → 0.0.5

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 (224) hide show
  1. package/package.json +3 -2
  2. package/src/client/javascripts/application.js +87 -0
  3. package/src/client/javascripts/file-upload.js +386 -0
  4. package/src/client/stylesheets/_code.scss +33 -0
  5. package/src/client/stylesheets/_govuk-frontend.scss +4 -0
  6. package/src/client/stylesheets/_prose.scss +56 -0
  7. package/src/client/stylesheets/_service-banner.scss +24 -0
  8. package/src/client/stylesheets/_summary-list.scss +28 -0
  9. package/src/client/stylesheets/_tag-env.scss +24 -0
  10. package/src/client/stylesheets/application.scss +14 -0
  11. package/src/common/cookies.js +58 -0
  12. package/src/common/cookies.test.js +23 -0
  13. package/src/common/types.js +5 -0
  14. package/src/config/index.ts +271 -0
  15. package/src/index.ts +31 -0
  16. package/src/server/common/helpers/logging/logger-options.test.ts +50 -0
  17. package/src/server/common/helpers/logging/logger-options.ts +46 -0
  18. package/src/server/common/helpers/logging/logger.ts +7 -0
  19. package/src/server/common/helpers/logging/request-logger.ts +9 -0
  20. package/src/server/common/helpers/logging/request-tracing.js +10 -0
  21. package/src/server/common/helpers/redis-client.js +70 -0
  22. package/src/server/constants.js +1 -0
  23. package/src/server/forms/README.md +10 -0
  24. package/src/server/forms/components.json +1015 -0
  25. package/src/server/forms/report-a-terrorist.json +270 -0
  26. package/src/server/forms/runner-components-test.json +365 -0
  27. package/src/server/forms/test.json +581 -0
  28. package/src/server/index.test.ts +582 -0
  29. package/src/server/index.ts +140 -0
  30. package/src/server/plugins/blankie.test.ts +73 -0
  31. package/src/server/plugins/blankie.ts +48 -0
  32. package/src/server/plugins/crumb.ts +20 -0
  33. package/src/server/plugins/engine/README.md +87 -0
  34. package/src/server/plugins/engine/components/AutocompleteField.test.ts +294 -0
  35. package/src/server/plugins/engine/components/AutocompleteField.ts +49 -0
  36. package/src/server/plugins/engine/components/CheckboxesField.test.ts +379 -0
  37. package/src/server/plugins/engine/components/CheckboxesField.ts +106 -0
  38. package/src/server/plugins/engine/components/ComponentBase.ts +97 -0
  39. package/src/server/plugins/engine/components/ComponentCollection.ts +278 -0
  40. package/src/server/plugins/engine/components/DatePartsField.test.ts +822 -0
  41. package/src/server/plugins/engine/components/DatePartsField.ts +264 -0
  42. package/src/server/plugins/engine/components/Details.test.ts +49 -0
  43. package/src/server/plugins/engine/components/Details.ts +30 -0
  44. package/src/server/plugins/engine/components/EmailAddressField.test.ts +395 -0
  45. package/src/server/plugins/engine/components/EmailAddressField.ts +55 -0
  46. package/src/server/plugins/engine/components/FileUploadField.test.ts +778 -0
  47. package/src/server/plugins/engine/components/FileUploadField.ts +262 -0
  48. package/src/server/plugins/engine/components/FormComponent.ts +249 -0
  49. package/src/server/plugins/engine/components/Html.test.ts +48 -0
  50. package/src/server/plugins/engine/components/Html.ts +29 -0
  51. package/src/server/plugins/engine/components/InsetText.test.ts +48 -0
  52. package/src/server/plugins/engine/components/InsetText.ts +27 -0
  53. package/src/server/plugins/engine/components/List.test.ts +76 -0
  54. package/src/server/plugins/engine/components/List.ts +72 -0
  55. package/src/server/plugins/engine/components/ListFormComponent.ts +140 -0
  56. package/src/server/plugins/engine/components/MonthYearField.test.ts +567 -0
  57. package/src/server/plugins/engine/components/MonthYearField.ts +222 -0
  58. package/src/server/plugins/engine/components/MultilineTextField.test.ts +558 -0
  59. package/src/server/plugins/engine/components/MultilineTextField.ts +138 -0
  60. package/src/server/plugins/engine/components/NumberField.test.ts +701 -0
  61. package/src/server/plugins/engine/components/NumberField.ts +163 -0
  62. package/src/server/plugins/engine/components/RadiosField.test.ts +288 -0
  63. package/src/server/plugins/engine/components/RadiosField.ts +24 -0
  64. package/src/server/plugins/engine/components/SelectField.test.ts +288 -0
  65. package/src/server/plugins/engine/components/SelectField.ts +47 -0
  66. package/src/server/plugins/engine/components/SelectionControlField.ts +43 -0
  67. package/src/server/plugins/engine/components/TelephoneNumberField.test.ts +356 -0
  68. package/src/server/plugins/engine/components/TelephoneNumberField.ts +67 -0
  69. package/src/server/plugins/engine/components/TextField.test.ts +489 -0
  70. package/src/server/plugins/engine/components/TextField.ts +96 -0
  71. package/src/server/plugins/engine/components/UkAddressField.test.ts +623 -0
  72. package/src/server/plugins/engine/components/UkAddressField.ts +172 -0
  73. package/src/server/plugins/engine/components/YesNoField.test.ts +248 -0
  74. package/src/server/plugins/engine/components/YesNoField.ts +31 -0
  75. package/src/server/plugins/engine/components/constants.ts +1 -0
  76. package/src/server/plugins/engine/components/helpers.ts +330 -0
  77. package/src/server/plugins/engine/components/index.ts +24 -0
  78. package/src/server/plugins/engine/components/types.ts +117 -0
  79. package/src/server/plugins/engine/configureEnginePlugin.ts +47 -0
  80. package/src/server/plugins/engine/helpers.test.ts +791 -0
  81. package/src/server/plugins/engine/helpers.ts +379 -0
  82. package/src/server/plugins/engine/index.ts +7 -0
  83. package/src/server/plugins/engine/models/FormModel.test.ts +42 -0
  84. package/src/server/plugins/engine/models/FormModel.ts +443 -0
  85. package/src/server/plugins/engine/models/RepeatingSummaryViewModel.ts +0 -0
  86. package/src/server/plugins/engine/models/Section.ts +0 -0
  87. package/src/server/plugins/engine/models/SummaryViewModel.test.ts +209 -0
  88. package/src/server/plugins/engine/models/SummaryViewModel.ts +220 -0
  89. package/src/server/plugins/engine/models/index.ts +2 -0
  90. package/src/server/plugins/engine/models/types.ts +114 -0
  91. package/src/server/plugins/engine/outputFormatters/human/v1.test.ts +143 -0
  92. package/src/server/plugins/engine/outputFormatters/human/v1.ts +73 -0
  93. package/src/server/plugins/engine/outputFormatters/index.test.ts +17 -0
  94. package/src/server/plugins/engine/outputFormatters/index.ts +44 -0
  95. package/src/server/plugins/engine/outputFormatters/machine/v1.test.ts +229 -0
  96. package/src/server/plugins/engine/outputFormatters/machine/v1.ts +140 -0
  97. package/src/server/plugins/engine/outputFormatters/machine/v2.test.ts +229 -0
  98. package/src/server/plugins/engine/outputFormatters/machine/v2.ts +153 -0
  99. package/src/server/plugins/engine/pageControllers/FileUploadPageController.test.ts +1108 -0
  100. package/src/server/plugins/engine/pageControllers/FileUploadPageController.ts +446 -0
  101. package/src/server/plugins/engine/pageControllers/PageController.test.ts +205 -0
  102. package/src/server/plugins/engine/pageControllers/PageController.ts +176 -0
  103. package/src/server/plugins/engine/pageControllers/QuestionPageController.test.ts +1264 -0
  104. package/src/server/plugins/engine/pageControllers/QuestionPageController.ts +561 -0
  105. package/src/server/plugins/engine/pageControllers/README.md +28 -0
  106. package/src/server/plugins/engine/pageControllers/RepeatPageController.test.ts +264 -0
  107. package/src/server/plugins/engine/pageControllers/RepeatPageController.ts +458 -0
  108. package/src/server/plugins/engine/pageControllers/StartPageController.ts +18 -0
  109. package/src/server/plugins/engine/pageControllers/StatusPageController.ts +50 -0
  110. package/src/server/plugins/engine/pageControllers/SummaryPageController.ts +261 -0
  111. package/src/server/plugins/engine/pageControllers/TerminalController.test.ts +28 -0
  112. package/src/server/plugins/engine/pageControllers/TerminalPageController.ts +19 -0
  113. package/src/server/plugins/engine/pageControllers/helpers.test.ts +198 -0
  114. package/src/server/plugins/engine/pageControllers/helpers.ts +101 -0
  115. package/src/server/plugins/engine/pageControllers/index.ts +10 -0
  116. package/src/server/plugins/engine/pageControllers/validationOptions.ts +89 -0
  117. package/src/server/plugins/engine/plugin.ts +673 -0
  118. package/src/server/plugins/engine/services/formSubmissionService.js +46 -0
  119. package/src/server/plugins/engine/services/formsService.js +46 -0
  120. package/src/server/plugins/engine/services/formsService.test.js +90 -0
  121. package/src/server/plugins/engine/services/index.js +3 -0
  122. package/src/server/plugins/engine/services/notifyService.test.ts +132 -0
  123. package/src/server/plugins/engine/services/notifyService.ts +64 -0
  124. package/src/server/plugins/engine/services/uploadService.js +60 -0
  125. package/src/server/plugins/engine/types.ts +315 -0
  126. package/src/server/plugins/engine/views/components/autocompletefield.html +5 -0
  127. package/src/server/plugins/engine/views/components/checkboxesfield.html +5 -0
  128. package/src/server/plugins/engine/views/components/datepartsfield.html +5 -0
  129. package/src/server/plugins/engine/views/components/details.html +6 -0
  130. package/src/server/plugins/engine/views/components/emailaddressfield.html +5 -0
  131. package/src/server/plugins/engine/views/components/fileuploadfield-key.html +8 -0
  132. package/src/server/plugins/engine/views/components/fileuploadfield-value.html +3 -0
  133. package/src/server/plugins/engine/views/components/fileuploadfield.html +24 -0
  134. package/src/server/plugins/engine/views/components/html.html +3 -0
  135. package/src/server/plugins/engine/views/components/insettext.html +7 -0
  136. package/src/server/plugins/engine/views/components/list.html +36 -0
  137. package/src/server/plugins/engine/views/components/monthyearfield.html +5 -0
  138. package/src/server/plugins/engine/views/components/multilinetextfield.html +10 -0
  139. package/src/server/plugins/engine/views/components/numberfield.html +5 -0
  140. package/src/server/plugins/engine/views/components/radiosfield.html +5 -0
  141. package/src/server/plugins/engine/views/components/selectfield.html +5 -0
  142. package/src/server/plugins/engine/views/components/telephonenumberfield.html +5 -0
  143. package/src/server/plugins/engine/views/components/textfield.html +5 -0
  144. package/src/server/plugins/engine/views/components/ukaddressfield.html +25 -0
  145. package/src/server/plugins/engine/views/components/yesnofield.html +5 -0
  146. package/src/server/plugins/engine/views/file-upload.html +45 -0
  147. package/src/server/plugins/engine/views/index.html +39 -0
  148. package/src/server/plugins/engine/views/item-delete.html +56 -0
  149. package/src/server/plugins/engine/views/partials/components.html +6 -0
  150. package/src/server/plugins/engine/views/partials/conditional-components.html +3 -0
  151. package/src/server/plugins/engine/views/partials/debug.html +44 -0
  152. package/src/server/plugins/engine/views/partials/form.html +15 -0
  153. package/src/server/plugins/engine/views/partials/heading.html +16 -0
  154. package/src/server/plugins/engine/views/partials/preview-banner.html +32 -0
  155. package/src/server/plugins/engine/views/partials/preview-banner.test.js +122 -0
  156. package/src/server/plugins/engine/views/partials/warn-missing-notification-email.html +10 -0
  157. package/src/server/plugins/engine/views/repeat-list-summary.html +53 -0
  158. package/src/server/plugins/errorPages.ts +58 -0
  159. package/src/server/plugins/nunjucks/context.js +88 -0
  160. package/src/server/plugins/nunjucks/context.test.js +142 -0
  161. package/src/server/plugins/nunjucks/enviroment.test.js +201 -0
  162. package/src/server/plugins/nunjucks/environment.js +116 -0
  163. package/src/server/plugins/nunjucks/filters/answer.js +27 -0
  164. package/src/server/plugins/nunjucks/filters/answer.test.js +89 -0
  165. package/src/server/plugins/nunjucks/filters/evaluate.js +21 -0
  166. package/src/server/plugins/nunjucks/filters/field.js +28 -0
  167. package/src/server/plugins/nunjucks/filters/field.test.js +75 -0
  168. package/src/server/plugins/nunjucks/filters/highlight.js +11 -0
  169. package/src/server/plugins/nunjucks/filters/href.js +30 -0
  170. package/src/server/plugins/nunjucks/filters/href.test.js +80 -0
  171. package/src/server/plugins/nunjucks/filters/index.js +8 -0
  172. package/src/server/plugins/nunjucks/filters/inspect.js +15 -0
  173. package/src/server/plugins/nunjucks/filters/page.js +24 -0
  174. package/src/server/plugins/nunjucks/filters/page.test.js +65 -0
  175. package/src/server/plugins/nunjucks/index.js +3 -0
  176. package/src/server/plugins/nunjucks/plugin.js +40 -0
  177. package/src/server/plugins/nunjucks/render.js +42 -0
  178. package/src/server/plugins/nunjucks/types.js +40 -0
  179. package/src/server/plugins/pulse.ts +11 -0
  180. package/src/server/plugins/router.ts +201 -0
  181. package/src/server/plugins/session.ts +28 -0
  182. package/src/server/routes/health.js +13 -0
  183. package/src/server/routes/health.test.js +35 -0
  184. package/src/server/routes/index.test.ts +125 -0
  185. package/src/server/routes/index.ts +2 -0
  186. package/src/server/routes/public.ts +47 -0
  187. package/src/server/routes/types.ts +48 -0
  188. package/src/server/schemas/index.ts +34 -0
  189. package/src/server/secure-context.js +43 -0
  190. package/src/server/services/cacheService.test.ts +276 -0
  191. package/src/server/services/cacheService.ts +131 -0
  192. package/src/server/services/httpService.test.js +491 -0
  193. package/src/server/services/httpService.ts +50 -0
  194. package/src/server/services/index.ts +1 -0
  195. package/src/server/types.ts +54 -0
  196. package/src/server/utils/notify.test.ts +37 -0
  197. package/src/server/utils/notify.ts +50 -0
  198. package/src/server/utils/secure-context/get-trust-store-certs.js +11 -0
  199. package/src/server/utils/secure-context/get-trust-store-certs.test.js +19 -0
  200. package/src/server/utils/utils.js +24 -0
  201. package/src/server/utils/utils.test.js +54 -0
  202. package/src/server/views/404.html +16 -0
  203. package/src/server/views/500.html +19 -0
  204. package/src/server/views/components/debug/macro.njk +3 -0
  205. package/src/server/views/components/debug/template.njk +13 -0
  206. package/src/server/views/components/service-banner/macro.njk +3 -0
  207. package/src/server/views/components/service-banner/template.njk +20 -0
  208. package/src/server/views/components/service-banner/template.test.js +43 -0
  209. package/src/server/views/components/tag-env/macro.njk +3 -0
  210. package/src/server/views/components/tag-env/template.njk +30 -0
  211. package/src/server/views/components/tag-env/template.test.js +66 -0
  212. package/src/server/views/confirmation.html +19 -0
  213. package/src/server/views/help/accessibility-statement.html +58 -0
  214. package/src/server/views/help/cookie-preferences.html +57 -0
  215. package/src/server/views/help/cookies.html +71 -0
  216. package/src/server/views/help/get-support.html +37 -0
  217. package/src/server/views/help/privacy-notice.html +68 -0
  218. package/src/server/views/help/terms-and-conditions.html +83 -0
  219. package/src/server/views/layout.html +199 -0
  220. package/src/server/views/summary.html +50 -0
  221. package/src/typings/hapi/index.d.ts +95 -0
  222. package/src/typings/hapi-tracing/index.d.ts +6 -0
  223. package/src/typings/index.d.ts +3 -0
  224. package/src/typings/joi/index.d.ts +22 -0
@@ -0,0 +1,379 @@
1
+ import {
2
+ ControllerPath,
3
+ Engine,
4
+ type ComponentDef,
5
+ type Page
6
+ } from '@defra/forms-model'
7
+ import Boom from '@hapi/boom'
8
+ import { type ResponseToolkit } from '@hapi/hapi'
9
+ import { format, parseISO } from 'date-fns'
10
+ import { StatusCodes } from 'http-status-codes'
11
+ import { type Schema, type ValidationErrorItem } from 'joi'
12
+ import { Liquid } from 'liquidjs'
13
+
14
+ import { createLogger } from '~/src/server/common/helpers/logging/logger.js'
15
+ import { PREVIEW_PATH_PREFIX } from '~/src/server/constants.js'
16
+ import {
17
+ getAnswer,
18
+ type Field
19
+ } from '~/src/server/plugins/engine/components/helpers.js'
20
+ import { type FormModel } from '~/src/server/plugins/engine/models/FormModel.js'
21
+ import { type PageControllerClass } from '~/src/server/plugins/engine/pageControllers/helpers.js'
22
+ import {
23
+ type FormContext,
24
+ type FormContextRequest,
25
+ type FormSubmissionError
26
+ } from '~/src/server/plugins/engine/types.js'
27
+ import {
28
+ FormAction,
29
+ FormStatus,
30
+ type FormQuery,
31
+ type FormRequest,
32
+ type FormRequestPayload
33
+ } from '~/src/server/routes/types.js'
34
+
35
+ const logger = createLogger()
36
+
37
+ export const engine = new Liquid({
38
+ outputEscape: 'escape',
39
+ jsTruthy: true,
40
+ ownPropertyOnly: false
41
+ })
42
+
43
+ export interface GlobalScope {
44
+ context: FormContext
45
+ pages: Map<string, Page>
46
+ components: Map<string, ComponentDef>
47
+ }
48
+
49
+ engine.registerFilter('evaluate', function (template?: string) {
50
+ if (typeof template !== 'string') {
51
+ return template
52
+ }
53
+
54
+ const globals = this.context.globals as GlobalScope
55
+ const evaluated = evaluateTemplate(template, globals.context)
56
+
57
+ return evaluated
58
+ })
59
+
60
+ engine.registerFilter('page', function (path?: string) {
61
+ if (typeof path !== 'string') {
62
+ return
63
+ }
64
+
65
+ const globals = this.context.globals as GlobalScope
66
+ const pageDef = globals.pages.get(path)
67
+
68
+ return pageDef
69
+ })
70
+
71
+ engine.registerFilter('href', function (path: string, query?: FormQuery) {
72
+ if (typeof path !== 'string') {
73
+ return
74
+ }
75
+
76
+ const globals = this.context.globals as GlobalScope
77
+ const page = globals.context.pageMap.get(path)
78
+
79
+ if (page === undefined) {
80
+ return
81
+ }
82
+
83
+ return getPageHref(page, query)
84
+ })
85
+
86
+ engine.registerFilter('field', function (name: string) {
87
+ if (typeof name !== 'string') {
88
+ return
89
+ }
90
+
91
+ const globals = this.context.globals as GlobalScope
92
+ const componentDef = globals.components.get(name)
93
+
94
+ return componentDef
95
+ })
96
+
97
+ engine.registerFilter('answer', function (name: string) {
98
+ if (typeof name !== 'string') {
99
+ return
100
+ }
101
+
102
+ const globals = this.context.globals as GlobalScope
103
+ const component = globals.context.componentMap.get(name)
104
+
105
+ if (!component?.isFormComponent) {
106
+ return
107
+ }
108
+
109
+ const answer = getAnswer(component as Field, globals.context.relevantState)
110
+
111
+ return answer
112
+ })
113
+
114
+ export function proceed(
115
+ request: Pick<FormContextRequest, 'method' | 'payload' | 'query'>,
116
+ h: Pick<ResponseToolkit, 'redirect' | 'view'>,
117
+ nextUrl: string
118
+ ) {
119
+ const { method, payload, query } = request
120
+ const { returnUrl } = query
121
+
122
+ const isReturnAllowed =
123
+ payload && 'action' in payload
124
+ ? payload.action === FormAction.Continue ||
125
+ payload.action === FormAction.Validate
126
+ : false
127
+
128
+ // Redirect to return location (optional)
129
+ const response =
130
+ isReturnAllowed && isPathRelative(returnUrl)
131
+ ? h.redirect(returnUrl)
132
+ : h.redirect(redirectPath(nextUrl))
133
+
134
+ // Redirect POST to GET to avoid resubmission
135
+ return method === 'post'
136
+ ? response.code(StatusCodes.SEE_OTHER)
137
+ : response.code(StatusCodes.MOVED_TEMPORARILY)
138
+ }
139
+
140
+ /**
141
+ * Encodes a URL, returning undefined if the process fails.
142
+ */
143
+ export function encodeUrl(link?: string) {
144
+ if (link) {
145
+ try {
146
+ return new URL(link).toString() // escape the search params without breaking the ? and & reserved characters in rfc2368
147
+ } catch (err) {
148
+ logger.error(err, `Failed to encode ${link}`)
149
+ throw err
150
+ }
151
+ }
152
+ }
153
+
154
+ /**
155
+ * Get page href
156
+ */
157
+ export function getPageHref(
158
+ page: PageControllerClass,
159
+ query?: FormQuery
160
+ ): string
161
+
162
+ /**
163
+ * Get page href by path
164
+ */
165
+ export function getPageHref(
166
+ page: PageControllerClass,
167
+ path: string,
168
+ query?: FormQuery
169
+ ): string
170
+
171
+ export function getPageHref(
172
+ page: PageControllerClass,
173
+ pathOrQuery?: string | FormQuery,
174
+ queryOnly: FormQuery = {}
175
+ ) {
176
+ const path = typeof pathOrQuery === 'string' ? pathOrQuery : page.path
177
+ const query = typeof pathOrQuery === 'object' ? pathOrQuery : queryOnly
178
+
179
+ if (!isPathRelative(path)) {
180
+ throw Error(`Only relative URLs are allowed: ${path}`)
181
+ }
182
+
183
+ // Return path with page href as base
184
+ return redirectPath(page.getHref(path), query)
185
+ }
186
+
187
+ /**
188
+ * Get redirect path with optional query params
189
+ */
190
+ export function redirectPath(nextUrl: string, query: FormQuery = {}) {
191
+ const isRelative = isPathRelative(nextUrl)
192
+
193
+ // Filter string query params only
194
+ const params = Object.entries(query).filter(
195
+ (query): query is [string, string] => typeof query[1] === 'string'
196
+ )
197
+
198
+ // Build URL with relative path support
199
+ const url = isRelative
200
+ ? new URL(nextUrl, 'http://example.com')
201
+ : new URL(nextUrl)
202
+
203
+ // Append query params
204
+ for (const [name, value] of params) {
205
+ url.searchParams.set(name, value)
206
+ }
207
+
208
+ if (isRelative) {
209
+ return `${url.pathname}${url.search}`
210
+ }
211
+
212
+ return url.href
213
+ }
214
+
215
+ export function isPathRelative(path?: string) {
216
+ return (path ?? '').startsWith('/')
217
+ }
218
+
219
+ export function normalisePath(path = '') {
220
+ return path
221
+ .trim() // Trim empty spaces
222
+ .replace(/^\//, '') // Remove leading slash
223
+ .replace(/\/$/, '') // Remove trailing slash
224
+ }
225
+
226
+ export function getPage(
227
+ model: FormModel | undefined,
228
+ request: FormContextRequest
229
+ ) {
230
+ const { params } = request
231
+
232
+ const page = findPage(model, `/${params.path}`)
233
+
234
+ if (!page) {
235
+ throw Boom.notFound(`No page found for /${params.path}`)
236
+ }
237
+
238
+ return page
239
+ }
240
+
241
+ export function findPage(model: FormModel | undefined, path?: string) {
242
+ const findPath = `/${normalisePath(path)}`
243
+ return model?.pages.find(({ path }) => path === findPath)
244
+ }
245
+
246
+ export function getStartPath(model?: FormModel) {
247
+ if (model?.engine === Engine.V2) {
248
+ const startPath = normalisePath(model.def.pages.at(0)?.path)
249
+ return startPath ? `/${startPath}` : ControllerPath.Start
250
+ }
251
+
252
+ const startPath = normalisePath(model?.def.startPage)
253
+ return startPath ? `/${startPath}` : ControllerPath.Start
254
+ }
255
+
256
+ export function checkFormStatus(path: string) {
257
+ const isPreview = path.toLowerCase().startsWith(PREVIEW_PATH_PREFIX)
258
+
259
+ let state: FormStatus | undefined
260
+
261
+ if (isPreview) {
262
+ const previewState = path.split('/')[2]
263
+
264
+ for (const formState of Object.values(FormStatus)) {
265
+ if (previewState === formState.toString()) {
266
+ state = formState
267
+ break
268
+ }
269
+ }
270
+
271
+ if (!state) {
272
+ throw new Error(`Invalid form state: ${previewState}`)
273
+ }
274
+ }
275
+
276
+ return {
277
+ isPreview,
278
+ state: state ?? FormStatus.Live
279
+ }
280
+ }
281
+
282
+ export function checkEmailAddressForLiveFormSubmission(
283
+ emailAddress: string | undefined,
284
+ isPreview: boolean
285
+ ) {
286
+ if (!emailAddress && !isPreview) {
287
+ throw Boom.internal(
288
+ 'An email address is required to complete the form submission'
289
+ )
290
+ }
291
+ }
292
+
293
+ /**
294
+ * Parses the errors from {@link Schema.validate} so they can be rendered by govuk-frontend templates
295
+ * @param [details] - provided by {@link Schema.validate}
296
+ */
297
+ export function getErrors(
298
+ details?: ValidationErrorItem[]
299
+ ): FormSubmissionError[] | undefined {
300
+ if (!details?.length) {
301
+ return
302
+ }
303
+
304
+ return details.map(getError)
305
+ }
306
+
307
+ export function getError(detail: ValidationErrorItem): FormSubmissionError {
308
+ const { context, message, path } = detail
309
+
310
+ const name = context?.key ?? ''
311
+ const href = `#${name}`
312
+
313
+ const text = message.replace(
314
+ /\d{4}-[01]\d-[0-3]\dT[0-2]\d:[0-5]\d:[0-5]\d\.\d+([+-][0-2]\d:[0-5]\d|Z)/,
315
+ (text) => format(parseISO(text), 'd MMMM yyyy')
316
+ )
317
+
318
+ return {
319
+ path,
320
+ href,
321
+ name,
322
+ text,
323
+ context
324
+ }
325
+ }
326
+
327
+ /**
328
+ * A small helper to safely generate a crumb token.
329
+ * Checks that the crumb plugin is available, that crumb
330
+ * is not disabled on the current route, and that cookies/state are present.
331
+ */
332
+ export function safeGenerateCrumb(
333
+ request: FormRequest | FormRequestPayload | null
334
+ ): string | undefined {
335
+ // no request or no .state
336
+ if (!request?.state) {
337
+ return undefined
338
+ }
339
+
340
+ // crumb plugin or its generate method doesn’t exist
341
+ if (!request.server.plugins.crumb.generate) {
342
+ return undefined
343
+ }
344
+
345
+ // crumb is explicitly disabled for this route
346
+ if (request.route.settings.plugins?.crumb === false) {
347
+ return undefined
348
+ }
349
+
350
+ return request.server.plugins.crumb.generate(request)
351
+ }
352
+
353
+ /**
354
+ * Calculates an exponential backoff delay (in milliseconds) based on the current retry depth,
355
+ * using a base delay of 2000ms (2 seconds) and doubling for each additional depth, while capping the delay at 25,000ms (25 seconds).
356
+ * @param depth - The current retry depth (1, 2, 3, …)
357
+ * @returns The calculated delay in milliseconds.
358
+ */
359
+ export function getExponentialBackoffDelay(depth: number): number {
360
+ const BASE_DELAY_MS = 2000 // 2 seconds initial delay
361
+ const CAP_DELAY_MS = 25000 // cap each delay to 25 seconds
362
+ const delay = BASE_DELAY_MS * 2 ** (depth - 1)
363
+ return Math.min(delay, CAP_DELAY_MS)
364
+ }
365
+ export function evaluateTemplate(
366
+ template: string,
367
+ context: FormContext
368
+ ): string {
369
+ const globals: GlobalScope = {
370
+ context,
371
+ pages: context.pageDefMap,
372
+ components: context.componentDefMap
373
+ }
374
+
375
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-return
376
+ return engine.parseAndRenderSync(template, context.relevantState, {
377
+ globals
378
+ })
379
+ }
@@ -0,0 +1,7 @@
1
+ import { plugin } from '~/src/server/plugins/engine/plugin.js'
2
+
3
+ export { getPageHref } from '~/src/server/plugins/engine/helpers.js'
4
+ export { configureEnginePlugin } from '~/src/server/plugins/engine/configureEnginePlugin.js'
5
+ export { CacheService } from '~/src/server/services/index.js'
6
+
7
+ export default plugin
@@ -0,0 +1,42 @@
1
+ import { FormModel } from '~/src/server/plugins/engine/models/FormModel.js'
2
+ import { type FormContextRequest } from '~/src/server/plugins/engine/types.js'
3
+ import definition from '~/test/form/definitions/conditions-escaping.js'
4
+ import fieldsRequiredDefinition from '~/test/form/definitions/fields-required.js'
5
+
6
+ describe('FormModel', () => {
7
+ describe('Constructor', () => {
8
+ it("doesn't throw when conditions are passed with apostrophes", () => {
9
+ expect(
10
+ () => new FormModel(definition, { basePath: 'test' })
11
+ ).not.toThrow()
12
+ })
13
+ })
14
+
15
+ describe('getFormContext', () => {
16
+ it('clears a previous checkbox field value when the field is omitted from the payload', () => {
17
+ const formModel = new FormModel(fieldsRequiredDefinition, {
18
+ basePath: '/components'
19
+ })
20
+
21
+ const state = { checkboxesSingle: ['Arabian', 'Shetland'] }
22
+ const pageUrl = new URL('http://example.com/components/fields-required')
23
+
24
+ const request: FormContextRequest = {
25
+ method: 'post',
26
+ payload: { crumb: 'dummyCrumb', action: 'validate' },
27
+ query: {},
28
+ path: pageUrl.pathname,
29
+ params: { path: 'components', slug: 'fields-required' },
30
+ url: pageUrl,
31
+ app: { model: formModel }
32
+ }
33
+
34
+ const context = formModel.getFormContext(request, state)
35
+
36
+ expect(context.payload.checkboxesSingle).toEqual([])
37
+ expect(context.errors).toContainEqual(
38
+ expect.objectContaining({ name: 'checkboxesSingle' })
39
+ )
40
+ })
41
+ })
42
+ })