@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,71 +0,0 @@
1
- import { join } from 'node:path'
2
-
3
- import { within } from '@testing-library/dom'
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
-
10
- jest.mock('~/src/server/plugins/engine/services/formsService.js')
11
-
12
- describe(`Phase banner`, () => {
13
- /** @type {Server} */
14
- let server
15
-
16
- afterEach(async () => {
17
- await server.stop()
18
- })
19
-
20
- beforeEach(() => {
21
- jest.mocked(getFormMetadata).mockResolvedValue(fixtures.form.metadata)
22
- })
23
-
24
- test('shows the server phase tag by default', async () => {
25
- const basePath = '/phase-default'
26
-
27
- server = await createServer({
28
- formFileName: 'phase-default.json',
29
- formFilePath: join(import.meta.dirname, 'definitions')
30
- })
31
-
32
- await server.initialize()
33
-
34
- await renderResponse(server, {
35
- url: `${basePath}/first-page`
36
- })
37
-
38
- const $phaseBanner = /** @type {HTMLElement} */ (
39
- document.querySelector('.govuk-phase-banner')
40
- )
41
-
42
- const $phaseTag = within($phaseBanner).getByRole('strong')
43
- expect($phaseTag).toHaveTextContent('Beta')
44
- })
45
-
46
- test('shows the form phase tag if provided', async () => {
47
- const basePath = '/phase-alpha'
48
-
49
- server = await createServer({
50
- formFileName: 'phase-alpha.json',
51
- formFilePath: join(import.meta.dirname, 'definitions')
52
- })
53
-
54
- await server.initialize()
55
-
56
- await renderResponse(server, {
57
- url: `${basePath}/first-page`
58
- })
59
-
60
- const $phaseBanner = /** @type {HTMLElement} */ (
61
- document.querySelector('.govuk-phase-banner')
62
- )
63
-
64
- const $phaseTag = within($phaseBanner).getByRole('strong')
65
- expect($phaseTag).toHaveTextContent('Alpha')
66
- })
67
- })
68
-
69
- /**
70
- * @import { Server } from '@hapi/hapi'
71
- */
@@ -1,628 +0,0 @@
1
- import crypto from 'node:crypto'
2
- import { resolve } from 'node:path'
3
-
4
- import { hasRepeater } from '@defra/forms-model'
5
- import { within } from '@testing-library/dom'
6
- import { StatusCodes } from 'http-status-codes'
7
-
8
- import { createServer } from '~/src/server/index.js'
9
- import { isRepeatState } from '~/src/server/plugins/engine/components/FormComponent.js'
10
- import { submit } from '~/src/server/plugins/engine/services/formSubmissionService.js'
11
- import { getFormMetadata } from '~/src/server/plugins/engine/services/formsService.js'
12
- import { FormAction } from '~/src/server/routes/types.js'
13
- import * as fixtures from '~/test/fixtures/index.js'
14
- import { renderResponse } from '~/test/helpers/component-helpers.js'
15
- import { getCookieHeader } from '~/test/utils/get-cookie.js'
16
-
17
- jest.mock('~/src/server/utils/notify.ts')
18
- jest.mock('~/src/server/plugins/engine/services/formsService.js')
19
- jest.mock('~/src/server/plugins/engine/services/formSubmissionService.js')
20
-
21
- const basePath = '/repeat'
22
-
23
- /**
24
- * POST a new repeat item
25
- * @param {Server} server
26
- * @param {PageRepeat} repeatPage
27
- * @param {number} [expectedItemCount]
28
- * @param {OutgoingHttpHeaders} [headers]
29
- */
30
- async function createRepeatItem(
31
- server,
32
- repeatPage,
33
- expectedItemCount = 1,
34
- headers,
35
- itemId = crypto.randomUUID()
36
- ) {
37
- // Visit the item page
38
- await server.inject({
39
- url: `${basePath}/pizza-order/${itemId}`,
40
- headers
41
- })
42
-
43
- const res1 = await server.inject({
44
- url: `${basePath}/pizza-order/${itemId}`,
45
- method: 'POST',
46
- headers,
47
- payload: {
48
- toppings: 'Ham',
49
- quantity: 2
50
- }
51
- })
52
-
53
- expect(res1.statusCode).toBe(StatusCodes.SEE_OTHER)
54
- expect(res1.headers.location).toBe(`${basePath}/pizza-order/summary`)
55
-
56
- // Extract the session cookie
57
- const request = res1.request
58
- const { cacheService } = request.services([])
59
-
60
- const { name } = repeatPage.repeat.options
61
- const state = await cacheService.getState(request)
62
-
63
- if (!isRepeatState(state[name]) || state[name].length !== expectedItemCount) {
64
- throw new Error('Unexpected list state')
65
- }
66
-
67
- const item = state[name].at(-1)
68
-
69
- if (!item) {
70
- throw new Error('No item state found')
71
- }
72
-
73
- return {
74
- item,
75
- headers: headers ?? getCookieHeader(res1, 'session'),
76
- redirectLocation: res1.headers.location
77
- }
78
- }
79
-
80
- describe('Repeat GET tests', () => {
81
- /** @type {Server} */
82
- let server
83
-
84
- /** @type {PageRepeat} */
85
- let repeatPage
86
-
87
- // Create server before each test
88
- beforeAll(async () => {
89
- server = await createServer({
90
- formFileName: 'repeat.js',
91
- formFilePath: resolve(import.meta.dirname, '../form/definitions')
92
- })
93
-
94
- const model = server.app.model
95
- const page = model?.def.pages[0]
96
-
97
- if (!hasRepeater(page)) {
98
- throw new Error('Unexpected controller type')
99
- }
100
-
101
- repeatPage = page
102
-
103
- await server.initialize()
104
- })
105
-
106
- beforeEach(() => {
107
- jest.mocked(getFormMetadata).mockResolvedValue(fixtures.form.metadata)
108
- })
109
-
110
- afterAll(async () => {
111
- await server.stop()
112
- })
113
-
114
- test('GET /pizza-order returns 302 to add another', async () => {
115
- const res = await server.inject({
116
- url: `${basePath}/pizza-order`
117
- })
118
-
119
- expect(res.statusCode).toBe(StatusCodes.MOVED_TEMPORARILY)
120
- expect(res.headers.location).toMatch(/^\/repeat\/pizza-order\/[0-9a-f-]+$/)
121
- })
122
-
123
- test('GET /pizza-order with 1 item returns 302 to repeater summary', async () => {
124
- const { headers } = await createRepeatItem(server, repeatPage, 1)
125
-
126
- const res = await server.inject({
127
- url: `${basePath}/pizza-order`,
128
- headers
129
- })
130
-
131
- expect(res.statusCode).toBe(StatusCodes.MOVED_TEMPORARILY)
132
- expect(res.headers.location).toMatch(`${basePath}/pizza-order/summary`)
133
- })
134
-
135
- test('GET /pizza-order with 2 items returns 302 to repeater summary', async () => {
136
- const { headers } = await createRepeatItem(server, repeatPage, 1)
137
-
138
- await createRepeatItem(server, repeatPage, 2, headers)
139
-
140
- const res = await server.inject({
141
- url: `${basePath}/pizza-order`,
142
- headers
143
- })
144
-
145
- expect(res.statusCode).toBe(StatusCodes.MOVED_TEMPORARILY)
146
- expect(res.headers.location).toMatch(`${basePath}/pizza-order/summary`)
147
- })
148
-
149
- test('GET /pizza-order with 3 items returns 302 to repeater summary', async () => {
150
- const { headers } = await createRepeatItem(server, repeatPage, 1)
151
-
152
- await createRepeatItem(server, repeatPage, 2, headers)
153
- await createRepeatItem(server, repeatPage, 3, headers)
154
-
155
- const res = await server.inject({
156
- url: `${basePath}/pizza-order`,
157
- headers
158
- })
159
-
160
- expect(res.statusCode).toBe(StatusCodes.MOVED_TEMPORARILY)
161
- expect(res.headers.location).toBe(`${basePath}/pizza-order/summary`)
162
- })
163
-
164
- test('GET /pizza-order/summary returns 302 to add another', async () => {
165
- const res = await server.inject({
166
- url: `${basePath}/pizza-order/summary`
167
- })
168
-
169
- expect(res.statusCode).toBe(StatusCodes.MOVED_TEMPORARILY)
170
- expect(res.headers.location).toMatch(/^\/repeat\/pizza-order\/[0-9a-f-]+$/)
171
- })
172
-
173
- test('GET /pizza-order/{id} returns 200', async () => {
174
- const { item, headers } = await createRepeatItem(server, repeatPage)
175
-
176
- const { container, response } = await renderResponse(server, {
177
- url: `${basePath}/pizza-order/${item.itemId}`,
178
- headers
179
- })
180
-
181
- expect(response.statusCode).toBe(StatusCodes.OK)
182
-
183
- const $heading1 = container.getByRole('heading', {
184
- name: 'Pizza order',
185
- level: 1
186
- })
187
-
188
- const $heading2 = container.getByRole('heading', {
189
- name: 'Food: Pizza 1',
190
- level: 2
191
- })
192
-
193
- expect($heading1).toBeInTheDocument()
194
- expect($heading1).toHaveClass('govuk-heading-l')
195
-
196
- expect($heading2).toBeInTheDocument()
197
- expect($heading2).toHaveClass('govuk-caption-l')
198
- })
199
-
200
- test('GET /pizza-order/{id} returns 200 with back link', async () => {
201
- const { item, headers, redirectLocation } = await createRepeatItem(
202
- server,
203
- repeatPage
204
- )
205
-
206
- if (!redirectLocation) {
207
- throw new Error('Unexpected empty redirectLocation')
208
- }
209
-
210
- // Visit the summary page
211
- await server.inject({
212
- url: redirectLocation,
213
- headers
214
- })
215
-
216
- const { container, response } = await renderResponse(server, {
217
- url: `${basePath}/pizza-order/${item.itemId}?returnUrl=${encodeURIComponent(`${basePath}/pizza-order/summary`)}`,
218
- headers
219
- })
220
-
221
- expect(response.statusCode).toBe(StatusCodes.OK)
222
-
223
- const $heading1 = container.getByRole('heading', {
224
- name: 'Pizza order',
225
- level: 1
226
- })
227
-
228
- const $heading2 = container.getByRole('heading', {
229
- name: 'Food: Pizza 1',
230
- level: 2
231
- })
232
-
233
- expect($heading1).toBeInTheDocument()
234
- expect($heading1).toHaveClass('govuk-heading-l')
235
-
236
- expect($heading2).toBeInTheDocument()
237
- expect($heading2).toHaveClass('govuk-caption-l')
238
-
239
- const $backLink = container.getByRole('link', {
240
- name: 'Go back to add another'
241
- })
242
-
243
- expect($backLink).toBeInTheDocument()
244
- expect($backLink).toHaveAttribute('href', `${basePath}/pizza-order/summary`)
245
- })
246
-
247
- test('GET /pizza-order/{id} with 1 item returns 200', async () => {
248
- const { headers } = await createRepeatItem(server, repeatPage, 1)
249
-
250
- const itemId = '00000000-0000-0000-0000-000000000000'
251
- jest.spyOn(crypto, 'randomUUID').mockReturnValue(itemId)
252
-
253
- const res = await server.inject({
254
- url: `${basePath}/pizza-order/${itemId}`,
255
- headers
256
- })
257
-
258
- expect(res.statusCode).toBe(StatusCodes.OK)
259
- })
260
-
261
- test('GET /pizza-order/{id} with 2 items returns 200', async () => {
262
- const { headers } = await createRepeatItem(server, repeatPage, 1)
263
- await createRepeatItem(server, repeatPage, 2, headers)
264
-
265
- const itemId = '00000000-0000-0000-0000-000000000000'
266
- jest.spyOn(crypto, 'randomUUID').mockReturnValue(itemId)
267
-
268
- const res = await server.inject({
269
- url: `${basePath}/pizza-order/${itemId}`,
270
- headers
271
- })
272
-
273
- expect(res.statusCode).toBe(StatusCodes.OK)
274
- })
275
-
276
- test('GET /pizza-order/{id} with 3 items returns 200', async () => {
277
- const { headers } = await createRepeatItem(server, repeatPage, 1)
278
-
279
- await createRepeatItem(server, repeatPage, 2, headers)
280
- await createRepeatItem(server, repeatPage, 3, headers)
281
-
282
- const itemId = '00000000-0000-0000-0000-000000000000'
283
- jest.spyOn(crypto, 'randomUUID').mockReturnValue(itemId)
284
-
285
- const res = await server.inject({
286
- url: `${basePath}/pizza-order/${itemId}`,
287
- headers
288
- })
289
-
290
- expect(res.statusCode).toBe(StatusCodes.OK)
291
- })
292
-
293
- test('GET /pizza-order/{id}/confirm-delete with 1 item returns 404', async () => {
294
- const { item, headers } = await createRepeatItem(server, repeatPage)
295
-
296
- const res = await server.inject({
297
- url: `${basePath}/pizza-order/${item.itemId}/confirm-delete`,
298
- headers
299
- })
300
-
301
- expect(res.statusCode).toBe(StatusCodes.NOT_FOUND)
302
- })
303
-
304
- test('GET /pizza-order/{id}/confirm-delete with 2 items returns 200', async () => {
305
- const { item, headers } = await createRepeatItem(server, repeatPage)
306
- await createRepeatItem(server, repeatPage, 2, headers)
307
-
308
- const res = await server.inject({
309
- url: `${basePath}/pizza-order/${item.itemId}/confirm-delete`,
310
- headers
311
- })
312
-
313
- expect(res.statusCode).toBe(StatusCodes.OK)
314
- })
315
-
316
- test('GET /pizza-order/{id}/confirm-delete with unknown itemId returns 404', async () => {
317
- const res = await server.inject({
318
- url: `${basePath}/pizza-order/00000000-0000-0000-0000-000000000000/confirm-delete`
319
- })
320
-
321
- expect(res.statusCode).toBe(StatusCodes.NOT_FOUND)
322
- })
323
-
324
- test('GET /pizza-order/summary with 1 item returns 200', async () => {
325
- const { headers } = await createRepeatItem(server, repeatPage)
326
-
327
- const res = await server.inject({
328
- url: `${basePath}/pizza-order/summary`,
329
- headers
330
- })
331
-
332
- expect(res.statusCode).toBe(StatusCodes.OK)
333
- })
334
-
335
- test('GET /pizza-order/summary with 2 items returns 200', async () => {
336
- const { headers } = await createRepeatItem(server, repeatPage)
337
- await createRepeatItem(server, repeatPage, 2, headers)
338
-
339
- const res = await server.inject({
340
- url: `${basePath}/pizza-order/summary`,
341
- headers
342
- })
343
-
344
- expect(res.statusCode).toBe(StatusCodes.OK)
345
- })
346
- })
347
-
348
- describe('Repeat POST tests', () => {
349
- /** @type {Server} */
350
- let server
351
-
352
- /** @type {PageRepeat} */
353
- let repeatPage
354
-
355
- // Create server before each test
356
- beforeAll(async () => {
357
- server = await createServer({
358
- formFileName: 'repeat.js',
359
- formFilePath: resolve(import.meta.dirname, '../form/definitions')
360
- })
361
-
362
- const model = server.app.model
363
- const page = model?.def.pages[0]
364
-
365
- if (!hasRepeater(page)) {
366
- throw new Error('Unexpected controller type')
367
- }
368
-
369
- repeatPage = page
370
-
371
- await server.initialize()
372
- })
373
-
374
- beforeEach(() => {
375
- jest.mocked(getFormMetadata).mockResolvedValue(fixtures.form.metadata)
376
- })
377
-
378
- afterAll(async () => {
379
- await server.stop()
380
- })
381
-
382
- test('POST /pizza-order/{id} returns 303', async () => {
383
- const { item, headers } = await createRepeatItem(server, repeatPage)
384
-
385
- const res = await server.inject({
386
- url: `${basePath}/pizza-order/${item.itemId}`,
387
- method: 'POST',
388
- headers,
389
- payload: {
390
- toppings: 'Ham',
391
- quantity: 3
392
- }
393
- })
394
-
395
- expect(res.statusCode).toBe(StatusCodes.SEE_OTHER)
396
- expect(res.headers.location).toMatch(/^\/repeat\/pizza-order\/summary?/)
397
- })
398
-
399
- test('POST /pizza-order/{id}/confirm-delete with 1 item returns 404', async () => {
400
- const { item, headers } = await createRepeatItem(server, repeatPage)
401
-
402
- const res = await server.inject({
403
- url: `${basePath}/pizza-order/${item.itemId}/confirm-delete`,
404
- method: 'POST',
405
- headers,
406
- payload: {
407
- confirm: true
408
- }
409
- })
410
-
411
- expect(res.statusCode).toBe(StatusCodes.NOT_FOUND)
412
- })
413
-
414
- test('POST /pizza-order/{id}/confirm-delete with 2 items returns 303', async () => {
415
- const { item, headers } = await createRepeatItem(server, repeatPage)
416
- await createRepeatItem(server, repeatPage, 2, headers)
417
-
418
- const res = await server.inject({
419
- url: `${basePath}/pizza-order/${item.itemId}/confirm-delete`,
420
- method: 'POST',
421
- headers,
422
- payload: {
423
- confirm: true
424
- }
425
- })
426
-
427
- expect(res.statusCode).toBe(StatusCodes.SEE_OTHER)
428
- expect(res.headers.location).toMatch(/^\/repeat\/pizza-order\/summary/)
429
- })
430
-
431
- test('POST /pizza-order/summary ADD_ANOTHER returns 303', async () => {
432
- const itemId = '00000000-0000-0000-0000-000000000000'
433
- jest.spyOn(crypto, 'randomUUID').mockReturnValue(itemId)
434
-
435
- const res = await server.inject({
436
- url: `${basePath}/pizza-order/summary`,
437
- method: 'POST',
438
- payload: {
439
- action: FormAction.AddAnother
440
- }
441
- })
442
-
443
- expect(res.statusCode).toBe(StatusCodes.SEE_OTHER)
444
- expect(res.headers.location).toBe(`${basePath}/pizza-order/${itemId}`)
445
- })
446
-
447
- test('POST /pizza-order/summary with 1 item ADD_ANOTHER returns 303', async () => {
448
- const itemId = '00000000-0000-0000-0000-000000000000'
449
- jest.spyOn(crypto, 'randomUUID').mockReturnValue(itemId)
450
-
451
- const { headers } = await createRepeatItem(server, repeatPage)
452
-
453
- const res = await server.inject({
454
- url: `${basePath}/pizza-order/summary`,
455
- method: 'POST',
456
- headers,
457
- payload: {
458
- action: FormAction.AddAnother
459
- }
460
- })
461
-
462
- expect(res.statusCode).toBe(StatusCodes.SEE_OTHER)
463
- expect(res.headers.location).toBe(`${basePath}/pizza-order/${itemId}`)
464
- })
465
-
466
- test('POST /pizza-order/summary with 2 items ADD_ANOTHER returns 303', async () => {
467
- const itemId = '00000000-0000-0000-0000-000000000000'
468
- jest.spyOn(crypto, 'randomUUID').mockReturnValue(itemId)
469
-
470
- const { headers } = await createRepeatItem(server, repeatPage)
471
-
472
- await createRepeatItem(server, repeatPage, 1, headers)
473
-
474
- const res = await server.inject({
475
- url: `${basePath}/pizza-order/summary`,
476
- method: 'POST',
477
- headers,
478
- payload: {
479
- action: FormAction.AddAnother
480
- }
481
- })
482
-
483
- expect(res.statusCode).toBe(StatusCodes.SEE_OTHER)
484
- expect(res.headers.location).toBe(`${basePath}/pizza-order/${itemId}`)
485
- })
486
-
487
- test('POST /pizza-order/summary with 3 items ADD_ANOTHER returns 200 with errors over schema.max', async () => {
488
- const { headers } = await createRepeatItem(server, repeatPage)
489
-
490
- await createRepeatItem(server, repeatPage, 2, headers)
491
- await createRepeatItem(server, repeatPage, 3, headers)
492
-
493
- const { container, response } = await renderResponse(server, {
494
- url: `${basePath}/pizza-order/summary`,
495
- method: 'POST',
496
- headers,
497
- payload: {
498
- action: FormAction.AddAnother
499
- }
500
- })
501
-
502
- expect(response.statusCode).toBe(StatusCodes.OK)
503
-
504
- const $errorSummary = container.getByRole('alert')
505
- const $errorItems = within($errorSummary).getAllByRole('listitem')
506
-
507
- const $heading = within($errorSummary).getByRole('heading', {
508
- name: 'There is a problem',
509
- level: 2
510
- })
511
-
512
- expect($heading).toBeInTheDocument()
513
- expect($errorItems[0]).toHaveTextContent('You can only add up to 3 Pizzas')
514
- })
515
-
516
- test('POST /pizza-order/summary CONTINUE with 1 item returns 200 with errors under schema.min', async () => {
517
- const { headers } = await createRepeatItem(server, repeatPage)
518
-
519
- const { container, response } = await renderResponse(server, {
520
- url: `${basePath}/pizza-order/summary`,
521
- method: 'POST',
522
- headers,
523
- payload: {
524
- action: FormAction.Continue
525
- }
526
- })
527
-
528
- expect(response.statusCode).toBe(StatusCodes.OK)
529
-
530
- const $errorSummary = container.getByRole('alert')
531
- const $errorItems = within($errorSummary).getAllByRole('listitem')
532
-
533
- const $heading = within($errorSummary).getByRole('heading', {
534
- name: 'There is a problem',
535
- level: 2
536
- })
537
-
538
- expect($heading).toBeInTheDocument()
539
- expect($errorItems[0]).toHaveTextContent('You must add at least 2 Pizzas')
540
- })
541
-
542
- test('POST /pizza-order/summary with 2 items CONTINUE returns 303 to /summary', async () => {
543
- const { headers } = await createRepeatItem(server, repeatPage)
544
-
545
- await createRepeatItem(server, repeatPage, 2, headers)
546
-
547
- const res1 = await server.inject({
548
- url: `${basePath}/pizza-order/summary`,
549
- method: 'POST',
550
- headers,
551
- payload: {
552
- action: FormAction.Continue
553
- }
554
- })
555
-
556
- expect(res1.statusCode).toBe(StatusCodes.SEE_OTHER)
557
- expect(res1.headers.location).toBe(`${basePath}/summary`)
558
-
559
- const { container, response: res2 } = await renderResponse(server, {
560
- url: `${basePath}/summary`,
561
- headers
562
- })
563
-
564
- expect(res2.statusCode).toBe(StatusCodes.OK)
565
-
566
- const $values = container
567
- .getAllByRole('definition')
568
- .filter(({ classList }) =>
569
- classList.contains('govuk-summary-list__value')
570
- )
571
-
572
- expect($values[0]).toHaveTextContent('You added 2 Pizzas')
573
-
574
- // POST the summary page
575
- await server.inject({
576
- url: `${basePath}/summary`,
577
- method: 'POST',
578
- headers,
579
- payload: {
580
- action: FormAction.Send
581
- }
582
- })
583
-
584
- expect(submit).toHaveBeenCalledWith({
585
- main: [],
586
- repeaters: [
587
- {
588
- name: 'pizza',
589
- title: 'Pizza',
590
- value: [
591
- [
592
- {
593
- name: 'toppings',
594
- title: 'Toppings',
595
- value: 'Ham'
596
- },
597
- {
598
- name: 'quantity',
599
- title: 'Quantity',
600
- value: '2'
601
- }
602
- ],
603
- [
604
- {
605
- name: 'toppings',
606
- title: 'Toppings',
607
- value: 'Ham'
608
- },
609
- {
610
- name: 'quantity',
611
- title: 'Quantity',
612
- value: '2'
613
- }
614
- ]
615
- ]
616
- }
617
- ],
618
- retrievalKey: 'enrique.chase@defra.gov.uk',
619
- sessionId: expect.any(String)
620
- })
621
- })
622
- })
623
-
624
- /**
625
- * @import { Server } from '@hapi/hapi'
626
- * @import { PageRepeat } from '@defra/forms-model'
627
- * @import { OutgoingHttpHeaders } from 'node:http'
628
- */