@ministryofjustice/hmpps-connect-dps-components 2.2.0 → 3.0.0-beta.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/dist/allocationService.d.ts +14 -2
- package/dist/allocationService.test.d.ts +1 -0
- package/dist/caseLoadService.d.ts +14 -2
- package/dist/caseLoadService.test.d.ts +1 -0
- package/dist/componentsService.d.ts +24 -2
- package/dist/componentsService.test.d.ts +1 -0
- package/dist/data/allocationsApi/allocationsApiClient.d.ts +6 -6
- package/dist/data/componentApi/componentApiClient.d.ts +6 -6
- package/dist/data/prisonApi/prisonApiClient.d.ts +6 -6
- package/dist/index.cjs +266 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.ts +141 -66
- package/dist/index.esm.js +259 -0
- package/dist/index.esm.js.map +1 -0
- package/dist/middleware/getFrontendComponents.d.ts +3 -0
- package/dist/middleware/retrieveAllocationJobResponsibilities.d.ts +3 -0
- package/dist/middleware/retrieveCaseLoadData.d.ts +3 -0
- package/dist/types/ConnectDpsComponentLogger.d.ts +2 -0
- package/dist/types/public/middleware/index.d.ts +1 -0
- package/dist/types/public/services/index.d.ts +3 -0
- package/dist/utils/fallbacks.d.ts +9 -3
- package/dist/utils/updateCsp.d.ts +1 -1
- package/dist/utils/updateCsp.test.d.ts +1 -0
- package/package.json +34 -47
- package/dist/allocationService.js +0 -48
- package/dist/allocationService.js.map +0 -1
- package/dist/caseLoadService.js +0 -55
- package/dist/caseLoadService.js.map +0 -1
- package/dist/componentsService.js +0 -61
- package/dist/componentsService.js.map +0 -1
- package/dist/config.d.ts +0 -14
- package/dist/config.js +0 -28
- package/dist/config.js.map +0 -1
- package/dist/data/allocationsApi/allocationsApiClient.js +0 -23
- package/dist/data/allocationsApi/allocationsApiClient.js.map +0 -1
- package/dist/data/componentApi/componentApiClient.js +0 -23
- package/dist/data/componentApi/componentApiClient.js.map +0 -1
- package/dist/data/prisonApi/prisonApiClient.js +0 -24
- package/dist/data/prisonApi/prisonApiClient.js.map +0 -1
- package/dist/index.js +0 -72
- package/dist/index.js.map +0 -1
- package/dist/types/AllocationJobResponsibility.js +0 -3
- package/dist/types/AllocationJobResponsibility.js.map +0 -1
- package/dist/types/AvailableComponent.js +0 -3
- package/dist/types/AvailableComponent.js.map +0 -1
- package/dist/types/CaseLoad.js +0 -3
- package/dist/types/CaseLoad.js.map +0 -1
- package/dist/types/CaseLoadOptions.js +0 -3
- package/dist/types/CaseLoadOptions.js.map +0 -1
- package/dist/types/Component.js +0 -3
- package/dist/types/Component.js.map +0 -1
- package/dist/types/HeaderFooterSharedData.js +0 -3
- package/dist/types/HeaderFooterSharedData.js.map +0 -1
- package/dist/types/HmppsUser.js +0 -3
- package/dist/types/HmppsUser.js.map +0 -1
- package/dist/types/RequestOptions.d.ts +0 -12
- package/dist/types/RequestOptions.js +0 -3
- package/dist/types/RequestOptions.js.map +0 -1
- package/dist/types/Service.js +0 -3
- package/dist/types/Service.js.map +0 -1
- package/dist/types/TimeoutOptions.js +0 -3
- package/dist/types/TimeoutOptions.js.map +0 -1
- package/dist/utils/fallbacks.js +0 -36
- package/dist/utils/fallbacks.js.map +0 -1
- package/dist/utils/updateCsp.js +0 -25
- package/dist/utils/updateCsp.js.map +0 -1
|
@@ -1,3 +1,15 @@
|
|
|
1
1
|
import { type RequestHandler } from 'express';
|
|
2
|
-
import
|
|
3
|
-
|
|
2
|
+
import { ApiConfig, AuthenticationClient } from '@ministryofjustice/hmpps-rest-client';
|
|
3
|
+
import AllocationsApiClient from './data/allocationsApi/allocationsApiClient';
|
|
4
|
+
import { ConnectDpsComponentLogger } from './types/ConnectDpsComponentLogger';
|
|
5
|
+
export default class AllocationService {
|
|
6
|
+
private readonly logger;
|
|
7
|
+
private readonly allocationsApiClient;
|
|
8
|
+
constructor(logger: ConnectDpsComponentLogger, allocationsApiClient: AllocationsApiClient);
|
|
9
|
+
static create({ logger, allocationsApiConfig, authenticationClient, }: {
|
|
10
|
+
logger?: ConnectDpsComponentLogger;
|
|
11
|
+
allocationsApiConfig: ApiConfig;
|
|
12
|
+
authenticationClient: AuthenticationClient;
|
|
13
|
+
}): AllocationService;
|
|
14
|
+
retrieveAllocationJobResponsibilities(): RequestHandler;
|
|
15
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -1,3 +1,15 @@
|
|
|
1
1
|
import { type RequestHandler } from 'express';
|
|
2
|
-
import
|
|
3
|
-
|
|
2
|
+
import { ApiConfig, AuthenticationClient } from '@ministryofjustice/hmpps-rest-client';
|
|
3
|
+
import PrisonApiClient from './data/prisonApi/prisonApiClient';
|
|
4
|
+
import { ConnectDpsComponentLogger } from './types/ConnectDpsComponentLogger';
|
|
5
|
+
export default class CaseLoadService {
|
|
6
|
+
private readonly logger;
|
|
7
|
+
private readonly prisonApiClient;
|
|
8
|
+
constructor(logger: ConnectDpsComponentLogger, prisonApiClient: PrisonApiClient);
|
|
9
|
+
static create({ logger, prisonApiConfig, authenticationClient, }: {
|
|
10
|
+
logger?: ConnectDpsComponentLogger;
|
|
11
|
+
prisonApiConfig: ApiConfig;
|
|
12
|
+
authenticationClient: AuthenticationClient;
|
|
13
|
+
}): CaseLoadService;
|
|
14
|
+
retrieveCaseLoadData(): RequestHandler;
|
|
15
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -1,3 +1,25 @@
|
|
|
1
1
|
import { type RequestHandler } from 'express';
|
|
2
|
-
import
|
|
3
|
-
|
|
2
|
+
import { ApiConfig, AuthenticationClient } from '@ministryofjustice/hmpps-rest-client';
|
|
3
|
+
import ComponentApiClient from './data/componentApi/componentApiClient';
|
|
4
|
+
import { ConnectDpsComponentLogger } from './types/ConnectDpsComponentLogger';
|
|
5
|
+
export interface FrontentComponentRequestOptions {
|
|
6
|
+
authUrl?: string;
|
|
7
|
+
supportUrl?: string;
|
|
8
|
+
environmentName?: 'DEV' | 'PRE-PRODUCTION' | 'PRODUCTION';
|
|
9
|
+
includeSharedData?: boolean;
|
|
10
|
+
useFallbacksByDefault?: boolean;
|
|
11
|
+
}
|
|
12
|
+
export default class ComponentsService {
|
|
13
|
+
private readonly logger;
|
|
14
|
+
private readonly componentApiConfig;
|
|
15
|
+
private readonly componentApiClient;
|
|
16
|
+
private readonly dpsUrl;
|
|
17
|
+
constructor(logger: ConnectDpsComponentLogger, componentApiConfig: ApiConfig, componentApiClient: ComponentApiClient, dpsUrl: string);
|
|
18
|
+
static create({ logger, componentApiConfig, authenticationClient, dpsUrl, }: {
|
|
19
|
+
logger?: ConnectDpsComponentLogger;
|
|
20
|
+
componentApiConfig: ApiConfig;
|
|
21
|
+
authenticationClient: AuthenticationClient;
|
|
22
|
+
dpsUrl: string;
|
|
23
|
+
}): ComponentsService;
|
|
24
|
+
getFrontendComponents(requestOptions: FrontentComponentRequestOptions): RequestHandler;
|
|
25
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -1,10 +1,10 @@
|
|
|
1
|
-
import
|
|
2
|
-
import TimeoutOptions from '../../types/TimeoutOptions';
|
|
1
|
+
import { ApiConfig, AuthenticationClient, RestClient } from '@ministryofjustice/hmpps-rest-client';
|
|
3
2
|
import { PrisonUser } from '../../types/HmppsUser';
|
|
4
3
|
import { AllocationJobResponsibility } from '../../types/AllocationJobResponsibility';
|
|
5
|
-
|
|
6
|
-
|
|
4
|
+
import { ConnectDpsComponentLogger } from '../../types/ConnectDpsComponentLogger';
|
|
5
|
+
export default class AllocationsApiClient extends RestClient {
|
|
6
|
+
constructor(logger: ConnectDpsComponentLogger, config: ApiConfig, authenticationClient: AuthenticationClient);
|
|
7
|
+
getStaffAllocationPolicies(user: PrisonUser): Promise<{
|
|
7
8
|
policies: AllocationJobResponsibility[];
|
|
8
9
|
}>;
|
|
9
|
-
}
|
|
10
|
-
export default _default;
|
|
10
|
+
}
|
|
@@ -1,12 +1,12 @@
|
|
|
1
|
-
import
|
|
1
|
+
import { ApiConfig, AuthenticationClient, RestClient } from '@ministryofjustice/hmpps-rest-client';
|
|
2
2
|
import AvailableComponent from '../../types/AvailableComponent';
|
|
3
3
|
import Component from '../../types/Component';
|
|
4
|
-
import
|
|
4
|
+
import { ConnectDpsComponentLogger } from '../../types/ConnectDpsComponentLogger';
|
|
5
5
|
import { ComponentsSharedData } from '../../types/HeaderFooterSharedData';
|
|
6
6
|
export type ComponentsApiResponse<T extends AvailableComponent[]> = Record<T[number], Component> & {
|
|
7
7
|
meta: ComponentsSharedData[T[number]];
|
|
8
8
|
};
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
9
|
+
export default class ComponentApiClient extends RestClient {
|
|
10
|
+
constructor(logger: ConnectDpsComponentLogger, config: ApiConfig, authenticationClient: AuthenticationClient);
|
|
11
|
+
getComponents<T extends AvailableComponent[]>(userToken: string): Promise<ComponentsApiResponse<T>>;
|
|
12
|
+
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
import
|
|
2
|
-
import TimeoutOptions from '../../types/TimeoutOptions';
|
|
1
|
+
import { ApiConfig, AuthenticationClient, RestClient } from '@ministryofjustice/hmpps-rest-client';
|
|
3
2
|
import CaseLoad from '../../types/CaseLoad';
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
3
|
+
import { ConnectDpsComponentLogger } from '../../types/ConnectDpsComponentLogger';
|
|
4
|
+
export default class PrisonApiClient extends RestClient {
|
|
5
|
+
constructor(logger: ConnectDpsComponentLogger, config: ApiConfig, authenticationClient: AuthenticationClient);
|
|
6
|
+
getUserCaseLoads(userToken: string): Promise<CaseLoad[]>;
|
|
7
|
+
}
|
package/dist/index.cjs
ADDED
|
@@ -0,0 +1,266 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
var hmppsRestClient = require('@ministryofjustice/hmpps-rest-client');
|
|
4
|
+
var nunjucks = require('nunjucks');
|
|
5
|
+
|
|
6
|
+
class ComponentApiClient extends hmppsRestClient.RestClient {
|
|
7
|
+
constructor(logger, config, authenticationClient) {
|
|
8
|
+
super('Component API Client', config, logger, authenticationClient);
|
|
9
|
+
}
|
|
10
|
+
async getComponents(userToken) {
|
|
11
|
+
return this.get({
|
|
12
|
+
path: `/components`,
|
|
13
|
+
query: 'component=header&component=footer',
|
|
14
|
+
headers: { 'x-user-token': userToken },
|
|
15
|
+
}, hmppsRestClient.asSystem());
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
function getFallbackHeader(user, dpsUrl, { environmentName, authUrl, supportUrl }) {
|
|
20
|
+
return nunjucks.render('dpsComponents/header-bar.njk', {
|
|
21
|
+
isPrisonUser: !user || user.authSource === 'nomis',
|
|
22
|
+
user,
|
|
23
|
+
dpsUrl,
|
|
24
|
+
environmentName,
|
|
25
|
+
authUrl,
|
|
26
|
+
supportUrl,
|
|
27
|
+
name: initialiseName(user?.displayName),
|
|
28
|
+
});
|
|
29
|
+
}
|
|
30
|
+
function getFallbackFooter(user, { authUrl, supportUrl }) {
|
|
31
|
+
return nunjucks.render('dpsComponents/footer.njk', {
|
|
32
|
+
isPrisonUser: !user || user.authSource === 'nomis',
|
|
33
|
+
supportUrl,
|
|
34
|
+
authUrl,
|
|
35
|
+
});
|
|
36
|
+
}
|
|
37
|
+
function initialiseName(fullName) {
|
|
38
|
+
if (!fullName)
|
|
39
|
+
return null;
|
|
40
|
+
const array = fullName.split(' ');
|
|
41
|
+
return `${array[0][0]}. ${array.reverse()[0]}`;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
function updateCsp(feComponentsUrl, res) {
|
|
45
|
+
const csp = res.getHeaders()['content-security-policy'];
|
|
46
|
+
const allDirectives = csp?.split(';') ?? [];
|
|
47
|
+
const directivesToUpdate = ['script-src', 'style-src', 'img-src', 'font-src'];
|
|
48
|
+
const updatedCspDirectives = allDirectives.map(directive => {
|
|
49
|
+
// if directive is not in cspToUpdate or includes fe components url already return as is
|
|
50
|
+
if (directive.includes(feComponentsUrl) || !directivesToUpdate.some(p => directive.includes(`${p} `)))
|
|
51
|
+
return directive;
|
|
52
|
+
// if directive is in cspToUpdate and does not have fe components url, add in
|
|
53
|
+
return `${directive} ${feComponentsUrl}`;
|
|
54
|
+
});
|
|
55
|
+
const requiredAndNotPresent = directivesToUpdate
|
|
56
|
+
.filter(p => !updatedCspDirectives.find(directive => directive.includes(`${p} `)))
|
|
57
|
+
.map(p => `${p} 'self' ${feComponentsUrl}`);
|
|
58
|
+
res.set('content-security-policy', [...updatedCspDirectives, ...requiredAndNotPresent].join(';'));
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
const defaultOptions = {
|
|
62
|
+
includeSharedData: false,
|
|
63
|
+
useFallbacksByDefault: false,
|
|
64
|
+
};
|
|
65
|
+
class ComponentsService {
|
|
66
|
+
logger;
|
|
67
|
+
componentApiConfig;
|
|
68
|
+
componentApiClient;
|
|
69
|
+
dpsUrl;
|
|
70
|
+
constructor(logger, componentApiConfig, componentApiClient, dpsUrl) {
|
|
71
|
+
this.logger = logger;
|
|
72
|
+
this.componentApiConfig = componentApiConfig;
|
|
73
|
+
this.componentApiClient = componentApiClient;
|
|
74
|
+
this.dpsUrl = dpsUrl;
|
|
75
|
+
}
|
|
76
|
+
static create({ logger = console, componentApiConfig, authenticationClient, dpsUrl, }) {
|
|
77
|
+
return new ComponentsService(logger, componentApiConfig, new ComponentApiClient(logger, componentApiConfig, authenticationClient), dpsUrl);
|
|
78
|
+
}
|
|
79
|
+
getFrontendComponents(requestOptions) {
|
|
80
|
+
const requestOptionsWithDefaults = {
|
|
81
|
+
...defaultOptions,
|
|
82
|
+
...requestOptions,
|
|
83
|
+
};
|
|
84
|
+
const { includeSharedData, useFallbacksByDefault } = requestOptionsWithDefaults;
|
|
85
|
+
return async (_req, res, next) => {
|
|
86
|
+
const useFallbacks = (user) => {
|
|
87
|
+
res.locals.feComponents = {
|
|
88
|
+
header: getFallbackHeader(user, this.dpsUrl, {
|
|
89
|
+
environmentName: requestOptionsWithDefaults.environmentName,
|
|
90
|
+
authUrl: requestOptionsWithDefaults.authUrl,
|
|
91
|
+
supportUrl: requestOptionsWithDefaults.supportUrl,
|
|
92
|
+
}),
|
|
93
|
+
footer: getFallbackFooter(user, {
|
|
94
|
+
authUrl: requestOptionsWithDefaults.authUrl,
|
|
95
|
+
supportUrl: requestOptionsWithDefaults.supportUrl,
|
|
96
|
+
}),
|
|
97
|
+
cssIncludes: [],
|
|
98
|
+
jsIncludes: [],
|
|
99
|
+
};
|
|
100
|
+
};
|
|
101
|
+
if (!res.locals.user) {
|
|
102
|
+
this.logger.info('Using fallback frontend components when no user in context');
|
|
103
|
+
useFallbacks(null);
|
|
104
|
+
return next();
|
|
105
|
+
}
|
|
106
|
+
if (useFallbacksByDefault) {
|
|
107
|
+
this.logger.info('Using fallback frontend components by default');
|
|
108
|
+
useFallbacks(res.locals.user);
|
|
109
|
+
return next();
|
|
110
|
+
}
|
|
111
|
+
try {
|
|
112
|
+
const { header, footer, meta } = await this.componentApiClient.getComponents(res.locals.user.token);
|
|
113
|
+
res.locals.feComponents = {
|
|
114
|
+
header: header.html,
|
|
115
|
+
footer: footer.html,
|
|
116
|
+
cssIncludes: [...header.css, ...footer.css],
|
|
117
|
+
jsIncludes: [...header.javascript, ...footer.javascript],
|
|
118
|
+
};
|
|
119
|
+
if (includeSharedData) {
|
|
120
|
+
res.locals.feComponents.sharedData = meta;
|
|
121
|
+
}
|
|
122
|
+
updateCsp(this.componentApiConfig.url, res);
|
|
123
|
+
return next();
|
|
124
|
+
}
|
|
125
|
+
catch (_error) {
|
|
126
|
+
this.logger.error('Failed to retrieve front end components, using fallbacks');
|
|
127
|
+
useFallbacks(res.locals.user);
|
|
128
|
+
return next();
|
|
129
|
+
}
|
|
130
|
+
};
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
function getFrontendComponents$1({ logger = console, componentApiConfig, authenticationClient, dpsUrl, }) {
|
|
135
|
+
const service = ComponentsService.create({ logger, componentApiConfig, authenticationClient, dpsUrl });
|
|
136
|
+
return requestOptions => service.getFrontendComponents(requestOptions);
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
class PrisonApiClient extends hmppsRestClient.RestClient {
|
|
140
|
+
constructor(logger, config, authenticationClient) {
|
|
141
|
+
super('Prison API Client', config, logger, authenticationClient);
|
|
142
|
+
}
|
|
143
|
+
async getUserCaseLoads(userToken) {
|
|
144
|
+
return this.get({
|
|
145
|
+
path: '/api/users/me/caseloads',
|
|
146
|
+
query: { allCaseloads: true },
|
|
147
|
+
}, hmppsRestClient.asUser(userToken));
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
class CaseLoadService {
|
|
152
|
+
logger;
|
|
153
|
+
prisonApiClient;
|
|
154
|
+
constructor(logger, prisonApiClient) {
|
|
155
|
+
this.logger = logger;
|
|
156
|
+
this.prisonApiClient = prisonApiClient;
|
|
157
|
+
}
|
|
158
|
+
static create({ logger = console, prisonApiConfig, authenticationClient, }) {
|
|
159
|
+
return new CaseLoadService(logger, new PrisonApiClient(logger, prisonApiConfig, authenticationClient));
|
|
160
|
+
}
|
|
161
|
+
retrieveCaseLoadData() {
|
|
162
|
+
return async (req, res, next) => {
|
|
163
|
+
if (!req.session)
|
|
164
|
+
throw new Error('User session required in order to cache case loads');
|
|
165
|
+
if (res.locals.user && res.locals.user.token && res.locals.user.authSource === 'nomis') {
|
|
166
|
+
try {
|
|
167
|
+
// Update cache with values from res.feComponents.sharedData if present
|
|
168
|
+
if (res.locals.feComponents && res.locals.feComponents.sharedData) {
|
|
169
|
+
req.session.caseLoads = res.locals.feComponents.sharedData.caseLoads;
|
|
170
|
+
req.session.activeCaseLoad = res.locals.feComponents.sharedData.activeCaseLoad;
|
|
171
|
+
req.session.activeCaseLoadId = res.locals.feComponents.sharedData.activeCaseLoad?.caseLoadId;
|
|
172
|
+
}
|
|
173
|
+
// If cache is empty, fetch data from Prison API
|
|
174
|
+
if (!req.session.caseLoads) {
|
|
175
|
+
this.logger.info(`Falling back to Prison API to retrieve case loads for: ${res.locals.user.username}`);
|
|
176
|
+
const userCaseLoads = await this.prisonApiClient.getUserCaseLoads(res.locals.user.token);
|
|
177
|
+
const caseLoads = userCaseLoads.filter(caseload => caseload.type !== 'APP');
|
|
178
|
+
const activeCaseLoad = caseLoads.find((caseLoad) => caseLoad.currentlyActive);
|
|
179
|
+
req.session.caseLoads = caseLoads;
|
|
180
|
+
req.session.activeCaseLoad = activeCaseLoad;
|
|
181
|
+
req.session.activeCaseLoadId = activeCaseLoad?.caseLoadId;
|
|
182
|
+
}
|
|
183
|
+
// Populate res.locals.user with values from cache
|
|
184
|
+
res.locals.user.caseLoads = req.session.caseLoads;
|
|
185
|
+
res.locals.user.activeCaseLoad = req.session.activeCaseLoad;
|
|
186
|
+
res.locals.user.activeCaseLoadId = req.session.activeCaseLoadId;
|
|
187
|
+
}
|
|
188
|
+
catch (error) {
|
|
189
|
+
this.logger.error(error, `Failed to retrieve case loads for: ${res.locals.user.username}`);
|
|
190
|
+
return next(error);
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
return next();
|
|
194
|
+
};
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
function retrieveCaseLoadData({ logger = console, prisonApiConfig, authenticationClient, }) {
|
|
199
|
+
const service = CaseLoadService.create({ logger, prisonApiConfig, authenticationClient });
|
|
200
|
+
return () => service.retrieveCaseLoadData();
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
class AllocationsApiClient extends hmppsRestClient.RestClient {
|
|
204
|
+
constructor(logger, config, authenticationClient) {
|
|
205
|
+
super('Allocations API Client', config, logger, authenticationClient);
|
|
206
|
+
}
|
|
207
|
+
async getStaffAllocationPolicies(user) {
|
|
208
|
+
return this.get({
|
|
209
|
+
path: `/prisons/${user.activeCaseLoadId}/staff/${user.userId}/job-classifications`,
|
|
210
|
+
}, hmppsRestClient.asSystem());
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
class AllocationService {
|
|
215
|
+
logger;
|
|
216
|
+
allocationsApiClient;
|
|
217
|
+
constructor(logger, allocationsApiClient) {
|
|
218
|
+
this.logger = logger;
|
|
219
|
+
this.allocationsApiClient = allocationsApiClient;
|
|
220
|
+
}
|
|
221
|
+
static create({ logger = console, allocationsApiConfig, authenticationClient, }) {
|
|
222
|
+
return new AllocationService(logger, new AllocationsApiClient(logger, allocationsApiConfig, authenticationClient));
|
|
223
|
+
}
|
|
224
|
+
retrieveAllocationJobResponsibilities() {
|
|
225
|
+
return async (req, res, next) => {
|
|
226
|
+
if (!req.session)
|
|
227
|
+
throw new Error('User session required in order to cache allocation job responsibilities');
|
|
228
|
+
if (!res.locals.user.token)
|
|
229
|
+
throw new Error('Caseload details needs to be populated before retrieving allocation job responsibilities. Please run retrieveCaseLoadData before retrieveAllocationJobResponsibilities.');
|
|
230
|
+
if (res.locals.user && res.locals.user.authSource === 'nomis') {
|
|
231
|
+
try {
|
|
232
|
+
// Update cache with values from res.feComponents.sharedData if present
|
|
233
|
+
if (res.locals.feComponents && res.locals.feComponents.sharedData) {
|
|
234
|
+
req.session.allocationJobResponsibilities = res.locals.feComponents.sharedData.allocationJobResponsibilities;
|
|
235
|
+
}
|
|
236
|
+
// If cache is empty, fetch data from Prison API
|
|
237
|
+
if (!req.session.allocationJobResponsibilities) {
|
|
238
|
+
this.logger.info(`Falling back to Allocations API to retrieve job responsibilities for: ${res.locals.user.username}`);
|
|
239
|
+
const allocationPolicies = await this.allocationsApiClient.getStaffAllocationPolicies(res.locals.user);
|
|
240
|
+
req.session.allocationJobResponsibilities = allocationPolicies.policies;
|
|
241
|
+
}
|
|
242
|
+
// Populate res.locals.user with values from cache
|
|
243
|
+
res.locals.user.allocationJobResponsibilities = req.session.allocationJobResponsibilities;
|
|
244
|
+
}
|
|
245
|
+
catch (error) {
|
|
246
|
+
this.logger.error(error, `Failed to retrieve allocation job responsibilities for: ${res.locals.user.username}`);
|
|
247
|
+
return next(error);
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
return next();
|
|
251
|
+
};
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
function getFrontendComponents({ logger = console, allocationsApiConfig, authenticationClient, }) {
|
|
256
|
+
const service = AllocationService.create({ logger, allocationsApiConfig, authenticationClient });
|
|
257
|
+
return () => service.retrieveAllocationJobResponsibilities();
|
|
258
|
+
}
|
|
259
|
+
|
|
260
|
+
exports.AllocationService = AllocationService;
|
|
261
|
+
exports.CaseLoadService = CaseLoadService;
|
|
262
|
+
exports.ComponentsService = ComponentsService;
|
|
263
|
+
exports.getFrontendComponents = getFrontendComponents$1;
|
|
264
|
+
exports.retrieveAllocationJobResponsibilities = getFrontendComponents;
|
|
265
|
+
exports.retrieveCaseLoadData = retrieveCaseLoadData;
|
|
266
|
+
//# sourceMappingURL=index.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.cjs","sources":["../src/data/componentApi/componentApiClient.ts","../src/utils/fallbacks.ts","../src/utils/updateCsp.ts","../src/componentsService.ts","../src/middleware/getFrontendComponents.ts","../src/data/prisonApi/prisonApiClient.ts","../src/caseLoadService.ts","../src/middleware/retrieveCaseLoadData.ts","../src/data/allocationsApi/allocationsApiClient.ts","../src/allocationService.ts","../src/middleware/retrieveAllocationJobResponsibilities.ts"],"sourcesContent":["import { ApiConfig, asSystem, AuthenticationClient, RestClient } from '@ministryofjustice/hmpps-rest-client'\nimport AvailableComponent from '../../types/AvailableComponent'\nimport Component from '../../types/Component'\nimport { ConnectDpsComponentLogger } from '../../types/ConnectDpsComponentLogger'\nimport { ComponentsSharedData } from '../../types/HeaderFooterSharedData'\n\nexport type ComponentsApiResponse<T extends AvailableComponent[]> = Record<T[number], Component> & {\n meta: ComponentsSharedData[T[number]] // TODO: rename 'meta' in the API response\n}\n\nexport default class ComponentApiClient extends RestClient {\n constructor(logger: ConnectDpsComponentLogger, config: ApiConfig, authenticationClient: AuthenticationClient) {\n super('Component API Client', config, logger, authenticationClient)\n }\n\n async getComponents<T extends AvailableComponent[]>(userToken: string): Promise<ComponentsApiResponse<T>> {\n return this.get<ComponentsApiResponse<T>>(\n {\n path: `/components`,\n query: 'component=header&component=footer',\n headers: { 'x-user-token': userToken },\n },\n asSystem(),\n )\n }\n}\n","import nunjucks from 'nunjucks'\nimport { HmppsUser } from '../types/HmppsUser'\n\nexport function getFallbackHeader(\n user: HmppsUser | null,\n dpsUrl: string,\n { environmentName, authUrl, supportUrl }: { environmentName?: string; authUrl?: string; supportUrl?: string },\n): string {\n return nunjucks.render('dpsComponents/header-bar.njk', {\n isPrisonUser: !user || user.authSource === 'nomis',\n user,\n dpsUrl,\n environmentName,\n authUrl,\n supportUrl,\n name: initialiseName(user?.displayName),\n })\n}\n\nexport function getFallbackFooter(\n user: HmppsUser | null,\n { authUrl, supportUrl }: { authUrl?: string; supportUrl?: string },\n): string {\n return nunjucks.render('dpsComponents/footer.njk', {\n isPrisonUser: !user || user.authSource === 'nomis',\n supportUrl,\n authUrl,\n })\n}\n\nfunction initialiseName(fullName?: string): string | null {\n if (!fullName) return null\n\n const array = fullName.split(' ')\n return `${array[0][0]}. ${array.reverse()[0]}`\n}\n","import { type Response } from 'express'\n\nexport default function updateCsp(feComponentsUrl: string, res: Response) {\n const csp = res.getHeaders()['content-security-policy']\n const allDirectives = csp?.split(';') ?? []\n const directivesToUpdate = ['script-src', 'style-src', 'img-src', 'font-src']\n\n const updatedCspDirectives = allDirectives.map(directive => {\n // if directive is not in cspToUpdate or includes fe components url already return as is\n if (directive.includes(feComponentsUrl as string) || !directivesToUpdate.some(p => directive.includes(`${p} `)))\n return directive\n\n // if directive is in cspToUpdate and does not have fe components url, add in\n return `${directive} ${feComponentsUrl}`\n })\n\n const requiredAndNotPresent = directivesToUpdate\n .filter(p => !updatedCspDirectives.find(directive => directive.includes(`${p} `)))\n .map(p => `${p} 'self' ${feComponentsUrl}`)\n\n res.set('content-security-policy', [...updatedCspDirectives, ...requiredAndNotPresent].join(';'))\n}\n","import { type NextFunction, type Request, type Response, type RequestHandler } from 'express'\nimport { ApiConfig, AuthenticationClient } from '@ministryofjustice/hmpps-rest-client'\nimport ComponentApiClient from './data/componentApi/componentApiClient'\nimport { getFallbackFooter, getFallbackHeader } from './utils/fallbacks'\nimport updateCsp from './utils/updateCsp'\nimport { HmppsUser } from './types/HmppsUser'\nimport { ConnectDpsComponentLogger } from './types/ConnectDpsComponentLogger'\n\nexport interface FrontentComponentRequestOptions {\n authUrl?: string\n supportUrl?: string\n environmentName?: 'DEV' | 'PRE-PRODUCTION' | 'PRODUCTION'\n includeSharedData?: boolean\n useFallbacksByDefault?: boolean\n}\n\nconst defaultOptions: Partial<FrontentComponentRequestOptions> = {\n includeSharedData: false,\n useFallbacksByDefault: false,\n}\n\nexport default class ComponentsService {\n constructor(\n private readonly logger: ConnectDpsComponentLogger,\n private readonly componentApiConfig: ApiConfig,\n private readonly componentApiClient: ComponentApiClient,\n private readonly dpsUrl: string,\n ) {}\n\n static create({\n logger = console,\n componentApiConfig,\n authenticationClient,\n dpsUrl,\n }: {\n logger?: ConnectDpsComponentLogger\n componentApiConfig: ApiConfig\n authenticationClient: AuthenticationClient\n dpsUrl: string\n }) {\n return new ComponentsService(\n logger,\n componentApiConfig,\n new ComponentApiClient(logger, componentApiConfig, authenticationClient),\n dpsUrl,\n )\n }\n\n getFrontendComponents(requestOptions: FrontentComponentRequestOptions): RequestHandler {\n const requestOptionsWithDefaults = {\n ...defaultOptions,\n ...requestOptions,\n }\n const { includeSharedData, useFallbacksByDefault } = requestOptionsWithDefaults\n\n return async (_req: Request, res: Response, next: NextFunction) => {\n const useFallbacks = (user: HmppsUser | null) => {\n res.locals.feComponents = {\n header: getFallbackHeader(user, this.dpsUrl, {\n environmentName: requestOptionsWithDefaults.environmentName,\n authUrl: requestOptionsWithDefaults.authUrl,\n supportUrl: requestOptionsWithDefaults.supportUrl,\n }),\n footer: getFallbackFooter(user, {\n authUrl: requestOptionsWithDefaults.authUrl,\n supportUrl: requestOptionsWithDefaults.supportUrl,\n }),\n cssIncludes: [],\n jsIncludes: [],\n }\n }\n\n if (!res.locals.user) {\n this.logger.info('Using fallback frontend components when no user in context')\n useFallbacks(null)\n return next()\n }\n\n if (useFallbacksByDefault) {\n this.logger.info('Using fallback frontend components by default')\n useFallbacks(res.locals.user)\n return next()\n }\n\n try {\n const { header, footer, meta } = await this.componentApiClient.getComponents(res.locals.user.token as string)\n\n res.locals.feComponents = {\n header: header.html,\n footer: footer.html,\n cssIncludes: [...header.css, ...footer.css],\n jsIncludes: [...header.javascript, ...footer.javascript],\n }\n\n if (includeSharedData) {\n res.locals.feComponents.sharedData = meta\n }\n\n updateCsp(this.componentApiConfig.url, res)\n\n return next()\n } catch (_error) {\n this.logger.error('Failed to retrieve front end components, using fallbacks')\n useFallbacks(res.locals.user)\n return next()\n }\n }\n }\n}\n","import { type RequestHandler } from 'express'\nimport ComponentsService, { FrontentComponentRequestOptions } from '../componentsService'\n\nexport default function getFrontendComponents({\n logger = console,\n componentApiConfig,\n authenticationClient,\n dpsUrl,\n}: Parameters<typeof ComponentsService.create>[0]): (\n requestOptions: FrontentComponentRequestOptions,\n) => RequestHandler {\n const service = ComponentsService.create({ logger, componentApiConfig, authenticationClient, dpsUrl })\n return requestOptions => service.getFrontendComponents(requestOptions)\n}\n","import { ApiConfig, asUser, AuthenticationClient, RestClient } from '@ministryofjustice/hmpps-rest-client'\nimport CaseLoad from '../../types/CaseLoad'\nimport { ConnectDpsComponentLogger } from '../../types/ConnectDpsComponentLogger'\n\nexport default class PrisonApiClient extends RestClient {\n constructor(logger: ConnectDpsComponentLogger, config: ApiConfig, authenticationClient: AuthenticationClient) {\n super('Prison API Client', config, logger, authenticationClient)\n }\n\n async getUserCaseLoads(userToken: string): Promise<CaseLoad[]> {\n return this.get<CaseLoad[]>(\n {\n path: '/api/users/me/caseloads',\n query: { allCaseloads: true },\n },\n asUser(userToken),\n )\n }\n}\n","import { type RequestHandler } from 'express'\nimport { ApiConfig, AuthenticationClient } from '@ministryofjustice/hmpps-rest-client'\nimport CaseLoad from './types/CaseLoad'\nimport PrisonApiClient from './data/prisonApi/prisonApiClient'\nimport { ConnectDpsComponentLogger } from './types/ConnectDpsComponentLogger'\n\nexport default class CaseLoadService {\n constructor(\n private readonly logger: ConnectDpsComponentLogger,\n private readonly prisonApiClient: PrisonApiClient,\n ) {}\n\n static create({\n logger = console,\n prisonApiConfig,\n authenticationClient,\n }: {\n logger?: ConnectDpsComponentLogger\n prisonApiConfig: ApiConfig\n authenticationClient: AuthenticationClient\n }) {\n return new CaseLoadService(logger, new PrisonApiClient(logger, prisonApiConfig, authenticationClient))\n }\n\n retrieveCaseLoadData(): RequestHandler {\n return async (req, res, next) => {\n if (!req.session) throw new Error('User session required in order to cache case loads')\n\n if (res.locals.user && res.locals.user.token && res.locals.user.authSource === 'nomis') {\n try {\n // Update cache with values from res.feComponents.sharedData if present\n if (res.locals.feComponents && res.locals.feComponents.sharedData) {\n req.session.caseLoads = res.locals.feComponents.sharedData.caseLoads\n req.session.activeCaseLoad = res.locals.feComponents.sharedData.activeCaseLoad\n req.session.activeCaseLoadId = res.locals.feComponents.sharedData.activeCaseLoad?.caseLoadId\n }\n\n // If cache is empty, fetch data from Prison API\n if (!req.session.caseLoads) {\n this.logger.info(`Falling back to Prison API to retrieve case loads for: ${res.locals.user.username}`)\n const userCaseLoads = await this.prisonApiClient.getUserCaseLoads(res.locals.user.token)\n const caseLoads = userCaseLoads.filter(caseload => caseload.type !== 'APP')\n const activeCaseLoad = caseLoads.find((caseLoad: CaseLoad) => caseLoad.currentlyActive)\n\n req.session.caseLoads = caseLoads\n req.session.activeCaseLoad = activeCaseLoad\n req.session.activeCaseLoadId = activeCaseLoad?.caseLoadId\n }\n\n // Populate res.locals.user with values from cache\n res.locals.user.caseLoads = req.session.caseLoads\n res.locals.user.activeCaseLoad = req.session.activeCaseLoad\n res.locals.user.activeCaseLoadId = req.session.activeCaseLoadId\n } catch (error) {\n this.logger.error(error, `Failed to retrieve case loads for: ${res.locals.user.username}`)\n return next(error)\n }\n }\n\n return next()\n }\n }\n}\n","import { type RequestHandler } from 'express'\nimport CaseLoadService from '../caseLoadService'\n\nexport default function retrieveCaseLoadData({\n logger = console,\n prisonApiConfig,\n authenticationClient,\n}: Parameters<typeof CaseLoadService.create>[0]): () => RequestHandler {\n const service = CaseLoadService.create({ logger, prisonApiConfig, authenticationClient })\n return () => service.retrieveCaseLoadData()\n}\n","import { ApiConfig, asSystem, AuthenticationClient, RestClient } from '@ministryofjustice/hmpps-rest-client'\nimport { PrisonUser } from '../../types/HmppsUser'\nimport { AllocationJobResponsibility } from '../../types/AllocationJobResponsibility'\nimport { ConnectDpsComponentLogger } from '../../types/ConnectDpsComponentLogger'\n\nexport default class AllocationsApiClient extends RestClient {\n constructor(logger: ConnectDpsComponentLogger, config: ApiConfig, authenticationClient: AuthenticationClient) {\n super('Allocations API Client', config, logger, authenticationClient)\n }\n\n async getStaffAllocationPolicies(user: PrisonUser): Promise<{ policies: AllocationJobResponsibility[] }> {\n return this.get<{ policies: AllocationJobResponsibility[] }>(\n {\n path: `/prisons/${user.activeCaseLoadId}/staff/${user.userId}/job-classifications`,\n },\n asSystem(),\n )\n }\n}\n","import { type RequestHandler } from 'express'\nimport { ApiConfig, AuthenticationClient } from '@ministryofjustice/hmpps-rest-client'\nimport AllocationsApiClient from './data/allocationsApi/allocationsApiClient'\nimport { ConnectDpsComponentLogger } from './types/ConnectDpsComponentLogger'\n\nexport default class AllocationService {\n constructor(\n private readonly logger: ConnectDpsComponentLogger,\n private readonly allocationsApiClient: AllocationsApiClient,\n ) {}\n\n static create({\n logger = console,\n allocationsApiConfig,\n authenticationClient,\n }: {\n logger?: ConnectDpsComponentLogger\n allocationsApiConfig: ApiConfig\n authenticationClient: AuthenticationClient\n }) {\n return new AllocationService(logger, new AllocationsApiClient(logger, allocationsApiConfig, authenticationClient))\n }\n\n public retrieveAllocationJobResponsibilities(): RequestHandler {\n return async (req, res, next) => {\n if (!req.session) throw new Error('User session required in order to cache allocation job responsibilities')\n if (!res.locals.user.token)\n throw new Error(\n 'Caseload details needs to be populated before retrieving allocation job responsibilities. Please run retrieveCaseLoadData before retrieveAllocationJobResponsibilities.',\n )\n\n if (res.locals.user && res.locals.user.authSource === 'nomis') {\n try {\n // Update cache with values from res.feComponents.sharedData if present\n if (res.locals.feComponents && res.locals.feComponents.sharedData) {\n req.session.allocationJobResponsibilities = res.locals.feComponents.sharedData.allocationJobResponsibilities\n }\n\n // If cache is empty, fetch data from Prison API\n if (!req.session.allocationJobResponsibilities) {\n this.logger.info(\n `Falling back to Allocations API to retrieve job responsibilities for: ${res.locals.user.username}`,\n )\n const allocationPolicies = await this.allocationsApiClient.getStaffAllocationPolicies(res.locals.user)\n req.session.allocationJobResponsibilities = allocationPolicies.policies\n }\n\n // Populate res.locals.user with values from cache\n res.locals.user.allocationJobResponsibilities = req.session.allocationJobResponsibilities\n } catch (error) {\n this.logger.error(\n error,\n `Failed to retrieve allocation job responsibilities for: ${res.locals.user.username}`,\n )\n return next(error)\n }\n }\n\n return next()\n }\n }\n}\n","import { type RequestHandler } from 'express'\nimport AllocationService from '../allocationService'\n\nexport default function getFrontendComponents({\n logger = console,\n allocationsApiConfig,\n authenticationClient,\n}: Parameters<typeof AllocationService.create>[0]): () => RequestHandler {\n const service = AllocationService.create({ logger, allocationsApiConfig, authenticationClient })\n return () => service.retrieveAllocationJobResponsibilities()\n}\n"],"names":["RestClient","asSystem","getFrontendComponents","asUser"],"mappings":"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"}
|