Full enterprise web operating system including: - Next.js 14 frontend with App Router, i18n (DE/EN), shadcn/ui - NestJS 10 backend with Prisma, JWT auth, Swagger docs - Keycloak 24 SSO with role-based access control - HR module (employees, time tracking, absences, org chart) - LEAN module (3S planning, morning meeting SQCDM, skill matrix) - Integrations module (PlentyONE, Zulip, Todoist, FreeScout, Nextcloud, ecoDMS, GembaDocs) - Dashboard with customizable drag & drop widget grid - Docker Compose infrastructure (PostgreSQL 16, Redis 7, Keycloak 24) Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
45 lines
1.4 KiB
TypeScript
45 lines
1.4 KiB
TypeScript
import { Injectable, UnauthorizedException } from '@nestjs/common';
|
|
import { PassportStrategy } from '@nestjs/passport';
|
|
import { ExtractJwt, Strategy } from 'passport-jwt';
|
|
import { ConfigService } from '@nestjs/config';
|
|
import { JwtPayload } from '../interfaces/jwt-payload.interface';
|
|
import { UsersService } from '../../users/users.service';
|
|
|
|
@Injectable()
|
|
export class JwtStrategy extends PassportStrategy(Strategy) {
|
|
constructor(
|
|
private readonly configService: ConfigService,
|
|
private readonly usersService: UsersService,
|
|
) {
|
|
const secret = configService.get<string>('JWT_SECRET');
|
|
if (!secret) {
|
|
throw new Error('JWT_SECRET is not defined');
|
|
}
|
|
|
|
super({
|
|
jwtFromRequest: ExtractJwt.fromAuthHeaderAsBearerToken(),
|
|
ignoreExpiration: false,
|
|
secretOrKey: secret,
|
|
});
|
|
}
|
|
|
|
async validate(payload: JwtPayload): Promise<JwtPayload> {
|
|
// Optionally validate that the user still exists and is active
|
|
try {
|
|
const user = await this.usersService.findOne(payload.sub);
|
|
if (!user.isActive) {
|
|
throw new UnauthorizedException('User account is deactivated');
|
|
}
|
|
|
|
// Return the payload to be attached to the request
|
|
return {
|
|
sub: payload.sub,
|
|
email: payload.email,
|
|
roles: payload.roles,
|
|
};
|
|
} catch {
|
|
throw new UnauthorizedException('Invalid token');
|
|
}
|
|
}
|
|
}
|