@defra/forms-engine-plugin 0.0.3 → 0.0.4

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 (328) hide show
  1. package/.public/assets/fonts/bold-affa96571d-v2.woff +0 -0
  2. package/.public/assets/fonts/bold-b542beb274-v2.woff2 +0 -0
  3. package/.public/assets/fonts/light-94a07e06a1-v2.woff2 +0 -0
  4. package/.public/assets/fonts/light-f591b13f7d-v2.woff +0 -0
  5. package/.public/assets/images/favicon.ico +0 -0
  6. package/.public/assets/images/favicon.svg +1 -0
  7. package/.public/assets/images/govuk-crest.svg +1 -0
  8. package/.public/assets/images/govuk-icon-180.png +0 -0
  9. package/.public/assets/images/govuk-icon-192.png +0 -0
  10. package/.public/assets/images/govuk-icon-512.png +0 -0
  11. package/.public/assets/images/govuk-icon-mask.svg +1 -0
  12. package/.public/assets/images/govuk-opengraph-image.png +0 -0
  13. package/.public/assets/manifest.json +39 -0
  14. package/.public/assets-manifest.json +24 -0
  15. package/.public/javascripts/application.0fd8c18.min.js +3 -0
  16. package/.public/javascripts/application.0fd8c18.min.js.LICENSE.txt +58 -0
  17. package/.public/javascripts/application.0fd8c18.min.js.map +1 -0
  18. package/.public/javascripts/file-upload.b2f18f5.min.js +2 -0
  19. package/.public/javascripts/file-upload.b2f18f5.min.js.map +1 -0
  20. package/.public/javascripts/vendor/accessible-autocomplete.275d332.min.js +2 -0
  21. package/.public/javascripts/vendor/accessible-autocomplete.275d332.min.js.map +1 -0
  22. package/.public/stylesheets/application.e340021.min.css +14 -0
  23. package/.public/stylesheets/application.e340021.min.css.map +1 -0
  24. package/package.json +6 -2
  25. package/.browserslistrc +0 -16
  26. package/.editorconfig +0 -9
  27. package/.eslintrc.cjs +0 -266
  28. package/.github/dependabot.yml +0 -85
  29. package/.github/workflows/check-pull-request.yml +0 -209
  30. package/.github/workflows/pr-notifier.yml +0 -98
  31. package/.github/workflows/publish.yml +0 -111
  32. package/.husky/pre-commit +0 -1
  33. package/.lintstagedrc.js +0 -4
  34. package/.nvmrc +0 -1
  35. package/.prettierignore +0 -8
  36. package/.prettierrc.cjs +0 -26
  37. package/Dockerfile +0 -61
  38. package/Procfile +0 -1
  39. package/babel.config.cjs +0 -55
  40. package/compose/aws.env +0 -4
  41. package/compose/start-localstack.sh +0 -26
  42. package/docker-compose.yaml +0 -86
  43. package/globals.d.ts +0 -1
  44. package/jest.config.cjs +0 -54
  45. package/jest.environment.js +0 -4
  46. package/jest.setup.cjs +0 -14
  47. package/postcss.config.js +0 -26
  48. package/sonar-project.properties +0 -17
  49. package/src/client/javascripts/application.js +0 -87
  50. package/src/client/javascripts/file-upload.js +0 -386
  51. package/src/client/stylesheets/_code.scss +0 -33
  52. package/src/client/stylesheets/_govuk-frontend.scss +0 -4
  53. package/src/client/stylesheets/_prose.scss +0 -56
  54. package/src/client/stylesheets/_service-banner.scss +0 -24
  55. package/src/client/stylesheets/_summary-list.scss +0 -28
  56. package/src/client/stylesheets/_tag-env.scss +0 -24
  57. package/src/client/stylesheets/application.scss +0 -14
  58. package/src/common/cookies.js +0 -58
  59. package/src/common/cookies.test.js +0 -23
  60. package/src/common/types.js +0 -5
  61. package/src/config/index.ts +0 -271
  62. package/src/index.ts +0 -31
  63. package/src/server/common/helpers/logging/logger-options.test.ts +0 -50
  64. package/src/server/common/helpers/logging/logger-options.ts +0 -46
  65. package/src/server/common/helpers/logging/logger.ts +0 -7
  66. package/src/server/common/helpers/logging/request-logger.ts +0 -9
  67. package/src/server/common/helpers/logging/request-tracing.js +0 -10
  68. package/src/server/common/helpers/redis-client.js +0 -70
  69. package/src/server/constants.js +0 -1
  70. package/src/server/forms/README.md +0 -10
  71. package/src/server/forms/components.json +0 -1015
  72. package/src/server/forms/report-a-terrorist.json +0 -270
  73. package/src/server/forms/runner-components-test.json +0 -365
  74. package/src/server/forms/test.json +0 -581
  75. package/src/server/index.test.ts +0 -582
  76. package/src/server/index.ts +0 -140
  77. package/src/server/plugins/blankie.test.ts +0 -73
  78. package/src/server/plugins/blankie.ts +0 -48
  79. package/src/server/plugins/crumb.ts +0 -20
  80. package/src/server/plugins/engine/README.md +0 -87
  81. package/src/server/plugins/engine/components/AutocompleteField.test.ts +0 -294
  82. package/src/server/plugins/engine/components/AutocompleteField.ts +0 -49
  83. package/src/server/plugins/engine/components/CheckboxesField.test.ts +0 -379
  84. package/src/server/plugins/engine/components/CheckboxesField.ts +0 -106
  85. package/src/server/plugins/engine/components/ComponentBase.ts +0 -97
  86. package/src/server/plugins/engine/components/ComponentCollection.ts +0 -278
  87. package/src/server/plugins/engine/components/DatePartsField.test.ts +0 -822
  88. package/src/server/plugins/engine/components/DatePartsField.ts +0 -264
  89. package/src/server/plugins/engine/components/Details.test.ts +0 -49
  90. package/src/server/plugins/engine/components/Details.ts +0 -30
  91. package/src/server/plugins/engine/components/EmailAddressField.test.ts +0 -395
  92. package/src/server/plugins/engine/components/EmailAddressField.ts +0 -55
  93. package/src/server/plugins/engine/components/FileUploadField.test.ts +0 -778
  94. package/src/server/plugins/engine/components/FileUploadField.ts +0 -262
  95. package/src/server/plugins/engine/components/FormComponent.ts +0 -249
  96. package/src/server/plugins/engine/components/Html.test.ts +0 -48
  97. package/src/server/plugins/engine/components/Html.ts +0 -29
  98. package/src/server/plugins/engine/components/InsetText.test.ts +0 -48
  99. package/src/server/plugins/engine/components/InsetText.ts +0 -27
  100. package/src/server/plugins/engine/components/List.test.ts +0 -76
  101. package/src/server/plugins/engine/components/List.ts +0 -72
  102. package/src/server/plugins/engine/components/ListFormComponent.ts +0 -140
  103. package/src/server/plugins/engine/components/MonthYearField.test.ts +0 -567
  104. package/src/server/plugins/engine/components/MonthYearField.ts +0 -222
  105. package/src/server/plugins/engine/components/MultilineTextField.test.ts +0 -558
  106. package/src/server/plugins/engine/components/MultilineTextField.ts +0 -138
  107. package/src/server/plugins/engine/components/NumberField.test.ts +0 -701
  108. package/src/server/plugins/engine/components/NumberField.ts +0 -163
  109. package/src/server/plugins/engine/components/RadiosField.test.ts +0 -288
  110. package/src/server/plugins/engine/components/RadiosField.ts +0 -24
  111. package/src/server/plugins/engine/components/SelectField.test.ts +0 -288
  112. package/src/server/plugins/engine/components/SelectField.ts +0 -47
  113. package/src/server/plugins/engine/components/SelectionControlField.ts +0 -43
  114. package/src/server/plugins/engine/components/TelephoneNumberField.test.ts +0 -356
  115. package/src/server/plugins/engine/components/TelephoneNumberField.ts +0 -67
  116. package/src/server/plugins/engine/components/TextField.test.ts +0 -489
  117. package/src/server/plugins/engine/components/TextField.ts +0 -96
  118. package/src/server/plugins/engine/components/UkAddressField.test.ts +0 -623
  119. package/src/server/plugins/engine/components/UkAddressField.ts +0 -172
  120. package/src/server/plugins/engine/components/YesNoField.test.ts +0 -248
  121. package/src/server/plugins/engine/components/YesNoField.ts +0 -31
  122. package/src/server/plugins/engine/components/constants.ts +0 -1
  123. package/src/server/plugins/engine/components/helpers.ts +0 -330
  124. package/src/server/plugins/engine/components/index.ts +0 -24
  125. package/src/server/plugins/engine/components/types.ts +0 -117
  126. package/src/server/plugins/engine/configureEnginePlugin.ts +0 -47
  127. package/src/server/plugins/engine/helpers.test.ts +0 -791
  128. package/src/server/plugins/engine/helpers.ts +0 -379
  129. package/src/server/plugins/engine/index.ts +0 -7
  130. package/src/server/plugins/engine/models/FormModel.test.ts +0 -42
  131. package/src/server/plugins/engine/models/FormModel.ts +0 -443
  132. package/src/server/plugins/engine/models/RepeatingSummaryViewModel.ts +0 -0
  133. package/src/server/plugins/engine/models/Section.ts +0 -0
  134. package/src/server/plugins/engine/models/SummaryViewModel.test.ts +0 -209
  135. package/src/server/plugins/engine/models/SummaryViewModel.ts +0 -220
  136. package/src/server/plugins/engine/models/index.ts +0 -2
  137. package/src/server/plugins/engine/models/types.ts +0 -114
  138. package/src/server/plugins/engine/outputFormatters/human/v1.test.ts +0 -143
  139. package/src/server/plugins/engine/outputFormatters/human/v1.ts +0 -73
  140. package/src/server/plugins/engine/outputFormatters/index.test.ts +0 -17
  141. package/src/server/plugins/engine/outputFormatters/index.ts +0 -44
  142. package/src/server/plugins/engine/outputFormatters/machine/v1.test.ts +0 -229
  143. package/src/server/plugins/engine/outputFormatters/machine/v1.ts +0 -140
  144. package/src/server/plugins/engine/outputFormatters/machine/v2.test.ts +0 -229
  145. package/src/server/plugins/engine/outputFormatters/machine/v2.ts +0 -153
  146. package/src/server/plugins/engine/pageControllers/FileUploadPageController.test.ts +0 -1108
  147. package/src/server/plugins/engine/pageControllers/FileUploadPageController.ts +0 -446
  148. package/src/server/plugins/engine/pageControllers/PageController.test.ts +0 -205
  149. package/src/server/plugins/engine/pageControllers/PageController.ts +0 -176
  150. package/src/server/plugins/engine/pageControllers/QuestionPageController.test.ts +0 -1264
  151. package/src/server/plugins/engine/pageControllers/QuestionPageController.ts +0 -561
  152. package/src/server/plugins/engine/pageControllers/README.md +0 -28
  153. package/src/server/plugins/engine/pageControllers/RepeatPageController.test.ts +0 -264
  154. package/src/server/plugins/engine/pageControllers/RepeatPageController.ts +0 -458
  155. package/src/server/plugins/engine/pageControllers/StartPageController.ts +0 -18
  156. package/src/server/plugins/engine/pageControllers/StatusPageController.ts +0 -50
  157. package/src/server/plugins/engine/pageControllers/SummaryPageController.ts +0 -261
  158. package/src/server/plugins/engine/pageControllers/TerminalController.test.ts +0 -28
  159. package/src/server/plugins/engine/pageControllers/TerminalPageController.ts +0 -19
  160. package/src/server/plugins/engine/pageControllers/helpers.test.ts +0 -198
  161. package/src/server/plugins/engine/pageControllers/helpers.ts +0 -101
  162. package/src/server/plugins/engine/pageControllers/index.ts +0 -10
  163. package/src/server/plugins/engine/pageControllers/validationOptions.ts +0 -89
  164. package/src/server/plugins/engine/plugin.ts +0 -673
  165. package/src/server/plugins/engine/services/formSubmissionService.js +0 -46
  166. package/src/server/plugins/engine/services/formsService.js +0 -46
  167. package/src/server/plugins/engine/services/formsService.test.js +0 -90
  168. package/src/server/plugins/engine/services/index.js +0 -3
  169. package/src/server/plugins/engine/services/notifyService.test.ts +0 -132
  170. package/src/server/plugins/engine/services/notifyService.ts +0 -64
  171. package/src/server/plugins/engine/services/uploadService.js +0 -60
  172. package/src/server/plugins/engine/types.ts +0 -315
  173. package/src/server/plugins/engine/views/components/autocompletefield.html +0 -5
  174. package/src/server/plugins/engine/views/components/checkboxesfield.html +0 -5
  175. package/src/server/plugins/engine/views/components/datepartsfield.html +0 -5
  176. package/src/server/plugins/engine/views/components/details.html +0 -6
  177. package/src/server/plugins/engine/views/components/emailaddressfield.html +0 -5
  178. package/src/server/plugins/engine/views/components/fileuploadfield-key.html +0 -8
  179. package/src/server/plugins/engine/views/components/fileuploadfield-value.html +0 -3
  180. package/src/server/plugins/engine/views/components/fileuploadfield.html +0 -24
  181. package/src/server/plugins/engine/views/components/html.html +0 -3
  182. package/src/server/plugins/engine/views/components/insettext.html +0 -7
  183. package/src/server/plugins/engine/views/components/list.html +0 -36
  184. package/src/server/plugins/engine/views/components/monthyearfield.html +0 -5
  185. package/src/server/plugins/engine/views/components/multilinetextfield.html +0 -10
  186. package/src/server/plugins/engine/views/components/numberfield.html +0 -5
  187. package/src/server/plugins/engine/views/components/radiosfield.html +0 -5
  188. package/src/server/plugins/engine/views/components/selectfield.html +0 -5
  189. package/src/server/plugins/engine/views/components/telephonenumberfield.html +0 -5
  190. package/src/server/plugins/engine/views/components/textfield.html +0 -5
  191. package/src/server/plugins/engine/views/components/ukaddressfield.html +0 -25
  192. package/src/server/plugins/engine/views/components/yesnofield.html +0 -5
  193. package/src/server/plugins/engine/views/file-upload.html +0 -45
  194. package/src/server/plugins/engine/views/index.html +0 -39
  195. package/src/server/plugins/engine/views/item-delete.html +0 -56
  196. package/src/server/plugins/engine/views/partials/components.html +0 -6
  197. package/src/server/plugins/engine/views/partials/conditional-components.html +0 -3
  198. package/src/server/plugins/engine/views/partials/debug.html +0 -44
  199. package/src/server/plugins/engine/views/partials/form.html +0 -15
  200. package/src/server/plugins/engine/views/partials/heading.html +0 -16
  201. package/src/server/plugins/engine/views/partials/preview-banner.html +0 -32
  202. package/src/server/plugins/engine/views/partials/preview-banner.test.js +0 -122
  203. package/src/server/plugins/engine/views/partials/warn-missing-notification-email.html +0 -10
  204. package/src/server/plugins/engine/views/repeat-list-summary.html +0 -53
  205. package/src/server/plugins/errorPages.ts +0 -58
  206. package/src/server/plugins/nunjucks/context.js +0 -88
  207. package/src/server/plugins/nunjucks/context.test.js +0 -142
  208. package/src/server/plugins/nunjucks/enviroment.test.js +0 -201
  209. package/src/server/plugins/nunjucks/environment.js +0 -116
  210. package/src/server/plugins/nunjucks/filters/answer.js +0 -27
  211. package/src/server/plugins/nunjucks/filters/answer.test.js +0 -89
  212. package/src/server/plugins/nunjucks/filters/evaluate.js +0 -21
  213. package/src/server/plugins/nunjucks/filters/field.js +0 -28
  214. package/src/server/plugins/nunjucks/filters/field.test.js +0 -75
  215. package/src/server/plugins/nunjucks/filters/highlight.js +0 -11
  216. package/src/server/plugins/nunjucks/filters/href.js +0 -30
  217. package/src/server/plugins/nunjucks/filters/href.test.js +0 -80
  218. package/src/server/plugins/nunjucks/filters/index.js +0 -8
  219. package/src/server/plugins/nunjucks/filters/inspect.js +0 -15
  220. package/src/server/plugins/nunjucks/filters/page.js +0 -24
  221. package/src/server/plugins/nunjucks/filters/page.test.js +0 -65
  222. package/src/server/plugins/nunjucks/index.js +0 -3
  223. package/src/server/plugins/nunjucks/plugin.js +0 -40
  224. package/src/server/plugins/nunjucks/render.js +0 -42
  225. package/src/server/plugins/nunjucks/types.js +0 -40
  226. package/src/server/plugins/pulse.ts +0 -11
  227. package/src/server/plugins/router.ts +0 -201
  228. package/src/server/plugins/session.ts +0 -28
  229. package/src/server/routes/health.js +0 -13
  230. package/src/server/routes/health.test.js +0 -35
  231. package/src/server/routes/index.test.ts +0 -125
  232. package/src/server/routes/index.ts +0 -2
  233. package/src/server/routes/public.ts +0 -47
  234. package/src/server/routes/types.ts +0 -48
  235. package/src/server/schemas/index.ts +0 -34
  236. package/src/server/secure-context.js +0 -43
  237. package/src/server/services/cacheService.test.ts +0 -276
  238. package/src/server/services/cacheService.ts +0 -131
  239. package/src/server/services/httpService.test.js +0 -491
  240. package/src/server/services/httpService.ts +0 -50
  241. package/src/server/services/index.ts +0 -1
  242. package/src/server/types.ts +0 -54
  243. package/src/server/utils/notify.test.ts +0 -37
  244. package/src/server/utils/notify.ts +0 -50
  245. package/src/server/utils/secure-context/get-trust-store-certs.js +0 -11
  246. package/src/server/utils/secure-context/get-trust-store-certs.test.js +0 -19
  247. package/src/server/utils/utils.js +0 -24
  248. package/src/server/utils/utils.test.js +0 -54
  249. package/src/server/views/404.html +0 -16
  250. package/src/server/views/500.html +0 -19
  251. package/src/server/views/components/debug/macro.njk +0 -3
  252. package/src/server/views/components/debug/template.njk +0 -13
  253. package/src/server/views/components/service-banner/macro.njk +0 -3
  254. package/src/server/views/components/service-banner/template.njk +0 -20
  255. package/src/server/views/components/service-banner/template.test.js +0 -43
  256. package/src/server/views/components/tag-env/macro.njk +0 -3
  257. package/src/server/views/components/tag-env/template.njk +0 -30
  258. package/src/server/views/components/tag-env/template.test.js +0 -66
  259. package/src/server/views/confirmation.html +0 -19
  260. package/src/server/views/help/accessibility-statement.html +0 -58
  261. package/src/server/views/help/cookie-preferences.html +0 -57
  262. package/src/server/views/help/cookies.html +0 -71
  263. package/src/server/views/help/get-support.html +0 -37
  264. package/src/server/views/help/privacy-notice.html +0 -68
  265. package/src/server/views/help/terms-and-conditions.html +0 -83
  266. package/src/server/views/layout.html +0 -199
  267. package/src/server/views/summary.html +0 -50
  268. package/src/typings/hapi/index.d.ts +0 -95
  269. package/src/typings/hapi-tracing/index.d.ts +0 -6
  270. package/src/typings/index.d.ts +0 -3
  271. package/src/typings/joi/index.d.ts +0 -22
  272. package/stylelint.config.js +0 -10
  273. package/test/client/javascripts/file-upload.test.js +0 -1197
  274. package/test/condition/checkboxes.test.js +0 -112
  275. package/test/condition/radios.test.js +0 -112
  276. package/test/condition/text.test.js +0 -103
  277. package/test/fixtures/assets-manifest.json +0 -4
  278. package/test/fixtures/form.js +0 -86
  279. package/test/fixtures/index.js +0 -2
  280. package/test/fixtures/list.js +0 -92
  281. package/test/form/cookies.test.js +0 -338
  282. package/test/form/csrf.test.js +0 -87
  283. package/test/form/definitions/basic.js +0 -101
  284. package/test/form/definitions/blank.js +0 -10
  285. package/test/form/definitions/checkboxes.json +0 -88
  286. package/test/form/definitions/components.json +0 -452
  287. package/test/form/definitions/conditional-reveal.js +0 -140
  288. package/test/form/definitions/conditions-basic.js +0 -187
  289. package/test/form/definitions/conditions-complex.js +0 -338
  290. package/test/form/definitions/conditions-dates.js +0 -78
  291. package/test/form/definitions/conditions-escaping.js +0 -143
  292. package/test/form/definitions/demo-cph-number.js +0 -3099
  293. package/test/form/definitions/feedback.json +0 -45
  294. package/test/form/definitions/fields-optional.js +0 -402
  295. package/test/form/definitions/fields-required.js +0 -402
  296. package/test/form/definitions/file-upload-basic.js +0 -44
  297. package/test/form/definitions/file-upload.js +0 -66
  298. package/test/form/definitions/minimal.js +0 -39
  299. package/test/form/definitions/phase-alpha.json +0 -33
  300. package/test/form/definitions/phase-default.json +0 -26
  301. package/test/form/definitions/radios.json +0 -88
  302. package/test/form/definitions/repeat-mixed.js +0 -54
  303. package/test/form/definitions/repeat.js +0 -70
  304. package/test/form/definitions/status.json +0 -126
  305. package/test/form/definitions/templates.js +0 -183
  306. package/test/form/definitions/test.json +0 -581
  307. package/test/form/definitions/text.json +0 -75
  308. package/test/form/definitions/titles.json +0 -170
  309. package/test/form/definitions.test.js +0 -47
  310. package/test/form/exit-page.test.js +0 -210
  311. package/test/form/feedback.test.js +0 -68
  312. package/test/form/fields-optional.test.js +0 -237
  313. package/test/form/fields-required.test.js +0 -294
  314. package/test/form/file-upload.test.js +0 -313
  315. package/test/form/govuk-notify.test.js +0 -449
  316. package/test/form/journey-basic.test.js +0 -444
  317. package/test/form/persist-files.test.js +0 -227
  318. package/test/form/phase-banner.test.js +0 -71
  319. package/test/form/repeat.test.js +0 -628
  320. package/test/form/summary-submission-email.test.js +0 -95
  321. package/test/form/template.test.js +0 -288
  322. package/test/form/titles.test.js +0 -204
  323. package/test/helpers/component-helpers.js +0 -74
  324. package/test/utils/get-cookie.js +0 -42
  325. package/test/utils/get-form-definitions.js +0 -18
  326. package/tmp.pdf +0 -1
  327. package/tsconfig.json +0 -28
  328. package/webpack.config.js +0 -208
@@ -1,53 +0,0 @@
1
- {% extends "layout.html" %}
2
-
3
- {% from "govuk/components/error-summary/macro.njk" import govukErrorSummary %}
4
- {% from "govuk/components/button/macro.njk" import govukButton %}
5
- {% from "govuk/components/summary-list/macro.njk" import govukSummaryList %}
6
-
7
- {% block content %}
8
- <div class="govuk-grid-row">
9
- <div class="govuk-grid-column-two-thirds">
10
- {% if previewMode %}
11
- {% include "partials/preview-banner.html" %}
12
- {% endif %}
13
-
14
- {% if errors %}
15
- {{ govukErrorSummary({
16
- titleText: "There is a problem",
17
- errorList: errors
18
- }) }}
19
- {% endif %}
20
-
21
- {% include "partials/heading.html" %}
22
-
23
- {% for section in checkAnswers %}
24
- {{ govukSummaryList(section.summaryList) }}
25
- {% endfor %}
26
-
27
- <div class="govuk-button-group">
28
- <form method="post" novalidate>
29
- <input type="hidden" name="crumb" value="{{ crumb }}">
30
-
31
- {{ govukButton({
32
- text: "Continue",
33
- name: "action",
34
- value: "continue",
35
- preventDoubleClick: true
36
- }) }}
37
-
38
- {{ govukButton({
39
- text: "Add another " + repeatTitle,
40
- name: "action",
41
- value: "add-another",
42
- classes: "govuk-button--secondary",
43
- preventDoubleClick: true
44
- }) }}
45
- </form>
46
- </div>
47
- </div>
48
- </div>
49
-
50
- {% if config.cdpEnvironment == 'local' and context | length and not context.isForceAccess %}
51
- {% include "partials/debug.html" %}
52
- {% endif %}
53
- {% endblock %}
@@ -1,58 +0,0 @@
1
- import {
2
- type Request,
3
- type ResponseToolkit,
4
- type ServerRegisterPluginObject
5
- } from '@hapi/hapi'
6
- import { StatusCodes } from 'http-status-codes'
7
-
8
- /*
9
- * Add an `onPreResponse` listener to return error pages
10
- */
11
- export default {
12
- plugin: {
13
- name: 'error-pages',
14
- register(server) {
15
- server.ext('onPreResponse', (request: Request, h: ResponseToolkit) => {
16
- const response = request.response
17
-
18
- if ('isBoom' in response && response.isBoom) {
19
- // An error was raised during
20
- // processing the request
21
- const statusCode = response.output.statusCode
22
-
23
- // Check for a form model on the request
24
- // and use it to set the correct service name
25
- // and start page path. In the event of a error
26
- // happening inside a "form" level request, the header
27
- // then displays the contextual form text and href
28
- const model = request.app.model
29
- const viewModel = model
30
- ? {
31
- name: model.name,
32
- serviceUrl: `/${model.basePath}`
33
- }
34
- : undefined
35
-
36
- // In the event of 404
37
- // return the `404` view
38
- if (statusCode === StatusCodes.NOT_FOUND.valueOf()) {
39
- return h.view('404', viewModel).code(statusCode)
40
- }
41
-
42
- request.log('error', {
43
- statusCode,
44
- data: response.data,
45
- message: response.message,
46
- stack: response.stack
47
- })
48
-
49
- request.logger.error(response.stack)
50
-
51
- // The return the `500` view
52
- return h.view('500', viewModel).code(statusCode)
53
- }
54
- return h.continue
55
- })
56
- }
57
- }
58
- } satisfies ServerRegisterPluginObject<void>
@@ -1,88 +0,0 @@
1
- import { readFileSync } from 'node:fs'
2
- import { basename, join } from 'node:path'
3
-
4
- import Boom from '@hapi/boom'
5
- import { StatusCodes } from 'http-status-codes'
6
-
7
- import pkg from '~/package.json' with { type: 'json' }
8
- import { parseCookieConsent } from '~/src/common/cookies.js'
9
- import { config } from '~/src/config/index.js'
10
- import { createLogger } from '~/src/server/common/helpers/logging/logger.js'
11
- import { PREVIEW_PATH_PREFIX } from '~/src/server/constants.js'
12
- import {
13
- encodeUrl,
14
- safeGenerateCrumb
15
- } from '~/src/server/plugins/engine/helpers.js'
16
-
17
- const logger = createLogger()
18
-
19
- /** @type {Record<string, string> | undefined} */
20
- let webpackManifest
21
-
22
- /**
23
- * @param {FormRequest | FormRequestPayload | null} request
24
- */
25
- export function context(request) {
26
- const manifestPath = join(config.get('publicDir'), 'assets-manifest.json')
27
-
28
- if (!webpackManifest) {
29
- try {
30
- // eslint-disable-next-line -- Allow JSON type 'any'
31
- webpackManifest = JSON.parse(readFileSync(manifestPath, 'utf-8'))
32
- } catch {
33
- logger.error(`Webpack ${basename(manifestPath)} not found`)
34
- }
35
- }
36
-
37
- const { params, path, query = {}, response, state } = request ?? {}
38
-
39
- const isForceAccess = 'force' in query
40
- const isPreviewMode = path?.startsWith(PREVIEW_PATH_PREFIX)
41
-
42
- // Only add the slug in to the context if the response is OK.
43
- // Footer meta links are not rendered when the slug is missing.
44
- const isResponseOK =
45
- !Boom.isBoom(response) && response?.statusCode === StatusCodes.OK
46
-
47
- /** @type {ViewContext} */
48
- const ctx = {
49
- appVersion: pkg.version,
50
- assetPath: '/assets',
51
- config: {
52
- cdpEnvironment: config.get('cdpEnvironment'),
53
- designerUrl: config.get('designerUrl'),
54
- feedbackLink: encodeUrl(config.get('feedbackLink')),
55
- phaseTag: config.get('phaseTag'),
56
- serviceBannerText: config.get('serviceBannerText'),
57
- serviceName: config.get('serviceName'),
58
- serviceVersion: config.get('serviceVersion')
59
- },
60
- crumb: safeGenerateCrumb(request),
61
- cspNonce: request?.plugins.blankie?.nonces?.script,
62
- currentPath: request ? `${request.path}${request.url.search}` : undefined,
63
- previewMode: isPreviewMode ? params?.state : undefined,
64
- slug: isResponseOK ? params?.slug : undefined,
65
-
66
- getAssetPath: (asset = '') => {
67
- return `/${webpackManifest?.[asset] ?? asset}`
68
- }
69
- }
70
-
71
- if (!isForceAccess) {
72
- ctx.config.googleAnalyticsTrackingId = config.get(
73
- 'googleAnalyticsTrackingId'
74
- )
75
-
76
- if (typeof state?.cookieConsent === 'string') {
77
- ctx.cookieConsent = parseCookieConsent(state.cookieConsent)
78
- }
79
- }
80
-
81
- return ctx
82
- }
83
-
84
- /**
85
- * @import { CookieConsent } from '~/src/common/types.js'
86
- * @import { ViewContext } from '~/src/server/plugins/nunjucks/types.js'
87
- * @import { FormRequest, FormRequestPayload } from '~/src/server/routes/types.js'
88
- */
@@ -1,142 +0,0 @@
1
- import { tmpdir } from 'node:os'
2
-
3
- import { config } from '~/src/config/index.js'
4
- import { encodeUrl } from '~/src/server/plugins/engine/helpers.js'
5
- import { context } from '~/src/server/plugins/nunjucks/context.js'
6
-
7
- describe('Nunjucks context', () => {
8
- beforeEach(() => jest.resetModules())
9
-
10
- describe('Asset path', () => {
11
- it("should include 'assetPath' for GOV.UK Frontend icons", () => {
12
- const { assetPath } = context(null)
13
- expect(assetPath).toBe('/assets')
14
- })
15
- })
16
-
17
- describe('Asset helper', () => {
18
- it("should locate 'assets-manifest.json' assets", () => {
19
- const { getAssetPath } = context(null)
20
-
21
- expect(getAssetPath('example.scss')).toBe(
22
- '/stylesheets/example.xxxxxxx.min.css'
23
- )
24
-
25
- expect(getAssetPath('example.mjs')).toBe(
26
- '/javascripts/example.xxxxxxx.min.js'
27
- )
28
- })
29
-
30
- it("should return path when 'assets-manifest.json' is missing", async () => {
31
- await jest.isolateModulesAsync(async () => {
32
- const { config } = await import('~/src/config/index.js')
33
-
34
- // Import when isolated to avoid cache
35
- const { context } = await import(
36
- '~/src/server/plugins/nunjucks/context.js'
37
- )
38
-
39
- // Update config for missing manifest
40
- config.set('publicDir', tmpdir())
41
- const { getAssetPath } = context(null)
42
-
43
- // Uses original paths when missing
44
- expect(getAssetPath('example.scss')).toBe('/example.scss')
45
- expect(getAssetPath('example.mjs')).toBe('/example.mjs')
46
- })
47
- })
48
-
49
- it('should return path to unknown assets', () => {
50
- const { getAssetPath } = context(null)
51
-
52
- expect(getAssetPath()).toBe('/')
53
- expect(getAssetPath('example.jpg')).toBe('/example.jpg')
54
- expect(getAssetPath('example.gif')).toBe('/example.gif')
55
- })
56
- })
57
-
58
- describe('Config', () => {
59
- it('should include environment, phase tag and service info', () => {
60
- const ctx = context(null)
61
-
62
- expect(ctx.config).toEqual(
63
- expect.objectContaining({
64
- cdpEnvironment: config.get('cdpEnvironment'),
65
- feedbackLink: encodeUrl(config.get('feedbackLink')),
66
- googleAnalyticsTrackingId: config.get('googleAnalyticsTrackingId'),
67
- phaseTag: config.get('phaseTag'),
68
- serviceBannerText: config.get('serviceBannerText'),
69
- serviceName: config.get('serviceName'),
70
- serviceVersion: config.get('serviceVersion')
71
- })
72
- )
73
- })
74
- })
75
-
76
- describe('Crumb', () => {
77
- it('should handle malformed requests with missing state', () => {
78
- // While state should always exist in a valid Hapi request (it holds cookies),
79
- // we've seen malformed requests in production where it's missing
80
- const malformedRequest = /** @type {FormRequest} */ (
81
- /** @type {unknown} */ ({
82
- server: {
83
- plugins: {
84
- crumb: {
85
- generate: jest.fn()
86
- }
87
- }
88
- },
89
- plugins: {},
90
- route: {
91
- settings: {
92
- plugins: {}
93
- }
94
- },
95
- path: '/test',
96
- url: { search: '' }
97
- // state intentionally omitted to test real malformed requests
98
- })
99
- )
100
-
101
- const { crumb } = context(malformedRequest)
102
- expect(crumb).toBeUndefined()
103
- expect(
104
- malformedRequest.server.plugins.crumb.generate
105
- ).not.toHaveBeenCalled()
106
- })
107
-
108
- it('should generate crumb when state exists', () => {
109
- const mockCrumb = 'generated-crumb-value'
110
- const validRequest = /** @type {FormRequest} */ (
111
- /** @type {unknown} */ ({
112
- server: {
113
- plugins: {
114
- crumb: {
115
- generate: jest.fn().mockReturnValue(mockCrumb)
116
- }
117
- }
118
- },
119
- plugins: {},
120
- route: {
121
- settings: {
122
- plugins: {}
123
- }
124
- },
125
- path: '/test',
126
- url: { search: '' },
127
- state: {}
128
- })
129
- )
130
-
131
- const { crumb } = context(validRequest)
132
- expect(crumb).toBe(mockCrumb)
133
- expect(validRequest.server.plugins.crumb.generate).toHaveBeenCalledWith(
134
- validRequest
135
- )
136
- })
137
- })
138
- })
139
-
140
- /**
141
- * @import { FormRequest } from '~/src/server/routes/types.js'
142
- */
@@ -1,201 +0,0 @@
1
- import * as helpers from '~/src/server/plugins/engine/helpers.js'
2
- import { environment } from '~/src/server/plugins/nunjucks/environment.js'
3
-
4
- describe('Nunjucks environment', () => {
5
- describe('checkErrorTemplates function', () => {
6
- /** @type {Function} */
7
- let checkErrorTemplates
8
-
9
- beforeEach(() => {
10
- checkErrorTemplates = environment.getGlobal('checkErrorTemplates')
11
-
12
- jest
13
- .spyOn(helpers, 'evaluateTemplate')
14
- .mockImplementation((text) => `evaluated-${text}`)
15
- })
16
-
17
- afterEach(() => {
18
- jest.restoreAllMocks()
19
- })
20
-
21
- test('returns errors unchanged when context is not present', () => {
22
- const nunjucksCtx = {
23
- ctx: {}
24
- }
25
-
26
- const errors = [{ text: 'Error 1' }, { text: 'Error 2' }]
27
-
28
- const result = checkErrorTemplates.call(nunjucksCtx, errors)
29
-
30
- expect(result).toBe(errors)
31
- expect(helpers.evaluateTemplate).not.toHaveBeenCalled()
32
- })
33
-
34
- test('evaluates error texts when context is present', () => {
35
- const formContext = { someData: 'some-text' }
36
- const nunjucksCtx = {
37
- ctx: { context: formContext }
38
- }
39
-
40
- const errors = [{ text: 'Error 1' }, { text: 'Error 2' }]
41
-
42
- const result = checkErrorTemplates.call(nunjucksCtx, errors)
43
-
44
- expect(helpers.evaluateTemplate).toHaveBeenCalledTimes(2)
45
- expect(helpers.evaluateTemplate).toHaveBeenCalledWith(
46
- 'Error 1',
47
- formContext
48
- )
49
- expect(helpers.evaluateTemplate).toHaveBeenCalledWith(
50
- 'Error 2',
51
- formContext
52
- )
53
-
54
- expect(result).toEqual([
55
- { text: 'evaluated-Error 1' },
56
- { text: 'evaluated-Error 2' }
57
- ])
58
- })
59
- })
60
-
61
- describe('checkComponentTemplates function', () => {
62
- /** @type {Function} */
63
- let checkComponentTemplates
64
-
65
- beforeEach(() => {
66
- checkComponentTemplates = environment.getGlobal('checkComponentTemplates')
67
-
68
- jest
69
- .spyOn(helpers, 'evaluateTemplate')
70
- .mockImplementation((text) => `evaluated-${text}`)
71
- })
72
-
73
- afterEach(() => {
74
- jest.restoreAllMocks()
75
- })
76
-
77
- test('evaluates string content for Html components', () => {
78
- const formContext = { someData: 'some-text' }
79
- const nunjucksCtx = {
80
- ctx: { context: formContext }
81
- }
82
-
83
- const component = {
84
- type: 'Html',
85
- isFormComponent: false,
86
- model: {
87
- content: 'Some {{ context.someData }} content'
88
- }
89
- }
90
-
91
- const result = checkComponentTemplates.call(nunjucksCtx, component)
92
-
93
- expect(helpers.evaluateTemplate).toHaveBeenCalledWith(
94
- 'Some {{ context.someData }} content',
95
- formContext
96
- )
97
- expect(result.model.content).toBe(
98
- 'evaluated-Some {{ context.someData }} content'
99
- )
100
- })
101
-
102
- test('does not evaluate non-string content for Html components', () => {
103
- const formContext = { someData: 'some-text' }
104
- const nunjucksCtx = {
105
- ctx: { context: formContext }
106
- }
107
-
108
- const nonStringContent = { some: 'object' }
109
- const component = {
110
- type: 'Html',
111
- isFormComponent: false,
112
- model: {
113
- content: nonStringContent
114
- }
115
- }
116
-
117
- const result = checkComponentTemplates.call(nunjucksCtx, component)
118
-
119
- expect(helpers.evaluateTemplate).not.toHaveBeenCalled()
120
-
121
- expect(result.model.content).toBe(nonStringContent)
122
- })
123
-
124
- test('evaluates label text for form components', () => {
125
- const formContext = { someData: 'some-text' }
126
- const nunjucksCtx = {
127
- ctx: { context: formContext }
128
- }
129
-
130
- const component = {
131
- isFormComponent: true,
132
- model: {
133
- label: {
134
- text: 'Label with {{ context.someData }}'
135
- }
136
- }
137
- }
138
-
139
- const result = checkComponentTemplates.call(nunjucksCtx, component)
140
-
141
- expect(helpers.evaluateTemplate).toHaveBeenCalledWith(
142
- 'Label with {{ context.someData }}',
143
- formContext
144
- )
145
-
146
- expect(result.model.label?.text).toBe(
147
- 'evaluated-Label with {{ context.someData }}'
148
- )
149
- })
150
- })
151
-
152
- describe('evaluate function', () => {
153
- /** @type {Function} */
154
- let evaluateFunc
155
-
156
- beforeEach(() => {
157
- evaluateFunc = environment.getGlobal('evaluate')
158
-
159
- jest
160
- .spyOn(helpers, 'evaluateTemplate')
161
- .mockImplementation((text) => `evaluated-${text}`)
162
- })
163
-
164
- afterEach(() => {
165
- jest.restoreAllMocks()
166
- })
167
-
168
- test('evaluates template when context is present', () => {
169
- const formContext = { someData: 'some-text' }
170
- const nunjucksCtx = {
171
- ctx: { context: formContext }
172
- }
173
-
174
- const template = 'Template with {{ context.someData }}'
175
- const result = evaluateFunc.call(nunjucksCtx, template)
176
-
177
- expect(helpers.evaluateTemplate).toHaveBeenCalledWith(
178
- template,
179
- formContext
180
- )
181
- expect(result).toBe('evaluated-Template with {{ context.someData }}')
182
- })
183
-
184
- test('returns template unchanged when context is not present', () => {
185
- const nunjucksCtx = {
186
- ctx: {}
187
- }
188
-
189
- const template = 'Template with {{ context.someData }}'
190
- const result = evaluateFunc.call(nunjucksCtx, template)
191
-
192
- expect(helpers.evaluateTemplate).not.toHaveBeenCalled()
193
-
194
- expect(result).toBe(template)
195
- })
196
- })
197
- })
198
-
199
- /*
200
- * @import { ComponentViewModel } from '~/src/server/plugins/engine/components/types.js'
201
- */
@@ -1,116 +0,0 @@
1
- import { dirname, join } from 'node:path'
2
-
3
- import { ComponentType } from '@defra/forms-model'
4
- import nunjucks from 'nunjucks'
5
- import resolvePkg from 'resolve'
6
-
7
- import { config } from '~/src/config/index.js'
8
- import { evaluateTemplate } from '~/src/server/plugins/engine/helpers.js'
9
- import * as filters from '~/src/server/plugins/nunjucks/filters/index.js'
10
-
11
- const govukFrontendPath = dirname(
12
- resolvePkg.sync('govuk-frontend/package.json')
13
- )
14
-
15
- export const paths = [
16
- join(config.get('appDir'), 'plugins/engine/views'),
17
- join(config.get('appDir'), 'views')
18
- ]
19
-
20
- export const environment = nunjucks.configure(
21
- [...paths, join(govukFrontendPath, 'dist')],
22
- {
23
- trimBlocks: true,
24
- lstripBlocks: true,
25
- watch: config.get('isDevelopment'),
26
- noCache: config.get('isDevelopment')
27
- }
28
- )
29
-
30
- for (const [name, nunjucksFilter] of Object.entries(filters)) {
31
- environment.addFilter(name, nunjucksFilter)
32
- }
33
-
34
- /**
35
- * @this {NunjucksContext}
36
- * @param {FormSubmissionError[]} errors
37
- */
38
- export function checkErrorTemplates(errors) {
39
- const { context } = this.ctx
40
-
41
- if (!context) {
42
- return errors
43
- }
44
-
45
- errors.forEach((error) => {
46
- error.text = evaluateTemplate(error.text, context)
47
- })
48
-
49
- return errors
50
- }
51
-
52
- environment.addGlobal('checkErrorTemplates', checkErrorTemplates)
53
-
54
- /**
55
- * @this {NunjucksContext}
56
- * @param {ComponentViewModel} component
57
- */
58
- export function checkComponentTemplates(component) {
59
- const { context } = this.ctx
60
-
61
- if (!context) {
62
- return component
63
- }
64
-
65
- if (component.isFormComponent) {
66
- // Evaluate label/legend text
67
- if (component.model.fieldset?.legend?.text) {
68
- const legend = component.model.fieldset.legend
69
-
70
- legend.text = evaluateTemplate(legend.text, context)
71
- } else if (component.model.label?.text) {
72
- const label = component.model.label
73
-
74
- label.text = evaluateTemplate(label.text, context)
75
- } else {
76
- // No template evaluation needed for other component types
77
- }
78
-
79
- // Evaluate error message
80
- if (component.model.errorMessage?.text) {
81
- const message = component.model.errorMessage
82
-
83
- message.text = evaluateTemplate(message.text, context)
84
- }
85
- } else if (component.type === ComponentType.Html) {
86
- const content = component.model.content
87
-
88
- if (typeof content === 'string') {
89
- component.model.content = evaluateTemplate(content, context)
90
- }
91
- } else {
92
- // No template evaluation needed for other component types
93
- }
94
-
95
- return component
96
- }
97
-
98
- environment.addGlobal('checkComponentTemplates', checkComponentTemplates)
99
-
100
- /**
101
- * @this {NunjucksContext}
102
- * @param {string} template
103
- */
104
- export function evaluate(template) {
105
- const { context } = this.ctx
106
-
107
- return context ? evaluateTemplate(template, context) : template
108
- }
109
-
110
- environment.addGlobal('evaluate', evaluate)
111
-
112
- /**
113
- * @import { NunjucksContext } from '~/src/server/plugins/nunjucks/types.js'
114
- * @import { FormSubmissionError } from '~/src/server/plugins/engine/types.js'
115
- * @import { ComponentViewModel } from '~/src/server/plugins/engine/components/types.js'
116
- */
@@ -1,27 +0,0 @@
1
- import { getAnswer } from '~/src/server/plugins/engine/components/helpers.js'
2
-
3
- /**
4
- * Nunjucks filter to get the answer for a component
5
- * @this {NunjucksContext}
6
- * @param {string} name - The name of the component to check
7
- */
8
- export function answer(name) {
9
- const { context } = this.ctx
10
-
11
- if (!context) {
12
- return undefined
13
- }
14
-
15
- const component = context.componentMap.get(name)
16
-
17
- if (!component?.isFormComponent) {
18
- return undefined
19
- }
20
-
21
- return getAnswer(/** @type {Field} */ (component), context.relevantState)
22
- }
23
-
24
- /**
25
- * @import { NunjucksContext } from '~/src/server/plugins/nunjucks/types.js'
26
- * @import { Field } from '~/src/server/plugins/engine/components/helpers.js'
27
- */