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

feat(core): Add MFA #4767

Merged
merged 200 commits into from
Aug 24, 2023
Merged
Show file tree
Hide file tree
Changes from 11 commits
Commits
Show all changes
200 commits
Select commit Hold shift + click to select a range
b38a0e1
:zap: MFA initial setup
RicardoE105 Nov 29, 2022
c5aee5a
Sync master
RicardoE105 Dec 1, 2022
0bd40cf
Sync master
RicardoE105 Jan 9, 2023
7446db2
Sync master
RicardoE105 Jan 31, 2023
b528dbf
update lock file
RicardoE105 Jan 31, 2023
215b208
fix merge issues
RicardoE105 Jan 31, 2023
5818d29
:zap: Improvements
RicardoE105 Feb 2, 2023
75d5ae3
Improvements
RicardoE105 Feb 9, 2023
31a13fa
Improvements
RicardoE105 Feb 13, 2023
cbc284e
sync master
RicardoE105 Feb 14, 2023
a59ebc4
update lock file
RicardoE105 Feb 14, 2023
e2c9de9
sync master
RicardoE105 Feb 15, 2023
4c60a76
Improvements
RicardoE105 Feb 16, 2023
79c0668
:zap: Improvements
RicardoE105 Feb 17, 2023
3c79b13
:zap: Improvements
RicardoE105 Feb 17, 2023
5fc0706
:zap: Improvements
RicardoE105 Feb 19, 2023
033b808
Sync master
RicardoE105 Feb 20, 2023
9a3d8d4
update lock file
RicardoE105 Feb 20, 2023
805901e
:fire: Remove old MfaSetupView
RicardoE105 Feb 21, 2023
e00723d
:fire: Remove duplicated route
RicardoE105 Feb 21, 2023
bae5c09
:zap: Add loading state to modal
RicardoE105 Feb 21, 2023
280863d
:zap: Update MFA migration timestamp
RicardoE105 Feb 21, 2023
fd7d8df
:zap: Add adi improvements
RicardoE105 Feb 21, 2023
ec1f43a
:zap: simplify MFA migration
RicardoE105 Feb 21, 2023
950221b
:zap: Reset recovery codes once MFA is disabled
RicardoE105 Feb 21, 2023
805b856
:bug: Fix small bug
RicardoE105 Feb 21, 2023
d8463d4
:bug: Fix MFA enable success toast
RicardoE105 Feb 21, 2023
b4180b9
:zap: Show toast when all recovery codes were used
RicardoE105 Feb 21, 2023
596de6f
:zap: Add missing i18n labels
RicardoE105 Feb 21, 2023
904f001
:zap: Apply review feedback
RicardoE105 Feb 22, 2023
c864591
fix issue setting "hasRecoveryCodeLeft" flag
RicardoE105 Feb 22, 2023
c6f486a
:zap: Small improvement
RicardoE105 Feb 22, 2023
8d500e5
sync master
RicardoE105 Feb 22, 2023
58196fb
update lock file
RicardoE105 Feb 22, 2023
e0b2eb9
update MFA migration timestamp
RicardoE105 Feb 22, 2023
107ad54
update the lockfile
netroy Feb 23, 2023
c8deff6
fix the tests
netroy Feb 23, 2023
4a7bcc1
add postgres and mysql migrations
netroy Feb 23, 2023
4ba5c35
Remove br from MfaView
RicardoE105 Feb 23, 2023
76a30cf
Remove icon from infobox
RicardoE105 Feb 23, 2023
6a35b1f
:zap: Add tests
RicardoE105 Feb 23, 2023
eedc6e9
Add missing tests
RicardoE105 Feb 23, 2023
a8de3a7
:zap: Add telemetry
RicardoE105 Feb 24, 2023
259d4e8
Remove duplication
RicardoE105 Feb 24, 2023
79dab61
Small improvement
RicardoE105 Feb 24, 2023
bace508
Isolate speakeasy library
RicardoE105 Feb 24, 2023
8e882dc
:fire: Remove unnecessary constant
RicardoE105 Feb 26, 2023
417d338
Sync master
RicardoE105 Feb 26, 2023
64bb66d
:zap: encrypt secret and recovery codes
RicardoE105 Feb 26, 2023
106a32a
Update migrations timestamp
RicardoE105 Feb 27, 2023
e07099e
:zap: apply feedback
RicardoE105 Feb 28, 2023
85651f6
Merge branch 'master' into ado-21-sync-current-mfa-functionality-with
RicardoE105 Mar 1, 2023
1701a78
:bug: Fix merge issue
RicardoE105 Mar 1, 2023
53863b7
add e2e tests
RicardoE105 Mar 3, 2023
372d9d5
Sync master
RicardoE105 Mar 4, 2023
3c61dba
update lock file
RicardoE105 Mar 4, 2023
f0625fc
:zap: Only retrieve mfaSecret and mfaRecoveryCodes when needed
RicardoE105 Mar 8, 2023
2bbfefb
Merge branch 'master' into ado-21-sync-current-mfa-functionality-with
RicardoE105 Mar 8, 2023
e651f4e
:bug: Fix bug with protectFromPhishing function
RicardoE105 Mar 8, 2023
525139c
:fire: remove console.log
RicardoE105 Mar 8, 2023
c37fe76
adjust mfa telemetry
RicardoE105 Mar 8, 2023
4671db0
Merge branch 'master' into ado-21-sync-current-mfa-functionality-with
RicardoE105 Mar 10, 2023
adee046
Merge branch 'master' into ado-21-sync-current-mfa-functionality-with
RicardoE105 Mar 13, 2023
1c92fa0
rename phishingAttempt to isRedirectSafe
RicardoE105 Mar 13, 2023
ad37a49
remove unnecessary import
RicardoE105 Mar 13, 2023
b5dcad9
inject mfaService to authentication controller
RicardoE105 Mar 13, 2023
3f4cad8
fix linting issue
RicardoE105 Mar 14, 2023
c5332d2
fix formatting issue
RicardoE105 Mar 14, 2023
59ab944
Merge remote-tracking branch 'upstream/master' into ado-21-sync-curre…
netroy Mar 21, 2023
1cde167
Fix select for postgres and mysql
RicardoE105 Mar 22, 2023
ba6d3c0
Sync master
RicardoE105 Mar 28, 2023
9bf3a33
updat lock file
RicardoE105 Mar 28, 2023
7e575f3
Update migration timestamps
RicardoE105 Mar 28, 2023
a32f7ad
Extend verification window when enabling MFA
RicardoE105 Mar 28, 2023
0aeb530
Show error message when token validation window expired
RicardoE105 Mar 28, 2023
ea17d16
add mfa:disable CLI command
RicardoE105 Mar 28, 2023
cc0e1f1
Fix linting issues
RicardoE105 Mar 28, 2023
195f42e
Fix mysql migration
RicardoE105 Mar 28, 2023
52ed14b
fix linting issue
RicardoE105 Mar 28, 2023
6d647c5
add issuer to QR URL
RicardoE105 Mar 29, 2023
66ed032
cypress-otp should be a devDependency
netroy Mar 29, 2023
02fcfaf
Merge remote-tracking branch 'upstream/master' into ado-21-sync-curre…
netroy Mar 29, 2023
cf491c1
Merge remote-tracking branch 'upstream/master' into ado-21-sync-curre…
netroy Mar 30, 2023
a4dbfbb
Merge branch 'master' into ado-21-sync-current-mfa-functionality-with
RicardoE105 Apr 2, 2023
3ee10de
Sync master
RicardoE105 Apr 3, 2023
c8dd21f
Apply feedback
RicardoE105 Apr 4, 2023
7b4b86a
Make e2e tests independent
RicardoE105 Apr 4, 2023
12f30fe
Allow to disable MFA feature in instance
RicardoE105 Apr 4, 2023
75257f9
Sync master
RicardoE105 Apr 4, 2023
2f20029
access to mfa.enable optionally
RicardoE105 Apr 4, 2023
1087e5f
Ask for MFA token when resetting password
RicardoE105 Apr 5, 2023
317589a
fix linting issues
RicardoE105 Apr 5, 2023
713af91
Make change password tests pass
RicardoE105 Apr 5, 2023
ce075f6
Add BE tests for change password endpoints
RicardoE105 Apr 5, 2023
109c5aa
Apply formatting to MFA tests
RicardoE105 Apr 5, 2023
18f5fb8
Remove duplicated tests
RicardoE105 Apr 5, 2023
47c90c3
Fix me.controller.test to account for mfa changes
RicardoE105 Apr 5, 2023
7434abe
Tiny refactor to tests utils
RicardoE105 Apr 5, 2023
b2388c0
Remove test.only in mfa tests
RicardoE105 Apr 5, 2023
3dfefc0
fix linting issues
RicardoE105 Apr 5, 2023
251677b
Move MFA fields length to constant file
RicardoE105 Apr 5, 2023
7697a73
centralize MFA helpers in one service and refactor
RicardoE105 Apr 6, 2023
c55a2e1
Make change-password form labels sentence case
RicardoE105 Apr 6, 2023
542077d
Remove unnecessary import
RicardoE105 Apr 6, 2023
0e96f5c
Parse mfaEnable query-string parameter correctly
RicardoE105 Apr 6, 2023
b55ed7f
fix tests
RicardoE105 Apr 6, 2023
3a138bd
Switch speakeasy for otpauth
RicardoE105 Apr 6, 2023
d0c3050
Fix typing issues
RicardoE105 Apr 6, 2023
6502c54
Fix tests
RicardoE105 Apr 6, 2023
1ce51c2
Sync master
RicardoE105 Apr 11, 2023
7aa7788
update pnpm-lock.yaml
RicardoE105 Apr 11, 2023
2fdd38e
Sync master
RicardoE105 Apr 13, 2023
ca89223
Add mfa.update external hooks to MFA controller
RicardoE105 Apr 17, 2023
369ba48
Migrate modals to use createEventBus
RicardoE105 Apr 17, 2023
e6af4c8
Add fallback value to mfaRecoveryCodes
RicardoE105 Apr 17, 2023
009458c
Update mfa.update hook name
RicardoE105 Apr 17, 2023
72b4d7d
Sync master
RicardoE105 Apr 24, 2023
16ff58f
Fix issue with types
RicardoE105 Apr 24, 2023
badb578
Fix linting issues
RicardoE105 Apr 24, 2023
3974419
Sync master
RicardoE105 Apr 26, 2023
3e3f3a3
Update pnpm-lock.yaml
RicardoE105 Apr 26, 2023
d24e7c7
Sync master
RicardoE105 May 6, 2023
675d04f
fix imports
RicardoE105 May 6, 2023
50e84c3
fix migrations
RicardoE105 May 6, 2023
1e56c50
fix linting issues
RicardoE105 May 6, 2023
515dd48
move migration logging. this is implicitly done now
netroy May 7, 2023
17bf246
fix mfa tests
netroy May 7, 2023
ded6190
ADD back the mistakenly removed section in personal settings .
RicardoE105 May 8, 2023
f478884
Sync master
RicardoE105 May 16, 2023
bd7ccda
Fix issues after merge
RicardoE105 May 16, 2023
7214bec
fix linting issues
RicardoE105 May 17, 2023
7cdb484
Merge branch 'master' into ado-21-sync-current-mfa-functionality-with
RicardoE105 May 30, 2023
46ffa12
Add MFA bus
RicardoE105 May 30, 2023
721f54b
fix event bus mfaView
RicardoE105 May 30, 2023
f7ddf29
fix linting issues
RicardoE105 May 31, 2023
bece85a
fix linting issues
RicardoE105 May 31, 2023
9accb72
fix linting issue
RicardoE105 May 31, 2023
56d9b85
Merge branch 'master' into ado-21-sync-current-mfa-functionality-with
RicardoE105 Jun 6, 2023
b34f392
Sync master
RicardoE105 Jun 26, 2023
0b39b66
fix merging issue
RicardoE105 Jun 26, 2023
0227be4
fix more merging issues
RicardoE105 Jun 26, 2023
d4cd4bd
fix linting issue
RicardoE105 Jun 26, 2023
ba8bdb8
Sync master
RicardoE105 Jul 29, 2023
a9352ec
update lock file
RicardoE105 Jul 29, 2023
6277454
Fix merge issues
RicardoE105 Jul 29, 2023
a2f8f60
move migrations to common folder
RicardoE105 Jul 29, 2023
90c86f2
update version of qrCode library
RicardoE105 Jul 29, 2023
77b40c2
update tests to use testServer
RicardoE105 Aug 2, 2023
c50b504
more updates to tests
RicardoE105 Aug 2, 2023
94515b9
sync master
RicardoE105 Aug 2, 2023
4042d4f
more test fixes
RicardoE105 Aug 2, 2023
f58ee8c
update lock file
RicardoE105 Aug 3, 2023
ba24ca6
Merge branch 'master' into ado-21-sync-current-mfa-functionality-with
RicardoE105 Aug 3, 2023
ad7c228
fix linting issues
RicardoE105 Aug 3, 2023
58fc621
fix e2e tests
RicardoE105 Aug 3, 2023
2688fd2
fix format issues
RicardoE105 Aug 3, 2023
08a0b17
remove console.logs
RicardoE105 Aug 3, 2023
1f09920
Merge branch 'master' into ado-21-sync-current-mfa-functionality-with
RicardoE105 Aug 4, 2023
0743241
Merge branch 'master' into ado-21-sync-current-mfa-functionality-with
RicardoE105 Aug 11, 2023
0856a88
fix linting issue
RicardoE105 Aug 11, 2023
07a2df0
ignore cypress download folder
RicardoE105 Aug 17, 2023
20bd360
fix typo
RicardoE105 Aug 17, 2023
a95b4ed
remove unneccesary default
RicardoE105 Aug 17, 2023
90d944d
remove calls to hook user.mfa.update
RicardoE105 Aug 17, 2023
51aabe2
remove duplicated interface
RicardoE105 Aug 17, 2023
8bd4b54
rename getRawSecretAndRecoveryCodes to getSecretAndRecoveryCodes
RicardoE105 Aug 17, 2023
caa467d
add max length to secret and recovery code columns
RicardoE105 Aug 17, 2023
aea64ad
copy secret directly instead of hiding it the html
RicardoE105 Aug 17, 2023
7f0705c
log telemetry event before redirecting
RicardoE105 Aug 17, 2023
19b45cc
fix typo
RicardoE105 Aug 17, 2023
2fdd211
fix tests description
RicardoE105 Aug 17, 2023
f5b6938
update e2e tests
RicardoE105 Aug 17, 2023
cdcc2d3
sync master
RicardoE105 Aug 17, 2023
518e550
update lock file
RicardoE105 Aug 17, 2023
74fc696
fix linting issues
RicardoE105 Aug 18, 2023
a8fd72a
use string instead of varchar
RicardoE105 Aug 18, 2023
d18610a
use mfaService to create mfa test user
RicardoE105 Aug 18, 2023
2d12efc
fix linting issue
RicardoE105 Aug 18, 2023
ba9430a
fix i18n component with i18n-t
RicardoE105 Aug 18, 2023
fa9b8cf
fix linting issue
RicardoE105 Aug 18, 2023
0439b8f
bump library version
RicardoE105 Aug 18, 2023
77da37a
revert mistakenly committed change to auth.controller
RicardoE105 Aug 18, 2023
1d948fb
remove unnecessary call to userStore.addUsers
RicardoE105 Aug 18, 2023
41aa73f
css improvement
RicardoE105 Aug 18, 2023
631ff55
here fine
RicardoE105 Aug 19, 2023
6a96df1
delete MFA route and move MFA component to sign-in view
RicardoE105 Aug 20, 2023
bf8ccf5
delete mfa route and fix linting issue
RicardoE105 Aug 20, 2023
613f27e
do not ask for mfa token to change password if user already signed in
RicardoE105 Aug 20, 2023
c199dea
general improvements
RicardoE105 Aug 20, 2023
ffc28e7
add variables to css
RicardoE105 Aug 21, 2023
e33ba93
add mfaEnabled flag to password reset
RicardoE105 Aug 21, 2023
b8cd736
use text instead of string
RicardoE105 Aug 21, 2023
af5942a
sync master
RicardoE105 Aug 22, 2023
c97b926
no need to clear credentials
RicardoE105 Aug 22, 2023
2e1300a
improvement
RicardoE105 Aug 22, 2023
94de010
add back mistakenly removed toast
RicardoE105 Aug 22, 2023
4418e7c
sync master
RicardoE105 Aug 22, 2023
91ea4a6
fix merge issue
RicardoE105 Aug 22, 2023
1ac980a
sync master
RicardoE105 Aug 23, 2023
b7818fe
fix e2e tests
RicardoE105 Aug 23, 2023
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
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@
"test:e2e:all": "cross-env E2E_TESTS=true start-server-and-test start http://localhost:5678/favicon.ico 'cypress run --headless'"
},
"dependencies": {
"n8n": "*"
"n8n": "^0.210.1"
RicardoE105 marked this conversation as resolved.
Show resolved Hide resolved
},
"devDependencies": {
"@n8n_io/eslint-config": "*",
Expand Down
2 changes: 2 additions & 0 deletions packages/cli/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,7 @@
"@rudderstack/rudder-sdk-node": "1.0.6",
"@sentry/integrations": "^7.28.1",
"@sentry/node": "^7.28.1",
"@types/speakeasy": "^2.0.7",
"axios": "^0.21.1",
"basic-auth": "^2.0.1",
"bcryptjs": "^2.4.3",
Expand Down Expand Up @@ -190,6 +191,7 @@
"shelljs": "^0.8.5",
"source-map-support": "^0.5.21",
"sqlite3": "^5.1.4",
"speakeasy": "^2.0.0",
"sse-channel": "^4.0.0",
"swagger-ui-express": "^4.3.0",
"syslog-client": "^1.1.1",
Expand Down
73 changes: 73 additions & 0 deletions packages/cli/src/Mfa/MfaController.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
/* eslint-disable import/no-cycle */
import express from 'express';
import * as speakeasy from 'speakeasy';
import { Db } from '..';
import type { AuthenticatedRequest } from '../requests';
import type { MFA } from './types';
import { v4 as uuid } from 'uuid';

export const mfaController = express.Router();
RicardoE105 marked this conversation as resolved.
Show resolved Hide resolved

/**
* GET /mfa/qr
*/
mfaController.get('/qr', async (req: AuthenticatedRequest, res: express.Response) => {
const secret = speakeasy.generateSecret({
RicardoE105 marked this conversation as resolved.
Show resolved Hide resolved
issuer: 'n8n',
name: req.user.email,
otpauth_url: true,
});

const codes = Array.from(Array(5)).map(() => uuid());

await Db.collections.User.update(req.user.id, {
mfaSecret: secret.base32,
mfaRecoveryCodes: codes.join('|'),
});

return res
.status(200)
.json({ data: { secret: secret.base32, qrCode: secret.otpauth_url, recoveryCodes: codes } });
});

/**
* POST /mfa/enable
*/
mfaController.post('/enable', async (req: MFA.activate, res: express.Response) => {
const { id } = req.user;

await Db.collections.User.update(id, { mfaEnabled: true });

return res.status(200).json();
});

/**
* POST /mfa/disable
*/
mfaController.delete('/disable', async (req: AuthenticatedRequest, res: express.Response) => {
await Db.collections.User.update(req.user.id, { mfaEnabled: false, mfaSecret: '' });

return res.status(200).json();
});

/**
* POST /mfa/verify
*/
mfaController.post('/verify', async (req: MFA.verify, res: express.Response) => {
const { mfaSecret: secret, id } = req.user;
const { token } = req.body;

const user = await Db.collections.User.findOneBy({ id });

if (!user) return res.status(400).json();

const verified = speakeasy.totp.verify({
secret: secret ?? '',
encoding: 'base32',
token,
});

if (!verified) return res.status(400).json();

return res.status(200).json();
});
20 changes: 20 additions & 0 deletions packages/cli/src/Mfa/helpers.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import type { User } from '@/databases/entities/User';
import * as speakeasy from 'speakeasy';

export const validateMfaToken = (user: User, mfaToken: string) => {
return speakeasy.totp.verify({
secret: user.mfaSecret ?? '',
encoding: 'base32',
token: mfaToken,
});
};

export const validateMfaRecoveryCode = (user: User, mfaRecoveryCode: string) => {
if (user?.mfaRecoveryCodes) {
const recoveryCodes = user.mfaRecoveryCodes.split('|');
if (recoveryCodes.includes(mfaRecoveryCode)) {
return true;
}
}
return false;
};
24 changes: 24 additions & 0 deletions packages/cli/src/Mfa/types.d.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
/* eslint-disable @typescript-eslint/naming-convention */
import type express from 'express';
import type { User } from '../databases/entities/User';

export type AuthenticatedRequest<
RouteParams = {},
ResponseBody = {},
RequestBody = {},
RequestQuery = {},
> = express.Request<RouteParams, ResponseBody, RequestBody, RequestQuery> & {
user: User;
};

export declare namespace MFA {
type verify = AuthenticatedRequest<{}, {}, { token: string }, {}>;
type activate = AuthenticatedRequest<{}, {}, { token: string }, {}>;
type config = AuthenticatedRequest<{}, {}, { login: { enabled: boolean } }, {}>;
type validateRecoveryCode = AuthenticatedRequest<
{},
{},
{ recoveryCode: { enabled: boolean } },
{}
>;
}
4 changes: 2 additions & 2 deletions packages/cli/src/ResponseHelper.ts
Original file line number Diff line number Diff line change
Expand Up @@ -46,8 +46,8 @@ export class BadRequestError extends ResponseError {
}

export class AuthError extends ResponseError {
constructor(message: string) {
super(message, 401);
constructor(message: string, errorCode?: number) {
super(message, 401, errorCode);
}
}

Expand Down
7 changes: 7 additions & 0 deletions packages/cli/src/Server.ts
Original file line number Diff line number Diff line change
Expand Up @@ -145,6 +145,7 @@ import { AbstractServer } from './AbstractServer';
import { configureMetrics } from './metrics';
import { setupBasicAuth } from './middlewares/basicAuth';
import { setupExternalJWTAuth } from './middlewares/externalJWTAuth';
import { mfaController } from './Mfa/MfaController';

const exec = promisify(callbackExec);

Expand Down Expand Up @@ -448,6 +449,12 @@ class Server extends AbstractServer {
this.app.use(`/${this.restEndpoint}/nodes`, nodesController);
}

// ----------------------------------------
// MFA
// ----------------------------------------

this.app.use(`/${this.restEndpoint}/mfa`, mfaController);

// ----------------------------------------
// Workflow
// ----------------------------------------
Expand Down
15 changes: 13 additions & 2 deletions packages/cli/src/controllers/auth.controller.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import validator from 'validator';
import { Get, Post, RestController } from '@/decorators';
import { AuthError, BadRequestError, InternalServerError } from '@/ResponseHelper';
import { BadRequestError, InternalServerError, AuthError } from '@/ResponseHelper';
import { sanitizeUser } from '@/UserManagement/UserManagementHelper';
import { issueCookie, resolveJwt } from '@/auth/jwt';
import { AUTH_COOKIE_NAME } from '@/constants';
Expand All @@ -13,6 +13,7 @@ import { In } from 'typeorm';
import type { Config } from '@/config';
import type { PublicUser, IDatabaseCollections, IInternalHooksClass } from '@/Interfaces';
import { handleEmailLogin, handleLdapLogin } from '@/auth';
import { validateMfaRecoveryCode, validateMfaToken } from '@/Mfa/helpers';

@RestController()
export class AuthController {
Expand Down Expand Up @@ -47,14 +48,24 @@ export class AuthController {
*/
@Post('/login')
async login(req: LoginRequest, res: Response): Promise<PublicUser> {
const { email, password } = req.body;
const { email, password, mfaToken = '', mfaRecoveryCode = '' } = req.body;
if (!email) throw new Error('Email is required to log in');
if (!password) throw new Error('Password is required to log in');

const user =
(await handleLdapLogin(email, password)) ?? (await handleEmailLogin(email, password));

if (user) {
console.log(user);
console.log(mfaToken);
console.log('valid mFA TOKEN', validateMfaToken(user, mfaToken));
if (
user.mfaEnabled &&
!(validateMfaToken(user, mfaToken) || validateMfaRecoveryCode(user, mfaRecoveryCode))
) {
throw new AuthError('MFA Error', 998);
}

await issueCookie(res, user);
return sanitizeUser(user);
}
Expand Down
9 changes: 9 additions & 0 deletions packages/cli/src/databases/entities/User.ts
Original file line number Diff line number Diff line change
Expand Up @@ -103,6 +103,15 @@ export class User extends AbstractEntity implements IUser {
@Index({ unique: true })
apiKey?: string | null;

@Column({ type: String, nullable: true })
RicardoE105 marked this conversation as resolved.
Show resolved Hide resolved
mfaSecret?: string | null;

@Column({ type: Boolean, default: false })
mfaEnabled?: boolean;

@Column({ type: String, nullable: true })
RicardoE105 marked this conversation as resolved.
Show resolved Hide resolved
mfaRecoveryCodes?: string | null;

/**
* Whether the user is pending setup completion.
*/
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
import { MigrationInterface, QueryRunner } from 'typeorm';
import { getTablePrefix, logMigrationEnd, logMigrationStart } from '@db/utils/migrationHelpers';
export class AddMfaColumns1669730543736 implements MigrationInterface {
RicardoE105 marked this conversation as resolved.
Show resolved Hide resolved
name = 'AddMfaColumns1669730543736';

async up(queryRunner: QueryRunner): Promise<void> {
logMigrationStart(this.name);

const tablePrefix = getTablePrefix();

await queryRunner.query('PRAGMA foreign_keys=OFF');

await queryRunner.query(
`CREATE TABLE "temporary_user" (
"id" varchar PRIMARY KEY NOT NULL,
"email" varchar(255),
"firstName" varchar(32),
"lastName" varchar(32),
"password" varchar,
"resetPasswordToken" varchar,
"resetPasswordTokenExpiration" integer DEFAULT NULL,
"mfaEnabled" boolean DEFAULT false,
"mfaSecret" varchar DEFAULT NULL,
"mfaRecoveryCodes" varchar DEFAULT NULL,
"personalizationAnswers" text,
"createdAt" datetime(3) NOT NULL DEFAULT (STRFTIME('%Y-%m-%d %H:%M:%f', 'NOW')),
"updatedAt" datetime(3) NOT NULL DEFAULT (STRFTIME('%Y-%m-%d %H:%M:%f', 'NOW')),
"globalRoleId" integer NOT NULL,
"settings" text,
"apiKey" varchar DEFAULT NULL,
CONSTRAINT "FK_${tablePrefix}f0609be844f9200ff4365b1bb3d" FOREIGN KEY ("globalRoleId") REFERENCES "${tablePrefix}role" ("id") ON DELETE NO ACTION ON UPDATE NO ACTION)`,
);

await queryRunner.query(
`INSERT INTO "temporary_user"(
"id",
"email",
"firstName",
"lastName",
"password",
"resetPasswordToken",
"resetPasswordTokenExpiration",
"personalizationAnswers",
"createdAt",
"updatedAt",
"globalRoleId",
"settings",
"apiKey"
) SELECT
"id",
"email",
"firstName",
"lastName",
"password",
"resetPasswordToken",
"resetPasswordTokenExpiration",
"personalizationAnswers",
"createdAt",
"updatedAt",
"globalRoleId",
"settings",
"apiKey"
FROM "${tablePrefix}user"`,
);

await queryRunner.query(`DROP TABLE "${tablePrefix}user"`);

await queryRunner.query(`ALTER TABLE "temporary_user" RENAME TO "${tablePrefix}user"`);

await queryRunner.query('PRAGMA foreign_keys=ON');

logMigrationEnd(this.name);
}

async down(queryRunner: QueryRunner): Promise<void> {}
}
2 changes: 2 additions & 0 deletions packages/cli/src/databases/migrations/sqlite/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ import { MessageEventBusDestinations1671535397530 } from './1671535397530-Messag
import { DeleteExecutionsWithWorkflows1673268682475 } from './1673268682475-DeleteExecutionsWithWorkflows';
import { CreateLdapEntities1674509946020 } from './1674509946020-CreateLdapEntities';
import { PurgeInvalidWorkflowConnections1675940580449 } from './1675940580449-PurgeInvalidWorkflowConnections';
import { AddMfaColumns1669730543736 } from './1669730543736-addMfaColumns';

const sqliteMigrations = [
InitialMigration1588102412422,
Expand Down Expand Up @@ -60,6 +61,7 @@ const sqliteMigrations = [
DeleteExecutionsWithWorkflows1673268682475,
CreateLdapEntities1674509946020,
PurgeInvalidWorkflowConnections1675940580449,
AddMfaColumns1669730543736,
];

export { sqliteMigrations };
2 changes: 2 additions & 0 deletions packages/cli/src/requests.ts
Original file line number Diff line number Diff line change
Expand Up @@ -250,6 +250,8 @@ export type LoginRequest = AuthlessRequest<
{
email: string;
password: string;
mfaToken?: string;
mfaRecoveryCode?: string;
}
>;

Expand Down
1 change: 1 addition & 0 deletions packages/editor-ui/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,7 @@
"prettier": "^2.8.3",
"prismjs": "^1.17.1",
"stream-browserify": "^3.0.0",
"qrcode.vue": "^1.7.0",
"timeago.js": "^4.0.2",
"uuid": "^8.3.2",
"v-click-outside": "^3.1.2",
Expand Down
1 change: 1 addition & 0 deletions packages/editor-ui/src/Interface.ts
Original file line number Diff line number Diff line change
Expand Up @@ -543,6 +543,7 @@ export interface IUser extends IUserResponse {
inviteAcceptUrl?: string;
fullName?: string;
createdAt?: Date;
mfaEnabled: boolean;
}

export interface IVersionNotificationSettings {
Expand Down
20 changes: 20 additions & 0 deletions packages/editor-ui/src/api/mfa.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import { IRestApiContext } from '@/Interface';
import { makeRestApiRequest } from '@/utils';

export function getMfaQr(
context: IRestApiContext,
): Promise<{ qrCode: string; secret: string; recoveryCodes: string[] }> {
return makeRestApiRequest(context, 'GET', '/mfa/qr');
}

export function enableMfa(context: IRestApiContext): Promise<void> {
return makeRestApiRequest(context, 'POST', '/mfa/enable');
}

export function verifyMfaToken(context: IRestApiContext, data: { token: string }): Promise<void> {
return makeRestApiRequest(context, 'POST', '/mfa/verify', data);
}

export function disableMfa(context: IRestApiContext): Promise<void> {
return makeRestApiRequest(context, 'DELETE', '/mfa/disable');
}
2 changes: 1 addition & 1 deletion packages/editor-ui/src/api/users.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ export function getCurrentUser(context: IRestApiContext): Promise<IUserResponse

export function login(
context: IRestApiContext,
params: { email: string; password: string },
params: { email: string; password: string; mfaToken?: string; mfaRecoveryToken?: string },
): Promise<IUserResponse> {
return makeRestApiRequest(context, 'POST', '/login', params);
}
Expand Down
Loading