Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Upgrading dependencies #66

Merged
merged 1 commit into from
Jan 6, 2022
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1,883 changes: 1,050 additions & 833 deletions package-lock.json

Large diffs are not rendered by default.

62 changes: 31 additions & 31 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
"scripts": {
"prepare": "husky install",
"copy-views": "cp -R server/views dist/server/",
"compile-sass": "sass --no-source-map --load-path=node_modules/govuk-frontend --load-path=node_modules/@ministryofjustice/frontend ./assets/sass/application.sass:./assets/stylesheets/application.css ./assets/sass/application-ie8.sass:./assets/stylesheets/application-ie8.css --style compressed",
"compile-sass": "sass --quiet-deps --no-source-map --load-path=node_modules/govuk-frontend --load-path=node_modules/@ministryofjustice/frontend --load-path=. ./assets/sass/application.sass:./assets/stylesheets/application.css ./assets/sass/application-ie8.sass:./assets/stylesheets/application-ie8.css --style compressed",
Copy link
Contributor Author

@andrewrlee andrewrlee Jan 4, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

reduce logging on errors related to this issue

Also need to add new load path for . as govuk frontend now explicitly reference govuk node modules as a dependency related to this: ministryofjustice/moj-frontend#147

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we need the second one? Or do we only import bits of govuk? ...and a separate issue, what point are we ditching IE8 support? I don't believe it's in our supported browsers.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We need all 3 - or we have to start referring to things with full paths.
The new load path is used only to bodge around the change in the moj-frontend and how they refer to assets from the gov uk frontend library.

I think we still have users in probation who use IE8, we did the last time I checked..

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the explanation @andrewrlee 👍

"watch-ts": "tsc -w",
"watch-views": "nodemon --watch server/views -e html,njk -x npm run copy-views",
"watch-node": "DEBUG=gov-starter-server* nodemon --watch dist/ dist/server.js | bunyan -o short",
Expand Down Expand Up @@ -84,78 +84,78 @@
]
},
"dependencies": {
"@ministryofjustice/frontend": "^1.0.1",
"agentkeepalive": "^4.1.4",
"applicationinsights": "^2.1.9",
"body-parser": "^1.19.0",
"@ministryofjustice/frontend": "^1.1.0",
"agentkeepalive": "^4.2.0",
"applicationinsights": "^2.2.0",
"body-parser": "^1.19.1",
"bunyan": "^1.8.15",
"bunyan-format": "^0.2.1",
"compression": "^1.7.4",
"connect-flash": "^0.1.1",
"connect-redis": "^6.0.0",
"cookie-session": "^1.4.0",
"cookie-session": "^2.0.0",
"csurf": "^1.11.0",
"dotenv": "^10.0.0",
"express": "^4.17.1",
"express": "^4.17.2",
"express-request-id": "^1.4.1",
"express-session": "^1.17.2",
"govuk-frontend": "^3.14.0",
"helmet": "^4.6.0",
"http-errors": "^1.8.1",
"govuk-frontend": "^4.0.0",
"helmet": "^5.0.1",
"http-errors": "^2.0.0",
"jquery": "^3.6.0",
"jwt-decode": "^3.1.2",
"nocache": "^3.0.1",
"nunjucks": "^3.2.3",
"passport": "^0.5.0",
"passport": "^0.5.2",
"passport-oauth2": "^1.6.1",
"redis": "^3.1.2",
"redis": "^4.0.1",
"superagent": "^6.1.0"
},
"devDependencies": {
"@types/bunyan": "^1.8.8",
"@types/bunyan-format": "^0.2.4",
"@types/compression": "^1.7.2",
"@types/connect-flash": "0.0.37",
"@types/connect-redis": "^0.0.17",
"@types/cookie-session": "^2.0.43",
"@types/connect-redis": "^0.0.18",
"@types/cookie-session": "^2.0.44",
"@types/csurf": "^1.11.2",
"@types/express-request-id": "^1.4.3",
"@types/express-session": "^1.17.4",
"@types/http-errors": "^1.8.1",
"@types/jest": "^27.0.3",
"@types/jest": "^27.4.0",
"@types/jsonwebtoken": "^8.5.6",
"@types/node": "^16.11.11",
"@types/nunjucks": "^3.2.0",
"@types/node": "^16.11.12",
"@types/nunjucks": "^3.2.1",
"@types/passport": "^1.0.7",
"@types/passport-oauth2": "^1.4.11",
"@types/redis": "^2.8.32",
"@types/superagent": "^4.1.13",
"@types/redis": "^4.0.10",
"@types/superagent": "^4.1.14",
"@types/supertest": "^2.0.11",
"@typescript-eslint/eslint-plugin": "^5.5.0",
"@typescript-eslint/parser": "^5.5.0",
"concurrently": "^6.4.0",
"cypress": "^9.1.0",
"@typescript-eslint/eslint-plugin": "^5.9.0",
"@typescript-eslint/parser": "^5.9.0",
"concurrently": "^7.0.0",
"cypress": "^9.2.0",
"cypress-multi-reporters": "^1.5.0",
"eslint": "^8.3.0",
"eslint": "^8.6.0",
"eslint-config-airbnb-base": "^15.0.0",
"eslint-config-prettier": "^8.3.0",
"eslint-import-resolver-typescript": "^2.5.0",
"eslint-plugin-cypress": "^2.12.1",
"eslint-plugin-import": "2.25.3",
"eslint-plugin-import": "2.25.4",
"eslint-plugin-prettier": "^4.0.0",
"husky": "^7.0.4",
"jest": "^27.4.3",
"jest": "^27.4.5",
"jest-html-reporter": "^3.4.2",
"jest-junit": "^13.0.0",
"jsonwebtoken": "^8.5.1",
"lint-staged": "^12.1.2",
"lint-staged": "^12.1.5",
"mocha-junit-reporter": "^2.0.2",
"nock": "^13.2.1",
"nodemon": "^2.0.15",
"prettier": "^2.5.0",
"sass": "^1.44.0",
"prettier": "^2.5.1",
"sass": "^1.45.2",
"supertest": "^6.1.6",
"ts-jest": "^27.0.7",
"typescript": "^4.5.2"
"ts-jest": "^27.1.2",
"typescript": "^4.5.4"
}
}
2 changes: 1 addition & 1 deletion server/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ export default {
https: production,
staticResourceCacheDuration: 20,
redis: {
host: process.env.REDIS_HOST,
host: get('REDIS_HOST', 'localhost', requiredInProduction),
port: parseInt(process.env.REDIS_PORT, 10) || 6379,
password: process.env.REDIS_AUTH_TOKEN,
tls_enabled: get('REDIS_TLS_ENABLED', 'false'),
Expand Down
18 changes: 18 additions & 0 deletions server/data/redisClient.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
import { createClient } from 'redis'

import config from '../config'

export type RedisClient = ReturnType<typeof createClient>

const url =
config.redis.tls_enabled === 'true'
? `rediss://${config.redis.host}:${config.redis.port}`
: `redis://${config.redis.host}:${config.redis.port}`

export const createRedisClient = (legacyMode = false): RedisClient => {
return createClient({
url,
password: config.redis.password,
legacyMode,
})
}
42 changes: 32 additions & 10 deletions server/data/tokenStore.test.ts
Original file line number Diff line number Diff line change
@@ -1,11 +1,13 @@
import { RedisClient } from 'redis'
import { RedisClient } from './redisClient'
import TokenStore from './tokenStore'

const redisClient = {
on: jest.fn(),
get: jest.fn(),
set: jest.fn(),
}
on: jest.fn(),
connect: jest.fn(),
isOpen: true,
} as unknown as jest.Mocked<RedisClient>

describe('tokenStore', () => {
let tokenStore: TokenStore
Expand All @@ -18,17 +20,37 @@ describe('tokenStore', () => {
jest.resetAllMocks()
})

it('Can retrieve token', async () => {
redisClient.get.mockImplementation((key, callback) => callback(null, 'token-1'))
describe('get token', () => {
it('Can retrieve token', async () => {
redisClient.get.mockResolvedValue('token-1')

await expect(tokenStore.getToken('user-1')).resolves.toBe('token-1')

expect(redisClient.get).toHaveBeenCalledWith('systemToken:user-1')
})

await expect(tokenStore.getToken('user-1')).resolves.toBe('token-1')
it('Connects when no connection calling getToken', async () => {
;(redisClient as unknown as Record<string, boolean>).isOpen = false

expect(redisClient.get).toHaveBeenCalledWith('user-1', expect.any(Function))
await tokenStore.getToken('user-1')

expect(redisClient.connect).toHaveBeenCalledWith()
})
})

it('Can set token', async () => {
await tokenStore.setToken('user-1', 'token-1', 10)
describe('set token', () => {
it('Can set token', async () => {
await tokenStore.setToken('user-1', 'token-1', 10)

expect(redisClient.set).toHaveBeenCalledWith('systemToken:user-1', 'token-1', { EX: 10 })
})

it('Connects when no connection calling set token', async () => {
;(redisClient as unknown as Record<string, boolean>).isOpen = false

await tokenStore.setToken('user-1', 'token-1', 10)

expect(redisClient.set).toHaveBeenCalledWith('user-1', 'token-1', 'EX', 10, expect.any(Function))
expect(redisClient.connect).toHaveBeenCalledWith()
})
})
})
35 changes: 13 additions & 22 deletions server/data/tokenStore.ts
Original file line number Diff line number Diff line change
@@ -1,38 +1,29 @@
import redis from 'redis'
import { promisify } from 'util'
import type { RedisClient } from './redisClient'

import logger from '../../logger'
import config from '../config'

const createRedisClient = () => {
return redis.createClient({
port: config.redis.port,
password: config.redis.password,
host: config.redis.host,
tls: config.redis.tls_enabled === 'true' ? {} : false,
prefix: 'systemToken:',
})
}

export default class TokenStore {
private getRedisAsync: (key: string) => Promise<string>
private readonly prefix = 'systemToken:'

private setRedisAsync: (key: string, value: string, mode: string, durationSeconds: number) => Promise<void>

constructor(redisClient: redis.RedisClient = createRedisClient()) {
redisClient.on('error', error => {
constructor(private readonly client: RedisClient) {
client.on('error', error => {
logger.error(error, `Redis error`)
})
}

this.getRedisAsync = promisify(redisClient.get).bind(redisClient)
this.setRedisAsync = promisify(redisClient.set).bind(redisClient)
private async ensureConnected() {
if (!this.client.isOpen) {
await this.client.connect()
}
}

public async setToken(key: string, token: string, durationSeconds: number): Promise<void> {
this.setRedisAsync(key, token, 'EX', durationSeconds)
await this.ensureConnected()
await this.client.set(`${this.prefix}${key}`, token, { EX: durationSeconds })
}

public async getToken(key: string): Promise<string> {
return this.getRedisAsync(key)
await this.ensureConnected()
return this.client.get(`${this.prefix}${key}`)
}
}
3 changes: 2 additions & 1 deletion server/index.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,10 @@
import createApp from './app'
import HmppsAuthClient from './data/hmppsAuthClient'
import { createRedisClient } from './data/redisClient'
import TokenStore from './data/tokenStore'
import UserService from './services/userService'

const hmppsAuthClient = new HmppsAuthClient(new TokenStore())
const hmppsAuthClient = new HmppsAuthClient(new TokenStore(createRedisClient()))
const userService = new UserService(hmppsAuthClient)

const app = createApp(userService)
Expand Down
17 changes: 6 additions & 11 deletions server/middleware/setUpWebSession.ts
Original file line number Diff line number Diff line change
@@ -1,25 +1,20 @@
import redis from 'redis'
import session from 'express-session'
import connectRedis from 'connect-redis'
import connectRedis, { Client } from 'connect-redis'
import addRequestId from 'express-request-id'
import express, { Router } from 'express'

import { createRedisClient } from '../data/redisClient'
import config from '../config'

const RedisStore = connectRedis(session)

const client = redis.createClient({
port: config.redis.port,
password: config.redis.password,
host: config.redis.host,
tls: config.redis.tls_enabled === 'true' ? {} : false,
})

export default function setUpWebSession(): Router {
const client = createRedisClient(true)
client.connect()

const router = express.Router()
router.use(
session({
store: new RedisStore({ client }),
store: new RedisStore({ client: client as unknown as Client }),
Copy link
Contributor Author

@andrewrlee andrewrlee Jan 4, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

bit gnarly - waiting on connect-redis to update to new version of redis client:
tj/connect-redis#337

cookie: { secure: config.https, sameSite: 'lax', maxAge: config.session.expiryMinutes * 60 * 1000 },
secret: config.session.secret,
resave: false, // redis implements touch so shouldn't need this
Expand Down