fix: resolve all ESLint errors in backend and frontend

Backend 958→0 errors, frontend 98→0 errors. Builds and tsc clean.

Echte Fixes:
- Auth: AuthenticatedUser/AuthenticatedRequest, JwtStrategy + alle 5
  Controller von `@Request() req: any` auf typisierten Request umgestellt
- Error-Handling: neuer getErrorMessage/Stack/Code/getResponseData-Helper;
  alle 50 `catch (err: any)`-Blöcke auf `unknown` + Helper umgestellt
- 24 echte Bugs: require-await, require-imports→ES-Imports, useless-escape,
  misused-promises, tote Imports/Vars, leere catch-Blöcke kommentiert
- document-pipeline: OCR-Ergebnis wird nicht gespeichert (als TODO markiert)

Pragmatisch auf warn herabgestuft (untypisierte Paperless-NGX-API):
no-unsafe-*, restrict-template-expressions, no-base-to-string,
no-explicit-any (FE), react-refresh/only-export-components

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
2026-06-09 21:33:37 +02:00
parent d96e06e86d
commit 07dfd7e840
43 changed files with 399 additions and 204 deletions
@@ -0,0 +1,24 @@
import type { Request } from 'express';
import type { Permission } from './permissions.enum';
/** Vom JwtStrategy.validate() an `request.user` angehängte Identität. */
export interface AuthenticatedUser {
userId: string;
email: string;
name: string;
preferredUsername: string | null;
groups: string[];
permissions: Permission[];
}
/** Vom ApiKeyGuard an `request.apiKeyMetadata` angehängte Schlüssel-Info. */
export interface ApiKeyMetadata {
id: number;
name: string;
}
/** Express-Request mit der durch die Auth-Guards gesetzten Identität. */
export interface AuthenticatedRequest extends Request {
user?: AuthenticatedUser;
apiKeyMetadata?: ApiKeyMetadata;
}
+12 -10
View File
@@ -4,6 +4,15 @@ import { Strategy, ExtractJwt } from 'passport-jwt';
import { ConfigService } from '@nestjs/config';
import { passportJwtSecret } from 'jwks-rsa';
import { mapGroupsToPermissions } from './permissions.enum';
import type { AuthenticatedUser } from './authenticated-request';
interface JwtPayload {
sub: string;
email: string;
name?: string;
preferred_username?: string;
groups?: string[];
}
@Injectable()
export class JwtStrategy extends PassportStrategy(Strategy, 'jwt') {
@@ -24,19 +33,12 @@ export class JwtStrategy extends PassportStrategy(Strategy, 'jwt') {
});
}
validate(payload: any): {
userId: string;
email: string;
name: string;
preferredUsername: string | null;
groups: string[];
permissions: any[];
} {
const groups = payload.groups || [];
validate(payload: JwtPayload): AuthenticatedUser {
const groups = payload.groups ?? [];
return {
userId: payload.sub,
email: payload.email,
name: payload.name || payload.preferred_username,
name: payload.name || payload.preferred_username || '',
preferredUsername: payload.preferred_username ?? null,
groups: groups,
permissions: mapGroupsToPermissions(groups),