@defra/forms-engine-plugin 0.0.2 → 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 (330) 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/.server/server/plugins/nunjucks/environment.js +3 -3
  25. package/.server/server/plugins/nunjucks/environment.js.map +1 -1
  26. package/package.json +6 -2
  27. package/.browserslistrc +0 -16
  28. package/.editorconfig +0 -9
  29. package/.eslintrc.cjs +0 -266
  30. package/.github/dependabot.yml +0 -85
  31. package/.github/workflows/check-pull-request.yml +0 -209
  32. package/.github/workflows/pr-notifier.yml +0 -98
  33. package/.github/workflows/publish.yml +0 -111
  34. package/.husky/pre-commit +0 -1
  35. package/.lintstagedrc.js +0 -4
  36. package/.nvmrc +0 -1
  37. package/.prettierignore +0 -8
  38. package/.prettierrc.cjs +0 -26
  39. package/Dockerfile +0 -61
  40. package/Procfile +0 -1
  41. package/babel.config.cjs +0 -55
  42. package/compose/aws.env +0 -4
  43. package/compose/start-localstack.sh +0 -26
  44. package/docker-compose.yaml +0 -86
  45. package/globals.d.ts +0 -1
  46. package/jest.config.cjs +0 -54
  47. package/jest.environment.js +0 -4
  48. package/jest.setup.cjs +0 -14
  49. package/postcss.config.js +0 -26
  50. package/sonar-project.properties +0 -17
  51. package/src/client/javascripts/application.js +0 -87
  52. package/src/client/javascripts/file-upload.js +0 -386
  53. package/src/client/stylesheets/_code.scss +0 -33
  54. package/src/client/stylesheets/_govuk-frontend.scss +0 -4
  55. package/src/client/stylesheets/_prose.scss +0 -56
  56. package/src/client/stylesheets/_service-banner.scss +0 -24
  57. package/src/client/stylesheets/_summary-list.scss +0 -28
  58. package/src/client/stylesheets/_tag-env.scss +0 -24
  59. package/src/client/stylesheets/application.scss +0 -14
  60. package/src/common/cookies.js +0 -58
  61. package/src/common/cookies.test.js +0 -23
  62. package/src/common/types.js +0 -5
  63. package/src/config/index.ts +0 -271
  64. package/src/index.ts +0 -31
  65. package/src/server/common/helpers/logging/logger-options.test.ts +0 -50
  66. package/src/server/common/helpers/logging/logger-options.ts +0 -46
  67. package/src/server/common/helpers/logging/logger.ts +0 -7
  68. package/src/server/common/helpers/logging/request-logger.ts +0 -9
  69. package/src/server/common/helpers/logging/request-tracing.js +0 -10
  70. package/src/server/common/helpers/redis-client.js +0 -70
  71. package/src/server/constants.js +0 -1
  72. package/src/server/forms/README.md +0 -10
  73. package/src/server/forms/components.json +0 -1015
  74. package/src/server/forms/report-a-terrorist.json +0 -270
  75. package/src/server/forms/runner-components-test.json +0 -365
  76. package/src/server/forms/test.json +0 -581
  77. package/src/server/index.test.ts +0 -582
  78. package/src/server/index.ts +0 -140
  79. package/src/server/plugins/blankie.test.ts +0 -73
  80. package/src/server/plugins/blankie.ts +0 -48
  81. package/src/server/plugins/crumb.ts +0 -20
  82. package/src/server/plugins/engine/README.md +0 -87
  83. package/src/server/plugins/engine/components/AutocompleteField.test.ts +0 -294
  84. package/src/server/plugins/engine/components/AutocompleteField.ts +0 -49
  85. package/src/server/plugins/engine/components/CheckboxesField.test.ts +0 -379
  86. package/src/server/plugins/engine/components/CheckboxesField.ts +0 -106
  87. package/src/server/plugins/engine/components/ComponentBase.ts +0 -97
  88. package/src/server/plugins/engine/components/ComponentCollection.ts +0 -278
  89. package/src/server/plugins/engine/components/DatePartsField.test.ts +0 -822
  90. package/src/server/plugins/engine/components/DatePartsField.ts +0 -264
  91. package/src/server/plugins/engine/components/Details.test.ts +0 -49
  92. package/src/server/plugins/engine/components/Details.ts +0 -30
  93. package/src/server/plugins/engine/components/EmailAddressField.test.ts +0 -395
  94. package/src/server/plugins/engine/components/EmailAddressField.ts +0 -55
  95. package/src/server/plugins/engine/components/FileUploadField.test.ts +0 -778
  96. package/src/server/plugins/engine/components/FileUploadField.ts +0 -262
  97. package/src/server/plugins/engine/components/FormComponent.ts +0 -249
  98. package/src/server/plugins/engine/components/Html.test.ts +0 -48
  99. package/src/server/plugins/engine/components/Html.ts +0 -29
  100. package/src/server/plugins/engine/components/InsetText.test.ts +0 -48
  101. package/src/server/plugins/engine/components/InsetText.ts +0 -27
  102. package/src/server/plugins/engine/components/List.test.ts +0 -76
  103. package/src/server/plugins/engine/components/List.ts +0 -72
  104. package/src/server/plugins/engine/components/ListFormComponent.ts +0 -140
  105. package/src/server/plugins/engine/components/MonthYearField.test.ts +0 -567
  106. package/src/server/plugins/engine/components/MonthYearField.ts +0 -222
  107. package/src/server/plugins/engine/components/MultilineTextField.test.ts +0 -558
  108. package/src/server/plugins/engine/components/MultilineTextField.ts +0 -138
  109. package/src/server/plugins/engine/components/NumberField.test.ts +0 -701
  110. package/src/server/plugins/engine/components/NumberField.ts +0 -163
  111. package/src/server/plugins/engine/components/RadiosField.test.ts +0 -288
  112. package/src/server/plugins/engine/components/RadiosField.ts +0 -24
  113. package/src/server/plugins/engine/components/SelectField.test.ts +0 -288
  114. package/src/server/plugins/engine/components/SelectField.ts +0 -47
  115. package/src/server/plugins/engine/components/SelectionControlField.ts +0 -43
  116. package/src/server/plugins/engine/components/TelephoneNumberField.test.ts +0 -356
  117. package/src/server/plugins/engine/components/TelephoneNumberField.ts +0 -67
  118. package/src/server/plugins/engine/components/TextField.test.ts +0 -489
  119. package/src/server/plugins/engine/components/TextField.ts +0 -96
  120. package/src/server/plugins/engine/components/UkAddressField.test.ts +0 -623
  121. package/src/server/plugins/engine/components/UkAddressField.ts +0 -172
  122. package/src/server/plugins/engine/components/YesNoField.test.ts +0 -248
  123. package/src/server/plugins/engine/components/YesNoField.ts +0 -31
  124. package/src/server/plugins/engine/components/constants.ts +0 -1
  125. package/src/server/plugins/engine/components/helpers.ts +0 -330
  126. package/src/server/plugins/engine/components/index.ts +0 -24
  127. package/src/server/plugins/engine/components/types.ts +0 -117
  128. package/src/server/plugins/engine/configureEnginePlugin.ts +0 -47
  129. package/src/server/plugins/engine/helpers.test.ts +0 -791
  130. package/src/server/plugins/engine/helpers.ts +0 -379
  131. package/src/server/plugins/engine/index.ts +0 -7
  132. package/src/server/plugins/engine/models/FormModel.test.ts +0 -42
  133. package/src/server/plugins/engine/models/FormModel.ts +0 -443
  134. package/src/server/plugins/engine/models/RepeatingSummaryViewModel.ts +0 -0
  135. package/src/server/plugins/engine/models/Section.ts +0 -0
  136. package/src/server/plugins/engine/models/SummaryViewModel.test.ts +0 -209
  137. package/src/server/plugins/engine/models/SummaryViewModel.ts +0 -220
  138. package/src/server/plugins/engine/models/index.ts +0 -2
  139. package/src/server/plugins/engine/models/types.ts +0 -114
  140. package/src/server/plugins/engine/outputFormatters/human/v1.test.ts +0 -143
  141. package/src/server/plugins/engine/outputFormatters/human/v1.ts +0 -73
  142. package/src/server/plugins/engine/outputFormatters/index.test.ts +0 -17
  143. package/src/server/plugins/engine/outputFormatters/index.ts +0 -44
  144. package/src/server/plugins/engine/outputFormatters/machine/v1.test.ts +0 -229
  145. package/src/server/plugins/engine/outputFormatters/machine/v1.ts +0 -140
  146. package/src/server/plugins/engine/outputFormatters/machine/v2.test.ts +0 -229
  147. package/src/server/plugins/engine/outputFormatters/machine/v2.ts +0 -153
  148. package/src/server/plugins/engine/pageControllers/FileUploadPageController.test.ts +0 -1108
  149. package/src/server/plugins/engine/pageControllers/FileUploadPageController.ts +0 -446
  150. package/src/server/plugins/engine/pageControllers/PageController.test.ts +0 -205
  151. package/src/server/plugins/engine/pageControllers/PageController.ts +0 -176
  152. package/src/server/plugins/engine/pageControllers/QuestionPageController.test.ts +0 -1264
  153. package/src/server/plugins/engine/pageControllers/QuestionPageController.ts +0 -561
  154. package/src/server/plugins/engine/pageControllers/README.md +0 -28
  155. package/src/server/plugins/engine/pageControllers/RepeatPageController.test.ts +0 -264
  156. package/src/server/plugins/engine/pageControllers/RepeatPageController.ts +0 -458
  157. package/src/server/plugins/engine/pageControllers/StartPageController.ts +0 -18
  158. package/src/server/plugins/engine/pageControllers/StatusPageController.ts +0 -50
  159. package/src/server/plugins/engine/pageControllers/SummaryPageController.ts +0 -261
  160. package/src/server/plugins/engine/pageControllers/TerminalController.test.ts +0 -28
  161. package/src/server/plugins/engine/pageControllers/TerminalPageController.ts +0 -19
  162. package/src/server/plugins/engine/pageControllers/helpers.test.ts +0 -198
  163. package/src/server/plugins/engine/pageControllers/helpers.ts +0 -101
  164. package/src/server/plugins/engine/pageControllers/index.ts +0 -10
  165. package/src/server/plugins/engine/pageControllers/validationOptions.ts +0 -89
  166. package/src/server/plugins/engine/plugin.ts +0 -673
  167. package/src/server/plugins/engine/services/formSubmissionService.js +0 -46
  168. package/src/server/plugins/engine/services/formsService.js +0 -46
  169. package/src/server/plugins/engine/services/formsService.test.js +0 -90
  170. package/src/server/plugins/engine/services/index.js +0 -3
  171. package/src/server/plugins/engine/services/notifyService.test.ts +0 -132
  172. package/src/server/plugins/engine/services/notifyService.ts +0 -64
  173. package/src/server/plugins/engine/services/uploadService.js +0 -60
  174. package/src/server/plugins/engine/types.ts +0 -315
  175. package/src/server/plugins/engine/views/components/autocompletefield.html +0 -5
  176. package/src/server/plugins/engine/views/components/checkboxesfield.html +0 -5
  177. package/src/server/plugins/engine/views/components/datepartsfield.html +0 -5
  178. package/src/server/plugins/engine/views/components/details.html +0 -6
  179. package/src/server/plugins/engine/views/components/emailaddressfield.html +0 -5
  180. package/src/server/plugins/engine/views/components/fileuploadfield-key.html +0 -8
  181. package/src/server/plugins/engine/views/components/fileuploadfield-value.html +0 -3
  182. package/src/server/plugins/engine/views/components/fileuploadfield.html +0 -24
  183. package/src/server/plugins/engine/views/components/html.html +0 -3
  184. package/src/server/plugins/engine/views/components/insettext.html +0 -7
  185. package/src/server/plugins/engine/views/components/list.html +0 -36
  186. package/src/server/plugins/engine/views/components/monthyearfield.html +0 -5
  187. package/src/server/plugins/engine/views/components/multilinetextfield.html +0 -10
  188. package/src/server/plugins/engine/views/components/numberfield.html +0 -5
  189. package/src/server/plugins/engine/views/components/radiosfield.html +0 -5
  190. package/src/server/plugins/engine/views/components/selectfield.html +0 -5
  191. package/src/server/plugins/engine/views/components/telephonenumberfield.html +0 -5
  192. package/src/server/plugins/engine/views/components/textfield.html +0 -5
  193. package/src/server/plugins/engine/views/components/ukaddressfield.html +0 -25
  194. package/src/server/plugins/engine/views/components/yesnofield.html +0 -5
  195. package/src/server/plugins/engine/views/file-upload.html +0 -45
  196. package/src/server/plugins/engine/views/index.html +0 -39
  197. package/src/server/plugins/engine/views/item-delete.html +0 -56
  198. package/src/server/plugins/engine/views/partials/components.html +0 -6
  199. package/src/server/plugins/engine/views/partials/conditional-components.html +0 -3
  200. package/src/server/plugins/engine/views/partials/debug.html +0 -44
  201. package/src/server/plugins/engine/views/partials/form.html +0 -15
  202. package/src/server/plugins/engine/views/partials/heading.html +0 -16
  203. package/src/server/plugins/engine/views/partials/preview-banner.html +0 -32
  204. package/src/server/plugins/engine/views/partials/preview-banner.test.js +0 -122
  205. package/src/server/plugins/engine/views/partials/warn-missing-notification-email.html +0 -10
  206. package/src/server/plugins/engine/views/repeat-list-summary.html +0 -53
  207. package/src/server/plugins/errorPages.ts +0 -58
  208. package/src/server/plugins/nunjucks/context.js +0 -88
  209. package/src/server/plugins/nunjucks/context.test.js +0 -142
  210. package/src/server/plugins/nunjucks/enviroment.test.js +0 -201
  211. package/src/server/plugins/nunjucks/environment.js +0 -116
  212. package/src/server/plugins/nunjucks/filters/answer.js +0 -27
  213. package/src/server/plugins/nunjucks/filters/answer.test.js +0 -89
  214. package/src/server/plugins/nunjucks/filters/evaluate.js +0 -21
  215. package/src/server/plugins/nunjucks/filters/field.js +0 -28
  216. package/src/server/plugins/nunjucks/filters/field.test.js +0 -75
  217. package/src/server/plugins/nunjucks/filters/highlight.js +0 -11
  218. package/src/server/plugins/nunjucks/filters/href.js +0 -30
  219. package/src/server/plugins/nunjucks/filters/href.test.js +0 -80
  220. package/src/server/plugins/nunjucks/filters/index.js +0 -8
  221. package/src/server/plugins/nunjucks/filters/inspect.js +0 -15
  222. package/src/server/plugins/nunjucks/filters/page.js +0 -24
  223. package/src/server/plugins/nunjucks/filters/page.test.js +0 -65
  224. package/src/server/plugins/nunjucks/index.js +0 -3
  225. package/src/server/plugins/nunjucks/plugin.js +0 -40
  226. package/src/server/plugins/nunjucks/render.js +0 -42
  227. package/src/server/plugins/nunjucks/types.js +0 -40
  228. package/src/server/plugins/pulse.ts +0 -11
  229. package/src/server/plugins/router.ts +0 -201
  230. package/src/server/plugins/session.ts +0 -28
  231. package/src/server/routes/health.js +0 -13
  232. package/src/server/routes/health.test.js +0 -35
  233. package/src/server/routes/index.test.ts +0 -125
  234. package/src/server/routes/index.ts +0 -2
  235. package/src/server/routes/public.ts +0 -47
  236. package/src/server/routes/types.ts +0 -48
  237. package/src/server/schemas/index.ts +0 -34
  238. package/src/server/secure-context.js +0 -43
  239. package/src/server/services/cacheService.test.ts +0 -276
  240. package/src/server/services/cacheService.ts +0 -131
  241. package/src/server/services/httpService.test.js +0 -491
  242. package/src/server/services/httpService.ts +0 -50
  243. package/src/server/services/index.ts +0 -1
  244. package/src/server/types.ts +0 -54
  245. package/src/server/utils/notify.test.ts +0 -37
  246. package/src/server/utils/notify.ts +0 -50
  247. package/src/server/utils/secure-context/get-trust-store-certs.js +0 -11
  248. package/src/server/utils/secure-context/get-trust-store-certs.test.js +0 -19
  249. package/src/server/utils/utils.js +0 -24
  250. package/src/server/utils/utils.test.js +0 -54
  251. package/src/server/views/404.html +0 -16
  252. package/src/server/views/500.html +0 -19
  253. package/src/server/views/components/debug/macro.njk +0 -3
  254. package/src/server/views/components/debug/template.njk +0 -13
  255. package/src/server/views/components/service-banner/macro.njk +0 -3
  256. package/src/server/views/components/service-banner/template.njk +0 -20
  257. package/src/server/views/components/service-banner/template.test.js +0 -43
  258. package/src/server/views/components/tag-env/macro.njk +0 -3
  259. package/src/server/views/components/tag-env/template.njk +0 -30
  260. package/src/server/views/components/tag-env/template.test.js +0 -66
  261. package/src/server/views/confirmation.html +0 -19
  262. package/src/server/views/help/accessibility-statement.html +0 -58
  263. package/src/server/views/help/cookie-preferences.html +0 -57
  264. package/src/server/views/help/cookies.html +0 -71
  265. package/src/server/views/help/get-support.html +0 -37
  266. package/src/server/views/help/privacy-notice.html +0 -68
  267. package/src/server/views/help/terms-and-conditions.html +0 -83
  268. package/src/server/views/layout.html +0 -199
  269. package/src/server/views/summary.html +0 -50
  270. package/src/typings/hapi/index.d.ts +0 -95
  271. package/src/typings/hapi-tracing/index.d.ts +0 -6
  272. package/src/typings/index.d.ts +0 -3
  273. package/src/typings/joi/index.d.ts +0 -22
  274. package/stylelint.config.js +0 -10
  275. package/test/client/javascripts/file-upload.test.js +0 -1197
  276. package/test/condition/checkboxes.test.js +0 -112
  277. package/test/condition/radios.test.js +0 -112
  278. package/test/condition/text.test.js +0 -103
  279. package/test/fixtures/assets-manifest.json +0 -4
  280. package/test/fixtures/form.js +0 -86
  281. package/test/fixtures/index.js +0 -2
  282. package/test/fixtures/list.js +0 -92
  283. package/test/form/cookies.test.js +0 -338
  284. package/test/form/csrf.test.js +0 -87
  285. package/test/form/definitions/basic.js +0 -101
  286. package/test/form/definitions/blank.js +0 -10
  287. package/test/form/definitions/checkboxes.json +0 -88
  288. package/test/form/definitions/components.json +0 -452
  289. package/test/form/definitions/conditional-reveal.js +0 -140
  290. package/test/form/definitions/conditions-basic.js +0 -187
  291. package/test/form/definitions/conditions-complex.js +0 -338
  292. package/test/form/definitions/conditions-dates.js +0 -78
  293. package/test/form/definitions/conditions-escaping.js +0 -143
  294. package/test/form/definitions/demo-cph-number.js +0 -3099
  295. package/test/form/definitions/feedback.json +0 -45
  296. package/test/form/definitions/fields-optional.js +0 -402
  297. package/test/form/definitions/fields-required.js +0 -402
  298. package/test/form/definitions/file-upload-basic.js +0 -44
  299. package/test/form/definitions/file-upload.js +0 -66
  300. package/test/form/definitions/minimal.js +0 -39
  301. package/test/form/definitions/phase-alpha.json +0 -33
  302. package/test/form/definitions/phase-default.json +0 -26
  303. package/test/form/definitions/radios.json +0 -88
  304. package/test/form/definitions/repeat-mixed.js +0 -54
  305. package/test/form/definitions/repeat.js +0 -70
  306. package/test/form/definitions/status.json +0 -126
  307. package/test/form/definitions/templates.js +0 -183
  308. package/test/form/definitions/test.json +0 -581
  309. package/test/form/definitions/text.json +0 -75
  310. package/test/form/definitions/titles.json +0 -170
  311. package/test/form/definitions.test.js +0 -47
  312. package/test/form/exit-page.test.js +0 -210
  313. package/test/form/feedback.test.js +0 -68
  314. package/test/form/fields-optional.test.js +0 -237
  315. package/test/form/fields-required.test.js +0 -294
  316. package/test/form/file-upload.test.js +0 -313
  317. package/test/form/govuk-notify.test.js +0 -449
  318. package/test/form/journey-basic.test.js +0 -444
  319. package/test/form/persist-files.test.js +0 -227
  320. package/test/form/phase-banner.test.js +0 -71
  321. package/test/form/repeat.test.js +0 -628
  322. package/test/form/summary-submission-email.test.js +0 -95
  323. package/test/form/template.test.js +0 -288
  324. package/test/form/titles.test.js +0 -204
  325. package/test/helpers/component-helpers.js +0 -74
  326. package/test/utils/get-cookie.js +0 -42
  327. package/test/utils/get-form-definitions.js +0 -18
  328. package/tmp.pdf +0 -1
  329. package/tsconfig.json +0 -28
  330. 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
- */