@defra/forms-engine-plugin 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (615) hide show
  1. package/.browserslistrc +16 -0
  2. package/.editorconfig +9 -0
  3. package/.eslintrc.cjs +266 -0
  4. package/.github/dependabot.yml +85 -0
  5. package/.github/workflows/check-pull-request.yml +209 -0
  6. package/.github/workflows/pr-notifier.yml +98 -0
  7. package/.github/workflows/publish.yml +111 -0
  8. package/.husky/pre-commit +1 -0
  9. package/.lintstagedrc.js +4 -0
  10. package/.nvmrc +1 -0
  11. package/.prettierignore +8 -0
  12. package/.prettierrc.cjs +26 -0
  13. package/.server/client/javascripts/application.js +68 -0
  14. package/.server/client/javascripts/application.js.map +1 -0
  15. package/.server/client/javascripts/file-upload.js +292 -0
  16. package/.server/client/javascripts/file-upload.js.map +1 -0
  17. package/.server/client/stylesheets/_code.scss +33 -0
  18. package/.server/client/stylesheets/_govuk-frontend.scss +4 -0
  19. package/.server/client/stylesheets/_prose.scss +56 -0
  20. package/.server/client/stylesheets/_service-banner.scss +24 -0
  21. package/.server/client/stylesheets/_summary-list.scss +28 -0
  22. package/.server/client/stylesheets/_tag-env.scss +24 -0
  23. package/.server/client/stylesheets/application.scss +14 -0
  24. package/.server/common/cookies.js +55 -0
  25. package/.server/common/cookies.js.map +1 -0
  26. package/.server/common/cookies.test.js +15 -0
  27. package/.server/common/cookies.test.js.map +1 -0
  28. package/.server/common/types.js +6 -0
  29. package/.server/common/types.js.map +1 -0
  30. package/.server/config/index.js +241 -0
  31. package/.server/config/index.js.map +1 -0
  32. package/.server/index.js +25 -0
  33. package/.server/index.js.map +1 -0
  34. package/.server/server/common/helpers/logging/logger-options.js +44 -0
  35. package/.server/server/common/helpers/logging/logger-options.js.map +1 -0
  36. package/.server/server/common/helpers/logging/logger.js +6 -0
  37. package/.server/server/common/helpers/logging/logger.js.map +1 -0
  38. package/.server/server/common/helpers/logging/request-logger.js +7 -0
  39. package/.server/server/common/helpers/logging/request-logger.js.map +1 -0
  40. package/.server/server/common/helpers/logging/request-tracing.js +12 -0
  41. package/.server/server/common/helpers/logging/request-tracing.js.map +1 -0
  42. package/.server/server/common/helpers/redis-client.js +55 -0
  43. package/.server/server/common/helpers/redis-client.js.map +1 -0
  44. package/.server/server/constants.js +2 -0
  45. package/.server/server/constants.js.map +1 -0
  46. package/.server/server/forms/README.md +10 -0
  47. package/.server/server/forms/components.json +1015 -0
  48. package/.server/server/forms/report-a-terrorist.json +270 -0
  49. package/.server/server/forms/runner-components-test.json +365 -0
  50. package/.server/server/forms/test.json +581 -0
  51. package/.server/server/index.js +116 -0
  52. package/.server/server/index.js.map +1 -0
  53. package/.server/server/plugins/blankie.js +29 -0
  54. package/.server/server/plugins/blankie.js.map +1 -0
  55. package/.server/server/plugins/crumb.js +15 -0
  56. package/.server/server/plugins/crumb.js.map +1 -0
  57. package/.server/server/plugins/engine/README.md +87 -0
  58. package/.server/server/plugins/engine/components/AutocompleteField.js +37 -0
  59. package/.server/server/plugins/engine/components/AutocompleteField.js.map +1 -0
  60. package/.server/server/plugins/engine/components/CheckboxesField.js +79 -0
  61. package/.server/server/plugins/engine/components/CheckboxesField.js.map +1 -0
  62. package/.server/server/plugins/engine/components/ComponentBase.js +55 -0
  63. package/.server/server/plugins/engine/components/ComponentBase.js.map +1 -0
  64. package/.server/server/plugins/engine/components/ComponentCollection.js +198 -0
  65. package/.server/server/plugins/engine/components/ComponentCollection.js.map +1 -0
  66. package/.server/server/plugins/engine/components/DatePartsField.js +208 -0
  67. package/.server/server/plugins/engine/components/DatePartsField.js.map +1 -0
  68. package/.server/server/plugins/engine/components/Details.js +26 -0
  69. package/.server/server/plugins/engine/components/Details.js.map +1 -0
  70. package/.server/server/plugins/engine/components/EmailAddressField.js +40 -0
  71. package/.server/server/plugins/engine/components/EmailAddressField.js.map +1 -0
  72. package/.server/server/plugins/engine/components/FileUploadField.js +195 -0
  73. package/.server/server/plugins/engine/components/FileUploadField.js.map +1 -0
  74. package/.server/server/plugins/engine/components/FormComponent.js +210 -0
  75. package/.server/server/plugins/engine/components/FormComponent.js.map +1 -0
  76. package/.server/server/plugins/engine/components/Html.js +24 -0
  77. package/.server/server/plugins/engine/components/Html.js.map +1 -0
  78. package/.server/server/plugins/engine/components/InsetText.js +22 -0
  79. package/.server/server/plugins/engine/components/InsetText.js.map +1 -0
  80. package/.server/server/plugins/engine/components/List.js +65 -0
  81. package/.server/server/plugins/engine/components/List.js.map +1 -0
  82. package/.server/server/plugins/engine/components/ListFormComponent.js +87 -0
  83. package/.server/server/plugins/engine/components/ListFormComponent.js.map +1 -0
  84. package/.server/server/plugins/engine/components/MonthYearField.js +172 -0
  85. package/.server/server/plugins/engine/components/MonthYearField.js.map +1 -0
  86. package/.server/server/plugins/engine/components/MultilineTextField.js +115 -0
  87. package/.server/server/plugins/engine/components/MultilineTextField.js.map +1 -0
  88. package/.server/server/plugins/engine/components/NumberField.js +137 -0
  89. package/.server/server/plugins/engine/components/NumberField.js.map +1 -0
  90. package/.server/server/plugins/engine/components/RadiosField.js +18 -0
  91. package/.server/server/plugins/engine/components/RadiosField.js.map +1 -0
  92. package/.server/server/plugins/engine/components/SelectField.js +33 -0
  93. package/.server/server/plugins/engine/components/SelectField.js.map +1 -0
  94. package/.server/server/plugins/engine/components/SelectionControlField.js +43 -0
  95. package/.server/server/plugins/engine/components/SelectionControlField.js.map +1 -0
  96. package/.server/server/plugins/engine/components/TelephoneNumberField.js +43 -0
  97. package/.server/server/plugins/engine/components/TelephoneNumberField.js.map +1 -0
  98. package/.server/server/plugins/engine/components/TextField.js +63 -0
  99. package/.server/server/plugins/engine/components/TextField.js.map +1 -0
  100. package/.server/server/plugins/engine/components/UkAddressField.js +130 -0
  101. package/.server/server/plugins/engine/components/UkAddressField.js.map +1 -0
  102. package/.server/server/plugins/engine/components/YesNoField.js +28 -0
  103. package/.server/server/plugins/engine/components/YesNoField.js.map +1 -0
  104. package/.server/server/plugins/engine/components/constants.js +2 -0
  105. package/.server/server/plugins/engine/components/constants.js.map +1 -0
  106. package/.server/server/plugins/engine/components/helpers.js +219 -0
  107. package/.server/server/plugins/engine/components/helpers.js.map +1 -0
  108. package/.server/server/plugins/engine/components/index.js +25 -0
  109. package/.server/server/plugins/engine/components/index.js.map +1 -0
  110. package/.server/server/plugins/engine/components/types.js +2 -0
  111. package/.server/server/plugins/engine/components/types.js.map +1 -0
  112. package/.server/server/plugins/engine/configureEnginePlugin.js +44 -0
  113. package/.server/server/plugins/engine/configureEnginePlugin.js.map +1 -0
  114. package/.server/server/plugins/engine/helpers.js +268 -0
  115. package/.server/server/plugins/engine/helpers.js.map +1 -0
  116. package/.server/server/plugins/engine/index.js +6 -0
  117. package/.server/server/plugins/engine/index.js.map +1 -0
  118. package/.server/server/plugins/engine/models/FormModel.js +363 -0
  119. package/.server/server/plugins/engine/models/FormModel.js.map +1 -0
  120. package/.server/server/plugins/engine/models/RepeatingSummaryViewModel.js +2 -0
  121. package/.server/server/plugins/engine/models/RepeatingSummaryViewModel.js.map +1 -0
  122. package/.server/server/plugins/engine/models/Section.js +2 -0
  123. package/.server/server/plugins/engine/models/Section.js.map +1 -0
  124. package/.server/server/plugins/engine/models/SummaryViewModel.js +183 -0
  125. package/.server/server/plugins/engine/models/SummaryViewModel.js.map +1 -0
  126. package/.server/server/plugins/engine/models/index.js +3 -0
  127. package/.server/server/plugins/engine/models/index.js.map +1 -0
  128. package/.server/server/plugins/engine/models/types.js +2 -0
  129. package/.server/server/plugins/engine/models/types.js.map +1 -0
  130. package/.server/server/plugins/engine/outputFormatters/human/v1.js +44 -0
  131. package/.server/server/plugins/engine/outputFormatters/human/v1.js.map +1 -0
  132. package/.server/server/plugins/engine/outputFormatters/index.js +24 -0
  133. package/.server/server/plugins/engine/outputFormatters/index.js.map +1 -0
  134. package/.server/server/plugins/engine/outputFormatters/machine/v1.js +104 -0
  135. package/.server/server/plugins/engine/outputFormatters/machine/v1.js.map +1 -0
  136. package/.server/server/plugins/engine/outputFormatters/machine/v2.js +110 -0
  137. package/.server/server/plugins/engine/outputFormatters/machine/v2.js.map +1 -0
  138. package/.server/server/plugins/engine/pageControllers/FileUploadPageController.js +401 -0
  139. package/.server/server/plugins/engine/pageControllers/FileUploadPageController.js.map +1 -0
  140. package/.server/server/plugins/engine/pageControllers/PageController.js +133 -0
  141. package/.server/server/plugins/engine/pageControllers/PageController.js.map +1 -0
  142. package/.server/server/plugins/engine/pageControllers/QuestionPageController.js +482 -0
  143. package/.server/server/plugins/engine/pageControllers/QuestionPageController.js.map +1 -0
  144. package/.server/server/plugins/engine/pageControllers/README.md +28 -0
  145. package/.server/server/plugins/engine/pageControllers/RepeatPageController.js +372 -0
  146. package/.server/server/plugins/engine/pageControllers/RepeatPageController.js.map +1 -0
  147. package/.server/server/plugins/engine/pageControllers/StartPageController.js +16 -0
  148. package/.server/server/plugins/engine/pageControllers/StartPageController.js.map +1 -0
  149. package/.server/server/plugins/engine/pageControllers/StatusPageController.js +43 -0
  150. package/.server/server/plugins/engine/pageControllers/StatusPageController.js.map +1 -0
  151. package/.server/server/plugins/engine/pageControllers/SummaryPageController.js +195 -0
  152. package/.server/server/plugins/engine/pageControllers/SummaryPageController.js.map +1 -0
  153. package/.server/server/plugins/engine/pageControllers/TerminalPageController.js +8 -0
  154. package/.server/server/plugins/engine/pageControllers/TerminalPageController.js.map +1 -0
  155. package/.server/server/plugins/engine/pageControllers/helpers.js +69 -0
  156. package/.server/server/plugins/engine/pageControllers/helpers.js.map +1 -0
  157. package/.server/server/plugins/engine/pageControllers/index.js +10 -0
  158. package/.server/server/plugins/engine/pageControllers/index.js.map +1 -0
  159. package/.server/server/plugins/engine/pageControllers/validationOptions.js +78 -0
  160. package/.server/server/plugins/engine/pageControllers/validationOptions.js.map +1 -0
  161. package/.server/server/plugins/engine/plugin.js +562 -0
  162. package/.server/server/plugins/engine/plugin.js.map +1 -0
  163. package/.server/server/plugins/engine/services/formSubmissionService.js +39 -0
  164. package/.server/server/plugins/engine/services/formSubmissionService.js.map +1 -0
  165. package/.server/server/plugins/engine/services/formsService.js +41 -0
  166. package/.server/server/plugins/engine/services/formsService.js.map +1 -0
  167. package/.server/server/plugins/engine/services/formsService.test.js +71 -0
  168. package/.server/server/plugins/engine/services/formsService.test.js.map +1 -0
  169. package/.server/server/plugins/engine/services/index.js +4 -0
  170. package/.server/server/plugins/engine/services/index.js.map +1 -0
  171. package/.server/server/plugins/engine/services/notifyService.js +45 -0
  172. package/.server/server/plugins/engine/services/notifyService.js.map +1 -0
  173. package/.server/server/plugins/engine/services/uploadService.js +51 -0
  174. package/.server/server/plugins/engine/services/uploadService.js.map +1 -0
  175. package/.server/server/plugins/engine/types.js +53 -0
  176. package/.server/server/plugins/engine/types.js.map +1 -0
  177. package/.server/server/plugins/engine/views/components/autocompletefield.html +5 -0
  178. package/.server/server/plugins/engine/views/components/checkboxesfield.html +5 -0
  179. package/.server/server/plugins/engine/views/components/datepartsfield.html +5 -0
  180. package/.server/server/plugins/engine/views/components/details.html +6 -0
  181. package/.server/server/plugins/engine/views/components/emailaddressfield.html +5 -0
  182. package/.server/server/plugins/engine/views/components/fileuploadfield-key.html +8 -0
  183. package/.server/server/plugins/engine/views/components/fileuploadfield-value.html +3 -0
  184. package/.server/server/plugins/engine/views/components/fileuploadfield.html +24 -0
  185. package/.server/server/plugins/engine/views/components/html.html +3 -0
  186. package/.server/server/plugins/engine/views/components/insettext.html +7 -0
  187. package/.server/server/plugins/engine/views/components/list.html +36 -0
  188. package/.server/server/plugins/engine/views/components/monthyearfield.html +5 -0
  189. package/.server/server/plugins/engine/views/components/multilinetextfield.html +10 -0
  190. package/.server/server/plugins/engine/views/components/numberfield.html +5 -0
  191. package/.server/server/plugins/engine/views/components/radiosfield.html +5 -0
  192. package/.server/server/plugins/engine/views/components/selectfield.html +5 -0
  193. package/.server/server/plugins/engine/views/components/telephonenumberfield.html +5 -0
  194. package/.server/server/plugins/engine/views/components/textfield.html +5 -0
  195. package/.server/server/plugins/engine/views/components/ukaddressfield.html +25 -0
  196. package/.server/server/plugins/engine/views/components/yesnofield.html +5 -0
  197. package/.server/server/plugins/engine/views/file-upload.html +45 -0
  198. package/.server/server/plugins/engine/views/index.html +39 -0
  199. package/.server/server/plugins/engine/views/item-delete.html +56 -0
  200. package/.server/server/plugins/engine/views/partials/components.html +6 -0
  201. package/.server/server/plugins/engine/views/partials/conditional-components.html +3 -0
  202. package/.server/server/plugins/engine/views/partials/debug.html +44 -0
  203. package/.server/server/plugins/engine/views/partials/form.html +15 -0
  204. package/.server/server/plugins/engine/views/partials/heading.html +16 -0
  205. package/.server/server/plugins/engine/views/partials/preview-banner.html +32 -0
  206. package/.server/server/plugins/engine/views/partials/preview-banner.test.js +108 -0
  207. package/.server/server/plugins/engine/views/partials/preview-banner.test.js.map +1 -0
  208. package/.server/server/plugins/engine/views/partials/warn-missing-notification-email.html +10 -0
  209. package/.server/server/plugins/engine/views/repeat-list-summary.html +53 -0
  210. package/.server/server/plugins/errorPages.js +49 -0
  211. package/.server/server/plugins/errorPages.js.map +1 -0
  212. package/.server/server/plugins/nunjucks/context.js +79 -0
  213. package/.server/server/plugins/nunjucks/context.js.map +1 -0
  214. package/.server/server/plugins/nunjucks/context.test.js +134 -0
  215. package/.server/server/plugins/nunjucks/context.test.js.map +1 -0
  216. package/.server/server/plugins/nunjucks/enviroment.test.js +166 -0
  217. package/.server/server/plugins/nunjucks/enviroment.test.js.map +1 -0
  218. package/.server/server/plugins/nunjucks/environment.js +95 -0
  219. package/.server/server/plugins/nunjucks/environment.js.map +1 -0
  220. package/.server/server/plugins/nunjucks/filters/answer.js +26 -0
  221. package/.server/server/plugins/nunjucks/filters/answer.js.map +1 -0
  222. package/.server/server/plugins/nunjucks/filters/answer.test.js +70 -0
  223. package/.server/server/plugins/nunjucks/filters/answer.test.js.map +1 -0
  224. package/.server/server/plugins/nunjucks/filters/evaluate.js +22 -0
  225. package/.server/server/plugins/nunjucks/filters/evaluate.js.map +1 -0
  226. package/.server/server/plugins/nunjucks/filters/field.js +26 -0
  227. package/.server/server/plugins/nunjucks/filters/field.js.map +1 -0
  228. package/.server/server/plugins/nunjucks/filters/field.test.js +64 -0
  229. package/.server/server/plugins/nunjucks/filters/field.test.js.map +1 -0
  230. package/.server/server/plugins/nunjucks/filters/highlight.js +14 -0
  231. package/.server/server/plugins/nunjucks/filters/highlight.js.map +1 -0
  232. package/.server/server/plugins/nunjucks/filters/href.js +28 -0
  233. package/.server/server/plugins/nunjucks/filters/href.js.map +1 -0
  234. package/.server/server/plugins/nunjucks/filters/href.test.js +63 -0
  235. package/.server/server/plugins/nunjucks/filters/href.test.js.map +1 -0
  236. package/.server/server/plugins/nunjucks/filters/index.js +9 -0
  237. package/.server/server/plugins/nunjucks/filters/index.js.map +1 -0
  238. package/.server/server/plugins/nunjucks/filters/inspect.js +16 -0
  239. package/.server/server/plugins/nunjucks/filters/inspect.js.map +1 -0
  240. package/.server/server/plugins/nunjucks/filters/page.js +23 -0
  241. package/.server/server/plugins/nunjucks/filters/page.js.map +1 -0
  242. package/.server/server/plugins/nunjucks/filters/page.test.js +51 -0
  243. package/.server/server/plugins/nunjucks/filters/page.test.js.map +1 -0
  244. package/.server/server/plugins/nunjucks/index.js +4 -0
  245. package/.server/server/plugins/nunjucks/index.js.map +1 -0
  246. package/.server/server/plugins/nunjucks/plugin.js +38 -0
  247. package/.server/server/plugins/nunjucks/plugin.js.map +1 -0
  248. package/.server/server/plugins/nunjucks/render.js +41 -0
  249. package/.server/server/plugins/nunjucks/render.js.map +1 -0
  250. package/.server/server/plugins/nunjucks/types.js +41 -0
  251. package/.server/server/plugins/nunjucks/types.js.map +1 -0
  252. package/.server/server/plugins/pulse.js +8 -0
  253. package/.server/server/plugins/pulse.js.map +1 -0
  254. package/.server/server/plugins/router.js +169 -0
  255. package/.server/server/plugins/router.js.map +1 -0
  256. package/.server/server/plugins/session.js +28 -0
  257. package/.server/server/plugins/session.js.map +1 -0
  258. package/.server/server/routes/health.js +15 -0
  259. package/.server/server/routes/health.js.map +1 -0
  260. package/.server/server/routes/health.test.js +32 -0
  261. package/.server/server/routes/health.test.js.map +1 -0
  262. package/.server/server/routes/index.js +3 -0
  263. package/.server/server/routes/index.js.map +1 -0
  264. package/.server/server/routes/public.js +37 -0
  265. package/.server/server/routes/public.js.map +1 -0
  266. package/.server/server/routes/types.js +14 -0
  267. package/.server/server/routes/types.js.map +1 -0
  268. package/.server/server/schemas/index.js +15 -0
  269. package/.server/server/schemas/index.js.map +1 -0
  270. package/.server/server/secure-context.js +36 -0
  271. package/.server/server/secure-context.js.map +1 -0
  272. package/.server/server/services/cacheService.js +85 -0
  273. package/.server/server/services/cacheService.js.map +1 -0
  274. package/.server/server/services/httpService.js +44 -0
  275. package/.server/server/services/httpService.js.map +1 -0
  276. package/.server/server/services/httpService.test.js +504 -0
  277. package/.server/server/services/httpService.test.js.map +1 -0
  278. package/.server/server/services/index.js +2 -0
  279. package/.server/server/services/index.js.map +1 -0
  280. package/.server/server/types.js +2 -0
  281. package/.server/server/types.js.map +1 -0
  282. package/.server/server/utils/notify.js +40 -0
  283. package/.server/server/utils/notify.js.map +1 -0
  284. package/.server/server/utils/secure-context/get-trust-store-certs.js +7 -0
  285. package/.server/server/utils/secure-context/get-trust-store-certs.js.map +1 -0
  286. package/.server/server/utils/secure-context/get-trust-store-certs.test.js +14 -0
  287. package/.server/server/utils/secure-context/get-trust-store-certs.test.js.map +1 -0
  288. package/.server/server/utils/utils.js +20 -0
  289. package/.server/server/utils/utils.js.map +1 -0
  290. package/.server/server/utils/utils.test.js +49 -0
  291. package/.server/server/utils/utils.test.js.map +1 -0
  292. package/.server/server/views/404.html +16 -0
  293. package/.server/server/views/500.html +19 -0
  294. package/.server/server/views/components/debug/macro.njk +3 -0
  295. package/.server/server/views/components/debug/template.njk +13 -0
  296. package/.server/server/views/components/service-banner/macro.njk +3 -0
  297. package/.server/server/views/components/service-banner/template.njk +20 -0
  298. package/.server/server/views/components/service-banner/template.test.js +36 -0
  299. package/.server/server/views/components/service-banner/template.test.js.map +1 -0
  300. package/.server/server/views/components/tag-env/macro.njk +3 -0
  301. package/.server/server/views/components/tag-env/template.njk +30 -0
  302. package/.server/server/views/components/tag-env/template.test.js +59 -0
  303. package/.server/server/views/components/tag-env/template.test.js.map +1 -0
  304. package/.server/server/views/confirmation.html +19 -0
  305. package/.server/server/views/help/accessibility-statement.html +58 -0
  306. package/.server/server/views/help/cookie-preferences.html +57 -0
  307. package/.server/server/views/help/cookies.html +71 -0
  308. package/.server/server/views/help/get-support.html +37 -0
  309. package/.server/server/views/help/privacy-notice.html +68 -0
  310. package/.server/server/views/help/terms-and-conditions.html +83 -0
  311. package/.server/server/views/layout.html +199 -0
  312. package/.server/server/views/summary.html +50 -0
  313. package/.server/typings/hapi/index.d.js +2 -0
  314. package/.server/typings/hapi/index.d.js.map +1 -0
  315. package/.server/typings/hapi-tracing/index.d.js +2 -0
  316. package/.server/typings/hapi-tracing/index.d.js.map +1 -0
  317. package/.server/typings/index.d.js +2 -0
  318. package/.server/typings/index.d.js.map +1 -0
  319. package/.server/typings/joi/index.d.js +2 -0
  320. package/.server/typings/joi/index.d.js.map +1 -0
  321. package/Dockerfile +61 -0
  322. package/LICENCE +8 -0
  323. package/Procfile +1 -0
  324. package/README.md +87 -0
  325. package/babel.config.cjs +55 -0
  326. package/compose/aws.env +4 -0
  327. package/compose/start-localstack.sh +26 -0
  328. package/docker-compose.yaml +86 -0
  329. package/globals.d.ts +1 -0
  330. package/jest.config.cjs +54 -0
  331. package/jest.environment.js +4 -0
  332. package/jest.setup.cjs +14 -0
  333. package/package.json +163 -0
  334. package/postcss.config.js +26 -0
  335. package/sonar-project.properties +17 -0
  336. package/src/client/javascripts/application.js +87 -0
  337. package/src/client/javascripts/file-upload.js +386 -0
  338. package/src/client/stylesheets/_code.scss +33 -0
  339. package/src/client/stylesheets/_govuk-frontend.scss +4 -0
  340. package/src/client/stylesheets/_prose.scss +56 -0
  341. package/src/client/stylesheets/_service-banner.scss +24 -0
  342. package/src/client/stylesheets/_summary-list.scss +28 -0
  343. package/src/client/stylesheets/_tag-env.scss +24 -0
  344. package/src/client/stylesheets/application.scss +14 -0
  345. package/src/common/cookies.js +58 -0
  346. package/src/common/cookies.test.js +23 -0
  347. package/src/common/types.js +5 -0
  348. package/src/config/index.ts +271 -0
  349. package/src/index.ts +31 -0
  350. package/src/server/common/helpers/logging/logger-options.test.ts +50 -0
  351. package/src/server/common/helpers/logging/logger-options.ts +46 -0
  352. package/src/server/common/helpers/logging/logger.ts +7 -0
  353. package/src/server/common/helpers/logging/request-logger.ts +9 -0
  354. package/src/server/common/helpers/logging/request-tracing.js +10 -0
  355. package/src/server/common/helpers/redis-client.js +70 -0
  356. package/src/server/constants.js +1 -0
  357. package/src/server/forms/README.md +10 -0
  358. package/src/server/forms/components.json +1015 -0
  359. package/src/server/forms/report-a-terrorist.json +270 -0
  360. package/src/server/forms/runner-components-test.json +365 -0
  361. package/src/server/forms/test.json +581 -0
  362. package/src/server/index.test.ts +582 -0
  363. package/src/server/index.ts +140 -0
  364. package/src/server/plugins/blankie.test.ts +73 -0
  365. package/src/server/plugins/blankie.ts +48 -0
  366. package/src/server/plugins/crumb.ts +20 -0
  367. package/src/server/plugins/engine/README.md +87 -0
  368. package/src/server/plugins/engine/components/AutocompleteField.test.ts +294 -0
  369. package/src/server/plugins/engine/components/AutocompleteField.ts +49 -0
  370. package/src/server/plugins/engine/components/CheckboxesField.test.ts +379 -0
  371. package/src/server/plugins/engine/components/CheckboxesField.ts +106 -0
  372. package/src/server/plugins/engine/components/ComponentBase.ts +97 -0
  373. package/src/server/plugins/engine/components/ComponentCollection.ts +278 -0
  374. package/src/server/plugins/engine/components/DatePartsField.test.ts +822 -0
  375. package/src/server/plugins/engine/components/DatePartsField.ts +264 -0
  376. package/src/server/plugins/engine/components/Details.test.ts +49 -0
  377. package/src/server/plugins/engine/components/Details.ts +30 -0
  378. package/src/server/plugins/engine/components/EmailAddressField.test.ts +395 -0
  379. package/src/server/plugins/engine/components/EmailAddressField.ts +55 -0
  380. package/src/server/plugins/engine/components/FileUploadField.test.ts +778 -0
  381. package/src/server/plugins/engine/components/FileUploadField.ts +262 -0
  382. package/src/server/plugins/engine/components/FormComponent.ts +249 -0
  383. package/src/server/plugins/engine/components/Html.test.ts +48 -0
  384. package/src/server/plugins/engine/components/Html.ts +29 -0
  385. package/src/server/plugins/engine/components/InsetText.test.ts +48 -0
  386. package/src/server/plugins/engine/components/InsetText.ts +27 -0
  387. package/src/server/plugins/engine/components/List.test.ts +76 -0
  388. package/src/server/plugins/engine/components/List.ts +72 -0
  389. package/src/server/plugins/engine/components/ListFormComponent.ts +140 -0
  390. package/src/server/plugins/engine/components/MonthYearField.test.ts +567 -0
  391. package/src/server/plugins/engine/components/MonthYearField.ts +222 -0
  392. package/src/server/plugins/engine/components/MultilineTextField.test.ts +558 -0
  393. package/src/server/plugins/engine/components/MultilineTextField.ts +138 -0
  394. package/src/server/plugins/engine/components/NumberField.test.ts +701 -0
  395. package/src/server/plugins/engine/components/NumberField.ts +163 -0
  396. package/src/server/plugins/engine/components/RadiosField.test.ts +288 -0
  397. package/src/server/plugins/engine/components/RadiosField.ts +24 -0
  398. package/src/server/plugins/engine/components/SelectField.test.ts +288 -0
  399. package/src/server/plugins/engine/components/SelectField.ts +47 -0
  400. package/src/server/plugins/engine/components/SelectionControlField.ts +43 -0
  401. package/src/server/plugins/engine/components/TelephoneNumberField.test.ts +356 -0
  402. package/src/server/plugins/engine/components/TelephoneNumberField.ts +67 -0
  403. package/src/server/plugins/engine/components/TextField.test.ts +489 -0
  404. package/src/server/plugins/engine/components/TextField.ts +96 -0
  405. package/src/server/plugins/engine/components/UkAddressField.test.ts +623 -0
  406. package/src/server/plugins/engine/components/UkAddressField.ts +172 -0
  407. package/src/server/plugins/engine/components/YesNoField.test.ts +248 -0
  408. package/src/server/plugins/engine/components/YesNoField.ts +31 -0
  409. package/src/server/plugins/engine/components/constants.ts +1 -0
  410. package/src/server/plugins/engine/components/helpers.ts +330 -0
  411. package/src/server/plugins/engine/components/index.ts +24 -0
  412. package/src/server/plugins/engine/components/types.ts +117 -0
  413. package/src/server/plugins/engine/configureEnginePlugin.ts +47 -0
  414. package/src/server/plugins/engine/helpers.test.ts +791 -0
  415. package/src/server/plugins/engine/helpers.ts +379 -0
  416. package/src/server/plugins/engine/index.ts +7 -0
  417. package/src/server/plugins/engine/models/FormModel.test.ts +42 -0
  418. package/src/server/plugins/engine/models/FormModel.ts +443 -0
  419. package/src/server/plugins/engine/models/RepeatingSummaryViewModel.ts +0 -0
  420. package/src/server/plugins/engine/models/Section.ts +0 -0
  421. package/src/server/plugins/engine/models/SummaryViewModel.test.ts +209 -0
  422. package/src/server/plugins/engine/models/SummaryViewModel.ts +220 -0
  423. package/src/server/plugins/engine/models/index.ts +2 -0
  424. package/src/server/plugins/engine/models/types.ts +114 -0
  425. package/src/server/plugins/engine/outputFormatters/human/v1.test.ts +143 -0
  426. package/src/server/plugins/engine/outputFormatters/human/v1.ts +73 -0
  427. package/src/server/plugins/engine/outputFormatters/index.test.ts +17 -0
  428. package/src/server/plugins/engine/outputFormatters/index.ts +44 -0
  429. package/src/server/plugins/engine/outputFormatters/machine/v1.test.ts +229 -0
  430. package/src/server/plugins/engine/outputFormatters/machine/v1.ts +140 -0
  431. package/src/server/plugins/engine/outputFormatters/machine/v2.test.ts +229 -0
  432. package/src/server/plugins/engine/outputFormatters/machine/v2.ts +153 -0
  433. package/src/server/plugins/engine/pageControllers/FileUploadPageController.test.ts +1108 -0
  434. package/src/server/plugins/engine/pageControllers/FileUploadPageController.ts +446 -0
  435. package/src/server/plugins/engine/pageControllers/PageController.test.ts +205 -0
  436. package/src/server/plugins/engine/pageControllers/PageController.ts +176 -0
  437. package/src/server/plugins/engine/pageControllers/QuestionPageController.test.ts +1264 -0
  438. package/src/server/plugins/engine/pageControllers/QuestionPageController.ts +561 -0
  439. package/src/server/plugins/engine/pageControllers/README.md +28 -0
  440. package/src/server/plugins/engine/pageControllers/RepeatPageController.test.ts +264 -0
  441. package/src/server/plugins/engine/pageControllers/RepeatPageController.ts +458 -0
  442. package/src/server/plugins/engine/pageControllers/StartPageController.ts +18 -0
  443. package/src/server/plugins/engine/pageControllers/StatusPageController.ts +50 -0
  444. package/src/server/plugins/engine/pageControllers/SummaryPageController.ts +261 -0
  445. package/src/server/plugins/engine/pageControllers/TerminalController.test.ts +28 -0
  446. package/src/server/plugins/engine/pageControllers/TerminalPageController.ts +19 -0
  447. package/src/server/plugins/engine/pageControllers/helpers.test.ts +198 -0
  448. package/src/server/plugins/engine/pageControllers/helpers.ts +101 -0
  449. package/src/server/plugins/engine/pageControllers/index.ts +10 -0
  450. package/src/server/plugins/engine/pageControllers/validationOptions.ts +89 -0
  451. package/src/server/plugins/engine/plugin.ts +673 -0
  452. package/src/server/plugins/engine/services/formSubmissionService.js +46 -0
  453. package/src/server/plugins/engine/services/formsService.js +46 -0
  454. package/src/server/plugins/engine/services/formsService.test.js +90 -0
  455. package/src/server/plugins/engine/services/index.js +3 -0
  456. package/src/server/plugins/engine/services/notifyService.test.ts +132 -0
  457. package/src/server/plugins/engine/services/notifyService.ts +64 -0
  458. package/src/server/plugins/engine/services/uploadService.js +60 -0
  459. package/src/server/plugins/engine/types.ts +315 -0
  460. package/src/server/plugins/engine/views/components/autocompletefield.html +5 -0
  461. package/src/server/plugins/engine/views/components/checkboxesfield.html +5 -0
  462. package/src/server/plugins/engine/views/components/datepartsfield.html +5 -0
  463. package/src/server/plugins/engine/views/components/details.html +6 -0
  464. package/src/server/plugins/engine/views/components/emailaddressfield.html +5 -0
  465. package/src/server/plugins/engine/views/components/fileuploadfield-key.html +8 -0
  466. package/src/server/plugins/engine/views/components/fileuploadfield-value.html +3 -0
  467. package/src/server/plugins/engine/views/components/fileuploadfield.html +24 -0
  468. package/src/server/plugins/engine/views/components/html.html +3 -0
  469. package/src/server/plugins/engine/views/components/insettext.html +7 -0
  470. package/src/server/plugins/engine/views/components/list.html +36 -0
  471. package/src/server/plugins/engine/views/components/monthyearfield.html +5 -0
  472. package/src/server/plugins/engine/views/components/multilinetextfield.html +10 -0
  473. package/src/server/plugins/engine/views/components/numberfield.html +5 -0
  474. package/src/server/plugins/engine/views/components/radiosfield.html +5 -0
  475. package/src/server/plugins/engine/views/components/selectfield.html +5 -0
  476. package/src/server/plugins/engine/views/components/telephonenumberfield.html +5 -0
  477. package/src/server/plugins/engine/views/components/textfield.html +5 -0
  478. package/src/server/plugins/engine/views/components/ukaddressfield.html +25 -0
  479. package/src/server/plugins/engine/views/components/yesnofield.html +5 -0
  480. package/src/server/plugins/engine/views/file-upload.html +45 -0
  481. package/src/server/plugins/engine/views/index.html +39 -0
  482. package/src/server/plugins/engine/views/item-delete.html +56 -0
  483. package/src/server/plugins/engine/views/partials/components.html +6 -0
  484. package/src/server/plugins/engine/views/partials/conditional-components.html +3 -0
  485. package/src/server/plugins/engine/views/partials/debug.html +44 -0
  486. package/src/server/plugins/engine/views/partials/form.html +15 -0
  487. package/src/server/plugins/engine/views/partials/heading.html +16 -0
  488. package/src/server/plugins/engine/views/partials/preview-banner.html +32 -0
  489. package/src/server/plugins/engine/views/partials/preview-banner.test.js +122 -0
  490. package/src/server/plugins/engine/views/partials/warn-missing-notification-email.html +10 -0
  491. package/src/server/plugins/engine/views/repeat-list-summary.html +53 -0
  492. package/src/server/plugins/errorPages.ts +58 -0
  493. package/src/server/plugins/nunjucks/context.js +88 -0
  494. package/src/server/plugins/nunjucks/context.test.js +142 -0
  495. package/src/server/plugins/nunjucks/enviroment.test.js +201 -0
  496. package/src/server/plugins/nunjucks/environment.js +116 -0
  497. package/src/server/plugins/nunjucks/filters/answer.js +27 -0
  498. package/src/server/plugins/nunjucks/filters/answer.test.js +89 -0
  499. package/src/server/plugins/nunjucks/filters/evaluate.js +21 -0
  500. package/src/server/plugins/nunjucks/filters/field.js +28 -0
  501. package/src/server/plugins/nunjucks/filters/field.test.js +75 -0
  502. package/src/server/plugins/nunjucks/filters/highlight.js +11 -0
  503. package/src/server/plugins/nunjucks/filters/href.js +30 -0
  504. package/src/server/plugins/nunjucks/filters/href.test.js +80 -0
  505. package/src/server/plugins/nunjucks/filters/index.js +8 -0
  506. package/src/server/plugins/nunjucks/filters/inspect.js +15 -0
  507. package/src/server/plugins/nunjucks/filters/page.js +24 -0
  508. package/src/server/plugins/nunjucks/filters/page.test.js +65 -0
  509. package/src/server/plugins/nunjucks/index.js +3 -0
  510. package/src/server/plugins/nunjucks/plugin.js +40 -0
  511. package/src/server/plugins/nunjucks/render.js +42 -0
  512. package/src/server/plugins/nunjucks/types.js +40 -0
  513. package/src/server/plugins/pulse.ts +11 -0
  514. package/src/server/plugins/router.ts +201 -0
  515. package/src/server/plugins/session.ts +28 -0
  516. package/src/server/routes/health.js +13 -0
  517. package/src/server/routes/health.test.js +35 -0
  518. package/src/server/routes/index.test.ts +125 -0
  519. package/src/server/routes/index.ts +2 -0
  520. package/src/server/routes/public.ts +47 -0
  521. package/src/server/routes/types.ts +48 -0
  522. package/src/server/schemas/index.ts +34 -0
  523. package/src/server/secure-context.js +43 -0
  524. package/src/server/services/cacheService.test.ts +276 -0
  525. package/src/server/services/cacheService.ts +131 -0
  526. package/src/server/services/httpService.test.js +491 -0
  527. package/src/server/services/httpService.ts +50 -0
  528. package/src/server/services/index.ts +1 -0
  529. package/src/server/types.ts +54 -0
  530. package/src/server/utils/notify.test.ts +37 -0
  531. package/src/server/utils/notify.ts +50 -0
  532. package/src/server/utils/secure-context/get-trust-store-certs.js +11 -0
  533. package/src/server/utils/secure-context/get-trust-store-certs.test.js +19 -0
  534. package/src/server/utils/utils.js +24 -0
  535. package/src/server/utils/utils.test.js +54 -0
  536. package/src/server/views/404.html +16 -0
  537. package/src/server/views/500.html +19 -0
  538. package/src/server/views/components/debug/macro.njk +3 -0
  539. package/src/server/views/components/debug/template.njk +13 -0
  540. package/src/server/views/components/service-banner/macro.njk +3 -0
  541. package/src/server/views/components/service-banner/template.njk +20 -0
  542. package/src/server/views/components/service-banner/template.test.js +43 -0
  543. package/src/server/views/components/tag-env/macro.njk +3 -0
  544. package/src/server/views/components/tag-env/template.njk +30 -0
  545. package/src/server/views/components/tag-env/template.test.js +66 -0
  546. package/src/server/views/confirmation.html +19 -0
  547. package/src/server/views/help/accessibility-statement.html +58 -0
  548. package/src/server/views/help/cookie-preferences.html +57 -0
  549. package/src/server/views/help/cookies.html +71 -0
  550. package/src/server/views/help/get-support.html +37 -0
  551. package/src/server/views/help/privacy-notice.html +68 -0
  552. package/src/server/views/help/terms-and-conditions.html +83 -0
  553. package/src/server/views/layout.html +199 -0
  554. package/src/server/views/summary.html +50 -0
  555. package/src/typings/hapi/index.d.ts +95 -0
  556. package/src/typings/hapi-tracing/index.d.ts +6 -0
  557. package/src/typings/index.d.ts +3 -0
  558. package/src/typings/joi/index.d.ts +22 -0
  559. package/stylelint.config.js +10 -0
  560. package/test/client/javascripts/file-upload.test.js +1197 -0
  561. package/test/condition/checkboxes.test.js +112 -0
  562. package/test/condition/radios.test.js +112 -0
  563. package/test/condition/text.test.js +103 -0
  564. package/test/fixtures/assets-manifest.json +4 -0
  565. package/test/fixtures/form.js +86 -0
  566. package/test/fixtures/index.js +2 -0
  567. package/test/fixtures/list.js +92 -0
  568. package/test/form/cookies.test.js +338 -0
  569. package/test/form/csrf.test.js +87 -0
  570. package/test/form/definitions/basic.js +101 -0
  571. package/test/form/definitions/blank.js +10 -0
  572. package/test/form/definitions/checkboxes.json +88 -0
  573. package/test/form/definitions/components.json +452 -0
  574. package/test/form/definitions/conditional-reveal.js +140 -0
  575. package/test/form/definitions/conditions-basic.js +187 -0
  576. package/test/form/definitions/conditions-complex.js +338 -0
  577. package/test/form/definitions/conditions-dates.js +78 -0
  578. package/test/form/definitions/conditions-escaping.js +143 -0
  579. package/test/form/definitions/demo-cph-number.js +3099 -0
  580. package/test/form/definitions/feedback.json +45 -0
  581. package/test/form/definitions/fields-optional.js +402 -0
  582. package/test/form/definitions/fields-required.js +402 -0
  583. package/test/form/definitions/file-upload-basic.js +44 -0
  584. package/test/form/definitions/file-upload.js +66 -0
  585. package/test/form/definitions/minimal.js +39 -0
  586. package/test/form/definitions/phase-alpha.json +33 -0
  587. package/test/form/definitions/phase-default.json +26 -0
  588. package/test/form/definitions/radios.json +88 -0
  589. package/test/form/definitions/repeat-mixed.js +54 -0
  590. package/test/form/definitions/repeat.js +70 -0
  591. package/test/form/definitions/status.json +126 -0
  592. package/test/form/definitions/templates.js +183 -0
  593. package/test/form/definitions/test.json +581 -0
  594. package/test/form/definitions/text.json +75 -0
  595. package/test/form/definitions/titles.json +170 -0
  596. package/test/form/definitions.test.js +47 -0
  597. package/test/form/exit-page.test.js +210 -0
  598. package/test/form/feedback.test.js +68 -0
  599. package/test/form/fields-optional.test.js +237 -0
  600. package/test/form/fields-required.test.js +294 -0
  601. package/test/form/file-upload.test.js +313 -0
  602. package/test/form/govuk-notify.test.js +449 -0
  603. package/test/form/journey-basic.test.js +444 -0
  604. package/test/form/persist-files.test.js +227 -0
  605. package/test/form/phase-banner.test.js +71 -0
  606. package/test/form/repeat.test.js +628 -0
  607. package/test/form/summary-submission-email.test.js +95 -0
  608. package/test/form/template.test.js +288 -0
  609. package/test/form/titles.test.js +204 -0
  610. package/test/helpers/component-helpers.js +74 -0
  611. package/test/utils/get-cookie.js +42 -0
  612. package/test/utils/get-form-definitions.js +18 -0
  613. package/tmp.pdf +1 -0
  614. package/tsconfig.json +28 -0
  615. package/webpack.config.js +208 -0
@@ -0,0 +1,791 @@
1
+ import Boom from '@hapi/boom'
2
+ import { type ResponseObject, type ResponseToolkit } from '@hapi/hapi'
3
+ import { StatusCodes } from 'http-status-codes'
4
+ import { ValidationError } from 'joi'
5
+
6
+ import { PREVIEW_PATH_PREFIX } from '~/src/server/constants.js'
7
+ import {
8
+ checkEmailAddressForLiveFormSubmission,
9
+ checkFormStatus,
10
+ encodeUrl,
11
+ engine,
12
+ evaluateTemplate,
13
+ getErrors,
14
+ getExponentialBackoffDelay,
15
+ getPageHref,
16
+ proceed,
17
+ safeGenerateCrumb,
18
+ type GlobalScope
19
+ } from '~/src/server/plugins/engine/helpers.js'
20
+ import { FormModel } from '~/src/server/plugins/engine/models/FormModel.js'
21
+ import {
22
+ createPage,
23
+ type PageControllerClass
24
+ } from '~/src/server/plugins/engine/pageControllers/helpers.js'
25
+ import {
26
+ type FormContext,
27
+ type FormContextRequest
28
+ } from '~/src/server/plugins/engine/types.js'
29
+ import {
30
+ FormAction,
31
+ FormStatus,
32
+ type FormRequest
33
+ } from '~/src/server/routes/types.js'
34
+ import definition from '~/test/form/definitions/basic.js'
35
+ import templateDefinition from '~/test/form/definitions/templates.js'
36
+
37
+ interface NunjucksContext {
38
+ context: {
39
+ globals: GlobalScope
40
+ }
41
+ }
42
+
43
+ type EvaluateFilter = (this: NunjucksContext, template: unknown) => unknown
44
+ type HrefFilter = (this: NunjucksContext, path: string) => string | undefined
45
+
46
+ describe('Helpers', () => {
47
+ let page: PageControllerClass
48
+ let request: FormContextRequest
49
+ let h: Pick<ResponseToolkit, 'redirect' | 'view'>
50
+
51
+ beforeEach(() => {
52
+ const model = new FormModel(definition, {
53
+ basePath: 'test'
54
+ })
55
+
56
+ page = createPage(model, definition.pages[0])
57
+ const pageUrl = new URL(page.href, 'http://example.com')
58
+
59
+ request = {
60
+ method: 'get',
61
+ url: pageUrl,
62
+ path: pageUrl.pathname,
63
+ params: {
64
+ path: 'licence',
65
+ slug: 'test'
66
+ },
67
+ query: {},
68
+ app: { model }
69
+ }
70
+
71
+ const response = {
72
+ code: jest.fn().mockImplementation(() => response)
73
+ }
74
+
75
+ h = {
76
+ redirect: jest.fn().mockImplementation(() => response),
77
+ view: jest.fn()
78
+ }
79
+ })
80
+
81
+ describe('proceed', () => {
82
+ it.each([
83
+ {
84
+ href: 'https://www.gov.uk/help/privacy-notice',
85
+
86
+ request: {
87
+ method: 'get'
88
+ } satisfies Partial<FormContextRequest>,
89
+
90
+ redirect: {
91
+ statusCode: StatusCodes.MOVED_TEMPORARILY
92
+ } satisfies Partial<ResponseObject>
93
+ },
94
+ {
95
+ href: '/test/full-name',
96
+
97
+ request: {
98
+ method: 'get'
99
+ } satisfies Partial<FormContextRequest>,
100
+
101
+ redirect: {
102
+ statusCode: StatusCodes.MOVED_TEMPORARILY
103
+ } satisfies Partial<ResponseObject>
104
+ },
105
+ {
106
+ href: '/test/full-name',
107
+
108
+ request: {
109
+ method: 'post',
110
+ payload: {
111
+ action: FormAction.Validate
112
+ }
113
+ } satisfies Partial<FormContextRequest>,
114
+
115
+ redirect: {
116
+ statusCode: StatusCodes.SEE_OTHER
117
+ } satisfies Partial<ResponseObject>
118
+ }
119
+ ])(
120
+ 'should redirect to the path provided',
121
+ ({ href, redirect, ...options }) => {
122
+ request = { ...request, ...options.request }
123
+
124
+ const response = proceed(request, h, href)
125
+
126
+ expect(h.view).not.toHaveBeenCalled()
127
+ expect(h.redirect).toHaveBeenCalledWith(href)
128
+ expect(response.code).toHaveBeenCalledWith(redirect.statusCode)
129
+ }
130
+ )
131
+
132
+ it.each([
133
+ {
134
+ href: '/test/full-name',
135
+
136
+ request: {
137
+ method: 'post',
138
+ payload: {
139
+ action: FormAction.Validate
140
+ },
141
+ query: {
142
+ myParam1: 'myValue1',
143
+ myParam2: 'myValue2',
144
+ returnUrl: '/test/summary'
145
+ }
146
+ } satisfies Partial<FormContextRequest>,
147
+
148
+ redirect: {
149
+ statusCode: StatusCodes.SEE_OTHER
150
+ } satisfies Partial<ResponseObject>
151
+ }
152
+ ])(
153
+ "should redirect to the 'returnUrl' query param provided (relative paths)",
154
+ ({ href, redirect, ...options }) => {
155
+ request = { ...request, ...options.request }
156
+
157
+ const response = proceed(request, h, href)
158
+
159
+ expect(h.view).not.toHaveBeenCalled()
160
+ expect(h.redirect).toHaveBeenCalledWith(request.query.returnUrl)
161
+ expect(response.code).toHaveBeenCalledWith(redirect.statusCode)
162
+ }
163
+ )
164
+
165
+ it.each([
166
+ {
167
+ href: '/test/full-name',
168
+
169
+ request: {
170
+ method: 'get',
171
+ query: { returnUrl: 'slash-missing' }
172
+ } satisfies Partial<FormContextRequest>,
173
+
174
+ redirect: {
175
+ statusCode: StatusCodes.MOVED_TEMPORARILY
176
+ } satisfies Partial<ResponseObject>
177
+ },
178
+ {
179
+ href: '/test/full-name',
180
+
181
+ request: {
182
+ method: 'post',
183
+ payload: {
184
+ action: FormAction.Validate
185
+ },
186
+ query: { returnUrl: 'https://www.gov.uk/help/privacy-notice' }
187
+ } satisfies Partial<FormContextRequest>,
188
+
189
+ redirect: {
190
+ statusCode: StatusCodes.SEE_OTHER
191
+ } satisfies Partial<ResponseObject>
192
+ },
193
+ {
194
+ href: '/test/repeater/example',
195
+
196
+ request: {
197
+ method: 'post',
198
+ query: {
199
+ myParam1: 'myValue1',
200
+ myParam2: 'myValue2',
201
+ returnUrl: '/test/repeater/summary'
202
+ },
203
+ payload: {
204
+ action: FormAction.AddAnother
205
+ }
206
+ } satisfies Partial<FormContextRequest>,
207
+
208
+ redirect: {
209
+ statusCode: StatusCodes.MOVED_TEMPORARILY
210
+ } satisfies Partial<ResponseObject>
211
+ }
212
+ ])(
213
+ "should not redirect to the 'returnUrl' query param provided (other paths)",
214
+ ({ href, ...options }) => {
215
+ request = { ...request, ...options.request }
216
+
217
+ proceed(request, h, href)
218
+ expect(h.redirect).not.toHaveBeenCalledWith(request.query.returnUrl)
219
+ }
220
+ )
221
+ })
222
+
223
+ describe('encodeUrl', () => {
224
+ it.each([
225
+ {
226
+ input: 'http://example.com?myParam=has spaces&more£',
227
+ output: 'http://example.com/?myParam=has%20spaces&more%C2%A3'
228
+ },
229
+ {
230
+ input: 'mailto:hello@example.com?subject=has spaces&body=more£',
231
+ output: 'mailto:hello@example.com?subject=has%20spaces&body=more%C2%A3'
232
+ }
233
+ ])('should percent encode parameters', ({ input, output }) => {
234
+ const returned = encodeUrl(input)
235
+ expect(returned).toBe(output)
236
+ })
237
+
238
+ it('should return undefined when no url is provided', () => {
239
+ const returned = encodeUrl()
240
+ expect(returned).toBeUndefined()
241
+ })
242
+
243
+ it('should throw when invalid url is provided', () => {
244
+ expect(() => encodeUrl('not a url')).toThrow()
245
+ })
246
+ })
247
+
248
+ describe('getPageHref', () => {
249
+ it('should return page href', () => {
250
+ const returned = getPageHref(page)
251
+ expect(returned).toEqual(page.href)
252
+ })
253
+
254
+ it('should return page href (path override)', () => {
255
+ const nextPath = '/badgers/monkeys'
256
+ const nextHref = '/test/badgers/monkeys'
257
+
258
+ const returned = getPageHref(page, nextPath)
259
+ expect(returned).toEqual(nextHref)
260
+ })
261
+
262
+ it('should return page href without query params', () => {
263
+ request.query.myParam = 'myValue'
264
+ request.query.myParam2 = 'myValue2'
265
+
266
+ const returned = getPageHref(page)
267
+ expect(returned).toEqual(page.href)
268
+ })
269
+
270
+ it('should return page href (path override) without query params', () => {
271
+ request.query.myParam = 'myValue'
272
+ request.query.myParam2 = 'myValue2'
273
+
274
+ const nextPath = '/badgers/monkeys'
275
+ const nextHref = '/test/badgers/monkeys'
276
+
277
+ const returned = getPageHref(page, nextPath)
278
+ expect(returned).toEqual(nextHref)
279
+ })
280
+
281
+ it('should return page href with new query params', () => {
282
+ const returned = getPageHref(page, {
283
+ returnUrl: page.getSummaryPath(),
284
+ badger: 'monkeys'
285
+ })
286
+
287
+ expect(returned).toBe(
288
+ `${page.href}?returnUrl=${encodeURIComponent('/summary')}&badger=monkeys`
289
+ )
290
+ })
291
+
292
+ it('should return page href (path override) with new query params', () => {
293
+ const nextPath = '/badgers/monkeys'
294
+ const nextHref = '/test/badgers/monkeys'
295
+
296
+ const returned = getPageHref(page, nextPath, {
297
+ returnUrl: page.getSummaryPath(),
298
+ badger: 'monkeys'
299
+ })
300
+
301
+ expect(returned).toBe(
302
+ `${nextHref}?returnUrl=${encodeURIComponent('/summary')}&badger=monkeys`
303
+ )
304
+ })
305
+
306
+ it('should throw when absolute URL is provided', () => {
307
+ expect(() =>
308
+ getPageHref(page, 'https://www.gov.uk/help/privacy-notice')
309
+ ).toThrow('Only relative URLs are allowed')
310
+ })
311
+ })
312
+
313
+ describe('checkFormStatus', () => {
314
+ it('should return true/live for paths starting with PREVIEW_PATH_PREFIX and form is live', () => {
315
+ const path = `${PREVIEW_PATH_PREFIX}/live/another/segment`
316
+ expect(checkFormStatus(path)).toStrictEqual({
317
+ state: FormStatus.Live,
318
+ isPreview: true
319
+ })
320
+ })
321
+
322
+ it('should return false for paths not starting with PREVIEW_PATH_PREFIX', () => {
323
+ const path = '/some/other/path'
324
+ expect(checkFormStatus(path)).toStrictEqual({
325
+ state: FormStatus.Live,
326
+ isPreview: false
327
+ })
328
+ })
329
+
330
+ it('should be case insensitive and return draft when form is draft', () => {
331
+ const path = `${PREVIEW_PATH_PREFIX.toUpperCase()}/draft/path`
332
+ expect(checkFormStatus(path)).toStrictEqual({
333
+ state: FormStatus.Draft,
334
+ isPreview: true
335
+ })
336
+ })
337
+
338
+ it('should throw an error for invalid form state', () => {
339
+ const path = `${PREVIEW_PATH_PREFIX}/invalid-state`
340
+ expect(() => checkFormStatus(path)).toThrow(
341
+ 'Invalid form state: invalid-state'
342
+ )
343
+ })
344
+ })
345
+
346
+ describe('checkEmailAddressForLiveFormSubmission', () => {
347
+ it('should throw an error if emailAddress is undefined and isPreview is false', () => {
348
+ expect(() =>
349
+ checkEmailAddressForLiveFormSubmission(undefined, false)
350
+ ).toThrow(
351
+ Boom.internal(
352
+ 'An email address is required to complete the form submission'
353
+ )
354
+ )
355
+ })
356
+
357
+ it('should not throw an error if emailAddress is defined and isPreview is false', () => {
358
+ expect(() =>
359
+ checkEmailAddressForLiveFormSubmission('test@example.com', false)
360
+ ).not.toThrow()
361
+ })
362
+
363
+ it('should not throw an error if emailAddress is undefined and isPreview is true', () => {
364
+ expect(() =>
365
+ checkEmailAddressForLiveFormSubmission(undefined, true)
366
+ ).not.toThrow()
367
+ })
368
+
369
+ it('should not throw an error if emailAddress is defined and isPreview is true', () => {
370
+ expect(() =>
371
+ checkEmailAddressForLiveFormSubmission('test@example.com', true)
372
+ ).not.toThrow()
373
+ })
374
+ })
375
+
376
+ describe('getErrors', () => {
377
+ it('formats dates with ISO strings', () => {
378
+ const { details } = new ValidationError(
379
+ 'Date of marriage example',
380
+ [
381
+ {
382
+ message:
383
+ 'Date of marriage must be on or before 2021-12-25T00:00:00.000Z',
384
+ path: ['dateField'],
385
+ type: 'date.max',
386
+ context: {
387
+ key: 'dateField',
388
+ title: 'date of marriage'
389
+ }
390
+ }
391
+ ],
392
+ undefined
393
+ )
394
+
395
+ expect(getErrors(details)).toEqual([
396
+ {
397
+ path: ['dateField'],
398
+ href: '#dateField',
399
+ name: 'dateField',
400
+ text: 'Date of marriage must be on or before 25 December 2021',
401
+ context: {
402
+ key: 'dateField',
403
+ title: 'date of marriage'
404
+ }
405
+ }
406
+ ])
407
+ })
408
+
409
+ it('does not format the first letter to uppercase', () => {
410
+ const { details } = new ValidationError(
411
+ 'Date of marriage example',
412
+ [
413
+ {
414
+ message: 'something invalid',
415
+ path: ['yesNoField'],
416
+ type: 'string.pattern.base',
417
+ context: {
418
+ key: 'yesNoField'
419
+ }
420
+ }
421
+ ],
422
+ undefined
423
+ )
424
+
425
+ expect(getErrors(details)).toEqual([
426
+ {
427
+ path: ['yesNoField'],
428
+ href: '#yesNoField',
429
+ name: 'yesNoField',
430
+ text: 'something invalid',
431
+ context: {
432
+ key: 'yesNoField'
433
+ }
434
+ }
435
+ ])
436
+ })
437
+ })
438
+
439
+ describe('safeGenerateCrumb', () => {
440
+ it('should return undefined when request.state is missing (malformed request)', () => {
441
+ const malformedRequest = {
442
+ server: {
443
+ plugins: {
444
+ crumb: {
445
+ generate: jest.fn()
446
+ }
447
+ }
448
+ },
449
+ plugins: {},
450
+ route: { settings: { plugins: {} } },
451
+ path: '/test',
452
+ url: { search: '' }
453
+ // state intentionally omitted
454
+ } as unknown as FormRequest
455
+
456
+ const crumbToken = safeGenerateCrumb(malformedRequest)
457
+ expect(crumbToken).toBeUndefined()
458
+ expect(
459
+ malformedRequest.server.plugins.crumb.generate
460
+ ).not.toHaveBeenCalled()
461
+ })
462
+
463
+ it('should return undefined if crumb is disabled in route settings', () => {
464
+ const requestWithDisabledCrumb = {
465
+ server: {
466
+ plugins: {
467
+ crumb: {
468
+ generate: jest.fn().mockReturnValue('test-token')
469
+ }
470
+ }
471
+ },
472
+ plugins: {},
473
+ route: { settings: { plugins: { crumb: false } } },
474
+ path: '/test',
475
+ url: { search: '' },
476
+ state: {}
477
+ } as unknown as FormRequest
478
+
479
+ const crumbToken = safeGenerateCrumb(requestWithDisabledCrumb)
480
+ expect(crumbToken).toBeUndefined()
481
+ expect(
482
+ requestWithDisabledCrumb.server.plugins.crumb.generate
483
+ ).not.toHaveBeenCalled()
484
+ })
485
+
486
+ it('should generate crumb when state exists and crumb plugin is available', () => {
487
+ const mockCrumb = 'generated-crumb-value'
488
+ const validRequest = {
489
+ server: {
490
+ plugins: {
491
+ crumb: {
492
+ generate: jest.fn().mockReturnValue(mockCrumb)
493
+ }
494
+ }
495
+ },
496
+ plugins: {},
497
+ route: { settings: { plugins: {} } },
498
+ path: '/test',
499
+ url: { search: '' },
500
+ state: {}
501
+ } as unknown as FormRequest
502
+
503
+ const crumbToken = safeGenerateCrumb(validRequest)
504
+ expect(crumbToken).toBe(mockCrumb)
505
+ expect(validRequest.server.plugins.crumb.generate).toHaveBeenCalledWith(
506
+ validRequest
507
+ )
508
+ })
509
+ })
510
+
511
+ describe('getExponentialBackoffDelay', () => {
512
+ it.each([
513
+ { depth: 1, expected: 2000 },
514
+ { depth: 2, expected: 4000 },
515
+ { depth: 3, expected: 8000 },
516
+ { depth: 4, expected: 16000 },
517
+ { depth: 5, expected: 25000 },
518
+ { depth: 6, expected: 25000 },
519
+ { depth: 7, expected: 25000 }
520
+ ])(
521
+ 'should calculate correct delay for depth $depth',
522
+ ({ depth, expected }) => {
523
+ expect(getExponentialBackoffDelay(depth)).toBe(expected)
524
+ }
525
+ )
526
+
527
+ it('should handle depth of 0', () => {
528
+ expect(getExponentialBackoffDelay(0)).toBe(1000)
529
+ })
530
+
531
+ it('should handle negative depth', () => {
532
+ expect(getExponentialBackoffDelay(-1)).toBe(500)
533
+ })
534
+
535
+ it('should cap at 25 seconds (25000ms) even for large depths', () => {
536
+ // For depth 10: 2000 * 2^(9) would be too high, so it should be capped
537
+ expect(getExponentialBackoffDelay(10)).toBe(25000)
538
+ expect(getExponentialBackoffDelay(20)).toBe(25000)
539
+ })
540
+ })
541
+
542
+ describe('evaluateTemplate', () => {
543
+ let model: FormModel
544
+ let formContext: FormContext
545
+
546
+ beforeEach(() => {
547
+ model = new FormModel(templateDefinition, {
548
+ basePath: 'template'
549
+ })
550
+
551
+ formContext = {
552
+ evaluationState: {},
553
+ relevantState: {},
554
+ relevantPages: [],
555
+ payload: {},
556
+ state: {},
557
+ paths: [],
558
+ isForceAccess: false,
559
+ data: {},
560
+ pageDefMap: model.pageDefMap,
561
+ listDefMap: model.listDefMap,
562
+ componentDefMap: model.componentDefMap,
563
+ pageMap: model.pageMap,
564
+ componentMap: model.componentMap
565
+ }
566
+ })
567
+
568
+ it('should replace placeholders with values from form context relevantState', () => {
569
+ Object.assign(formContext.relevantState, {
570
+ WmHfSb: 'Enrique Chase'
571
+ })
572
+
573
+ const areYouInEngland = templateDefinition.pages[2]
574
+ expect(areYouInEngland.title).toBe('Are you in England, {{ WmHfSb }}?')
575
+
576
+ const result = evaluateTemplate(areYouInEngland.title, formContext)
577
+ expect(result).toBe('Are you in England, Enrique Chase?')
578
+ })
579
+
580
+ it('should replace placeholders with values from form context data', () => {
581
+ Object.assign(formContext.data, {
582
+ score: 'Low'
583
+ })
584
+
585
+ const result = evaluateTemplate(
586
+ 'Your score is: {{ context.data.score }}',
587
+ formContext
588
+ )
589
+
590
+ expect(result).toBe('Your score is: Low')
591
+ })
592
+
593
+ it('evaluate filter should evaluate a liquid template', () => {
594
+ Object.assign(formContext.relevantState, {
595
+ WmHfSb: 'Enrique Chase'
596
+ })
597
+
598
+ const result = evaluateTemplate(
599
+ '{{ "Hello, {{ WmHfSb }}!" | evaluate }}',
600
+ formContext
601
+ )
602
+
603
+ expect(result).toBe('Hello, Enrique Chase!')
604
+ })
605
+
606
+ it('page filter should return the page definition', () => {
607
+ // @ts-expect-error - spyOn type issue
608
+ const filterSpy = jest.spyOn(engine.filters, 'page')
609
+ const result = evaluateTemplate(
610
+ '{%- assign startPageDef = "/start" | page -%}{{ startPageDef.title }}',
611
+ formContext
612
+ )
613
+
614
+ expect(filterSpy).toHaveBeenCalledWith('/start')
615
+ expect(result).toBe('Start page')
616
+ })
617
+
618
+ it('page filter should return empty when anything but a string is passed', () => {
619
+ // @ts-expect-error - spyOn type issue
620
+ const pageFilterSpy = jest.spyOn(engine.filters, 'page')
621
+
622
+ let result = evaluateTemplate('{{ 0 | page }}', formContext)
623
+ expect(pageFilterSpy).toHaveBeenLastCalledWith(0)
624
+ expect(result).toBe('')
625
+
626
+ result = evaluateTemplate('{{ undefined | page }}', formContext)
627
+ expect(pageFilterSpy).toHaveBeenLastCalledWith(undefined)
628
+ expect(result).toBe('')
629
+
630
+ result = evaluateTemplate('{{ null | page }}', formContext)
631
+ expect(result).toBe('')
632
+
633
+ result = evaluateTemplate('{{ false | page }}', formContext)
634
+ expect(pageFilterSpy).toHaveBeenLastCalledWith(false)
635
+ expect(result).toBe('')
636
+
637
+ result = evaluateTemplate('{{ [] | page }}', formContext)
638
+ expect(result).toBe('')
639
+ })
640
+
641
+ it('href filter should return the page href', () => {
642
+ // @ts-expect-error - spyOn type issue
643
+ const filterSpy = jest.spyOn(engine.filters, 'href')
644
+ const result = evaluateTemplate('{{ "/full-name" | href }}', formContext)
645
+
646
+ expect(filterSpy).toHaveBeenCalledWith('/full-name')
647
+ expect(result).toBe('/template/full-name')
648
+ })
649
+
650
+ it('href filter should return empty when no page passed', () => {
651
+ // @ts-expect-error - spyOn type issue
652
+ const pageFilterSpy = jest.spyOn(engine.filters, 'href')
653
+
654
+ const result = evaluateTemplate('{{ undefined | href }}', formContext)
655
+ expect(pageFilterSpy).toHaveBeenLastCalledWith(undefined)
656
+ expect(result).toBe('')
657
+ })
658
+
659
+ it('field filter should return the component definition', () => {
660
+ // @ts-expect-error - spyOn type issue
661
+ const filterSpy = jest.spyOn(engine.filters, 'field')
662
+ const result = evaluateTemplate(
663
+ '{%- assign fullNameComponentDef = "WmHfSb" | field -%}{{ fullNameComponentDef.title }}',
664
+ formContext
665
+ )
666
+
667
+ expect(filterSpy).toHaveBeenCalledWith('WmHfSb')
668
+ expect(result).toBe('What&#39;s your full name?')
669
+ })
670
+
671
+ it('field filter should return empty when anything but a string is passed', () => {
672
+ // @ts-expect-error - spyOn type issue
673
+ const pageFilterSpy = jest.spyOn(engine.filters, 'field')
674
+
675
+ let result = evaluateTemplate('{{ 0 | field }}', formContext)
676
+ expect(pageFilterSpy).toHaveBeenLastCalledWith(0)
677
+ expect(result).toBe('')
678
+
679
+ result = evaluateTemplate('{{ undefined | field }}', formContext)
680
+ expect(pageFilterSpy).toHaveBeenLastCalledWith(undefined)
681
+ expect(result).toBe('')
682
+
683
+ result = evaluateTemplate('{{ null | field }}', formContext)
684
+ expect(result).toBe('')
685
+
686
+ result = evaluateTemplate('{{ false | field }}', formContext)
687
+ expect(pageFilterSpy).toHaveBeenLastCalledWith(false)
688
+ expect(result).toBe('')
689
+
690
+ result = evaluateTemplate('{{ [] | field }}', formContext)
691
+ expect(result).toBe('')
692
+ })
693
+
694
+ it('answer filter should return the formatted submitted answer', () => {
695
+ Object.assign(formContext.relevantState, {
696
+ TKsWbP: true,
697
+ WmHfSb: 'Enrique Chase'
698
+ })
699
+
700
+ // @ts-expect-error - spyOn type issue
701
+ const filterSpy = jest.spyOn(engine.filters, 'answer')
702
+
703
+ let result = evaluateTemplate("{{ 'TKsWbP' | answer }}", formContext)
704
+ expect(filterSpy).toHaveBeenCalledWith('TKsWbP')
705
+ expect(result).toBe('Yes')
706
+
707
+ result = evaluateTemplate("{{ 'WmHfSb' | answer }}", formContext)
708
+ expect(filterSpy).toHaveBeenCalledWith('WmHfSb')
709
+ expect(result).toBe('Enrique Chase')
710
+ })
711
+
712
+ it('answer filter should return empty when anything but a string is passed', () => {
713
+ // @ts-expect-error - spyOn type issue
714
+ const pageFilterSpy = jest.spyOn(engine.filters, 'answer')
715
+
716
+ let result = evaluateTemplate('{{ 0 | answer }}', formContext)
717
+ expect(pageFilterSpy).toHaveBeenLastCalledWith(0)
718
+ expect(result).toBe('')
719
+
720
+ result = evaluateTemplate('{{ undefined | answer }}', formContext)
721
+ expect(pageFilterSpy).toHaveBeenLastCalledWith(undefined)
722
+ expect(result).toBe('')
723
+
724
+ result = evaluateTemplate('{{ null | answer }}', formContext)
725
+ expect(result).toBe('')
726
+
727
+ result = evaluateTemplate('{{ false | answer }}', formContext)
728
+ expect(pageFilterSpy).toHaveBeenLastCalledWith(false)
729
+ expect(result).toBe('')
730
+
731
+ result = evaluateTemplate('{{ [] | answer }}', formContext)
732
+ expect(result).toBe('')
733
+ })
734
+
735
+ it('answer filter should return empty when non-form component name is passed', () => {
736
+ // @ts-expect-error - spyOn type issue
737
+ const pageFilterSpy = jest.spyOn(engine.filters, 'answer')
738
+
739
+ const result = evaluateTemplate('{{ "FGyiLS" | answer }}', formContext)
740
+ expect(pageFilterSpy).toHaveBeenLastCalledWith('FGyiLS')
741
+ expect(result).toBe('')
742
+ })
743
+ })
744
+
745
+ describe('Nunjucks filters', () => {
746
+ describe('evaluate filter', () => {
747
+ it('returns non-string values unchanged', () => {
748
+ const mockContext: NunjucksContext = {
749
+ context: {
750
+ globals: {
751
+ context: { pageMap: new Map() } as FormContext,
752
+ pages: new Map(),
753
+ components: new Map()
754
+ }
755
+ }
756
+ }
757
+
758
+ const numResult = (
759
+ engine.filters.evaluate as unknown as EvaluateFilter
760
+ ).call(mockContext, 123)
761
+ expect(numResult).toBe(123)
762
+
763
+ const objResult = (
764
+ engine.filters.evaluate as unknown as EvaluateFilter
765
+ ).call(mockContext, { foo: 'bar' })
766
+ expect(objResult).toEqual({ foo: 'bar' })
767
+ })
768
+ })
769
+
770
+ describe('href filter', () => {
771
+ it('returns undefined when page is undefined', () => {
772
+ const mockContext: NunjucksContext = {
773
+ context: {
774
+ globals: {
775
+ context: { pageMap: new Map() } as FormContext,
776
+ pages: new Map(),
777
+ components: new Map()
778
+ }
779
+ }
780
+ }
781
+
782
+ const result = (engine.filters.href as unknown as HrefFilter).call(
783
+ mockContext,
784
+ '/some-page'
785
+ )
786
+
787
+ expect(result).toBeUndefined()
788
+ })
789
+ })
790
+ })
791
+ })