@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,447 @@
1
+ import { ComponentType, type PageFileUpload } from '@defra/forms-model'
2
+ import Boom from '@hapi/boom'
3
+ import { type ResponseToolkit } from '@hapi/hapi'
4
+ import { wait } from '@hapi/hoek'
5
+ import { type ValidationErrorItem } from 'joi'
6
+
7
+ import {
8
+ tempItemSchema,
9
+ type FileUploadField
10
+ } from '~/src/server/plugins/engine/components/FileUploadField.js'
11
+ import {
12
+ getCacheService,
13
+ getError,
14
+ getExponentialBackoffDelay
15
+ } from '~/src/server/plugins/engine/helpers.js'
16
+ import { type FormModel } from '~/src/server/plugins/engine/models/index.js'
17
+ import { QuestionPageController } from '~/src/server/plugins/engine/pageControllers/QuestionPageController.js'
18
+ import { getProxyUrlForLocalDevelopment } from '~/src/server/plugins/engine/pageControllers/helpers.js'
19
+ import {
20
+ getUploadStatus,
21
+ initiateUpload
22
+ } from '~/src/server/plugins/engine/services/uploadService.js'
23
+ import {
24
+ FileStatus,
25
+ UploadStatus,
26
+ type FeaturedFormPageViewModel,
27
+ type FileState,
28
+ type FormContext,
29
+ type FormContextRequest,
30
+ type FormSubmissionError,
31
+ type FormSubmissionState,
32
+ type ItemDeletePageViewModel,
33
+ type UploadInitiateResponse,
34
+ type UploadStatusFileResponse
35
+ } from '~/src/server/plugins/engine/types.js'
36
+ import {
37
+ type FormRequest,
38
+ type FormRequestPayload
39
+ } from '~/src/server/routes/types.js'
40
+
41
+ const MAX_UPLOADS = 25
42
+ const CDP_UPLOAD_TIMEOUT_MS = 60000 // 1 minute
43
+
44
+ export function prepareStatus(status: UploadStatusFileResponse) {
45
+ const file = status.form.file
46
+ const isPending = file.fileStatus === FileStatus.pending
47
+
48
+ if (!file.errorMessage && isPending) {
49
+ file.errorMessage = 'The selected file has not fully uploaded'
50
+ }
51
+
52
+ return status
53
+ }
54
+
55
+ function prepareFileState(fileState: FileState) {
56
+ prepareStatus(fileState.status)
57
+
58
+ return fileState
59
+ }
60
+
61
+ export class FileUploadPageController extends QuestionPageController {
62
+ declare pageDef: PageFileUpload
63
+
64
+ fileUpload: FileUploadField
65
+ fileDeleteViewName = 'item-delete'
66
+
67
+ constructor(model: FormModel, pageDef: PageFileUpload) {
68
+ super(model, pageDef)
69
+
70
+ const { collection } = this
71
+
72
+ // Get the file upload fields from the collection
73
+ const fileUploads = collection.fields.filter(
74
+ (field): field is FileUploadField =>
75
+ field.type === ComponentType.FileUploadField
76
+ )
77
+
78
+ const fileUpload = fileUploads.at(0)
79
+
80
+ // Assert we have exactly 1 file upload component
81
+ if (!fileUpload || fileUploads.length > 1) {
82
+ throw Boom.badImplementation(
83
+ `Expected 1 FileUploadFieldComponent in FileUploadPageController '${pageDef.path}'`
84
+ )
85
+ }
86
+
87
+ // Assert the file upload component is the first form component
88
+ if (collection.fields.indexOf(fileUpload) !== 0) {
89
+ throw Boom.badImplementation(
90
+ `Expected '${fileUpload.name}' to be the first form component in FileUploadPageController '${pageDef.path}'`
91
+ )
92
+ }
93
+
94
+ // Assign the file upload component to the controller
95
+ this.fileUpload = fileUpload
96
+ this.viewName = 'file-upload'
97
+ }
98
+
99
+ getFormDataFromState(
100
+ request: FormContextRequest | undefined,
101
+ state: FormSubmissionState
102
+ ) {
103
+ const { fileUpload } = this
104
+
105
+ const payload = super.getFormDataFromState(request, state)
106
+ const files = this.getFilesFromState(state)
107
+
108
+ // Append the files to the payload
109
+ payload[fileUpload.name] = files.length ? files : undefined
110
+
111
+ return payload
112
+ }
113
+
114
+ async getState(request: FormRequest | FormRequestPayload) {
115
+ const { fileUpload } = this
116
+
117
+ // Get the actual state
118
+ const state = await super.getState(request)
119
+ const files = this.getFilesFromState(state)
120
+
121
+ // Overwrite the files with those in the upload state
122
+ state[fileUpload.name] = files
123
+
124
+ return this.refreshUpload(request, state)
125
+ }
126
+
127
+ /**
128
+ * Get the uploaded files from state.
129
+ */
130
+ getFilesFromState(state: FormSubmissionState) {
131
+ const { path } = this
132
+
133
+ const uploadState = state.upload?.[path]
134
+ return uploadState?.files ?? []
135
+ }
136
+
137
+ /**
138
+ * Get the initiated upload from state.
139
+ */
140
+ getUploadFromState(state: FormSubmissionState) {
141
+ const { path } = this
142
+
143
+ const uploadState = state.upload?.[path]
144
+ return uploadState?.upload
145
+ }
146
+
147
+ makeGetItemDeleteRouteHandler() {
148
+ return (
149
+ request: FormRequest,
150
+ context: FormContext,
151
+ h: Pick<ResponseToolkit, 'redirect' | 'view'>
152
+ ) => {
153
+ const { viewModel } = this
154
+ const { params } = request
155
+ const { state } = context
156
+
157
+ const files = this.getFilesFromState(state)
158
+
159
+ const fileToRemove = files.find(
160
+ ({ uploadId }) => uploadId === params.itemId
161
+ )
162
+
163
+ if (!fileToRemove) {
164
+ throw Boom.notFound('File to delete not found')
165
+ }
166
+
167
+ const { filename } = fileToRemove.status.form.file
168
+
169
+ return h.view(this.fileDeleteViewName, {
170
+ ...viewModel,
171
+ context,
172
+ backLink: this.getBackLink(request, context),
173
+ pageTitle: `Are you sure you want to remove this file?`,
174
+ itemTitle: filename,
175
+ confirmation: { text: 'You cannot recover removed files.' },
176
+ buttonConfirm: { text: 'Remove file' },
177
+ buttonCancel: { text: 'Cancel' }
178
+ } satisfies ItemDeletePageViewModel)
179
+ }
180
+ }
181
+
182
+ makePostItemDeleteRouteHandler() {
183
+ return async (
184
+ request: FormRequestPayload,
185
+ context: FormContext,
186
+ h: Pick<ResponseToolkit, 'redirect' | 'view'>
187
+ ) => {
188
+ const { path } = this
189
+ const { state } = context
190
+
191
+ const { confirm } = this.getFormParams(request)
192
+
193
+ // Check for any removed files in the POST payload
194
+ if (confirm) {
195
+ await this.checkRemovedFiles(request, state)
196
+ return this.proceed(request, h, path)
197
+ }
198
+
199
+ return this.proceed(request, h)
200
+ }
201
+ }
202
+
203
+ getErrors(details?: ValidationErrorItem[]) {
204
+ const { fileUpload } = this
205
+
206
+ if (details) {
207
+ const errors: FormSubmissionError[] = []
208
+
209
+ details.forEach((error) => {
210
+ const isUploadError = error.path[0] === fileUpload.name
211
+ const isUploadRootError = isUploadError && error.path.length === 1
212
+
213
+ if (!isUploadError || isUploadRootError) {
214
+ // The error is for the root of the upload or another
215
+ // field on the page so defer to the getError helper
216
+ errors.push(getError(error))
217
+ } else {
218
+ const { context, path, type } = error
219
+
220
+ if (type === 'object.unknown' && path.at(-1) === 'errorMessage') {
221
+ const value = context?.value as string | undefined
222
+
223
+ if (value) {
224
+ const name = fileUpload.name
225
+ const text = typeof value === 'string' ? value : 'Unknown error'
226
+ const href = `#${name}`
227
+
228
+ errors.push({ path, href, name, text })
229
+ }
230
+ }
231
+ }
232
+ })
233
+
234
+ return errors
235
+ }
236
+ }
237
+
238
+ getViewModel(
239
+ request: FormContextRequest,
240
+ context: FormContext
241
+ ): FeaturedFormPageViewModel {
242
+ const { fileUpload } = this
243
+ const { state } = context
244
+
245
+ const upload = this.getUploadFromState(state)
246
+
247
+ const viewModel = super.getViewModel(request, context)
248
+ const { components } = viewModel
249
+
250
+ // Featured form component
251
+ const [formComponent] = components.filter(
252
+ ({ model }) => model.id === fileUpload.name
253
+ )
254
+
255
+ const index = components.indexOf(formComponent)
256
+
257
+ const proxyUrl = getProxyUrlForLocalDevelopment(upload?.uploadUrl)
258
+
259
+ return {
260
+ ...viewModel,
261
+ formAction: upload?.uploadUrl,
262
+ uploadId: upload?.uploadId,
263
+ formComponent,
264
+
265
+ // Split out components before/after
266
+ componentsBefore: components.slice(0, index),
267
+ components: components.slice(index),
268
+ proxyUrl
269
+ }
270
+ }
271
+
272
+ /**
273
+ * Refreshes the CDP upload and files in the
274
+ * state and checks for any removed files.
275
+ *
276
+ * If an upload exists and hasn't been consumed
277
+ * it gets re-used, otherwise we initiate a new one.
278
+ * @param request - the hapi request
279
+ * @param state - the form state
280
+ */
281
+ private async refreshUpload(
282
+ request: FormRequest | FormRequestPayload,
283
+ state: FormSubmissionState
284
+ ) {
285
+ state = await this.checkUploadStatus(request, state)
286
+
287
+ return state
288
+ }
289
+
290
+ /**
291
+ * If an upload exists and hasn't been consumed
292
+ * it gets re-used, otherwise a new one is initiated.
293
+ * @param request - the hapi request
294
+ * @param state - the form state
295
+ * @param depth - the number of retries so far
296
+ */
297
+ private async checkUploadStatus(
298
+ request: FormRequest | FormRequestPayload,
299
+ state: FormSubmissionState,
300
+ depth = 1
301
+ ): Promise<FormSubmissionState> {
302
+ const upload = this.getUploadFromState(state)
303
+ const files = this.getFilesFromState(state)
304
+
305
+ // If no upload exists, initiate a new one.
306
+ if (!upload?.uploadId) {
307
+ return this.initiateAndStoreNewUpload(request, state)
308
+ }
309
+
310
+ const uploadId = upload.uploadId
311
+ const statusResponse = await getUploadStatus(uploadId)
312
+ if (!statusResponse) {
313
+ throw Boom.badRequest(
314
+ `Unexpected empty response from getUploadStatus for ${uploadId}`
315
+ )
316
+ }
317
+
318
+ // Re-use the upload if it is still in the "initiated" state.
319
+ if (statusResponse.uploadStatus === UploadStatus.initiated) {
320
+ return state
321
+ }
322
+
323
+ if (statusResponse.uploadStatus === UploadStatus.pending) {
324
+ // Using exponential backoff delays:
325
+ // Depth 1: 2000ms, Depth 2: 4000ms, Depth 3: 8000ms, Depth 4: 16000ms, Depth 5+: 30000ms (capped)
326
+ // A depth of 5 (or more) implies cumulative delays roughly reaching 55 seconds.
327
+ if (depth >= 5) {
328
+ request.logger.error(
329
+ `Exceeded cumulative retry delay for ${uploadId} (depth: ${depth}). Re-initiating a new upload.`
330
+ )
331
+ await this.initiateAndStoreNewUpload(request, state)
332
+ throw Boom.gatewayTimeout(
333
+ `Timed out waiting for ${uploadId} after cumulative retries exceeding ${((CDP_UPLOAD_TIMEOUT_MS - 5000) / 1000).toFixed(0)} seconds`
334
+ )
335
+ }
336
+ const delay = getExponentialBackoffDelay(depth)
337
+ request.logger.info(
338
+ `Waiting ${delay / 1000} seconds for ${uploadId} to complete (depth: ${depth})`
339
+ )
340
+ await wait(delay)
341
+ return this.checkUploadStatus(request, state, depth + 1)
342
+ }
343
+
344
+ // Only add to files state if the file validates.
345
+ // This secures against html tampering of the file input
346
+ // by adding a 'multiple' attribute or it being
347
+ // changed to a simple text field or similar.
348
+ const validationResult = tempItemSchema.validate(
349
+ { uploadId, status: statusResponse },
350
+ { stripUnknown: true }
351
+ )
352
+ const error = validationResult.error
353
+ const fileState = validationResult.value as FileState
354
+
355
+ if (error) {
356
+ return this.initiateAndStoreNewUpload(request, state)
357
+ }
358
+
359
+ const file = fileState.status.form.file
360
+ if (file.fileStatus === FileStatus.complete) {
361
+ files.unshift(prepareFileState(fileState))
362
+ await this.mergeState(request, state, {
363
+ upload: { [this.path]: { files, upload } }
364
+ })
365
+ } else {
366
+ // Flash the error message.
367
+ const { fileUpload } = this
368
+ const cacheService = getCacheService(request.server)
369
+ const name = fileUpload.name
370
+ const text = file.errorMessage ?? 'Unknown error'
371
+ const errors: FormSubmissionError[] = [
372
+ { path: [name], href: `#${name}`, name, text }
373
+ ]
374
+ cacheService.setFlash(request, { errors })
375
+ }
376
+
377
+ return this.initiateAndStoreNewUpload(request, state)
378
+ }
379
+
380
+ /**
381
+ * Checks the payload for a file getting removed
382
+ * and removes it from the upload files if found
383
+ * @param request - the hapi request
384
+ * @param state - the form state
385
+ * @returns updated state if any files have been removed
386
+ */
387
+ private async checkRemovedFiles(
388
+ request: FormRequestPayload,
389
+ state: FormSubmissionState
390
+ ) {
391
+ const { path } = this
392
+ const { params } = request
393
+
394
+ const upload = this.getUploadFromState(state)
395
+ const files = this.getFilesFromState(state)
396
+
397
+ const filesUpdated = files.filter(
398
+ ({ uploadId }) => uploadId !== params.itemId
399
+ )
400
+
401
+ if (filesUpdated.length === files.length) {
402
+ return
403
+ }
404
+
405
+ await this.mergeState(request, state, {
406
+ upload: { [path]: { files: filesUpdated, upload } }
407
+ })
408
+ }
409
+
410
+ /**
411
+ * Initiates a CDP file upload and stores in the upload state
412
+ * @param request - the hapi request
413
+ * @param state - the form state
414
+ */
415
+ private async initiateAndStoreNewUpload(
416
+ request: FormRequest | FormRequestPayload,
417
+ state: FormSubmissionState
418
+ ) {
419
+ const { fileUpload, href, path } = this
420
+ const { options, schema } = fileUpload
421
+
422
+ const files = this.getFilesFromState(state)
423
+
424
+ // Reset the upload in state
425
+ let upload: UploadInitiateResponse | undefined
426
+
427
+ // Don't initiate anymore after minimum of `schema.max` or MAX_UPLOADS
428
+ const max = Math.min(schema.max ?? MAX_UPLOADS, MAX_UPLOADS)
429
+
430
+ if (files.length < max) {
431
+ const outputEmail =
432
+ this.model.def.outputEmail ?? 'defraforms@defra.gov.uk'
433
+
434
+ const newUpload = await initiateUpload(href, outputEmail, options.accept)
435
+
436
+ if (newUpload === undefined) {
437
+ throw Boom.badRequest('Unexpected empty response from initiateUpload')
438
+ }
439
+
440
+ upload = newUpload
441
+ }
442
+
443
+ return this.mergeState(request, state, {
444
+ upload: { [path]: { files, upload } }
445
+ })
446
+ }
447
+ }
@@ -0,0 +1,205 @@
1
+ import { type ResponseToolkit } from '@hapi/hapi'
2
+
3
+ import { FormModel } from '~/src/server/plugins/engine/models/FormModel.js'
4
+ import { PageController } from '~/src/server/plugins/engine/pageControllers/PageController.js'
5
+ import { type FormRequest } from '~/src/server/routes/types.js'
6
+ import definition from '~/test/form/definitions/basic.js'
7
+
8
+ describe('PageController', () => {
9
+ let model: FormModel
10
+ let controller1: PageController
11
+ let controller2: PageController
12
+
13
+ beforeEach(() => {
14
+ const { pages } = definition
15
+
16
+ const page1 = pages[0]
17
+ const page2 = pages[1]
18
+
19
+ model = new FormModel(definition, {
20
+ basePath: 'test'
21
+ })
22
+
23
+ controller1 = new PageController(model, page1)
24
+ controller2 = new PageController(model, page2)
25
+ })
26
+
27
+ describe('Properties', () => {
28
+ it('returns path', () => {
29
+ expect(controller1).toHaveProperty('path', '/licence')
30
+ expect(controller2).toHaveProperty('path', '/full-name')
31
+ })
32
+
33
+ it('returns href', () => {
34
+ expect(controller1).toHaveProperty('href', '/test/licence')
35
+ expect(controller2).toHaveProperty('href', '/test/full-name')
36
+ })
37
+
38
+ it('returns keys (empty)', () => {
39
+ expect(controller1).toHaveProperty('keys', [])
40
+ expect(controller2).toHaveProperty('keys', [])
41
+ })
42
+
43
+ it('returns the page section', () => {
44
+ expect(controller1).toHaveProperty('section', {
45
+ name: 'licenceDetails',
46
+ title: 'Licence details',
47
+ hideTitle: false
48
+ })
49
+
50
+ expect(controller2).toHaveProperty('section', {
51
+ name: 'personalDetails',
52
+ title: 'Personal details',
53
+ hideTitle: false
54
+ })
55
+ })
56
+
57
+ it('returns feedback link (from form definition)', () => {
58
+ expect(controller1).toHaveProperty('feedbackLink', undefined)
59
+
60
+ const emailAddress = 'test@feedback.cat'
61
+
62
+ model.def.feedback = {
63
+ emailAddress
64
+ }
65
+
66
+ expect(controller1).toHaveProperty(
67
+ 'feedbackLink',
68
+ `mailto:${emailAddress}`
69
+ )
70
+ })
71
+
72
+ it('returns phase tag (from form definition)', () => {
73
+ expect(controller1).toHaveProperty('phaseTag', undefined)
74
+
75
+ model.def.phaseBanner = {
76
+ phase: 'alpha'
77
+ }
78
+
79
+ expect(controller1).toHaveProperty('phaseTag', 'alpha')
80
+ })
81
+
82
+ it('sets default viewName to "index"', () => {
83
+ expect(controller1).toHaveProperty('viewName', 'index')
84
+ expect(controller2).toHaveProperty('viewName', 'index')
85
+ })
86
+
87
+ it('overrides viewName when pageDef.view is provided', () => {
88
+ const customPage = {
89
+ ...definition.pages[0],
90
+ view: 'custom-view'
91
+ }
92
+
93
+ const customController = new PageController(model, customPage)
94
+
95
+ expect(customController).toHaveProperty('viewName', 'custom-view')
96
+ })
97
+ })
98
+
99
+ describe('Path methods', () => {
100
+ describe('Link href', () => {
101
+ it('prefixes paths into link hrefs', () => {
102
+ const href1 = controller1.getHref('/')
103
+ const href2 = controller1.getHref('/page-one')
104
+
105
+ expect(href1).toBe('/test')
106
+ expect(href2).toBe('/test/page-one')
107
+ })
108
+ })
109
+
110
+ describe('Start path', () => {
111
+ it('returns path to start page', () => {
112
+ const startPath = controller1.getStartPath()
113
+ expect(startPath).toBe('/licence')
114
+ })
115
+
116
+ it('returns path to start page (default)', () => {
117
+ delete model.def.startPage
118
+
119
+ const startPath = controller1.getStartPath()
120
+ expect(startPath).toBe('/start')
121
+ })
122
+ })
123
+
124
+ describe('Summary path', () => {
125
+ it('returns path to summary page', () => {
126
+ const summaryPath = controller1.getSummaryPath()
127
+ expect(summaryPath).toBe('/summary')
128
+ })
129
+ })
130
+
131
+ describe('Status path', () => {
132
+ it('returns path to status page', () => {
133
+ const summaryPath = controller1.getStatusPath()
134
+ expect(summaryPath).toBe('/status')
135
+ })
136
+ })
137
+ })
138
+
139
+ describe('Route handlers', () => {
140
+ const page1Url = new URL('http://example.com/test/licence')
141
+
142
+ const request = {
143
+ method: 'get',
144
+ url: page1Url,
145
+ path: page1Url.pathname,
146
+ params: {
147
+ path: 'licence',
148
+ slug: 'test'
149
+ },
150
+ query: {},
151
+ app: { model }
152
+ } as FormRequest
153
+
154
+ const h: Pick<ResponseToolkit, 'redirect' | 'view'> = {
155
+ redirect: jest.fn(),
156
+ view: jest.fn()
157
+ }
158
+
159
+ it('returns default route options', () => {
160
+ expect(controller1.getRouteOptions).toEqual({})
161
+ expect(controller1.postRouteOptions).toEqual({})
162
+ })
163
+
164
+ it('supports GET route handler', async () => {
165
+ expect(() => controller1.makeGetRouteHandler()).not.toThrow()
166
+ expect(() => controller1.makeGetRouteHandler()).toBeInstanceOf(Function)
167
+
168
+ await controller1.makeGetRouteHandler()(
169
+ request,
170
+ model.getFormContext(request, {}),
171
+ h
172
+ )
173
+
174
+ await controller2.makeGetRouteHandler()(
175
+ request,
176
+ model.getFormContext(request, {}),
177
+ h
178
+ )
179
+
180
+ expect(h.view).toHaveBeenNthCalledWith(
181
+ 1,
182
+ controller1.viewName,
183
+ expect.objectContaining({
184
+ pageTitle: 'Buy a rod fishing licence',
185
+ sectionTitle: 'Licence details'
186
+ })
187
+ )
188
+
189
+ expect(h.view).toHaveBeenNthCalledWith(
190
+ 2,
191
+ controller1.viewName,
192
+ expect.objectContaining({
193
+ pageTitle: "What's your name?",
194
+ sectionTitle: 'Personal details'
195
+ })
196
+ )
197
+ })
198
+
199
+ it('does not support POST route handler', () => {
200
+ expect(() => controller1.makePostRouteHandler()).toThrow(
201
+ 'Unsupported POST route handler for this page'
202
+ )
203
+ })
204
+ })
205
+ })