fix(core): Initialize JWT Secret before it's used anywhere (#7707)

HELP-394
This commit is contained in:
कारतोफ्फेलस्क्रिप्ट™ 2023-11-15 12:17:18 +01:00 committed by GitHub
parent 5aee2b768f
commit 3460eb5eeb
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 75 additions and 53 deletions

View file

@ -1,4 +1,3 @@
import jwt from 'jsonwebtoken';
import type { Response } from 'express'; import type { Response } from 'express';
import { createHash } from 'crypto'; import { createHash } from 'crypto';
import { AUTH_COOKIE_NAME, RESPONSE_ERROR_MESSAGES } from '@/constants'; import { AUTH_COOKIE_NAME, RESPONSE_ERROR_MESSAGES } from '@/constants';
@ -9,6 +8,7 @@ import * as ResponseHelper from '@/ResponseHelper';
import { License } from '@/License'; import { License } from '@/License';
import { Container } from 'typedi'; import { Container } from 'typedi';
import { UserRepository } from '@db/repositories/user.repository'; import { UserRepository } from '@db/repositories/user.repository';
import { JwtService } from '@/services/jwt.service';
export function issueJWT(user: User): JwtToken { export function issueJWT(user: User): JwtToken {
const { id, email, password } = user; const { id, email, password } = user;
@ -34,7 +34,7 @@ export function issueJWT(user: User): JwtToken {
.digest('hex'); .digest('hex');
} }
const signedToken = jwt.sign(payload, config.getEnv('userManagement.jwtSecret'), { const signedToken = Container.get(JwtService).sign(payload, {
expiresIn: expiresIn / 1000 /* in seconds */, expiresIn: expiresIn / 1000 /* in seconds */,
}); });
@ -75,9 +75,9 @@ export async function resolveJwtContent(jwtPayload: JwtPayload): Promise<User> {
} }
export async function resolveJwt(token: string): Promise<User> { export async function resolveJwt(token: string): Promise<User> {
const jwtPayload = jwt.verify(token, config.getEnv('userManagement.jwtSecret'), { const jwtPayload: JwtPayload = Container.get(JwtService).verify(token, {
algorithms: ['HS256'], algorithms: ['HS256'],
}) as JwtPayload; });
return resolveJwtContent(jwtPayload); return resolveJwtContent(jwtPayload);
} }

View file

@ -13,7 +13,6 @@ import { promisify } from 'util';
import glob from 'fast-glob'; import glob from 'fast-glob';
import { sleep, jsonParse } from 'n8n-workflow'; import { sleep, jsonParse } from 'n8n-workflow';
import { createHash } from 'crypto';
import config from '@/config'; import config from '@/config';
import { ActiveExecutions } from '@/ActiveExecutions'; import { ActiveExecutions } from '@/ActiveExecutions';
@ -272,20 +271,6 @@ export class Start extends BaseCommand {
// eslint-disable-next-line @typescript-eslint/no-shadow // eslint-disable-next-line @typescript-eslint/no-shadow
const { flags } = this.parse(Start); const { flags } = this.parse(Start);
if (!config.getEnv('userManagement.jwtSecret')) {
// If we don't have a JWT secret set, generate
// one based and save to config.
const { encryptionKey } = this.instanceSettings;
// For a key off every other letter from encryption key
// CAREFUL: do not change this or it breaks all existing tokens.
let baseKey = '';
for (let i = 0; i < encryptionKey.length; i += 2) {
baseKey += encryptionKey[i];
}
config.set('userManagement.jwtSecret', createHash('sha256').update(baseKey).digest('hex'));
}
// Load settings from database and set them to config. // Load settings from database and set them to config.
const databaseSettings = await Container.get(SettingsRepository).findBy({ const databaseSettings = await Container.get(SettingsRepository).findBy({
loadOnStartup: true, loadOnStartup: true,

View file

@ -6,11 +6,11 @@ import { Strategy } from 'passport-jwt';
import { sync as globSync } from 'fast-glob'; import { sync as globSync } from 'fast-glob';
import type { JwtPayload } from '@/Interfaces'; import type { JwtPayload } from '@/Interfaces';
import type { AuthenticatedRequest } from '@/requests'; import type { AuthenticatedRequest } from '@/requests';
import config from '@/config';
import { AUTH_COOKIE_NAME, EDITOR_UI_DIST_DIR } from '@/constants'; import { AUTH_COOKIE_NAME, EDITOR_UI_DIST_DIR } from '@/constants';
import { issueCookie, resolveJwtContent } from '@/auth/jwt'; import { issueCookie, resolveJwtContent } from '@/auth/jwt';
import { canSkipAuth } from '@/decorators/registerController'; import { canSkipAuth } from '@/decorators/registerController';
import { Logger } from '@/Logger'; import { Logger } from '@/Logger';
import { JwtService } from '@/services/jwt.service';
const jwtFromRequest = (req: Request) => { const jwtFromRequest = (req: Request) => {
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
@ -21,7 +21,7 @@ const userManagementJwtAuth = (): RequestHandler => {
const jwtStrategy = new Strategy( const jwtStrategy = new Strategy(
{ {
jwtFromRequest, jwtFromRequest,
secretOrKey: config.getEnv('userManagement.jwtSecret'), secretOrKey: Container.get(JwtService).jwtSecret,
}, },
async (jwtPayload: JwtPayload, done) => { async (jwtPayload: JwtPayload, done) => {
try { try {

View file

@ -1,17 +1,34 @@
import { Service } from 'typedi'; import { Service } from 'typedi';
import * as jwt from 'jsonwebtoken'; import { createHash } from 'crypto';
import jwt from 'jsonwebtoken';
import { InstanceSettings } from 'n8n-core';
import config from '@/config'; import config from '@/config';
@Service() @Service()
export class JwtService { export class JwtService {
private readonly userManagementSecret = config.getEnv('userManagement.jwtSecret'); readonly jwtSecret = config.getEnv('userManagement.jwtSecret');
public signData(payload: object, options: jwt.SignOptions = {}): string { constructor({ encryptionKey }: InstanceSettings) {
return jwt.sign(payload, this.userManagementSecret, options); this.jwtSecret = config.getEnv('userManagement.jwtSecret');
if (!this.jwtSecret) {
// If we don't have a JWT secret set, generate one based on encryption key.
// For a key off every other letter from encryption key
// CAREFUL: do not change this or it breaks all existing tokens.
let baseKey = '';
for (let i = 0; i < encryptionKey.length; i += 2) {
baseKey += encryptionKey[i];
}
this.jwtSecret = createHash('sha256').update(baseKey).digest('hex');
config.set('userManagement.jwtSecret', this.jwtSecret);
}
} }
public verifyToken<T = JwtPayload>(token: string, options: jwt.VerifyOptions = {}) { public sign(payload: object, options: jwt.SignOptions = {}): string {
return jwt.verify(token, this.userManagementSecret, options) as T; return jwt.sign(payload, this.jwtSecret, options);
}
public verify<T = JwtPayload>(token: string, options: jwt.VerifyOptions = {}) {
return jwt.verify(token, this.jwtSecret, options) as T;
} }
} }

View file

@ -63,7 +63,7 @@ export class UserService {
} }
generatePasswordResetToken(user: User, expiresIn = '20m') { generatePasswordResetToken(user: User, expiresIn = '20m') {
return this.jwtService.signData( return this.jwtService.sign(
{ sub: user.id, passwordSha: createPasswordSha(user) }, { sub: user.id, passwordSha: createPasswordSha(user) },
{ expiresIn }, { expiresIn },
); );
@ -82,7 +82,7 @@ export class UserService {
async resolvePasswordResetToken(token: string): Promise<User | undefined> { async resolvePasswordResetToken(token: string): Promise<User | undefined> {
let decodedToken: JwtPayload & { passwordSha: string }; let decodedToken: JwtPayload & { passwordSha: string };
try { try {
decodedToken = this.jwtService.verifyToken(token); decodedToken = this.jwtService.verify(token);
} catch (e) { } catch (e) {
if (e instanceof TokenExpiredError) { if (e instanceof TokenExpiredError) {
this.logger.debug('Reset password token expired', { token }); this.logger.debug('Reset password token expired', { token });

View file

@ -156,7 +156,7 @@ describe('GET /resolve-password-token', () => {
}); });
test('should fail if user is not found', async () => { test('should fail if user is not found', async () => {
const token = jwtService.signData({ sub: uuid() }); const token = jwtService.sign({ sub: uuid() });
const response = await testServer.authlessAgent const response = await testServer.authlessAgent
.get('/resolve-password-token') .get('/resolve-password-token')

View file

@ -1,42 +1,62 @@
import jwt from 'jsonwebtoken';
import type { InstanceSettings } from 'n8n-core';
import { mock } from 'jest-mock-extended';
import config from '@/config'; import config from '@/config';
import { JwtService } from '@/services/jwt.service'; import { JwtService } from '@/services/jwt.service';
import { randomString } from '../../integration/shared/random';
import * as jwt from 'jsonwebtoken';
describe('JwtService', () => { describe('JwtService', () => {
config.set('userManagement.jwtSecret', randomString(5, 10)); const iat = 1699984313;
const jwtSecret = 'random-string';
const payload = { sub: 1 };
const signedToken =
'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOjEsImlhdCI6MTY5OTk4NDMxM30.xNZOAmcidW5ovEF_mwIOzCWkJ70FEO6MFNLK2QRDOeQ';
const jwtService = new JwtService(); const instanceSettings = mock<InstanceSettings>({ encryptionKey: 'test-key' });
beforeEach(() => { beforeEach(() => {
jest.clearAllMocks(); jest.clearAllMocks();
}); });
test('Should sign input with user management secret', async () => { describe('secret initialization', () => {
const userId = 1; it('should read the secret from config, when set', () => {
config.set('userManagement.jwtSecret', jwtSecret);
const token = jwtService.signData({ sub: userId }); const jwtService = new JwtService(instanceSettings);
expect(typeof token).toBe('string'); expect(jwtService.jwtSecret).toEqual(jwtSecret);
const secret = config.get('userManagement.jwtSecret');
const decodedToken = jwt.verify(token, secret);
expect(decodedToken).toHaveProperty('sub');
expect(decodedToken).toHaveProperty('iat');
expect(decodedToken?.sub).toBe(userId);
}); });
test('Should verify token with user management secret', async () => { it('should derive the secret from encryption key when not set in config', () => {
const userId = 1; config.set('userManagement.jwtSecret', '');
const jwtService = new JwtService(instanceSettings);
expect(jwtService.jwtSecret).toEqual(
'e9e2975005eddefbd31b2c04a0b0f2d9c37d9d718cf3676cddf76d65dec555cb',
);
});
});
const secret = config.get('userManagement.jwtSecret'); describe('with a secret set', () => {
config.set('userManagement.jwtSecret', jwtSecret);
const jwtService = new JwtService(instanceSettings);
const token = jwt.sign({ sub: userId }, secret); beforeAll(() => {
jest.useFakeTimers().setSystemTime(new Date(iat * 1000));
});
const decodedToken = jwt.verify(token, secret); afterAll(() => jest.useRealTimers());
expect(decodedToken).toHaveProperty('sub'); it('should sign', () => {
expect(decodedToken?.sub).toBe(userId); const token = jwtService.sign(payload);
expect(token).toEqual(signedToken);
});
it('should decode and verify payload', () => {
const decodedToken = jwtService.verify(signedToken);
expect(decodedToken.sub).toEqual(1);
expect(decodedToken.iat).toEqual(iat);
});
it('should throw an error on verify if the token is expired', () => {
const expiredToken = jwt.sign(payload, jwtSecret, { expiresIn: -10 });
expect(() => jwtService.verify(expiredToken)).toThrow(jwt.TokenExpiredError);
});
}); });
}); });