import { NextFunction, Request, Response } from 'express' import { BaseMiddleware } from 'inversify-express-utils' import { verify } from 'jsonwebtoken' import { CrossServiceTokenData } from '@standardnotes/security' import * as winston from 'winston' export class InversifyExpressAuthMiddleware extends BaseMiddleware { constructor( private authJWTSecret: string, private logger: winston.Logger, ) { super() } async handler(request: Request, response: Response, next: NextFunction): Promise { try { if (!request.header('X-Auth-Token')) { this.logger.debug('Missing X-Auth-Token header') return this.sendInvalidAuthResponse(response) } const authToken = request.header('X-Auth-Token') const decodedToken = verify(authToken, this.authJWTSecret, { algorithms: ['HS256'] }) response.locals.user = decodedToken.user response.locals.roles = decodedToken.roles response.locals.session = decodedToken.session response.locals.readOnlyAccess = decodedToken.session?.readonly_access ?? false response.locals.sharedVaultOwnerContext = decodedToken.shared_vault_owner_context response.locals.ongoingTransition = decodedToken.ongoing_transition return next() } catch (error) { this.logger.error(`Could not verify JWT Auth Token ${(error as Error).message}`) return this.sendInvalidAuthResponse(response) } } private sendInvalidAuthResponse(response: Response) { response.status(401).send({ error: { tag: 'invalid-auth', message: 'Invalid login credentials.', }, }) } }