@defra/forms-engine-plugin 0.0.4 → 0.0.6

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 (260) hide show
  1. package/.server/server/index.js +0 -4
  2. package/.server/server/index.js.map +1 -1
  3. package/.server/server/plugins/engine/helpers.js +3 -0
  4. package/.server/server/plugins/engine/helpers.js.map +1 -1
  5. package/.server/server/plugins/engine/index.js +27 -1
  6. package/.server/server/plugins/engine/index.js.map +1 -1
  7. package/.server/server/plugins/engine/pageControllers/FileUploadPageController.js +2 -4
  8. package/.server/server/plugins/engine/pageControllers/FileUploadPageController.js.map +1 -1
  9. package/.server/server/plugins/engine/pageControllers/QuestionPageController.js +4 -10
  10. package/.server/server/plugins/engine/pageControllers/QuestionPageController.js.map +1 -1
  11. package/.server/server/plugins/engine/pageControllers/StatusPageController.js +2 -3
  12. package/.server/server/plugins/engine/pageControllers/StatusPageController.js.map +1 -1
  13. package/.server/server/plugins/engine/pageControllers/SummaryPageController.js +2 -4
  14. package/.server/server/plugins/engine/pageControllers/SummaryPageController.js.map +1 -1
  15. package/.server/server/plugins/engine/plugin.js +65 -6
  16. package/.server/server/plugins/engine/plugin.js.map +1 -1
  17. package/.server/server/plugins/engine/types.js.map +1 -1
  18. package/.server/server/{views → plugins/engine/views}/components/service-banner/template.test.js +1 -1
  19. package/.server/server/plugins/engine/views/components/service-banner/template.test.js.map +1 -0
  20. package/.server/server/{views → plugins/engine/views}/components/tag-env/template.test.js +1 -1
  21. package/.server/server/plugins/engine/views/components/tag-env/template.test.js.map +1 -0
  22. package/.server/server/services/cacheService.js +5 -2
  23. package/.server/server/services/cacheService.js.map +1 -1
  24. package/.server/typings/hapi/index.d.js.map +1 -1
  25. package/README.md +215 -4
  26. package/package.json +3 -3
  27. package/src/client/javascripts/application.js +87 -0
  28. package/src/client/javascripts/file-upload.js +386 -0
  29. package/src/client/stylesheets/_code.scss +33 -0
  30. package/src/client/stylesheets/_govuk-frontend.scss +4 -0
  31. package/src/client/stylesheets/_prose.scss +56 -0
  32. package/src/client/stylesheets/_service-banner.scss +24 -0
  33. package/src/client/stylesheets/_summary-list.scss +28 -0
  34. package/src/client/stylesheets/_tag-env.scss +24 -0
  35. package/src/client/stylesheets/application.scss +14 -0
  36. package/src/common/cookies.js +58 -0
  37. package/src/common/cookies.test.js +23 -0
  38. package/src/common/types.js +5 -0
  39. package/src/config/index.ts +271 -0
  40. package/src/index.ts +31 -0
  41. package/src/server/common/helpers/logging/logger-options.test.ts +50 -0
  42. package/src/server/common/helpers/logging/logger-options.ts +46 -0
  43. package/src/server/common/helpers/logging/logger.ts +7 -0
  44. package/src/server/common/helpers/logging/request-logger.ts +9 -0
  45. package/src/server/common/helpers/logging/request-tracing.js +10 -0
  46. package/src/server/common/helpers/redis-client.js +70 -0
  47. package/src/server/constants.js +1 -0
  48. package/src/server/forms/README.md +10 -0
  49. package/src/server/forms/components.json +1015 -0
  50. package/src/server/forms/report-a-terrorist.json +270 -0
  51. package/src/server/forms/runner-components-test.json +365 -0
  52. package/src/server/forms/test.json +581 -0
  53. package/src/server/index.test.ts +582 -0
  54. package/src/server/index.ts +135 -0
  55. package/src/server/plugins/blankie.test.ts +73 -0
  56. package/src/server/plugins/blankie.ts +48 -0
  57. package/src/server/plugins/crumb.ts +20 -0
  58. package/src/server/plugins/engine/README.md +87 -0
  59. package/src/server/plugins/engine/components/AutocompleteField.test.ts +294 -0
  60. package/src/server/plugins/engine/components/AutocompleteField.ts +49 -0
  61. package/src/server/plugins/engine/components/CheckboxesField.test.ts +379 -0
  62. package/src/server/plugins/engine/components/CheckboxesField.ts +106 -0
  63. package/src/server/plugins/engine/components/ComponentBase.ts +97 -0
  64. package/src/server/plugins/engine/components/ComponentCollection.ts +278 -0
  65. package/src/server/plugins/engine/components/DatePartsField.test.ts +822 -0
  66. package/src/server/plugins/engine/components/DatePartsField.ts +264 -0
  67. package/src/server/plugins/engine/components/Details.test.ts +49 -0
  68. package/src/server/plugins/engine/components/Details.ts +30 -0
  69. package/src/server/plugins/engine/components/EmailAddressField.test.ts +395 -0
  70. package/src/server/plugins/engine/components/EmailAddressField.ts +55 -0
  71. package/src/server/plugins/engine/components/FileUploadField.test.ts +778 -0
  72. package/src/server/plugins/engine/components/FileUploadField.ts +262 -0
  73. package/src/server/plugins/engine/components/FormComponent.ts +249 -0
  74. package/src/server/plugins/engine/components/Html.test.ts +48 -0
  75. package/src/server/plugins/engine/components/Html.ts +29 -0
  76. package/src/server/plugins/engine/components/InsetText.test.ts +48 -0
  77. package/src/server/plugins/engine/components/InsetText.ts +27 -0
  78. package/src/server/plugins/engine/components/List.test.ts +76 -0
  79. package/src/server/plugins/engine/components/List.ts +72 -0
  80. package/src/server/plugins/engine/components/ListFormComponent.ts +140 -0
  81. package/src/server/plugins/engine/components/MonthYearField.test.ts +567 -0
  82. package/src/server/plugins/engine/components/MonthYearField.ts +222 -0
  83. package/src/server/plugins/engine/components/MultilineTextField.test.ts +558 -0
  84. package/src/server/plugins/engine/components/MultilineTextField.ts +138 -0
  85. package/src/server/plugins/engine/components/NumberField.test.ts +701 -0
  86. package/src/server/plugins/engine/components/NumberField.ts +163 -0
  87. package/src/server/plugins/engine/components/RadiosField.test.ts +288 -0
  88. package/src/server/plugins/engine/components/RadiosField.ts +24 -0
  89. package/src/server/plugins/engine/components/SelectField.test.ts +288 -0
  90. package/src/server/plugins/engine/components/SelectField.ts +47 -0
  91. package/src/server/plugins/engine/components/SelectionControlField.ts +43 -0
  92. package/src/server/plugins/engine/components/TelephoneNumberField.test.ts +356 -0
  93. package/src/server/plugins/engine/components/TelephoneNumberField.ts +67 -0
  94. package/src/server/plugins/engine/components/TextField.test.ts +489 -0
  95. package/src/server/plugins/engine/components/TextField.ts +96 -0
  96. package/src/server/plugins/engine/components/UkAddressField.test.ts +623 -0
  97. package/src/server/plugins/engine/components/UkAddressField.ts +172 -0
  98. package/src/server/plugins/engine/components/YesNoField.test.ts +248 -0
  99. package/src/server/plugins/engine/components/YesNoField.ts +31 -0
  100. package/src/server/plugins/engine/components/constants.ts +1 -0
  101. package/src/server/plugins/engine/components/helpers.ts +330 -0
  102. package/src/server/plugins/engine/components/index.ts +24 -0
  103. package/src/server/plugins/engine/components/types.ts +117 -0
  104. package/src/server/plugins/engine/configureEnginePlugin.ts +47 -0
  105. package/src/server/plugins/engine/helpers.test.ts +791 -0
  106. package/src/server/plugins/engine/helpers.ts +384 -0
  107. package/src/server/plugins/engine/index.ts +47 -0
  108. package/src/server/plugins/engine/models/FormModel.test.ts +42 -0
  109. package/src/server/plugins/engine/models/FormModel.ts +443 -0
  110. package/src/server/plugins/engine/models/RepeatingSummaryViewModel.ts +0 -0
  111. package/src/server/plugins/engine/models/Section.ts +0 -0
  112. package/src/server/plugins/engine/models/SummaryViewModel.test.ts +209 -0
  113. package/src/server/plugins/engine/models/SummaryViewModel.ts +220 -0
  114. package/src/server/plugins/engine/models/index.ts +2 -0
  115. package/src/server/plugins/engine/models/types.ts +114 -0
  116. package/src/server/plugins/engine/outputFormatters/human/v1.test.ts +143 -0
  117. package/src/server/plugins/engine/outputFormatters/human/v1.ts +73 -0
  118. package/src/server/plugins/engine/outputFormatters/index.test.ts +17 -0
  119. package/src/server/plugins/engine/outputFormatters/index.ts +44 -0
  120. package/src/server/plugins/engine/outputFormatters/machine/v1.test.ts +229 -0
  121. package/src/server/plugins/engine/outputFormatters/machine/v1.ts +140 -0
  122. package/src/server/plugins/engine/outputFormatters/machine/v2.test.ts +229 -0
  123. package/src/server/plugins/engine/outputFormatters/machine/v2.ts +153 -0
  124. package/src/server/plugins/engine/pageControllers/FileUploadPageController.test.ts +1116 -0
  125. package/src/server/plugins/engine/pageControllers/FileUploadPageController.ts +447 -0
  126. package/src/server/plugins/engine/pageControllers/PageController.test.ts +205 -0
  127. package/src/server/plugins/engine/pageControllers/PageController.ts +176 -0
  128. package/src/server/plugins/engine/pageControllers/QuestionPageController.test.ts +1264 -0
  129. package/src/server/plugins/engine/pageControllers/QuestionPageController.ts +565 -0
  130. package/src/server/plugins/engine/pageControllers/README.md +28 -0
  131. package/src/server/plugins/engine/pageControllers/RepeatPageController.test.ts +264 -0
  132. package/src/server/plugins/engine/pageControllers/RepeatPageController.ts +458 -0
  133. package/src/server/plugins/engine/pageControllers/StartPageController.ts +18 -0
  134. package/src/server/plugins/engine/pageControllers/StatusPageController.ts +51 -0
  135. package/src/server/plugins/engine/pageControllers/SummaryPageController.ts +262 -0
  136. package/src/server/plugins/engine/pageControllers/TerminalController.test.ts +28 -0
  137. package/src/server/plugins/engine/pageControllers/TerminalPageController.ts +19 -0
  138. package/src/server/plugins/engine/pageControllers/helpers.test.ts +198 -0
  139. package/src/server/plugins/engine/pageControllers/helpers.ts +101 -0
  140. package/src/server/plugins/engine/pageControllers/index.ts +10 -0
  141. package/src/server/plugins/engine/pageControllers/validationOptions.ts +89 -0
  142. package/src/server/plugins/engine/plugin.ts +753 -0
  143. package/src/server/plugins/engine/services/formSubmissionService.js +46 -0
  144. package/src/server/plugins/engine/services/formsService.js +46 -0
  145. package/src/server/plugins/engine/services/formsService.test.js +90 -0
  146. package/src/server/plugins/engine/services/index.js +3 -0
  147. package/src/server/plugins/engine/services/notifyService.test.ts +132 -0
  148. package/src/server/plugins/engine/services/notifyService.ts +64 -0
  149. package/src/server/plugins/engine/services/uploadService.js +60 -0
  150. package/src/server/plugins/engine/types.ts +317 -0
  151. package/src/server/plugins/engine/views/components/autocompletefield.html +5 -0
  152. package/src/server/plugins/engine/views/components/checkboxesfield.html +5 -0
  153. package/src/server/plugins/engine/views/components/datepartsfield.html +5 -0
  154. package/src/server/plugins/engine/views/components/debug/macro.njk +3 -0
  155. package/src/server/plugins/engine/views/components/debug/template.njk +13 -0
  156. package/src/server/plugins/engine/views/components/details.html +6 -0
  157. package/src/server/plugins/engine/views/components/emailaddressfield.html +5 -0
  158. package/src/server/plugins/engine/views/components/fileuploadfield-key.html +8 -0
  159. package/src/server/plugins/engine/views/components/fileuploadfield-value.html +3 -0
  160. package/src/server/plugins/engine/views/components/fileuploadfield.html +24 -0
  161. package/src/server/plugins/engine/views/components/html.html +3 -0
  162. package/src/server/plugins/engine/views/components/insettext.html +7 -0
  163. package/src/server/plugins/engine/views/components/list.html +36 -0
  164. package/src/server/plugins/engine/views/components/monthyearfield.html +5 -0
  165. package/src/server/plugins/engine/views/components/multilinetextfield.html +10 -0
  166. package/src/server/plugins/engine/views/components/numberfield.html +5 -0
  167. package/src/server/plugins/engine/views/components/radiosfield.html +5 -0
  168. package/src/server/plugins/engine/views/components/selectfield.html +5 -0
  169. package/src/server/plugins/engine/views/components/service-banner/macro.njk +3 -0
  170. package/src/server/plugins/engine/views/components/service-banner/template.njk +20 -0
  171. package/src/server/plugins/engine/views/components/service-banner/template.test.js +43 -0
  172. package/src/server/plugins/engine/views/components/tag-env/macro.njk +3 -0
  173. package/src/server/plugins/engine/views/components/tag-env/template.njk +30 -0
  174. package/src/server/plugins/engine/views/components/tag-env/template.test.js +66 -0
  175. package/src/server/plugins/engine/views/components/telephonenumberfield.html +5 -0
  176. package/src/server/plugins/engine/views/components/textfield.html +5 -0
  177. package/src/server/plugins/engine/views/components/ukaddressfield.html +25 -0
  178. package/src/server/plugins/engine/views/components/yesnofield.html +5 -0
  179. package/src/server/plugins/engine/views/confirmation.html +19 -0
  180. package/src/server/plugins/engine/views/file-upload.html +45 -0
  181. package/src/server/plugins/engine/views/index.html +39 -0
  182. package/src/server/plugins/engine/views/item-delete.html +56 -0
  183. package/src/server/plugins/engine/views/layout.html +199 -0
  184. package/src/server/plugins/engine/views/partials/components.html +6 -0
  185. package/src/server/plugins/engine/views/partials/conditional-components.html +3 -0
  186. package/src/server/plugins/engine/views/partials/debug.html +44 -0
  187. package/src/server/plugins/engine/views/partials/form.html +15 -0
  188. package/src/server/plugins/engine/views/partials/heading.html +16 -0
  189. package/src/server/plugins/engine/views/partials/preview-banner.html +32 -0
  190. package/src/server/plugins/engine/views/partials/preview-banner.test.js +122 -0
  191. package/src/server/plugins/engine/views/partials/warn-missing-notification-email.html +10 -0
  192. package/src/server/plugins/engine/views/repeat-list-summary.html +53 -0
  193. package/src/server/plugins/engine/views/summary.html +50 -0
  194. package/src/server/plugins/errorPages.ts +58 -0
  195. package/src/server/plugins/nunjucks/context.js +88 -0
  196. package/src/server/plugins/nunjucks/context.test.js +142 -0
  197. package/src/server/plugins/nunjucks/enviroment.test.js +201 -0
  198. package/src/server/plugins/nunjucks/environment.js +116 -0
  199. package/src/server/plugins/nunjucks/filters/answer.js +27 -0
  200. package/src/server/plugins/nunjucks/filters/answer.test.js +89 -0
  201. package/src/server/plugins/nunjucks/filters/evaluate.js +21 -0
  202. package/src/server/plugins/nunjucks/filters/field.js +28 -0
  203. package/src/server/plugins/nunjucks/filters/field.test.js +75 -0
  204. package/src/server/plugins/nunjucks/filters/highlight.js +11 -0
  205. package/src/server/plugins/nunjucks/filters/href.js +30 -0
  206. package/src/server/plugins/nunjucks/filters/href.test.js +80 -0
  207. package/src/server/plugins/nunjucks/filters/index.js +8 -0
  208. package/src/server/plugins/nunjucks/filters/inspect.js +15 -0
  209. package/src/server/plugins/nunjucks/filters/page.js +24 -0
  210. package/src/server/plugins/nunjucks/filters/page.test.js +65 -0
  211. package/src/server/plugins/nunjucks/index.js +3 -0
  212. package/src/server/plugins/nunjucks/plugin.js +40 -0
  213. package/src/server/plugins/nunjucks/render.js +42 -0
  214. package/src/server/plugins/nunjucks/types.js +40 -0
  215. package/src/server/plugins/pulse.ts +11 -0
  216. package/src/server/plugins/router.ts +201 -0
  217. package/src/server/plugins/session.ts +28 -0
  218. package/src/server/routes/health.js +13 -0
  219. package/src/server/routes/health.test.js +35 -0
  220. package/src/server/routes/index.test.ts +125 -0
  221. package/src/server/routes/index.ts +2 -0
  222. package/src/server/routes/public.ts +47 -0
  223. package/src/server/routes/types.ts +48 -0
  224. package/src/server/schemas/index.ts +34 -0
  225. package/src/server/secure-context.js +43 -0
  226. package/src/server/services/cacheService.test.ts +277 -0
  227. package/src/server/services/cacheService.ts +138 -0
  228. package/src/server/services/httpService.test.js +491 -0
  229. package/src/server/services/httpService.ts +50 -0
  230. package/src/server/services/index.ts +1 -0
  231. package/src/server/types.ts +54 -0
  232. package/src/server/utils/notify.test.ts +37 -0
  233. package/src/server/utils/notify.ts +50 -0
  234. package/src/server/utils/secure-context/get-trust-store-certs.js +11 -0
  235. package/src/server/utils/secure-context/get-trust-store-certs.test.js +19 -0
  236. package/src/server/utils/utils.js +24 -0
  237. package/src/server/utils/utils.test.js +54 -0
  238. package/src/server/views/404.html +16 -0
  239. package/src/server/views/500.html +19 -0
  240. package/src/server/views/help/accessibility-statement.html +58 -0
  241. package/src/server/views/help/cookie-preferences.html +57 -0
  242. package/src/server/views/help/cookies.html +71 -0
  243. package/src/server/views/help/get-support.html +37 -0
  244. package/src/server/views/help/privacy-notice.html +68 -0
  245. package/src/server/views/help/terms-and-conditions.html +83 -0
  246. package/src/typings/hapi/index.d.ts +87 -0
  247. package/src/typings/hapi-tracing/index.d.ts +6 -0
  248. package/src/typings/index.d.ts +3 -0
  249. package/src/typings/joi/index.d.ts +22 -0
  250. package/.server/server/views/components/service-banner/template.test.js.map +0 -1
  251. package/.server/server/views/components/tag-env/template.test.js.map +0 -1
  252. /package/.server/server/{views → plugins/engine/views}/components/debug/macro.njk +0 -0
  253. /package/.server/server/{views → plugins/engine/views}/components/debug/template.njk +0 -0
  254. /package/.server/server/{views → plugins/engine/views}/components/service-banner/macro.njk +0 -0
  255. /package/.server/server/{views → plugins/engine/views}/components/service-banner/template.njk +0 -0
  256. /package/.server/server/{views → plugins/engine/views}/components/tag-env/macro.njk +0 -0
  257. /package/.server/server/{views → plugins/engine/views}/components/tag-env/template.njk +0 -0
  258. /package/.server/server/{views → plugins/engine/views}/confirmation.html +0 -0
  259. /package/.server/server/{views → plugins/engine/views}/layout.html +0 -0
  260. /package/.server/server/{views → plugins/engine/views}/summary.html +0 -0
@@ -0,0 +1,220 @@
1
+ import { type Section } from '@defra/forms-model'
2
+
3
+ import {
4
+ getAnswer,
5
+ type Field
6
+ } from '~/src/server/plugins/engine/components/helpers.js'
7
+ import { type BackLink } from '~/src/server/plugins/engine/components/types.js'
8
+ import {
9
+ evaluateTemplate,
10
+ getError,
11
+ getPageHref
12
+ } from '~/src/server/plugins/engine/helpers.js'
13
+ import {
14
+ type Detail,
15
+ type DetailItem,
16
+ type DetailItemField,
17
+ type DetailItemRepeat
18
+ } from '~/src/server/plugins/engine/models/types.js'
19
+ import { RepeatPageController } from '~/src/server/plugins/engine/pageControllers/RepeatPageController.js'
20
+ import { type PageControllerClass } from '~/src/server/plugins/engine/pageControllers/helpers.js'
21
+ import { validationOptions as opts } from '~/src/server/plugins/engine/pageControllers/validationOptions.js'
22
+ import {
23
+ type CheckAnswers,
24
+ type FormContext,
25
+ type FormContextRequest,
26
+ type FormState,
27
+ type FormSubmissionError,
28
+ type SummaryListAction,
29
+ type SummaryListRow
30
+ } from '~/src/server/plugins/engine/types.js'
31
+
32
+ export class SummaryViewModel {
33
+ /**
34
+ * Responsible for parsing state values to the govuk-frontend summary list template
35
+ */
36
+
37
+ page: PageControllerClass
38
+ pageTitle: string
39
+ declaration?: string
40
+ details: Detail[]
41
+ checkAnswers: CheckAnswers[]
42
+ context: FormContext
43
+ name: string | undefined
44
+ backLink?: BackLink
45
+ feedbackLink?: string
46
+ phaseTag?: string
47
+ errors?: FormSubmissionError[]
48
+ serviceUrl: string
49
+ hasMissingNotificationEmail?: boolean
50
+
51
+ constructor(
52
+ request: FormContextRequest,
53
+ page: PageControllerClass,
54
+ context: FormContext
55
+ ) {
56
+ const { model } = page
57
+ const { basePath, def, sections } = model
58
+ const { isForceAccess } = context
59
+
60
+ this.page = page
61
+ this.pageTitle = page.title
62
+ this.serviceUrl = `/${basePath}`
63
+ this.name = def.name
64
+ this.declaration = def.declaration
65
+ this.context = context
66
+
67
+ const result = model
68
+ .makeFilteredSchema(this.context.relevantPages)
69
+ .validate(this.context.relevantState, { ...opts, stripUnknown: true })
70
+
71
+ // Format errors
72
+ this.errors = result.error?.details.map(getError)
73
+ this.details = this.summaryDetails(request, sections)
74
+
75
+ // Format check answers
76
+ this.checkAnswers = this.details.map((detail): CheckAnswers => {
77
+ const { title } = detail
78
+
79
+ const rows = detail.items.map((item): SummaryListRow => {
80
+ const items: SummaryListAction[] = []
81
+
82
+ // Remove summary list actions from previews
83
+ if (!isForceAccess) {
84
+ items.push({
85
+ href: item.href,
86
+ text: 'Change',
87
+ classes: 'govuk-link--no-visited-state',
88
+ visuallyHiddenText: item.label
89
+ })
90
+ }
91
+
92
+ return {
93
+ key: {
94
+ text: evaluateTemplate(item.title, context)
95
+ },
96
+ value: {
97
+ classes: 'app-prose-scope',
98
+ html: item.value || 'Not supplied'
99
+ },
100
+ actions: {
101
+ items
102
+ }
103
+ }
104
+ })
105
+
106
+ return {
107
+ title: title ? { text: title } : undefined,
108
+ summaryList: { rows }
109
+ }
110
+ })
111
+ }
112
+
113
+ private summaryDetails(request: FormContextRequest, sections: Section[]) {
114
+ const { context, errors } = this
115
+ const { relevantPages, state } = context
116
+
117
+ const details: Detail[] = []
118
+
119
+ ;[undefined, ...sections].forEach((section) => {
120
+ const items: DetailItem[] = []
121
+
122
+ const sectionPages = relevantPages.filter(
123
+ (page) => page.section === section
124
+ )
125
+
126
+ sectionPages.forEach((page) => {
127
+ const { collection, path } = page
128
+
129
+ if (page instanceof RepeatPageController) {
130
+ items.push(
131
+ ItemRepeat(page, state, {
132
+ path: page.getSummaryPath(request),
133
+ errors
134
+ })
135
+ )
136
+ } else {
137
+ for (const field of collection.fields) {
138
+ items.push(ItemField(page, state, field, { path, errors }))
139
+ }
140
+ }
141
+ })
142
+
143
+ if (items.length) {
144
+ details.push({
145
+ name: section?.name,
146
+ title: section?.title,
147
+ items
148
+ })
149
+ }
150
+ })
151
+
152
+ return details
153
+ }
154
+ }
155
+
156
+ /**
157
+ * Creates a repeater detail item
158
+ * @see {@link DetailItemField}
159
+ */
160
+ function ItemRepeat(
161
+ page: RepeatPageController,
162
+ state: FormState,
163
+ options: {
164
+ path: string
165
+ errors?: FormSubmissionError[]
166
+ }
167
+ ): DetailItemRepeat {
168
+ const { collection, repeat } = page
169
+ const { name, title } = repeat.options
170
+
171
+ const values = page.getListFromState(state)
172
+ const unit = values.length === 1 ? title : `${title}s`
173
+
174
+ return {
175
+ name,
176
+ label: title,
177
+ title: values.length ? `${unit} added` : unit,
178
+ value: values.length ? `You added ${values.length} ${unit}` : '',
179
+ href: getPageHref(page, options.path, {
180
+ returnUrl: getPageHref(page, page.getSummaryPath())
181
+ }),
182
+ state,
183
+ page,
184
+
185
+ // Repeater field detail items
186
+ subItems: values.map((repeatState) =>
187
+ collection.fields.map((field) =>
188
+ ItemField(page, repeatState, field, options)
189
+ )
190
+ )
191
+ }
192
+ }
193
+
194
+ /**
195
+ * Creates a form field detail item
196
+ * @see {@link DetailItemField}
197
+ */
198
+ function ItemField(
199
+ page: PageControllerClass,
200
+ state: FormState,
201
+ field: Field,
202
+ options: {
203
+ path: string
204
+ errors?: FormSubmissionError[]
205
+ }
206
+ ): DetailItemField {
207
+ return {
208
+ name: field.name,
209
+ label: field.title,
210
+ title: field.title,
211
+ error: field.getError(options.errors),
212
+ value: getAnswer(field, state),
213
+ href: getPageHref(page, options.path, {
214
+ returnUrl: getPageHref(page, page.getSummaryPath())
215
+ }),
216
+ state,
217
+ page,
218
+ field
219
+ }
220
+ }
@@ -0,0 +1,2 @@
1
+ export { FormModel } from '~/src/server/plugins/engine/models/FormModel.js'
2
+ export { SummaryViewModel } from '~/src/server/plugins/engine/models/SummaryViewModel.js'
@@ -0,0 +1,114 @@
1
+ import {
2
+ type ConditionWrapper,
3
+ type FormComponentsDef,
4
+ type Section
5
+ } from '@defra/forms-model'
6
+ import { type Expression } from 'expr-eval'
7
+
8
+ import {
9
+ type Field,
10
+ type getAnswer
11
+ } from '~/src/server/plugins/engine/components/helpers.js'
12
+ import { type RepeatPageController } from '~/src/server/plugins/engine/pageControllers/RepeatPageController.js'
13
+ import { type PageControllerClass } from '~/src/server/plugins/engine/pageControllers/helpers.js'
14
+ import {
15
+ type FormState,
16
+ type FormSubmissionError
17
+ } from '~/src/server/plugins/engine/types.js'
18
+
19
+ export type ExecutableCondition = ConditionWrapper & {
20
+ expr: Expression
21
+ fn: (evaluationState: FormState) => boolean
22
+ }
23
+
24
+ /**
25
+ * Used to render a row on a Summary List (check your answers)
26
+ */
27
+ export interface DetailItemBase {
28
+ /**
29
+ * Name of the component defined in the JSON
30
+ * @see {@link FormComponentsDef.name}
31
+ */
32
+ name: string
33
+
34
+ /**
35
+ * Field label, used for change link visually hidden text
36
+ * @see {@link FormComponentsDef.title}
37
+ */
38
+ label: string
39
+
40
+ /**
41
+ * Field change link
42
+ */
43
+ href: string
44
+
45
+ /**
46
+ * Form submission state (or repeat state for sub items)
47
+ */
48
+ state: FormState
49
+
50
+ /**
51
+ * Field submission state error, used to flag unanswered questions
52
+ * Shown as 'Complete all unanswered questions before submitting the form'
53
+ */
54
+ error?: FormSubmissionError
55
+ }
56
+
57
+ export interface DetailItemField extends DetailItemBase {
58
+ /**
59
+ * Field page controller instance
60
+ */
61
+ page: Exclude<PageControllerClass, RepeatPageController>
62
+
63
+ /**
64
+ * Check answers summary list key
65
+ * For example, 'Date of birth'
66
+ */
67
+ title: string
68
+
69
+ /**
70
+ * Check answers summary list value, formatted by {@link getAnswer}
71
+ * For example, date fields formatted as '25 December 2022'
72
+ */
73
+ value: string
74
+
75
+ /**
76
+ * Field component instance
77
+ */
78
+ field: Field
79
+ }
80
+
81
+ export interface DetailItemRepeat extends DetailItemBase {
82
+ /**
83
+ * Repeat page controller instance
84
+ */
85
+ page: RepeatPageController
86
+
87
+ /**
88
+ * Check answers summary list key
89
+ * For example, 'Pizza' or 'Pizza added'
90
+ */
91
+ title: string
92
+
93
+ /**
94
+ * Check answers summary list value
95
+ * For example, 'You added 2 Pizzas'
96
+ */
97
+ value: string
98
+
99
+ /**
100
+ * Repeater field detail items
101
+ */
102
+ subItems: DetailItemField[][]
103
+ }
104
+
105
+ export type DetailItem = DetailItemField | DetailItemRepeat
106
+
107
+ /**
108
+ * Used to render a row on a Summary List (check your answers)
109
+ */
110
+ export interface Detail {
111
+ name?: Section['name']
112
+ title?: Section['title']
113
+ items: DetailItem[]
114
+ }
@@ -0,0 +1,143 @@
1
+ import { addDays, format as dateFormat } from 'date-fns'
2
+ import { outdent } from 'outdent'
3
+
4
+ import { FormModel } from '~/src/server/plugins/engine/models/index.js'
5
+ import { format } from '~/src/server/plugins/engine/outputFormatters/human/v1.js'
6
+ import {
7
+ SummaryPageController,
8
+ getFormSubmissionData
9
+ } from '~/src/server/plugins/engine/pageControllers/SummaryPageController.js'
10
+ import { type FormContextRequest } from '~/src/server/plugins/engine/types.js'
11
+ import { FormStatus } from '~/src/server/routes/types.js'
12
+ import definition from '~/test/form/definitions/repeat-mixed.js'
13
+
14
+ const itemId1 = 'abc-123'
15
+ const itemId2 = 'xyz-987'
16
+
17
+ const submitResponse = {
18
+ message: 'Submit completed',
19
+ result: {
20
+ files: {
21
+ main: '00000000-0000-0000-0000-000000000000',
22
+ repeaters: {
23
+ pizza: '11111111-1111-1111-1111-111111111111'
24
+ }
25
+ }
26
+ }
27
+ }
28
+
29
+ const model = new FormModel(definition, {
30
+ basePath: 'test'
31
+ })
32
+
33
+ const state = {
34
+ orderType: 'delivery',
35
+ pizza: [
36
+ {
37
+ toppings: 'Ham',
38
+ quantity: 2,
39
+ itemId: itemId1
40
+ },
41
+ {
42
+ toppings: 'Pepperoni',
43
+ quantity: 1,
44
+ itemId: itemId2
45
+ }
46
+ ]
47
+ }
48
+
49
+ const pageDef = definition.pages[2]
50
+ const pageUrl = new URL('http://example.com/repeat/pizza-order/summary')
51
+
52
+ const controller = new SummaryPageController(model, pageDef)
53
+
54
+ const request = {
55
+ method: 'get',
56
+ url: pageUrl,
57
+ path: pageUrl.pathname,
58
+ params: {
59
+ path: 'pizza-order',
60
+ slug: 'repeat'
61
+ },
62
+ query: {},
63
+ app: { model }
64
+ } satisfies FormContextRequest
65
+
66
+ const context = model.getFormContext(request, state)
67
+ const summaryViewModel = controller.getSummaryViewModel(request, context)
68
+
69
+ const items = getFormSubmissionData(
70
+ summaryViewModel.context,
71
+ summaryViewModel.details
72
+ )
73
+
74
+ describe('getPersonalisation', () => {
75
+ it.each([
76
+ {
77
+ state: FormStatus.Live,
78
+ isPreview: false
79
+ },
80
+ {
81
+ state: FormStatus.Draft,
82
+ isPreview: true
83
+ }
84
+ ])('should personalise $state email', (formStatus) => {
85
+ const body = format(items, model, submitResponse, formStatus)
86
+
87
+ const dateNow = new Date()
88
+ const dateExpiry = addDays(dateNow, 90)
89
+
90
+ // Check for link expiry message
91
+ expect(body).toContain(
92
+ `^ For security reasons, the links in this email expire at ${dateFormat(dateExpiry, 'h:mmaaa')} on ${dateFormat(dateExpiry, 'eeee d MMMM yyyy')}`
93
+ )
94
+
95
+ // Check for form answers
96
+ expect(body).toContain(
97
+ outdent`
98
+ Form submitted at ${dateFormat(dateNow, 'h:mmaaa')} on ${dateFormat(dateNow, 'd MMMM yyyy')}.
99
+
100
+ ---
101
+
102
+ ## How would you like to receive your pizza?
103
+
104
+ Delivery
105
+
106
+ ---
107
+
108
+ ## Pizza
109
+
110
+ [Download Pizza \\(CSV\\)](https://forms-designer/file-download/11111111-1111-1111-1111-111111111111)
111
+
112
+ ---
113
+
114
+ [Download main form \\(CSV\\)](https://forms-designer/file-download/00000000-0000-0000-0000-000000000000)
115
+ `
116
+ )
117
+ })
118
+
119
+ it('should add test warnings to preview email only', () => {
120
+ const formStatus = {
121
+ state: FormStatus.Draft,
122
+ isPreview: true
123
+ }
124
+
125
+ const body1 = format(items, model, submitResponse, {
126
+ state: FormStatus.Live,
127
+ isPreview: false
128
+ })
129
+
130
+ const body2 = format(items, model, submitResponse, {
131
+ state: FormStatus.Draft,
132
+ isPreview: true
133
+ })
134
+
135
+ expect(body1).not.toContain(
136
+ `This is a test of the ${definition.name} ${formStatus.state} form`
137
+ )
138
+
139
+ expect(body2).toContain(
140
+ `This is a test of the ${definition.name} ${formStatus.state} form`
141
+ )
142
+ })
143
+ })
@@ -0,0 +1,73 @@
1
+ import { type SubmitResponsePayload } from '@defra/forms-model'
2
+ import { addDays, format as dateFormat } from 'date-fns'
3
+
4
+ import { config } from '~/src/config/index.js'
5
+ import {
6
+ escapeMarkdown,
7
+ getAnswer
8
+ } from '~/src/server/plugins/engine/components/helpers.js'
9
+ import { type checkFormStatus } from '~/src/server/plugins/engine/helpers.js'
10
+ import { type FormModel } from '~/src/server/plugins/engine/models/index.js'
11
+ import { type DetailItem } from '~/src/server/plugins/engine/models/types.js'
12
+
13
+ const designerUrl = config.get('designerUrl')
14
+
15
+ export function format(
16
+ items: DetailItem[],
17
+ model: FormModel,
18
+ submitResponse: SubmitResponsePayload,
19
+ formStatus: ReturnType<typeof checkFormStatus>
20
+ ) {
21
+ const { files } = submitResponse.result
22
+
23
+ const formName = escapeMarkdown(model.name)
24
+
25
+ /**
26
+ * @todo Refactor this below but the code to
27
+ * generate the question and answers works for now
28
+ */
29
+ const now = new Date()
30
+ const formattedNow = `${dateFormat(now, 'h:mmaaa')} on ${dateFormat(now, 'd MMMM yyyy')}`
31
+
32
+ const fileExpiryDate = addDays(now, 90)
33
+ const formattedExpiryDate = `${dateFormat(fileExpiryDate, 'h:mmaaa')} on ${dateFormat(fileExpiryDate, 'eeee d MMMM yyyy')}`
34
+
35
+ const lines: string[] = []
36
+
37
+ lines.push(
38
+ `^ For security reasons, the links in this email expire at ${escapeMarkdown(formattedExpiryDate)}\n`
39
+ )
40
+
41
+ if (formStatus.isPreview) {
42
+ lines.push(`This is a test of the ${formName} ${formStatus.state} form.\n`)
43
+ }
44
+
45
+ lines.push(`Form submitted at ${escapeMarkdown(formattedNow)}.\n`)
46
+ lines.push('---\n')
47
+
48
+ items.forEach((item) => {
49
+ const label = escapeMarkdown(item.label)
50
+
51
+ lines.push(`## ${label}\n`)
52
+
53
+ if ('subItems' in item) {
54
+ const filename = escapeMarkdown(`Download ${label} (CSV)`)
55
+ const fileId = files.repeaters[item.name]
56
+
57
+ lines.push(`[${filename}](${designerUrl}/file-download/${fileId})\n`)
58
+ } else {
59
+ lines.push(
60
+ getAnswer(item.field, item.state, {
61
+ format: 'email'
62
+ })
63
+ )
64
+ }
65
+
66
+ lines.push('---\n')
67
+ })
68
+
69
+ const filename = escapeMarkdown('Download main form (CSV)')
70
+ lines.push(`[${filename}](${designerUrl}/file-download/${files.main})\n`)
71
+
72
+ return lines.join('\n')
73
+ }
@@ -0,0 +1,17 @@
1
+ import { format as formatHumanV1 } from '~/src/server/plugins/engine/outputFormatters/human/v1.js'
2
+ import { getFormatter } from '~/src/server/plugins/engine/outputFormatters/index.js'
3
+
4
+ describe('Page controller helpers', () => {
5
+ it('should return a valid formatter if it exists', () => {
6
+ const formatter = getFormatter('human', '1')
7
+ expect(formatter).toBe(formatHumanV1)
8
+ })
9
+
10
+ it("should return an error if the audience doesn't exist", () => {
11
+ expect(() => getFormatter('foobar', '1')).toThrow('Unknown audience')
12
+ })
13
+
14
+ it("should return an error if the version doesn't exist", () => {
15
+ expect(() => getFormatter('human', '9999')).toThrow('Unknown version')
16
+ })
17
+ })
@@ -0,0 +1,44 @@
1
+ import { type SubmitResponsePayload } from '@defra/forms-model'
2
+
3
+ import { type checkFormStatus } from '~/src/server/plugins/engine/helpers.js'
4
+ import { type FormModel } from '~/src/server/plugins/engine/models/index.js'
5
+ import { type DetailItem } from '~/src/server/plugins/engine/models/types.js'
6
+ import { format as formatHumanV1 } from '~/src/server/plugins/engine/outputFormatters/human/v1.js'
7
+ import { format as formatMachineV1 } from '~/src/server/plugins/engine/outputFormatters/machine/v1.js'
8
+ import { format as formatMachineV2 } from '~/src/server/plugins/engine/outputFormatters/machine/v2.js'
9
+
10
+ type Formatter = (
11
+ items: DetailItem[],
12
+ model: FormModel,
13
+ submitResponse: SubmitResponsePayload,
14
+ formStatus: ReturnType<typeof checkFormStatus>
15
+ ) => string
16
+
17
+ const formatters: Record<
18
+ string,
19
+ Record<string, Formatter | undefined> | undefined
20
+ > = {
21
+ human: {
22
+ '1': formatHumanV1
23
+ },
24
+ machine: {
25
+ '1': formatMachineV1,
26
+ '2': formatMachineV2
27
+ }
28
+ }
29
+
30
+ export function getFormatter(audience: string, version: string) {
31
+ const versions = formatters[audience]
32
+
33
+ if (!versions) {
34
+ throw new Error('Unknown audience')
35
+ }
36
+
37
+ const formatter = versions[version]
38
+
39
+ if (!formatter) {
40
+ throw new Error('Unknown version')
41
+ }
42
+
43
+ return formatter
44
+ }