@defra/forms-engine-plugin 1.0.0 → 1.0.2

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 (47) hide show
  1. package/.server/server/plugins/engine/README.md +56 -0
  2. package/.server/server/plugins/engine/configureEnginePlugin.d.ts +2 -1
  3. package/.server/server/plugins/engine/configureEnginePlugin.js +1 -1
  4. package/.server/server/plugins/engine/configureEnginePlugin.js.map +1 -1
  5. package/.server/server/plugins/engine/plugin.d.ts +2 -27
  6. package/.server/server/plugins/engine/plugin.js +17 -594
  7. package/.server/server/plugins/engine/plugin.js.map +1 -1
  8. package/.server/server/plugins/engine/registrationOptions.d.ts +1 -0
  9. package/.server/server/plugins/engine/registrationOptions.js +2 -0
  10. package/.server/server/plugins/engine/registrationOptions.js.map +1 -0
  11. package/.server/server/plugins/engine/routes/file-upload.d.ts +4 -0
  12. package/.server/server/plugins/engine/routes/file-upload.js +41 -0
  13. package/.server/server/plugins/engine/routes/file-upload.js.map +1 -0
  14. package/.server/server/plugins/engine/routes/index.d.ts +7 -0
  15. package/.server/server/plugins/engine/routes/index.js +141 -0
  16. package/.server/server/plugins/engine/routes/index.js.map +1 -0
  17. package/.server/server/plugins/engine/routes/questions.d.ts +3 -0
  18. package/.server/server/plugins/engine/routes/questions.js +168 -0
  19. package/.server/server/plugins/engine/routes/questions.js.map +1 -0
  20. package/.server/server/plugins/engine/routes/repeaters/item-delete.d.ts +3 -0
  21. package/.server/server/plugins/engine/routes/repeaters/item-delete.js +106 -0
  22. package/.server/server/plugins/engine/routes/repeaters/item-delete.js.map +1 -0
  23. package/.server/server/plugins/engine/routes/repeaters/summary.d.ts +3 -0
  24. package/.server/server/plugins/engine/routes/repeaters/summary.js +98 -0
  25. package/.server/server/plugins/engine/routes/repeaters/summary.js.map +1 -0
  26. package/.server/server/plugins/engine/types.d.ts +19 -1
  27. package/.server/server/plugins/engine/types.js.map +1 -1
  28. package/.server/server/plugins/engine/vision.d.ts +12 -0
  29. package/.server/server/plugins/engine/vision.js +55 -0
  30. package/.server/server/plugins/engine/vision.js.map +1 -0
  31. package/.server/server/services/cacheService.d.ts +12 -3
  32. package/.server/server/services/cacheService.js +35 -8
  33. package/.server/server/services/cacheService.js.map +1 -1
  34. package/package.json +3 -3
  35. package/src/server/plugins/engine/README.md +56 -0
  36. package/src/server/plugins/engine/configureEnginePlugin.ts +3 -5
  37. package/src/server/plugins/engine/plugin.ts +35 -765
  38. package/src/server/plugins/engine/registrationOptions.ts +0 -0
  39. package/src/server/plugins/engine/routes/file-upload.ts +54 -0
  40. package/src/server/plugins/engine/routes/index.ts +187 -0
  41. package/src/server/plugins/engine/routes/questions.ts +208 -0
  42. package/src/server/plugins/engine/routes/repeaters/item-delete.ts +157 -0
  43. package/src/server/plugins/engine/routes/repeaters/summary.ts +137 -0
  44. package/src/server/plugins/engine/types.ts +26 -1
  45. package/src/server/plugins/engine/vision.ts +95 -0
  46. package/src/server/services/cacheService.test.ts +98 -2
  47. package/src/server/services/cacheService.ts +57 -8
@@ -0,0 +1,137 @@
1
+ // List summary GET route
2
+ import { slugSchema } from '@defra/forms-model'
3
+ import Boom from '@hapi/boom'
4
+ import {
5
+ type ResponseToolkit,
6
+ type RouteOptions,
7
+ type ServerRoute
8
+ } from '@hapi/hapi'
9
+ import Joi from 'joi'
10
+
11
+ import { RepeatPageController } from '~/src/server/plugins/engine/pageControllers/RepeatPageController.js'
12
+ import { redirectOrMakeHandler } from '~/src/server/plugins/engine/routes/index.js'
13
+ import {
14
+ type FormRequest,
15
+ type FormRequestPayload,
16
+ type FormRequestPayloadRefs,
17
+ type FormRequestRefs
18
+ } from '~/src/server/routes/types.js'
19
+ import {
20
+ actionSchema,
21
+ crumbSchema,
22
+ pathSchema,
23
+ stateSchema
24
+ } from '~/src/server/schemas/index.js'
25
+
26
+ function getHandler(
27
+ request: FormRequest,
28
+ h: Pick<ResponseToolkit, 'redirect' | 'view'>
29
+ ) {
30
+ const { params } = request
31
+
32
+ return redirectOrMakeHandler(request, h, (page, context) => {
33
+ if (!(page instanceof RepeatPageController)) {
34
+ throw Boom.notFound(`No repeater page found for /${params.path}`)
35
+ }
36
+
37
+ return page.makeGetListSummaryRouteHandler()(request, context, h)
38
+ })
39
+ }
40
+
41
+ function postHandler(
42
+ request: FormRequestPayload,
43
+ h: Pick<ResponseToolkit, 'redirect' | 'view'>
44
+ ) {
45
+ const { params } = request
46
+
47
+ return redirectOrMakeHandler(request, h, (page, context) => {
48
+ const { isForceAccess } = context
49
+
50
+ if (isForceAccess || !(page instanceof RepeatPageController)) {
51
+ throw Boom.notFound(`No repeater page found for /${params.path}`)
52
+ }
53
+
54
+ return page.makePostListSummaryRouteHandler()(request, context, h)
55
+ })
56
+ }
57
+
58
+ export function getRoutes(
59
+ getRouteOptions: RouteOptions<FormRequestRefs>,
60
+ postRouteOptions: RouteOptions<FormRequestPayloadRefs>
61
+ ): (ServerRoute<FormRequestRefs> | ServerRoute<FormRequestPayloadRefs>)[] {
62
+ return [
63
+ {
64
+ method: 'get',
65
+ path: '/{slug}/{path}/summary',
66
+ handler: getHandler,
67
+ options: {
68
+ ...getRouteOptions,
69
+ validate: {
70
+ params: Joi.object().keys({
71
+ slug: slugSchema,
72
+ path: pathSchema
73
+ })
74
+ }
75
+ }
76
+ },
77
+
78
+ {
79
+ method: 'get',
80
+ path: '/preview/{state}/{slug}/{path}/summary',
81
+ handler: getHandler,
82
+ options: {
83
+ ...getRouteOptions,
84
+ validate: {
85
+ params: Joi.object().keys({
86
+ state: stateSchema,
87
+ slug: slugSchema,
88
+ path: pathSchema
89
+ })
90
+ }
91
+ }
92
+ },
93
+
94
+ {
95
+ method: 'post',
96
+ path: '/{slug}/{path}/summary',
97
+ handler: postHandler,
98
+ options: {
99
+ ...postRouteOptions,
100
+ validate: {
101
+ params: Joi.object().keys({
102
+ slug: slugSchema,
103
+ path: pathSchema
104
+ }),
105
+ payload: Joi.object()
106
+ .keys({
107
+ crumb: crumbSchema,
108
+ action: actionSchema
109
+ })
110
+ .required()
111
+ }
112
+ }
113
+ },
114
+
115
+ {
116
+ method: 'post',
117
+ path: '/preview/{state}/{slug}/{path}/summary',
118
+ handler: postHandler,
119
+ options: {
120
+ ...postRouteOptions,
121
+ validate: {
122
+ params: Joi.object().keys({
123
+ state: stateSchema,
124
+ slug: slugSchema,
125
+ path: pathSchema
126
+ }),
127
+ payload: Joi.object()
128
+ .keys({
129
+ crumb: crumbSchema,
130
+ action: actionSchema
131
+ })
132
+ .required()
133
+ }
134
+ }
135
+ }
136
+ ]
137
+ }
@@ -4,6 +4,7 @@ import {
4
4
  type List,
5
5
  type Page
6
6
  } from '@defra/forms-model'
7
+ import { type PluginProperties, type Request } from '@hapi/hapi'
7
8
  import { type JoiExpression, type ValidationErrorItem } from 'joi'
8
9
 
9
10
  import { FormComponent } from '~/src/server/plugins/engine/components/FormComponent.js'
@@ -13,10 +14,16 @@ import {
13
14
  type ComponentText,
14
15
  type ComponentViewModel
15
16
  } from '~/src/server/plugins/engine/components/types.js'
17
+ import { type FormModel } from '~/src/server/plugins/engine/models/index.js'
16
18
  import { type PageController } from '~/src/server/plugins/engine/pageControllers/PageController.js'
17
19
  import { type PageControllerClass } from '~/src/server/plugins/engine/pageControllers/helpers.js'
18
20
  import { type ViewContext } from '~/src/server/plugins/nunjucks/types.js'
19
- import { type FormAction, type FormRequest } from '~/src/server/routes/types.js'
21
+ import {
22
+ type FormAction,
23
+ type FormRequest,
24
+ type FormRequestPayload
25
+ } from '~/src/server/routes/types.js'
26
+ import { type Services } from '~/src/server/types.js'
20
27
 
21
28
  /**
22
29
  * Form submission state stores the following in Redis:
@@ -325,3 +332,21 @@ export interface ErrorMessageTemplateList {
325
332
  baseErrors: ErrorMessageTemplate[]
326
333
  advancedSettingsErrors: ErrorMessageTemplate[]
327
334
  }
335
+
336
+ export interface PluginOptions {
337
+ model?: FormModel
338
+ services?: Services
339
+ controllers?: Record<string, typeof PageController>
340
+ cacheName?: string
341
+ filters?: Record<string, FilterFunction>
342
+ keyGenerator?: (request: Request | FormRequest | FormRequestPayload) => string
343
+ sessionHydrator?: (
344
+ request: Request | FormRequest | FormRequestPayload
345
+ ) => Promise<FormSubmissionState>
346
+ pluginPath?: string
347
+ nunjucks: {
348
+ baseLayoutPath: string
349
+ paths: string[]
350
+ }
351
+ viewContext: PluginProperties['forms-engine-plugin']['viewContext']
352
+ }
@@ -0,0 +1,95 @@
1
+ import { existsSync } from 'fs'
2
+ import { dirname, join } from 'path'
3
+ import { fileURLToPath } from 'url'
4
+
5
+ import { type Server } from '@hapi/hapi'
6
+ import vision from '@hapi/vision'
7
+ import nunjucks, { type Environment } from 'nunjucks'
8
+ import resolvePkg from 'resolve'
9
+
10
+ import {
11
+ VIEW_PATH,
12
+ context,
13
+ prepareNunjucksEnvironment
14
+ } from '~/src/server/plugins/engine/index.js'
15
+ import { type PluginOptions } from '~/src/server/plugins/engine/types.js'
16
+
17
+ export async function registerVision(
18
+ server: Server,
19
+ pluginOptions: PluginOptions
20
+ ) {
21
+ const packageRoot = findPackageRoot()
22
+ const govukFrontendPath = dirname(
23
+ resolvePkg.sync('govuk-frontend/package.json')
24
+ )
25
+
26
+ const viewPathResolved = join(packageRoot, VIEW_PATH)
27
+
28
+ const paths = [
29
+ ...pluginOptions.nunjucks.paths,
30
+ viewPathResolved,
31
+ join(govukFrontendPath, 'dist')
32
+ ]
33
+
34
+ await server.register({
35
+ plugin: vision,
36
+ options: {
37
+ engines: {
38
+ html: {
39
+ compile: (
40
+ path: string,
41
+ compileOptions: { environment: Environment }
42
+ ) => {
43
+ const template = nunjucks.compile(path, compileOptions.environment)
44
+
45
+ return (context: object | undefined) => {
46
+ return template.render(context)
47
+ }
48
+ },
49
+ prepare: (
50
+ options: EngineConfigurationObject,
51
+ next: (err?: Error) => void
52
+ ) => {
53
+ // Nunjucks also needs an additional path configuration
54
+ // to use the templates and macros from `govuk-frontend`
55
+ const environment = nunjucks.configure(paths)
56
+
57
+ // Applies custom filters and globals for nunjucks
58
+ // that are required by the `forms-engine-plugin`
59
+ prepareNunjucksEnvironment(environment, pluginOptions.filters)
60
+
61
+ options.compileOptions.environment = environment
62
+
63
+ next()
64
+ }
65
+ }
66
+ },
67
+ path: paths,
68
+ // Provides global context used with all templates
69
+ context
70
+ }
71
+ })
72
+ }
73
+
74
+ interface CompileOptions {
75
+ environment: Environment
76
+ }
77
+
78
+ export interface EngineConfigurationObject {
79
+ compileOptions: CompileOptions
80
+ }
81
+
82
+ export function findPackageRoot() {
83
+ const currentFileName = fileURLToPath(import.meta.url)
84
+ const currentDirectoryName = dirname(currentFileName)
85
+
86
+ let dir = currentDirectoryName
87
+ while (dir !== '/') {
88
+ if (existsSync(join(dir, 'package.json'))) {
89
+ return dir
90
+ }
91
+ dir = dirname(dir)
92
+ }
93
+
94
+ throw new Error('package.json not found in parent directories')
95
+ }
@@ -2,7 +2,11 @@ import { type Request, type Server } from '@hapi/hapi'
2
2
 
3
3
  import { config } from '~/src/config/index.js'
4
4
  import { type FormRequest } from '~/src/server/routes/types.js'
5
- import { CacheService, merge } from '~/src/server/services/cacheService.js'
5
+ import {
6
+ ADDITIONAL_IDENTIFIER,
7
+ CacheService,
8
+ merge
9
+ } from '~/src/server/services/cacheService.js'
6
10
 
7
11
  describe('CacheService', () => {
8
12
  let mockServer: Partial<Server>
@@ -31,7 +35,10 @@ describe('CacheService', () => {
31
35
  } as unknown as Server['logger']
32
36
  }
33
37
 
34
- cacheService = new CacheService(mockServer as Server)
38
+ cacheService = new CacheService({
39
+ server: mockServer as Server,
40
+ cacheName: 'test-cache'
41
+ })
35
42
  })
36
43
 
37
44
  describe('getState', () => {
@@ -69,6 +76,40 @@ describe('CacheService', () => {
69
76
  expect(result).toEqual({})
70
77
  })
71
78
  })
79
+
80
+ it('should rehydrate state using custom fetcher when cache is missed', async () => {
81
+ const rehydratedState = { rehydrated: true }
82
+
83
+ const customFetcher = jest.fn().mockResolvedValue(rehydratedState)
84
+
85
+ cacheService = new CacheService({
86
+ server: mockServer as Server,
87
+ cacheName: 'test-cache',
88
+ options: { sessionHydrator: customFetcher }
89
+ })
90
+
91
+ const mockRequest = {
92
+ yar: { id: 'session-id' },
93
+ params: { state: 's', slug: 'p' }
94
+ } as unknown as FormRequest
95
+
96
+ mockCache.get
97
+ .mockResolvedValueOnce(null)
98
+ .mockResolvedValueOnce(rehydratedState)
99
+
100
+ const result = await cacheService.getState(mockRequest)
101
+
102
+ expect(customFetcher).toHaveBeenCalledWith(mockRequest)
103
+ expect(mockCache.set).toHaveBeenCalledWith(
104
+ expect.objectContaining({
105
+ segment: 'cache',
106
+ id: expect.stringContaining('session-id')
107
+ }),
108
+ rehydratedState,
109
+ config.get('sessionTimeout')
110
+ )
111
+ expect(result).toEqual(rehydratedState)
112
+ })
72
113
  })
73
114
 
74
115
  describe('setState', () => {
@@ -108,6 +149,61 @@ describe('CacheService', () => {
108
149
  )
109
150
  })
110
151
  })
152
+
153
+ it('should use custom key generator if provided', async () => {
154
+ const customKey = 'my-custom-key'
155
+ const customKeyGenerator = jest.fn().mockReturnValue(customKey)
156
+
157
+ cacheService = new CacheService({
158
+ server: mockServer as Server,
159
+ cacheName: 'test-cache',
160
+ options: { keyGenerator: customKeyGenerator }
161
+ })
162
+
163
+ const mockRequest = {
164
+ yar: { id: 'some-session' },
165
+ params: { state: 'form1', slug: 'page1' }
166
+ } as unknown as FormRequest
167
+
168
+ await cacheService.setState(mockRequest, { test: 'value' })
169
+
170
+ expect(mockCache.set).toHaveBeenCalledWith(
171
+ {
172
+ segment: 'cache',
173
+ id: 'my-custom-key'
174
+ },
175
+ { test: 'value' },
176
+ expect.any(Number)
177
+ )
178
+ })
179
+
180
+ it('should append additionalIdentifier to custom key', () => {
181
+ const customKey = 'custom:key:base:'
182
+ const customKeyGenerator = jest.fn().mockReturnValue(customKey)
183
+
184
+ cacheService = new CacheService({
185
+ server: mockServer as Server,
186
+ cacheName: 'test-cache',
187
+ options: { keyGenerator: customKeyGenerator }
188
+ })
189
+
190
+ const mockRequest = {
191
+ yar: { id: 'session-id' },
192
+ params: { state: 'formA', slug: 'step1' }
193
+ } as unknown as FormRequest
194
+
195
+ const result = cacheService.Key(
196
+ mockRequest,
197
+ ADDITIONAL_IDENTIFIER.Confirmation
198
+ )
199
+
200
+ expect(result).toEqual({
201
+ segment: 'cache',
202
+ id: 'custom:key:base::confirmation'
203
+ })
204
+
205
+ expect(customKeyGenerator).toHaveBeenCalledWith(mockRequest)
206
+ })
111
207
  })
112
208
 
113
209
  describe('merge', () => {
@@ -16,7 +16,7 @@ import {
16
16
 
17
17
  const partition = 'cache'
18
18
 
19
- enum ADDITIONAL_IDENTIFIER {
19
+ export enum ADDITIONAL_IDENTIFIER {
20
20
  Confirmation = ':confirmation'
21
21
  }
22
22
 
@@ -25,16 +25,38 @@ export class CacheService {
25
25
  * This service is responsible for getting, storing or deleting a user's session data in the cache. This service has been registered by {@link createServer}
26
26
  */
27
27
  cache
28
+ generateKey?: (request: Request | FormRequest | FormRequestPayload) => string
29
+ customFetcher?: (
30
+ request: Request | FormRequest | FormRequestPayload
31
+ ) => Promise<FormSubmissionState | null>
32
+
28
33
  logger: Server['logger']
29
34
 
30
- constructor(server: Server, cacheName?: string) {
35
+ constructor({
36
+ server,
37
+ cacheName,
38
+ options
39
+ }: {
40
+ server: Server
41
+ cacheName?: string
42
+ options?: {
43
+ keyGenerator?: (
44
+ request: Request | FormRequest | FormRequestPayload
45
+ ) => string
46
+ sessionHydrator?: (
47
+ request: Request | FormRequest | FormRequestPayload
48
+ ) => Promise<FormSubmissionState | null>
49
+ }
50
+ }) {
51
+ const { keyGenerator, sessionHydrator } = options ?? {}
31
52
  if (!cacheName) {
32
53
  server.log(
33
54
  'warn',
34
55
  'You are using the default hapi cache. Please provide a cache name in plugin registration options.'
35
56
  )
36
57
  }
37
-
58
+ this.generateKey = keyGenerator ?? this.defaultKeyGenerator.bind(this)
59
+ this.customFetcher = sessionHydrator ?? undefined
38
60
  this.cache = server.cache({ cache: cacheName, segment: 'formSubmission' })
39
61
  this.logger = server.logger
40
62
  }
@@ -42,7 +64,21 @@ export class CacheService {
42
64
  async getState(
43
65
  request: Request | FormRequest | FormRequestPayload
44
66
  ): Promise<FormSubmissionState> {
45
- const cached = await this.cache.get(this.Key(request))
67
+ let cached = await this.cache.get(this.Key(request))
68
+
69
+ // If nothing in Redis, attempt to rehydrate from backend DB
70
+ if (!cached && this.customFetcher) {
71
+ const rehydrated = await this.customFetcher(request)
72
+
73
+ if (rehydrated != null) {
74
+ await this.cache.set(
75
+ this.Key(request),
76
+ rehydrated,
77
+ config.get('sessionTimeout')
78
+ )
79
+ cached = await this.getState(request)
80
+ }
81
+ }
46
82
 
47
83
  return cached ?? {}
48
84
  }
@@ -103,6 +139,18 @@ export class CacheService {
103
139
  request.yar.flash(key.id, message)
104
140
  }
105
141
 
142
+ private defaultKeyGenerator(
143
+ request: Request | FormRequest | FormRequestPayload
144
+ ): string {
145
+ if (!request.yar.id) {
146
+ throw new Error('No session ID found')
147
+ }
148
+
149
+ const state = request.params.state ?? ''
150
+ const slug = request.params.slug ?? ''
151
+ return `${request.yar.id}:${state}:${slug}:`
152
+ }
153
+
106
154
  /**
107
155
  * The key used to store user session data against.
108
156
  * If there are multiple forms on the same runner instance, for example `form-a` and `form-a-feedback` this will prevent CacheService from clearing data from `form-a` if a user gave feedback before they finished `form-a`
@@ -113,12 +161,13 @@ export class CacheService {
113
161
  request: Request | FormRequest | FormRequestPayload,
114
162
  additionalIdentifier?: ADDITIONAL_IDENTIFIER
115
163
  ) {
116
- if (!request.yar.id) {
117
- throw Error('No session ID found')
118
- }
164
+ const baseKey = this.generateKey
165
+ ? this.generateKey(request)
166
+ : this.defaultKeyGenerator(request)
167
+
119
168
  return {
120
169
  segment: partition,
121
- id: `${request.yar.id}:${request.params.state ?? ''}:${request.params.slug ?? ''}:${additionalIdentifier ?? ''}`
170
+ id: `${baseKey}${additionalIdentifier ?? ''}`
122
171
  }
123
172
  }
124
173
  }