@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.
- package/.server/server/plugins/engine/README.md +56 -0
- package/.server/server/plugins/engine/configureEnginePlugin.d.ts +2 -1
- package/.server/server/plugins/engine/configureEnginePlugin.js +1 -1
- package/.server/server/plugins/engine/configureEnginePlugin.js.map +1 -1
- package/.server/server/plugins/engine/plugin.d.ts +2 -27
- package/.server/server/plugins/engine/plugin.js +17 -594
- package/.server/server/plugins/engine/plugin.js.map +1 -1
- package/.server/server/plugins/engine/registrationOptions.d.ts +1 -0
- package/.server/server/plugins/engine/registrationOptions.js +2 -0
- package/.server/server/plugins/engine/registrationOptions.js.map +1 -0
- package/.server/server/plugins/engine/routes/file-upload.d.ts +4 -0
- package/.server/server/plugins/engine/routes/file-upload.js +41 -0
- package/.server/server/plugins/engine/routes/file-upload.js.map +1 -0
- package/.server/server/plugins/engine/routes/index.d.ts +7 -0
- package/.server/server/plugins/engine/routes/index.js +141 -0
- package/.server/server/plugins/engine/routes/index.js.map +1 -0
- package/.server/server/plugins/engine/routes/questions.d.ts +3 -0
- package/.server/server/plugins/engine/routes/questions.js +168 -0
- package/.server/server/plugins/engine/routes/questions.js.map +1 -0
- package/.server/server/plugins/engine/routes/repeaters/item-delete.d.ts +3 -0
- package/.server/server/plugins/engine/routes/repeaters/item-delete.js +106 -0
- package/.server/server/plugins/engine/routes/repeaters/item-delete.js.map +1 -0
- package/.server/server/plugins/engine/routes/repeaters/summary.d.ts +3 -0
- package/.server/server/plugins/engine/routes/repeaters/summary.js +98 -0
- package/.server/server/plugins/engine/routes/repeaters/summary.js.map +1 -0
- package/.server/server/plugins/engine/types.d.ts +19 -1
- package/.server/server/plugins/engine/types.js.map +1 -1
- package/.server/server/plugins/engine/vision.d.ts +12 -0
- package/.server/server/plugins/engine/vision.js +55 -0
- package/.server/server/plugins/engine/vision.js.map +1 -0
- package/.server/server/services/cacheService.d.ts +12 -3
- package/.server/server/services/cacheService.js +35 -8
- package/.server/server/services/cacheService.js.map +1 -1
- package/package.json +3 -3
- package/src/server/plugins/engine/README.md +56 -0
- package/src/server/plugins/engine/configureEnginePlugin.ts +3 -5
- package/src/server/plugins/engine/plugin.ts +35 -765
- package/src/server/plugins/engine/registrationOptions.ts +0 -0
- package/src/server/plugins/engine/routes/file-upload.ts +54 -0
- package/src/server/plugins/engine/routes/index.ts +187 -0
- package/src/server/plugins/engine/routes/questions.ts +208 -0
- package/src/server/plugins/engine/routes/repeaters/item-delete.ts +157 -0
- package/src/server/plugins/engine/routes/repeaters/summary.ts +137 -0
- package/src/server/plugins/engine/types.ts +26 -1
- package/src/server/plugins/engine/vision.ts +95 -0
- package/src/server/services/cacheService.test.ts +98 -2
- 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 {
|
|
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 {
|
|
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(
|
|
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(
|
|
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
|
-
|
|
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
|
-
|
|
117
|
-
|
|
118
|
-
|
|
164
|
+
const baseKey = this.generateKey
|
|
165
|
+
? this.generateKey(request)
|
|
166
|
+
: this.defaultKeyGenerator(request)
|
|
167
|
+
|
|
119
168
|
return {
|
|
120
169
|
segment: partition,
|
|
121
|
-
id: `${
|
|
170
|
+
id: `${baseKey}${additionalIdentifier ?? ''}`
|
|
122
171
|
}
|
|
123
172
|
}
|
|
124
173
|
}
|