@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,338 @@
1
+ import { join } from 'node:path'
2
+
3
+ import { within } from '@testing-library/dom'
4
+ import { StatusCodes } from 'http-status-codes'
5
+
6
+ import { createServer } from '~/src/server/index.js'
7
+ import { getFormMetadata } from '~/src/server/plugins/engine/services/formsService.js'
8
+ import * as fixtures from '~/test/fixtures/index.js'
9
+ import { renderResponse } from '~/test/helpers/component-helpers.js'
10
+ import { getCookieHeader } from '~/test/utils/get-cookie.js'
11
+
12
+ jest.mock('~/src/server/plugins/engine/services/formsService.js')
13
+
14
+ describe(`Cookie banner and analytics`, () => {
15
+ /** @type {Server} */
16
+ let server
17
+
18
+ beforeEach(() => {
19
+ jest.mocked(getFormMetadata).mockResolvedValue(fixtures.form.metadata)
20
+ })
21
+
22
+ afterEach(async () => {
23
+ await server.stop()
24
+ })
25
+
26
+ test.each([
27
+ '/basic/licence', // form pages
28
+ '/help/accessibility-statement/basic' // non-form pages
29
+ ])('shows the cookie banner by default', async (path) => {
30
+ server = await createServer({
31
+ formFileName: 'basic.js',
32
+ formFilePath: join(import.meta.dirname, 'definitions')
33
+ })
34
+ await server.initialize()
35
+
36
+ const options = {
37
+ method: 'GET',
38
+ url: path
39
+ }
40
+
41
+ const { container, document } = await renderResponse(server, options)
42
+
43
+ const $cookieBanner = container.queryByRole('region', {
44
+ name: 'Cookies on Submit a form to Defra'
45
+ })
46
+
47
+ const $gaScriptMain = document.getElementById('ga-tag-js-main')
48
+ const $gaScriptInit = document.getElementById('ga-tag-js-init')
49
+
50
+ expect($cookieBanner).toBeInTheDocument()
51
+ expect($gaScriptMain).not.toBeInTheDocument()
52
+ expect($gaScriptInit).not.toBeInTheDocument()
53
+ })
54
+
55
+ test.each([
56
+ // form pages
57
+ '/basic/licence',
58
+ // non-form pages
59
+ '/help/accessibility-statement/basic'
60
+ ])('confirms when the user has accepted analytics cookies', async (path) => {
61
+ server = await createServer({
62
+ formFileName: 'basic.js',
63
+ formFilePath: join(import.meta.dirname, 'definitions')
64
+ })
65
+
66
+ await server.initialize()
67
+
68
+ // set the cookie preferences
69
+ const sessionInitialisationResponse = await server.inject({
70
+ method: 'POST',
71
+ url: `/help/cookie-preferences/basic?returnUrl=${encodeURIComponent('/mypage')}`,
72
+ payload: {
73
+ 'cookies[analytics]': 'yes'
74
+ }
75
+ })
76
+
77
+ const headers = getCookieHeader(sessionInitialisationResponse, [
78
+ 'crumb',
79
+ 'session',
80
+ 'cookieConsent'
81
+ ])
82
+
83
+ const { container, document } = await renderResponse(server, {
84
+ method: 'GET',
85
+ url: path,
86
+ headers
87
+ })
88
+
89
+ const $cookieBanner = container.getByRole('region', {
90
+ name: 'Cookies on Submit a form to Defra'
91
+ })
92
+
93
+ const $confirmationText = within($cookieBanner).getByText(
94
+ 'You’ve accepted analytics cookies.',
95
+ { exact: false }
96
+ )
97
+
98
+ const $gaScriptMain = document.getElementById('ga-tag-js-main')
99
+ const $gaScriptInit = document.getElementById('ga-tag-js-init')
100
+
101
+ expect($cookieBanner).toBeInTheDocument()
102
+ expect($confirmationText).toBeInTheDocument()
103
+ expect($gaScriptMain).toBeInTheDocument()
104
+ expect($gaScriptInit).toBeInTheDocument()
105
+ })
106
+
107
+ test.each([
108
+ // form pages
109
+ '/basic/licence',
110
+ // non-form pages
111
+ '/help/accessibility-statement/basic'
112
+ ])('confirms when the user has rejected analytics cookies', async (path) => {
113
+ server = await createServer({
114
+ formFileName: 'basic.js',
115
+ formFilePath: join(import.meta.dirname, 'definitions')
116
+ })
117
+
118
+ await server.initialize()
119
+
120
+ // set the cookie preferences
121
+ const sessionInitialisationResponse = await server.inject({
122
+ method: 'POST',
123
+ url: `/help/cookie-preferences/basic?returnUrl=${encodeURIComponent('/mypage')}`,
124
+ payload: {
125
+ 'cookies[analytics]': 'no'
126
+ }
127
+ })
128
+
129
+ const headers = getCookieHeader(sessionInitialisationResponse, [
130
+ 'crumb',
131
+ 'session',
132
+ 'cookieConsent'
133
+ ])
134
+
135
+ const { container, document } = await renderResponse(server, {
136
+ method: 'GET',
137
+ url: path,
138
+ headers
139
+ })
140
+
141
+ const $cookieBanner = container.getByRole('region', {
142
+ name: 'Cookies on Submit a form to Defra'
143
+ })
144
+
145
+ const $confirmationText = within($cookieBanner).getByText(
146
+ 'You’ve rejected analytics cookies.',
147
+ { exact: false }
148
+ )
149
+
150
+ const $gaScriptMain = document.getElementById('ga-tag-js-main')
151
+ const $gaScriptInit = document.getElementById('ga-tag-js-init')
152
+
153
+ expect($cookieBanner).toBeInTheDocument()
154
+ expect($confirmationText).toBeInTheDocument()
155
+
156
+ expect($gaScriptMain).not.toBeInTheDocument()
157
+ expect($gaScriptInit).not.toBeInTheDocument()
158
+ })
159
+
160
+ test.each([
161
+ // form pages
162
+ '/basic/start',
163
+ // non-form pages
164
+ '/'
165
+ ])('hides the cookie banner once dismissed', async (path) => {
166
+ server = await createServer({
167
+ formFileName: 'basic.js',
168
+ formFilePath: join(import.meta.dirname, 'definitions')
169
+ })
170
+
171
+ await server.initialize()
172
+
173
+ // set the cookie preferences
174
+ const sessionInitialisationResponse = await server.inject({
175
+ method: 'POST',
176
+ url: `/help/cookie-preferences/basic?returnUrl=${encodeURIComponent('/mypage')}`,
177
+ payload: {
178
+ 'cookies[analytics]': 'yes',
179
+ 'cookies[dismissed]': 'yes'
180
+ }
181
+ })
182
+
183
+ const headers = getCookieHeader(sessionInitialisationResponse, [
184
+ 'crumb',
185
+ 'session',
186
+ 'cookieConsent'
187
+ ])
188
+
189
+ const { container } = await renderResponse(server, {
190
+ method: 'GET',
191
+ url: path,
192
+ headers
193
+ })
194
+
195
+ const $cookieBanner = container.queryByRole('region', {
196
+ name: 'Cookies on Submit a form to Defra'
197
+ })
198
+
199
+ expect($cookieBanner).not.toBeInTheDocument()
200
+ })
201
+ })
202
+
203
+ describe(`Cookie preferences`, () => {
204
+ /** @type {Server} */
205
+ let server
206
+
207
+ afterEach(async () => {
208
+ await server.stop()
209
+ })
210
+
211
+ test.each([
212
+ {
213
+ value: 'yes',
214
+ text: 'Yes'
215
+ },
216
+ {
217
+ value: 'no',
218
+ text: 'No'
219
+ }
220
+ ])(
221
+ 'selects the cookie preference automatically based on the user selection',
222
+ async ({ text, value }) => {
223
+ server = await createServer()
224
+ await server.initialize()
225
+
226
+ // set the cookie preferences
227
+ const sessionInitialisationResponse = await server.inject({
228
+ method: 'POST',
229
+ url: `/help/cookie-preferences/basic`,
230
+ payload: {
231
+ 'cookies[analytics]': value
232
+ }
233
+ })
234
+
235
+ const headers = {
236
+ Referer: '/help/cookie-preferences/basic',
237
+ ...getCookieHeader(sessionInitialisationResponse, [
238
+ 'crumb',
239
+ 'session',
240
+ 'cookieConsent'
241
+ ])
242
+ }
243
+
244
+ const { container } = await renderResponse(server, {
245
+ method: 'GET',
246
+ url: '/help/cookie-preferences/basic',
247
+ headers
248
+ })
249
+
250
+ const $input = container.getByRole('radio', {
251
+ name: text
252
+ })
253
+
254
+ const $successNotification = container.getByRole('alert', {
255
+ name: 'Success'
256
+ })
257
+
258
+ expect($input).toBeChecked()
259
+ expect($successNotification).toHaveTextContent(
260
+ 'You’ve set your cookie preferences.'
261
+ )
262
+ }
263
+ )
264
+
265
+ test("doesn't show the success banner if the user hasn't been posted from the cookie preferences page", async () => {
266
+ server = await createServer()
267
+ await server.initialize()
268
+
269
+ // set the cookie preferences
270
+ const sessionInitialisationResponse = await server.inject({
271
+ method: 'POST',
272
+ url: `/help/cookie-preferences/basic?returnUrl=${encodeURIComponent('/another-page')}`,
273
+ payload: {
274
+ 'cookies[analytics]': 'yes'
275
+ }
276
+ })
277
+
278
+ const headers = {
279
+ ...getCookieHeader(sessionInitialisationResponse, [
280
+ 'crumb',
281
+ 'session',
282
+ 'cookieConsent'
283
+ ])
284
+ }
285
+
286
+ const { container } = await renderResponse(server, {
287
+ method: 'GET',
288
+ url: '/help/cookie-preferences/basic',
289
+ headers
290
+ })
291
+
292
+ const $input = container.getByRole('radio', {
293
+ name: 'Yes'
294
+ })
295
+
296
+ const $successNotification = container.queryByRole('alert', {
297
+ name: 'Success'
298
+ })
299
+
300
+ expect($input).toBeChecked()
301
+ expect($successNotification).not.toBeInTheDocument()
302
+ })
303
+
304
+ test('defaults to no if one is not provided', async () => {
305
+ server = await createServer()
306
+ await server.initialize()
307
+
308
+ const { container } = await renderResponse(server, {
309
+ method: 'GET',
310
+ url: '/help/cookie-preferences/basic'
311
+ })
312
+
313
+ const $input = container.getByRole('radio', {
314
+ name: 'No'
315
+ })
316
+
317
+ expect($input).toBeChecked()
318
+ })
319
+
320
+ test('returns bad request for invalid redirect urls', async () => {
321
+ server = await createServer()
322
+ await server.initialize()
323
+
324
+ const { response } = await renderResponse(server, {
325
+ method: 'POST',
326
+ url: `/help/cookie-preferences/basic?returnUrl=${encodeURIComponent('https://my-malicious-url.com')}`,
327
+ payload: {
328
+ 'cookies[analytics]': 'yes'
329
+ }
330
+ })
331
+
332
+ expect(response.statusCode).toBe(StatusCodes.BAD_REQUEST)
333
+ })
334
+ })
335
+
336
+ /**
337
+ * @import { Server } from '@hapi/hapi'
338
+ */
@@ -0,0 +1,87 @@
1
+ import { join } from 'node:path'
2
+
3
+ import { StatusCodes } from 'http-status-codes'
4
+
5
+ import { createServer } from '~/src/server/index.js'
6
+ import { getFormMetadata } from '~/src/server/plugins/engine/services/formsService.js'
7
+ import * as fixtures from '~/test/fixtures/index.js'
8
+ import { renderResponse } from '~/test/helpers/component-helpers.js'
9
+ import { getCookie } from '~/test/utils/get-cookie.js'
10
+
11
+ const basePath = '/basic'
12
+
13
+ jest.mock('~/src/server/plugins/engine/services/formsService.js')
14
+
15
+ describe('CSRF', () => {
16
+ /** @type {Server} */
17
+ let server
18
+
19
+ // Create server before each test
20
+ beforeAll(async () => {
21
+ server = await createServer({
22
+ formFileName: 'basic.js',
23
+ formFilePath: join(import.meta.dirname, 'definitions'),
24
+ enforceCsrf: true
25
+ })
26
+
27
+ await server.initialize()
28
+ })
29
+
30
+ beforeEach(() => {
31
+ jest.mocked(getFormMetadata).mockResolvedValue(fixtures.form.metadata)
32
+ })
33
+
34
+ afterAll(async () => {
35
+ await server.stop()
36
+ })
37
+
38
+ test('get request returns CSRF header', async () => {
39
+ const { response } = await renderResponse(server, {
40
+ url: `${basePath}/licence`
41
+ })
42
+
43
+ expect(response.statusCode).toBe(StatusCodes.OK)
44
+
45
+ const csrfToken = getCookie(response, 'crumb')
46
+ expect(csrfToken).toBeTruthy()
47
+
48
+ const $input = document.querySelector('[name=crumb]')
49
+ expect($input).toBeInTheDocument()
50
+ expect($input).toHaveAttribute('type', 'hidden')
51
+ expect($input).toHaveValue(csrfToken)
52
+ })
53
+
54
+ test('post request without CSRF token returns 403 forbidden', async () => {
55
+ const response = await server.inject({
56
+ url: `${basePath}/licence`,
57
+ method: 'POST',
58
+ payload: {
59
+ licenceLength: '1'
60
+ }
61
+ })
62
+
63
+ expect(response.statusCode).toBe(StatusCodes.FORBIDDEN)
64
+ })
65
+
66
+ test('post request with CSRF token returns 303 redirect', async () => {
67
+ const csrfToken = 'dummy-token'
68
+
69
+ const response = await server.inject({
70
+ url: `${basePath}/licence`,
71
+ method: 'POST',
72
+ headers: {
73
+ cookie: `crumb=${csrfToken}`
74
+ },
75
+ payload: {
76
+ crumb: csrfToken,
77
+ licenceLength: '1'
78
+ }
79
+ })
80
+
81
+ expect(response.statusCode).toBe(StatusCodes.SEE_OTHER)
82
+ })
83
+ })
84
+
85
+ /**
86
+ * @import { Server, ServerInjectOptions } from '@hapi/hapi'
87
+ */
@@ -0,0 +1,101 @@
1
+ import {
2
+ ComponentType,
3
+ ControllerPath,
4
+ ControllerType
5
+ } from '@defra/forms-model'
6
+
7
+ export default /** @satisfies {FormDefinition} */ ({
8
+ name: 'Basic',
9
+ startPage: '/licence',
10
+ pages: /** @type {const} */ ([
11
+ {
12
+ title: 'Buy a rod fishing licence',
13
+ path: '/licence',
14
+ components: [
15
+ {
16
+ options: {
17
+ bold: true
18
+ },
19
+ type: ComponentType.RadiosField,
20
+ name: 'licenceLength',
21
+ title: 'Which fishing licence do you want to get?',
22
+ list: 'licenceLengthDays'
23
+ }
24
+ ],
25
+ section: 'licenceDetails',
26
+ next: [
27
+ {
28
+ path: '/full-name'
29
+ }
30
+ ]
31
+ },
32
+ {
33
+ title: "What's your name?",
34
+ path: '/full-name',
35
+ components: [
36
+ {
37
+ schema: {
38
+ max: 70
39
+ },
40
+ options: {},
41
+ type: ComponentType.TextField,
42
+ name: 'fullName',
43
+ title: "What's your name?"
44
+ }
45
+ ],
46
+ section: 'personalDetails',
47
+ next: [
48
+ {
49
+ path: '/summary'
50
+ }
51
+ ]
52
+ },
53
+ {
54
+ path: ControllerPath.Summary,
55
+ controller: ControllerType.Summary,
56
+ title: 'Summary'
57
+ }
58
+ ]),
59
+ sections: [
60
+ {
61
+ name: 'licenceDetails',
62
+ title: 'Licence details'
63
+ },
64
+ {
65
+ name: 'personalDetails',
66
+ title: 'Personal details'
67
+ }
68
+ ],
69
+ conditions: [],
70
+ lists: [
71
+ {
72
+ name: 'licenceLengthDays',
73
+ title: 'Licence length (days)',
74
+ type: 'number',
75
+ items: [
76
+ {
77
+ text: '1 day',
78
+ value: 1,
79
+ description: 'Valid for 24 hours from the start time that you select'
80
+ },
81
+ {
82
+ text: '8 day',
83
+ value: 8,
84
+ description:
85
+ 'Valid for 8 consecutive days from the start time that you select'
86
+ },
87
+ {
88
+ text: '12 months',
89
+ value: 365,
90
+ description:
91
+ '12-month licences are now valid for 365 days from their start date and can be purchased at any time during the year'
92
+ }
93
+ ]
94
+ }
95
+ ],
96
+ outputEmail: 'enrique.chase@defra.gov.uk'
97
+ })
98
+
99
+ /**
100
+ * @import { FormDefinition } from '@defra/forms-model'
101
+ */
@@ -0,0 +1,10 @@
1
+ export default /** @type {FormDefinition} */ ({
2
+ pages: [],
3
+ lists: [],
4
+ sections: [],
5
+ conditions: []
6
+ })
7
+
8
+ /**
9
+ * @import { FormDefinition } from '@defra/forms-model'
10
+ */
@@ -0,0 +1,88 @@
1
+ {
2
+ "metadata": {},
3
+ "startPage": "/first-page",
4
+ "pages": [
5
+ {
6
+ "title": "First page",
7
+ "path": "/first-page",
8
+ "components": [
9
+ {
10
+ "name": "wqJmSf",
11
+ "options": { "required": false },
12
+ "type": "CheckboxesField",
13
+ "title": "chooseHorseType",
14
+ "list": "PeVjPw",
15
+ "schema": {}
16
+ }
17
+ ],
18
+ "next": [
19
+ { "path": "/second-page", "condition": "ihuObd" },
20
+ { "path": "/third-page", "condition": "EcLLXH" }
21
+ ]
22
+ },
23
+ {
24
+ "path": "/second-page",
25
+ "title": "Second page",
26
+ "components": [],
27
+ "next": []
28
+ },
29
+ {
30
+ "path": "/third-page",
31
+ "title": "Third page",
32
+ "components": [],
33
+ "next": []
34
+ }
35
+ ],
36
+ "lists": [
37
+ {
38
+ "title": "horseType",
39
+ "name": "PeVjPw",
40
+ "type": "string",
41
+ "items": [
42
+ { "text": "Shire", "value": "shire" },
43
+ { "text": "Race", "value": "race" },
44
+ { "text": "Pantomime", "value": "pantomime" },
45
+ { "text": "Other", "value": "other" }
46
+ ]
47
+ }
48
+ ],
49
+ "sections": [],
50
+ "conditions": [
51
+ {
52
+ "displayName": "Not other",
53
+ "name": "ihuObd",
54
+ "value": {
55
+ "name": "Not other",
56
+ "conditions": [
57
+ {
58
+ "field": {
59
+ "name": "wqJmSf",
60
+ "type": "CheckboxesField",
61
+ "display": "chooseHorseType"
62
+ },
63
+ "operator": "does not contain",
64
+ "value": { "type": "Value", "value": "other", "display": "other" }
65
+ }
66
+ ]
67
+ }
68
+ },
69
+ {
70
+ "displayName": "Other",
71
+ "name": "EcLLXH",
72
+ "value": {
73
+ "name": "Other",
74
+ "conditions": [
75
+ {
76
+ "field": {
77
+ "name": "wqJmSf",
78
+ "type": "CheckboxesField",
79
+ "display": "chooseHorseType"
80
+ },
81
+ "operator": "contains",
82
+ "value": { "type": "Value", "value": "other", "display": "other" }
83
+ }
84
+ ]
85
+ }
86
+ }
87
+ ]
88
+ }