diff --git a/.DS_Store b/.DS_Store index 60b378a6..354e791a 100644 Binary files a/.DS_Store and b/.DS_Store differ diff --git a/ACTION_PLAN_LOGIN_FLOW.md b/ACTION_PLAN_LOGIN_FLOW.md deleted file mode 100644 index 7c93d3bc..00000000 --- a/ACTION_PLAN_LOGIN_FLOW.md +++ /dev/null @@ -1,425 +0,0 @@ -# Plan d'Action - Amélioration Flow de Connexion - -## 🎯 Objectifs - -1. **Améliorer l'UX** : Permettre SSO naturel pour les utilisateurs légitimes -2. **Sécuriser le logout** : S'assurer que les credentials sont demandés après logout -3. **Simplifier le code** : Réduire la complexité de détection session invalide -4. **Éliminer les race conditions** : Mécanisme robuste pour éviter auto-login après logout - ---- - -## 📋 Actions Immédiates (À faire en premier) - -### Action 1 : Supprimer `prompt=login` par défaut ⚡ - -**Fichier** : `app/api/auth/options.ts` - -**Changement** : -```typescript -// AVANT (ligne 147-155) -authorization: { - params: { - scope: "openid profile email roles", - prompt: "login" // ❌ Supprimer cette ligne - } -} - -// APRÈS -authorization: { - params: { - scope: "openid profile email roles", - // prompt: "login" supprimé - sera ajouté conditionnellement après logout - } -} -``` - -**Impact** : ✅ SSO fonctionne naturellement pour les utilisateurs légitimes - ---- - -### Action 2 : Créer route API pour marquer logout ⚡ - -**Nouveau fichier** : `app/api/auth/mark-logout/route.ts` - -```typescript -import { NextRequest, NextResponse } from 'next/server'; - -export async function POST(request: NextRequest) { - const response = NextResponse.json({ - success: true, - message: 'Logout marked successfully' - }); - - // Cookie HttpOnly pour marquer le logout (5 minutes) - response.cookies.set('force_login_prompt', 'true', { - httpOnly: true, - secure: process.env.NODE_ENV === 'production', - sameSite: 'lax', - path: '/', - maxAge: 300 // 5 minutes - }); - - return response; -} -``` - -**Impact** : ✅ Mécanisme robuste pour forcer login après logout - ---- - -### Action 3 : Modifier signout-handler pour utiliser la route ⚡ - -**Fichier** : `components/auth/signout-handler.tsx` - -**Changement** (après ligne 25) : -```typescript -// AVANT -clearKeycloakCookies(); - -// APRÈS -clearKeycloakCookies(); - -// Marquer le logout côté serveur -try { - await fetch('/api/auth/mark-logout', { - method: 'POST', - credentials: 'include', - }); -} catch (error) { - console.error('Error marking logout:', error); - // Continue même si ça échoue -} -``` - -**Répéter dans** : -- `components/main-nav.tsx` (ligne ~377) -- `components/layout/layout-wrapper.tsx` (ligne ~42) - -**Impact** : ✅ Flag serveur pour empêcher auto-login - ---- - -### Action 4 : Simplifier signin/page.tsx ⚡ - -**Fichier** : `app/signin/page.tsx` - -**Changement** : Remplacer la logique complexe (lignes 17-67) par : - -```typescript -useEffect(() => { - // Vérifier le cookie serveur pour forcer login - const forceLoginCookie = document.cookie - .split(';') - .find(c => c.trim().startsWith('force_login_prompt=')); - - // Si logout récent, forcer prompt=login - if (forceLoginCookie) { - // Supprimer le cookie - document.cookie = 'force_login_prompt=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;'; - - // Ne pas auto-login, attendre clic utilisateur - // Le bouton "Se connecter" forcera prompt=login - return; - } - - // Si déjà authentifié, rediriger - if (status === "authenticated" && session?.user) { - router.push("/"); - return; - } - - // Si non authentifié et pas de flag logout, auto-login (SSO naturel) - if (status === "unauthenticated" && !forceLoginCookie) { - const timer = setTimeout(() => { - if (status === "unauthenticated") { - signIn("keycloak", { callbackUrl: "/" }); - } - }, 1000); - return () => clearTimeout(timer); - } -}, [status, session, router]); -``` - -**ET** modifier le bouton "Se connecter" (ligne ~202) : - -```typescript - -``` - -**Impact** : ✅ Code plus simple et maintenable - ---- - -### Action 5 : Ajouter prompt=login conditionnel dans options.ts ⚡ - -**Fichier** : `app/api/auth/options.ts` - -**Changement** : Modifier la configuration KeycloakProvider pour accepter un paramètre custom : - -```typescript -KeycloakProvider({ - clientId: getRequiredEnvVar("KEYCLOAK_CLIENT_ID"), - clientSecret: getRequiredEnvVar("KEYCLOAK_CLIENT_SECRET"), - issuer: getRequiredEnvVar("KEYCLOAK_ISSUER"), - authorization: { - params: { - scope: "openid profile email roles", - // prompt sera ajouté dynamiquement si force_login=true dans l'URL - } - }, - // ... profile callback ... -}) -``` - -**ET** créer une route custom pour signin qui ajoute prompt : - -**Nouveau fichier** : `app/api/auth/signin/keycloak/route.ts` - -```typescript -import { NextRequest, NextResponse } from 'next/server'; -import { getServerSession } from 'next-auth/next'; -import { authOptions } from '../../options'; - -export async function GET(request: NextRequest) { - const searchParams = request.nextUrl.searchParams; - const forceLogin = searchParams.get('force_login') === 'true'; - const callbackUrl = searchParams.get('callbackUrl') || '/'; - - // Rediriger vers NextAuth signin avec prompt si nécessaire - const signinUrl = new URL('/api/auth/signin/keycloak', request.nextUrl.origin); - signinUrl.searchParams.set('callbackUrl', callbackUrl); - - if (forceLogin) { - // Ajouter prompt=login dans l'URL de redirection Keycloak - // Note: NextAuth ne supporte pas directement, il faut modifier l'URL après - // Solution alternative: Utiliser un middleware ou modifier options dynamiquement - } - - return NextResponse.redirect(signinUrl); -} -``` - -**OU** Solution plus simple : Modifier directement dans `options.ts` pour lire un cookie : - -```typescript -// Dans options.ts, modifier authorization params dynamiquement -authorization: { - params: (provider, action, request) => { - const forceLogin = request?.cookies?.get('force_login_prompt')?.value === 'true'; - return { - scope: "openid profile email roles", - ...(forceLogin ? { prompt: "login" } : {}), - }; - } -} -``` - -**Note** : NextAuth v4 ne supporte pas `params` comme fonction. Solution alternative : - -**Modifier** `app/api/auth/options.ts` pour utiliser `authorization.url` : - -```typescript -KeycloakProvider({ - // ... config ... - authorization: { - params: { - scope: "openid profile email roles", - }, - // Ajouter prompt dynamiquement via URL personnalisée - url: (params) => { - // Vérifier si on doit forcer login (via cookie ou autre moyen) - const url = new URL(`${process.env.KEYCLOAK_ISSUER}/protocol/openid-connect/auth`); - url.searchParams.set('client_id', process.env.KEYCLOAK_CLIENT_ID!); - url.searchParams.set('redirect_uri', params.redirect_uri); - url.searchParams.set('response_type', 'code'); - url.searchParams.set('scope', 'openid profile email roles'); - url.searchParams.set('state', params.state); - - // Ajouter prompt si nécessaire (à vérifier via cookie dans le callback) - // Note: Plus complexe, nécessite de passer le flag via state - return url.toString(); - } - } -}) -``` - -**Solution RECOMMANDÉE (plus simple)** : Utiliser un paramètre dans l'URL de callback et le vérifier dans le callback JWT : - -```typescript -// Dans signin/page.tsx, lors du clic sur "Se connecter" -const url = new URL(window.location.origin + '/api/auth/signin/keycloak'); -url.searchParams.set('callbackUrl', '/'); -url.searchParams.set('force_login', 'true'); -// Stocker dans sessionStorage pour le callback -sessionStorage.setItem('force_login', 'true'); -window.location.href = url.toString(); - -// Dans options.ts, callback jwt, vérifier sessionStorage n'est pas possible côté serveur -// Solution: Passer via state OAuth -``` - -**MEILLEURE SOLUTION** : Utiliser un cookie avant le signIn : - -```typescript -// Dans signin/page.tsx, bouton "Se connecter" -onClick={() => { - // Créer cookie pour forcer login - document.cookie = 'force_login_prompt=true; path=/; max-age=300'; - // Puis signIn normal - signIn("keycloak", { callbackUrl: "/" }); -}} - -// Dans options.ts, lire le cookie dans authorization params -// Note: NextAuth ne permet pas d'accéder aux cookies dans params -// Solution: Middleware ou route custom -``` - -**SOLUTION FINALE RECOMMANDÉE** : Créer une route API custom qui gère le signin avec prompt conditionnel : - -```typescript -// app/api/auth/custom-signin/route.ts -import { NextRequest, NextResponse } from 'next/server'; - -export async function GET(request: NextRequest) { - const searchParams = request.nextUrl.searchParams; - const forceLogin = searchParams.get('force_login') === 'true'; - const callbackUrl = searchParams.get('callbackUrl') || '/'; - - // Construire l'URL Keycloak avec prompt si nécessaire - const keycloakIssuer = process.env.KEYCLOAK_ISSUER!; - const clientId = process.env.KEYCLOAK_CLIENT_ID!; - const redirectUri = `${request.nextUrl.origin}/api/auth/callback/keycloak`; - - const authUrl = new URL(`${keycloakIssuer}/protocol/openid-connect/auth`); - authUrl.searchParams.set('client_id', clientId); - authUrl.searchParams.set('redirect_uri', redirectUri); - authUrl.searchParams.set('response_type', 'code'); - authUrl.searchParams.set('scope', 'openid profile email roles'); - authUrl.searchParams.set('state', generateState()); // Générer state - - if (forceLogin) { - authUrl.searchParams.set('prompt', 'login'); - } - - return NextResponse.redirect(authUrl.toString()); -} -``` - -**Impact** : ✅ Prompt login seulement après logout - ---- - -## 🔧 Actions Secondaires (Après les actions immédiates) - -### Action 6 : Configurer explicitement les cookies NextAuth - -**Fichier** : `app/api/auth/options.ts` - -**Ajouter** après `session: { ... }` : - -```typescript -cookies: { - sessionToken: { - name: `next-auth.session-token`, - options: { - httpOnly: true, - sameSite: 'lax', - path: '/', - secure: process.env.NEXTAUTH_URL?.startsWith('https://') ?? false, - }, - }, - // ... autres cookies si nécessaire -}, -``` - ---- - -### Action 7 : Améliorer Keycloak logout URL - -**Fichiers** : -- `components/auth/signout-handler.tsx` -- `components/main-nav.tsx` -- `components/layout/layout-wrapper.tsx` - -**Changement** (ligne ~58-76) : - -```typescript -// AVANT -keycloakLogoutUrl.searchParams.append('kc_action', 'LOGOUT'); - -// APRÈS -keycloakLogoutUrl.searchParams.append('kc_action', 'LOGOUT'); -// Ajouter client_id pour forcer logout client spécifique -if (process.env.NEXT_PUBLIC_KEYCLOAK_CLIENT_ID) { - keycloakLogoutUrl.searchParams.append('client_id', - process.env.NEXT_PUBLIC_KEYCLOAK_CLIENT_ID); -} -``` - ---- - -### Action 8 : Améliorer gestion erreur refresh token - -**Fichier** : `app/api/auth/options.ts` - -**Changement** : Voir détails dans `IMPROVEMENTS_LOGIN_FLOW.md` section "Problème 7" - ---- - -## ✅ Checklist d'Implémentation - -### Phase 1 : Corrections Critiques (1-2 heures) -- [ ] Action 1 : Supprimer `prompt=login` par défaut -- [ ] Action 2 : Créer route `/api/auth/mark-logout` -- [ ] Action 3 : Modifier signout-handler pour utiliser la route -- [ ] Action 4 : Simplifier signin/page.tsx -- [ ] Action 5 : Ajouter prompt=login conditionnel - -### Phase 2 : Améliorations (1 heure) -- [ ] Action 6 : Configurer explicitement les cookies -- [ ] Action 7 : Améliorer Keycloak logout URL -- [ ] Action 8 : Améliorer gestion erreur refresh - -### Phase 3 : Tests (30 minutes) -- [ ] Tester login première visite (SSO doit fonctionner) -- [ ] Tester login après logout (credentials doivent être demandés) -- [ ] Tester logout depuis dashboard -- [ ] Tester logout depuis iframe -- [ ] Tester expiration session - ---- - -## 🎯 Résultat Attendu - -### Avant -- ❌ Toujours demander credentials (même première visite) -- ❌ Logique complexe de détection session invalide -- ❌ Race conditions possibles -- ❌ Cookies Keycloak peuvent persister - -### Après -- ✅ SSO naturel pour utilisateurs légitimes -- ✅ Credentials demandés seulement après logout -- ✅ Détection session invalide simple et robuste -- ✅ Pas de race conditions -- ✅ Meilleure gestion des cookies - ---- - -**Document créé le** : $(date) -**Priorité** : Actions immédiates à faire en premier - diff --git a/AUDIT_API_N8N_CONNECTION.md b/AUDIT_API_N8N_CONNECTION.md new file mode 100644 index 00000000..e97bc943 --- /dev/null +++ b/AUDIT_API_N8N_CONNECTION.md @@ -0,0 +1,724 @@ +# 🔍 Audit Développeur Senior - Connexion API Next.js ↔️ N8N (Missions) + +**Date**: $(date) +**Auteur**: Audit Développeur Senior +**Objectif**: Vérifier et documenter la connexion entre Next.js et N8N pour la gestion des missions + +--- + +## 📋 Table des Matières + +1. [Architecture Globale](#architecture-globale) +2. [Flux de Communication](#flux-de-communication) +3. [Endpoints API](#endpoints-api) +4. [Configuration Requise](#configuration-requise) +5. [Sécurité](#sécurité) +6. [Points Critiques à Vérifier](#points-critiques-à-vérifier) +7. [Problèmes Potentiels et Solutions](#problèmes-potentiels-et-solutions) +8. [Tests et Validation](#tests-et-validation) +9. [Recommandations](#recommandations) + +--- + +## 🏗️ Architecture Globale + +### Vue d'ensemble + +``` +┌─────────────┐ ┌─────────────┐ ┌─────────────┐ +│ Next.js │────────▶│ N8N │────────▶│ Intégrations│ +│ (API) │ │ (Workflow) │ │ (Gitea, etc)│ +└─────────────┘ └─────────────┘ └─────────────┘ + │ │ + │ │ + └─────────────────────────┘ + (Callback) +``` + +### Composants Principaux + +1. **Next.js API Routes** + - `POST /api/missions` - Création de mission + - `POST /api/missions/mission-created` - Callback de N8N + - `GET /api/missions` - Liste des missions + +2. **Service N8N** (`lib/services/n8n-service.ts`) + - Envoi de données vers N8N + - Gestion des webhooks + - Gestion des erreurs + +3. **N8N Workflows** + - Webhook de réception: `/webhook/mission-created` + - Création des intégrations externes + - Callback vers Next.js + +--- + +## 🔄 Flux de Communication + +### 1. Création d'une Mission (Next.js → N8N → Next.js) + +``` +┌─────────────────────────────────────────────────────────────────┐ +│ ÉTAPE 1: Création Mission dans Next.js │ +└─────────────────────────────────────────────────────────────────┘ +POST /api/missions + ↓ +1. Validation des données +2. Création en base de données (Prisma) +3. Upload des fichiers (logo, attachments) vers Minio +4. Vérification des fichiers + ↓ +┌─────────────────────────────────────────────────────────────────┐ +│ ÉTAPE 2: Envoi vers N8N │ +└─────────────────────────────────────────────────────────────────┘ +POST https://brain.slm-lab.net/webhook/mission-created +Headers: + - Content-Type: application/json + - x-api-key: {N8N_API_KEY} +Body: + { + missionId: "uuid", + name: "...", + oddScope: [...], + services: [...], + config: { + N8N_API_KEY: "...", + MISSION_API_URL: "https://api.slm-lab.net/api" + }, + ... + } + ↓ +┌─────────────────────────────────────────────────────────────────┐ +│ ÉTAPE 3: Traitement N8N │ +└─────────────────────────────────────────────────────────────────┘ +N8N Workflow: + 1. Réception webhook + 2. Création Gitea repository (si service "Gite") + 3. Création Leantime project (si service "Leantime") + 4. Création Outline collection (si service "Documentation") + 5. Création RocketChat channel (si service "RocketChat") + 6. Préparation des données de callback + ↓ +┌─────────────────────────────────────────────────────────────────┐ +│ ÉTAPE 4: Callback N8N → Next.js │ +└─────────────────────────────────────────────────────────────────┘ +POST {MISSION_API_URL}/api/missions/mission-created +Headers: + - Content-Type: application/json + - x-api-key: {N8N_API_KEY} (depuis config.N8N_API_KEY) +Body: + { + missionId: "uuid", + gitRepoUrl: "...", + leantimeProjectId: "...", + documentationCollectionId: "...", + rocketchatChannelId: "..." + } + ↓ +┌─────────────────────────────────────────────────────────────────┐ +│ ÉTAPE 5: Mise à jour Mission dans Next.js │ +└─────────────────────────────────────────────────────────────────┘ +Validation API key +Recherche mission par missionId +Mise à jour des champs d'intégration: + - giteaRepositoryUrl + - leantimeProjectId + - outlineCollectionId + - rocketChatChannelId +``` + +--- + +## 🔌 Endpoints API + +### 1. POST /api/missions + +**Fichier**: `app/api/missions/route.ts` + +**Fonction**: Créer une nouvelle mission et déclencher le workflow N8N + +**Authentification**: +- Session utilisateur requise (via `getServerSession`) +- Vérification: `checkAuth(request)` + +**Body attendu**: +```typescript +{ + name: string; + oddScope: string[]; + niveau?: string; + intention?: string; + missionType?: string; + services?: string[]; + guardians?: Record; + volunteers?: string[]; + logo?: { data: string; name?: string; type?: string }; + attachments?: Array<{ data: string; name?: string; type?: string }>; +} +``` + +**Réponse**: +```json +{ + "success": true, + "mission": { ... }, + "message": "Mission created successfully with all integrations" +} +``` + +**Points critiques**: +- ✅ Mission créée en base AVANT l'envoi à N8N +- ✅ Fichiers uploadés et vérifiés AVANT l'envoi à N8N +- ✅ `missionId` inclus dans les données envoyées à N8N +- ✅ `config.N8N_API_KEY` et `config.MISSION_API_URL` inclus + +--- + +### 2. POST /api/missions/mission-created + +**Fichier**: `app/api/missions/mission-created/route.ts` + +**Fonction**: Recevoir les IDs d'intégration de N8N et mettre à jour la mission + +**Authentification**: +- **API Key** via header `x-api-key` +- **PAS** de session utilisateur requise (N8N n'a pas de session) + +**Headers requis**: +``` +x-api-key: {N8N_API_KEY} +Content-Type: application/json +``` + +**Body attendu**: +```typescript +{ + missionId: string; // ✅ Préféré (plus fiable) + // OU (fallback pour compatibilité) + name: string; + creatorId: string; + + // IDs d'intégration (optionnels) + gitRepoUrl?: string; + leantimeProjectId?: string | number; + documentationCollectionId?: string; + rocketchatChannelId?: string; +} +``` + +**Réponse succès**: +```json +{ + "success": true, + "message": "Mission updated successfully", + "mission": { + "id": "...", + "name": "...", + "giteaRepositoryUrl": "...", + "leantimeProjectId": "...", + "outlineCollectionId": "...", + "rocketChatChannelId": "..." + } +} +``` + +**Codes d'erreur**: +- `401` - API key invalide ou manquante +- `400` - Champs requis manquants +- `404` - Mission non trouvée +- `500` - Erreur serveur + +**Points critiques**: +- ✅ Validation stricte de l'API key +- ✅ Recherche par `missionId` (préféré) ou `name + creatorId` (fallback) +- ✅ Conversion `leantimeProjectId` de number vers string si nécessaire +- ✅ Mise à jour uniquement des champs fournis + +--- + +## ⚙️ Configuration Requise + +### Variables d'Environnement + +#### 1. N8N_API_KEY (OBLIGATOIRE) +```env +N8N_API_KEY=LwgeE1ntADD20OuWC88S3pR0EaO7FtO4 +``` + +**Usage**: +- Envoyé à N8N dans `config.N8N_API_KEY` +- N8N l'utilise pour authentifier le callback +- Vérifié côté serveur dans `/api/missions/mission-created` + +**Où configurer**: +- `.env.local` (développement) +- Variables d'environnement production (CapRover, Vercel, Docker, etc.) + +**Vérification**: +```typescript +// Erreur si non défini +if (!process.env.N8N_API_KEY) { + logger.error('N8N_API_KEY is not set in environment variables'); +} +``` + +--- + +#### 2. N8N_WEBHOOK_URL (Optionnel) +```env +N8N_WEBHOOK_URL=https://brain.slm-lab.net/webhook/mission-created +``` + +**Valeur par défaut**: `https://brain.slm-lab.net/webhook/mission-created` + +**Usage**: URL du webhook N8N pour la création de mission + +--- + +#### 3. NEXT_PUBLIC_API_URL (Recommandé) +```env +NEXT_PUBLIC_API_URL=https://api.slm-lab.net/api +``` + +**Usage**: +- Envoyé à N8N dans `config.MISSION_API_URL` +- N8N l'utilise pour construire l'URL du callback +- Format attendu: `{MISSION_API_URL}/api/missions/mission-created` + +**Valeur par défaut**: `https://api.slm-lab.net/api` + +--- + +#### 4. N8N_ROLLBACK_WEBHOOK_URL (Optionnel) +```env +N8N_ROLLBACK_WEBHOOK_URL=https://brain.slm-lab.net/webhook/mission-rollback +``` + +**Usage**: URL du webhook N8N pour le rollback de mission + +--- + +### Configuration N8N Workflow + +#### Webhook de Réception + +**Path**: `mission-created` +**URL complète**: `https://brain.slm-lab.net/webhook/mission-created` +**Méthode**: `POST` +**Status**: Doit être **ACTIF** (toggle vert dans N8N) + +--- + +#### Node "Save Mission To API" + +**URL**: +``` +{{ $node['Process Mission Data'].json.config.MISSION_API_URL }}/api/missions/mission-created +``` + +**Méthode**: `POST` + +**Headers**: +``` +Content-Type: application/json +x-api-key: {{ $node['Process Mission Data'].json.config.N8N_API_KEY }} +``` + +**Body**: +```json +{ + "missionId": "{{ $node['Process Mission Data'].json.missionId }}", + "gitRepoUrl": "{{ $node['Create Git Repo'].json.url }}", + "leantimeProjectId": "{{ $node['Create Leantime Project'].json.id }}", + "documentationCollectionId": "{{ $node['Create Outline Collection'].json.id }}", + "rocketchatChannelId": "{{ $node['Create RocketChat Channel'].json.id }}" +} +``` + +**Points critiques**: +- ✅ Utiliser `config.MISSION_API_URL` (pas d'URL en dur) +- ✅ Utiliser `config.N8N_API_KEY` (pas de clé en dur) +- ✅ Inclure `missionId` dans le body +- ✅ Inclure tous les IDs d'intégration créés + +--- + +## 🔒 Sécurité + +### 1. Authentification API Key + +**Mécanisme**: +- N8N envoie `x-api-key` header +- Next.js compare avec `process.env.N8N_API_KEY` +- Si différent → `401 Unauthorized` + +**Code de validation** (`app/api/missions/mission-created/route.ts:42`): +```typescript +const apiKey = request.headers.get('x-api-key'); +const expectedApiKey = process.env.N8N_API_KEY; + +if (apiKey !== expectedApiKey) { + logger.error('Invalid API key', { + received: apiKey ? 'present' : 'missing', + expected: expectedApiKey ? 'configured' : 'missing' + }); + return NextResponse.json({ error: 'Unauthorized' }, { status: 401 }); +} +``` + +**Points critiques**: +- ✅ Comparaison stricte (pas de hash, clé en clair) +- ✅ Logging des tentatives invalides +- ✅ Pas de fallback si clé manquante + +--- + +### 2. Transmission de la Clé API + +**Flux**: +1. Next.js lit `process.env.N8N_API_KEY` +2. Next.js envoie à N8N dans `config.N8N_API_KEY` +3. N8N stocke temporairement dans le workflow +4. N8N renvoie dans header `x-api-key` lors du callback + +**Risque**: Si `N8N_API_KEY` est `undefined` au moment de l'envoi: +- N8N reçoit `undefined` ou chaîne vide +- N8N envoie chaîne vide dans le header +- Next.js rejette avec `401` + +**Solution**: Vérifier que `N8N_API_KEY` est défini avant l'envoi à N8N + +--- + +### 3. Validation des Données + +**Côté Next.js**: +- ✅ Validation des champs requis +- ✅ Recherche de mission par `missionId` (plus sûr que `name + creatorId`) +- ✅ Conversion de types (number → string pour `leantimeProjectId`) + +**Côté N8N**: +- ⚠️ Pas de validation visible dans le code Next.js +- ⚠️ N8N doit valider les données avant création des intégrations + +--- + +## ⚠️ Points Critiques à Vérifier + +### 1. Configuration Environnement + +- [ ] `N8N_API_KEY` est défini dans l'environnement +- [ ] `N8N_API_KEY` a la même valeur partout (dev, staging, prod) +- [ ] `NEXT_PUBLIC_API_URL` pointe vers la bonne URL +- [ ] Application redémarrée après modification des variables + +--- + +### 2. Workflow N8N + +- [ ] Workflow est **ACTIF** (toggle vert) +- [ ] Webhook path est correct: `mission-created` +- [ ] Node "Save Mission To API" utilise `config.MISSION_API_URL` +- [ ] Node "Save Mission To API" utilise `config.N8N_API_KEY` +- [ ] Node "Save Mission To API" inclut `missionId` dans le body +- [ ] Tous les IDs d'intégration sont inclus dans le callback + +--- + +### 3. Flux de Données + +- [ ] `missionId` est envoyé à N8N lors de la création +- [ ] `missionId` est renvoyé par N8N dans le callback +- [ ] Les IDs d'intégration sont correctement mappés: + - `gitRepoUrl` → `giteaRepositoryUrl` + - `leantimeProjectId` → `leantimeProjectId` (string) + - `documentationCollectionId` → `outlineCollectionId` + - `rocketchatChannelId` → `rocketChatChannelId` + +--- + +### 4. Gestion d'Erreurs + +- [ ] Erreurs N8N sont loggées +- [ ] Rollback en cas d'échec (si configuré) +- [ ] Messages d'erreur clairs pour debugging +- [ ] Pas de données sensibles dans les logs + +--- + +## 🐛 Problèmes Potentiels et Solutions + +### Problème 1: 401 Unauthorized + +**Symptômes**: +``` +Invalid API key { received: 'present', expected: 'configured' } +``` + +**Causes possibles**: +1. `N8N_API_KEY` non défini dans l'environnement +2. `N8N_API_KEY` différent entre Next.js et N8N +3. N8N envoie une clé vide ou `undefined` + +**Solutions**: +1. Vérifier que `N8N_API_KEY` est défini: + ```bash + echo $N8N_API_KEY + ``` +2. Vérifier la valeur dans N8N: + - Ouvrir l'exécution du workflow + - Vérifier `config.N8N_API_KEY` dans "Process Mission Data" +3. S'assurer que la même clé est utilisée partout + +--- + +### Problème 2: 404 Mission Not Found + +**Symptômes**: +``` +Mission not found { missionId: "...", name: "...", creatorId: "..." } +``` + +**Causes possibles**: +1. `missionId` non envoyé par N8N +2. `missionId` incorrect +3. Mission supprimée entre temps + +**Solutions**: +1. Vérifier que N8N envoie `missionId`: + ```json + { + "missionId": "{{ $node['Process Mission Data'].json.missionId }}" + } + ``` +2. Vérifier que Next.js envoie `missionId` à N8N: + ```typescript + config: { + missionId: mission.id // ✅ Inclus dans n8nData + } + ``` +3. Utiliser le fallback `name + creatorId` si `missionId` manquant + +--- + +### Problème 3: 500 Server Configuration Error + +**Symptômes**: +``` +N8N_API_KEY not configured in environment +``` + +**Cause**: `process.env.N8N_API_KEY` est `undefined` + +**Solution**: +1. Ajouter `N8N_API_KEY` à `.env.local` ou variables d'environnement +2. Redémarrer l'application +3. Vérifier avec un endpoint de test + +--- + +### Problème 4: 404 Webhook Not Registered + +**Symptômes**: +``` +404 Error: The requested webhook "mission-created" is not registered. +Hint: Click the 'Execute workflow' button on the canvas, then try again. +``` + +**Cause**: Workflow N8N n'est pas actif + +**Solution**: +1. Ouvrir le workflow dans N8N +2. Activer le toggle "Active" (devrait être vert) +3. Vérifier que le webhook node est actif + +--- + +### Problème 5: IDs d'Intégration Non Sauvegardés + +**Symptômes**: +- Mission créée mais `giteaRepositoryUrl`, `leantimeProjectId`, etc. sont `null` + +**Causes possibles**: +1. N8N ne rappelle pas `/api/missions/mission-created` +2. N8N rappelle mais avec des IDs manquants +3. Erreur lors de la mise à jour en base + +**Solutions**: +1. Vérifier les logs N8N (Executions) +2. Vérifier que le node "Save Mission To API" s'exécute +3. Vérifier les logs Next.js pour "Mission Created Webhook Received" +4. Vérifier que tous les IDs sont inclus dans le body du callback + +--- + +## 🧪 Tests et Validation + +### Test 1: Vérifier Configuration + +**Endpoint de test** (à créer): +```typescript +// app/api/test-n8n-config/route.ts +import { NextResponse } from 'next/server'; + +export async function GET() { + return NextResponse.json({ + hasN8NApiKey: !!process.env.N8N_API_KEY, + n8nApiKeyLength: process.env.N8N_API_KEY?.length || 0, + n8nWebhookUrl: process.env.N8N_WEBHOOK_URL || 'https://brain.slm-lab.net/webhook/mission-created', + missionApiUrl: process.env.NEXT_PUBLIC_API_URL || 'https://api.slm-lab.net/api' + }); +} +``` + +**Usage**: `GET /api/test-n8n-config` + +--- + +### Test 2: Tester Webhook N8N + +```bash +curl -X POST https://brain.slm-lab.net/webhook/mission-created \ + -H "Content-Type: application/json" \ + -d '{"test": "data"}' +``` + +**Résultats attendus**: +- ✅ `200/400/500` avec erreur workflow: Webhook actif +- ❌ `404` avec "webhook not registered": Webhook inactif + +--- + +### Test 3: Tester Callback Endpoint + +```bash +curl -X POST https://api.slm-lab.net/api/missions/mission-created \ + -H "Content-Type: application/json" \ + -H "x-api-key: YOUR_N8N_API_KEY" \ + -d '{ + "missionId": "test-mission-id", + "gitRepoUrl": "https://git.example.com/repo", + "leantimeProjectId": "123" + }' +``` + +**Résultats attendus**: +- ✅ `200` avec `success: true`: API key valide +- ❌ `401`: API key invalide +- ❌ `404`: Mission non trouvée (normal si missionId de test) + +--- + +### Test 4: Créer une Mission Complète + +1. Créer une mission via le frontend +2. Vérifier les logs Next.js: + - ✅ "Mission created successfully" + - ✅ "Starting N8N workflow" + - ✅ "N8N workflow result { success: true }" +3. Vérifier les logs N8N (Executions): + - ✅ Workflow exécuté avec succès + - ✅ Node "Save Mission To API" exécuté +4. Vérifier la base de données: + - ✅ Mission a les IDs d'intégration sauvegardés + +--- + +## 💡 Recommandations + +### 1. Amélioration de la Sécurité + +**Problème actuel**: Clé API en clair, comparaison simple + +**Recommandations**: +- [ ] Utiliser un système de tokens avec expiration +- [ ] Implémenter un système de signature HMAC +- [ ] Ajouter un rate limiting sur `/api/missions/mission-created` +- [ ] Logging des tentatives d'accès invalides avec IP + +--- + +### 2. Amélioration de la Robustesse + +**Problème actuel**: Pas de retry automatique si N8N échoue + +**Recommandations**: +- [ ] Implémenter un système de retry avec backoff exponentiel +- [ ] Queue de messages pour les callbacks manqués +- [ ] Webhook de santé pour vérifier que N8N est accessible +- [ ] Timeout configurable pour les appels N8N + +--- + +### 3. Amélioration du Debugging + +**Problème actuel**: Logs dispersés, pas de traçabilité complète + +**Recommandations**: +- [ ] Ajouter un `correlationId` pour tracer une mission de bout en bout +- [ ] Logs structurés avec contexte complet +- [ ] Dashboard de monitoring des intégrations +- [ ] Alertes en cas d'échec répété + +--- + +### 4. Amélioration de la Documentation + +**Recommandations**: +- [ ] Documenter le format exact attendu par N8N +- [ ] Exemples de payloads complets +- [ ] Diagrammes de séquence détaillés +- [ ] Guide de troubleshooting avec cas réels + +--- + +### 5. Tests Automatisés + +**Recommandations**: +- [ ] Tests unitaires pour `N8nService` +- [ ] Tests d'intégration pour les endpoints API +- [ ] Tests E2E avec mock N8N +- [ ] Tests de charge pour vérifier la scalabilité + +--- + +## 📝 Checklist de Vérification Rapide + +### Configuration +- [ ] `N8N_API_KEY` défini et identique partout +- [ ] `NEXT_PUBLIC_API_URL` pointe vers la bonne URL +- [ ] Application redémarrée après modifications + +### N8N Workflow +- [ ] Workflow actif (toggle vert) +- [ ] Webhook path: `mission-created` +- [ ] Node "Save Mission To API" configuré correctement +- [ ] `missionId` inclus dans le callback + +### Code Next.js +- [ ] `missionId` envoyé à N8N lors de la création +- [ ] Validation API key fonctionnelle +- [ ] Mapping des champs correct +- [ ] Gestion d'erreurs appropriée + +### Tests +- [ ] Test de création de mission réussi +- [ ] IDs d'intégration sauvegardés en base +- [ ] Logs sans erreurs critiques + +--- + +## 🔗 Références + +- **Service N8N**: `lib/services/n8n-service.ts` +- **Endpoint création**: `app/api/missions/route.ts` +- **Endpoint callback**: `app/api/missions/mission-created/route.ts` +- **Documentation N8N**: Voir fichiers `N8N_*.md` dans le projet + +--- + +**Document créé le**: $(date) +**Dernière mise à jour**: $(date) +**Version**: 1.0 + diff --git a/AUTHENTICATION_FIXES.md b/AUTHENTICATION_FIXES.md deleted file mode 100644 index c6f8d69b..00000000 --- a/AUTHENTICATION_FIXES.md +++ /dev/null @@ -1,224 +0,0 @@ -# Authentication Flow Fixes - -## Issues Fixed - -### 1. Logout Loop Issue ✅ - -**Problem**: -- User couldn't log out - infinite redirect loop -- Sign-in page auto-triggered Keycloak login even when user was already authenticated -- Keycloak session cookies weren't cleared, causing immediate re-authentication - -**Root Cause**: -- `/signin` page had `useEffect(() => { signIn("keycloak") }, [])` that always triggered login -- No check for existing authentication status -- Keycloak logout endpoint was never called, leaving Keycloak cookies valid - -**Fix Applied**: -1. **Sign-in page** (`app/signin/page.tsx`): - - Added check for existing session before triggering login - - If user is already authenticated, redirect to home - - Only trigger Keycloak login if status is "unauthenticated" - -2. **Sign-out handler** (`components/auth/signout-handler.tsx`): - - Now properly calls Keycloak logout endpoint - - Uses ID token for proper logout - - Clears both NextAuth and Keycloak cookies - -3. **Main navigation logout** (`components/main-nav.tsx`): - - Fixed to use `idToken` instead of `accessToken` for Keycloak logout - - Proper logout flow with Keycloak endpoint - ---- - -### 2. Iframe Applications Logging Out ✅ - -**Problem**: -- Iframe applications were logging out even when user was still authenticated in dashboard -- Desynchronization between NextAuth session and Keycloak session - -**Root Cause**: -- Sign-out only cleared NextAuth cookies -- Keycloak session cookies remained valid but could expire independently -- Iframe apps rely on Keycloak cookies for SSO -- When Keycloak cookies expired/invalidated, iframes logged out but dashboard stayed logged in - -**Fix Applied**: -1. **ID Token Storage** (`app/api/auth/options.ts`): - - Now stores `idToken` from Keycloak in JWT - - Exposes `idToken` in session object - - Preserves ID token during token refresh - -2. **Proper Keycloak Logout**: - - Sign-out now calls Keycloak logout endpoint with `id_token_hint` - - This properly invalidates Keycloak session and clears Keycloak cookies - - Ensures synchronization between dashboard and iframe apps - -3. **Type Definitions** (`types/next-auth.d.ts`): - - Added `idToken` to Session and JWT interfaces - - Type-safe access to ID token - ---- - -## Changes Made - -### Files Modified - -1. **`app/api/auth/options.ts`** - - Added `idToken` to JWT interface - - Store `account.id_token` in JWT during initial authentication - - Expose `idToken` in session callback - - Preserve `idToken` during token refresh - -2. **`app/signin/page.tsx`** - - Added session status check - - Prevent auto-login if already authenticated - - Redirect authenticated users to home - -3. **`components/auth/signout-handler.tsx`** - - Call Keycloak logout endpoint with ID token - - Proper logout flow that clears both NextAuth and Keycloak sessions - -4. **`components/main-nav.tsx`** - - Fixed logout button to use `idToken` instead of `accessToken` - - Proper Keycloak logout flow - -5. **`types/next-auth.d.ts`** - - Added `idToken?: string` to Session interface - - Added `idToken?: string` to JWT interface (both modules) - ---- - -## How It Works Now - -### Sign-In Flow (Fixed) - -``` -1. User navigates to /signin -2. Check session status: - - If authenticated → Redirect to / - - If unauthenticated → Trigger Keycloak login -3. After Keycloak authentication: - - Store tokens (access, refresh, ID token) - - Initialize storage - - Redirect to dashboard -``` - -### Sign-Out Flow (Fixed) - -``` -1. User clicks logout -2. Sign out from NextAuth (clears NextAuth cookies) -3. Call Keycloak logout endpoint: - - URL: ${KEYCLOAK_ISSUER}/protocol/openid-connect/logout - - Parameters: - * post_logout_redirect_uri: /signin - * id_token_hint: -4. Keycloak clears its session and cookies -5. Redirect to /signin (no auto-login loop) -``` - -### Iframe SSO (Fixed) - -``` -1. User authenticates in dashboard -2. Keycloak sets session cookies -3. Iframe apps read Keycloak cookies -4. When user logs out: - - Keycloak logout endpoint is called - - Keycloak cookies are cleared - - Iframe apps lose access (synchronized logout) -``` - ---- - -## Environment Variables Required - -Ensure these are set: - -```bash -# Required for logout -NEXT_PUBLIC_KEYCLOAK_ISSUER=https://keycloak.example.com/realms/neah - -# Already required for authentication -KEYCLOAK_CLIENT_ID=neah-dashboard -KEYCLOAK_CLIENT_SECRET= -KEYCLOAK_ISSUER=https://keycloak.example.com/realms/neah -NEXTAUTH_URL=https://dashboard.example.com -NEXTAUTH_SECRET= -``` - -**Important**: `NEXT_PUBLIC_KEYCLOAK_ISSUER` must be set for client-side logout to work. - ---- - -## Testing Checklist - -### Logout Flow -- [ ] Click logout button -- [ ] Should redirect to Keycloak logout -- [ ] Should redirect back to /signin -- [ ] Should NOT auto-login (no loop) -- [ ] Should be able to manually log in again - -### Sign-In Flow -- [ ] Navigate to /signin when not authenticated -- [ ] Should trigger Keycloak login -- [ ] Navigate to /signin when already authenticated -- [ ] Should redirect to / (no auto-login trigger) - -### Iframe SSO -- [ ] Log in to dashboard -- [ ] Open iframe application -- [ ] Should be automatically authenticated -- [ ] Log out from dashboard -- [ ] Iframe application should also lose authentication -- [ ] Refresh iframe - should require login - ---- - -## Additional Notes - -### ID Token vs Access Token - -- **Access Token**: Used for API calls to Keycloak-protected resources -- **ID Token**: Used for user identification and logout -- **Refresh Token**: Used to get new access tokens - -The ID token is required for proper Keycloak logout. It tells Keycloak which session to invalidate. - -### Cookie Synchronization - -The fix ensures that: -1. NextAuth cookies are cleared (dashboard logout) -2. Keycloak cookies are cleared (via logout endpoint) -3. Both happen in sequence, maintaining synchronization - -### Token Refresh - -During token refresh, the ID token is preserved (Keycloak doesn't issue new ID tokens on refresh). This ensures logout continues to work even after token refreshes. - ---- - -## Troubleshooting - -### If logout still loops: - -1. Check browser console for errors -2. Verify `NEXT_PUBLIC_KEYCLOAK_ISSUER` is set correctly -3. Check that Keycloak logout endpoint is accessible -4. Verify ID token is present in session: `console.log(session?.idToken)` - -### If iframes still log out independently: - -1. Check Keycloak cookie domain configuration -2. Verify iframe apps are configured to use same Keycloak realm -3. Check browser cookie settings (third-party cookies may be blocked) -4. Verify Keycloak session timeout settings - ---- - -**Date**: 2024 -**Status**: ✅ Fixed -**Version**: 1.0 - diff --git a/AUTHENTICATION_FLOW_AUDIT.md b/AUTHENTICATION_FLOW_AUDIT.md deleted file mode 100644 index c5a4be4e..00000000 --- a/AUTHENTICATION_FLOW_AUDIT.md +++ /dev/null @@ -1,988 +0,0 @@ -# Authentication Flow Audit - NextAuth with Keycloak & SSO for Iframe Applications - -## Executive Summary - -This document provides a comprehensive audit of the authentication architecture in the Neah dashboard application. The system uses **NextAuth.js v4** with **Keycloak** as the OAuth provider, implementing JWT-based sessions and supporting Single Sign-On (SSO) for multiple iframe-embedded applications via cookie-based authentication. - ---- - -## Architecture Overview - -### Components -1. **NextAuth.js** - Authentication framework -2. **Keycloak** - Identity Provider (IdP) via OAuth 2.0/OpenID Connect -3. **JWT Strategy** - Session management (no database sessions) -4. **Iframe Applications** - Multiple embedded applications using SSO via cookies -5. **Keycloak Admin Client** - Server-side user management - ---- - -## 1. Authentication Entry Points - -### 1.1 Sign-In Page (`/app/signin/page.tsx`) - -**Location**: `app/signin/page.tsx` - -**Flow**: -```typescript -1. User navigates to /signin -2. Component automatically triggers: signIn("keycloak", { callbackUrl: "/" }) -3. Redirects to Keycloak authorization endpoint -4. After Keycloak authentication, initializes storage via /api/storage/init -5. Redirects to home page -``` - -**Key Methods**: -- `signIn("keycloak")` - NextAuth client-side method -- Automatic redirect to Keycloak OAuth flow -- Storage initialization after successful authentication - -**Dependencies**: -- `next-auth/react` - Client-side NextAuth hooks -- Storage API endpoint for user space initialization - ---- - -## 2. NextAuth Configuration - -### 2.1 Route Handler (`/app/api/auth/[...nextauth]/route.ts`) - -**Location**: `app/api/auth/[...nextauth]/route.ts` - -**Purpose**: NextAuth API route handler for all authentication endpoints - -**Endpoints Handled**: -- `GET/POST /api/auth/signin` - Sign in -- `GET/POST /api/auth/signout` - Sign out -- `GET /api/auth/session` - Get current session -- `GET /api/auth/csrf` - CSRF token -- `GET /api/auth/providers` - Available providers -- `GET /api/auth/callback/keycloak` - OAuth callback - -**Implementation**: -```typescript -import NextAuth from "next-auth"; -import { authOptions } from "../options"; - -const handler = NextAuth(authOptions); -export { handler as GET, handler as POST }; -``` - ---- - -### 2.2 Auth Options Configuration (`/app/api/auth/options.ts`) - -**Location**: `app/api/auth/options.ts` - -**This is the core authentication configuration file.** - -#### 2.2.1 Keycloak Provider Setup - -```typescript -KeycloakProvider({ - clientId: getRequiredEnvVar("KEYCLOAK_CLIENT_ID"), - clientSecret: getRequiredEnvVar("KEYCLOAK_CLIENT_SECRET"), - issuer: getRequiredEnvVar("KEYCLOAK_ISSUER"), - authorization: { - params: { - scope: "openid profile email roles" // Requested OAuth scopes - } - }, - profile(profile) { /* Profile transformation */ } -}) -``` - -**Environment Variables Required**: -- `KEYCLOAK_CLIENT_ID` - OAuth client identifier -- `KEYCLOAK_CLIENT_SECRET` - OAuth client secret -- `KEYCLOAK_ISSUER` - Keycloak realm issuer URL (e.g., `https://keycloak.example.com/realms/neah`) - -**OAuth Scopes Requested**: -- `openid` - OpenID Connect core -- `profile` - User profile information -- `email` - User email address -- `roles` - User roles from Keycloak realm - -#### 2.2.2 Profile Callback - -**Location**: Lines 109-137 in `options.ts` - -**Purpose**: Transforms Keycloak user profile into NextAuth user object - -**Process**: -1. Receives Keycloak profile with `realm_access.roles` -2. Extracts roles from `realm_access.roles` array -3. Cleans roles by: - - Removing `ROLE_` prefix (if present) - - Converting to lowercase -4. Maps Keycloak profile fields to NextAuth user: - - `sub` → `id` - - `name` or `preferred_username` → `name` - - `email` → `email` - - `given_name` → `first_name` - - `family_name` → `last_name` - - `preferred_username` → `username` - - Cleaned roles → `role[]` - -**Code Flow**: -```typescript -profile(profile) { - const roles = profile.realm_access?.roles || []; - const cleanRoles = roles.map((role: string) => - role.replace(/^ROLE_/, '').toLowerCase() - ); - - return { - id: profile.sub, - name: profile.name ?? profile.preferred_username, - email: profile.email, - first_name: profile.given_name ?? '', - last_name: profile.family_name ?? '', - username: profile.preferred_username ?? profile.email?.split('@')[0] ?? '', - role: cleanRoles, - } -} -``` - -#### 2.2.3 Session Configuration - -**Location**: Lines 140-143 - -```typescript -session: { - strategy: "jwt", // JWT-based sessions (no database) - maxAge: 30 * 24 * 60 * 60, // 30 days -} -``` - -**Characteristics**: -- **Strategy**: JWT (stateless, no database lookups) -- **Max Age**: 30 days (2,592,000 seconds) -- **Storage**: Encrypted JWT stored in HTTP-only cookies - -#### 2.2.4 JWT Callback - -**Location**: Lines 145-181 - -**Purpose**: Handles JWT token creation and refresh - -**Flow**: - -**Initial Authentication (account & profile present)**: -```typescript -if (account && profile) { - 1. Extract roles from Keycloak profile - 2. Clean roles (remove ROLE_ prefix, lowercase) - 3. Store in JWT token: - - accessToken: account.access_token (Keycloak access token) - - refreshToken: account.refresh_token (Keycloak refresh token) - - accessTokenExpires: account.expires_at (expiration timestamp) - - sub: Keycloak user ID - - role: cleaned roles array - - username, first_name, last_name: from profile -} -``` - -**Subsequent Requests (token refresh check)**: -```typescript -else if (token.accessToken) { - 1. Decode JWT to extract roles (if not already in token) - 2. Check if token is expired: - - If expired: Call refreshAccessToken() - - If valid: Return existing token -} -``` - -**Token Expiration Check**: -```typescript -if (Date.now() < (token.accessTokenExpires as number) * 1000) { - return token; // Token still valid -} -return refreshAccessToken(token); // Token expired, refresh -``` - -**Note**: There's a **BUG** in line 176 - it multiplies `accessTokenExpires` by 1000, but `expires_at` from Keycloak is already in seconds since epoch. This should be checked. - -#### 2.2.5 Token Refresh Function - -**Location**: Lines 64-96 - -**Purpose**: Refreshes expired Keycloak access tokens - -**Implementation**: -```typescript -async function refreshAccessToken(token: JWT) { - 1. POST to Keycloak token endpoint: - - URL: ${KEYCLOAK_ISSUER}/protocol/openid-connect/token - - Method: POST - - Body: - * client_id: KEYCLOAK_CLIENT_ID - * client_secret: KEYCLOAK_CLIENT_SECRET - * grant_type: refresh_token - * refresh_token: token.refreshToken - - 2. On Success: - - Update accessToken - - Update refreshToken (if new one provided) - - Update accessTokenExpires: Date.now() + expires_in * 1000 - - 3. On Error: - - Set token.error = "RefreshAccessTokenError" - - Return token with error flag -} -``` - -**Error Handling**: Sets `token.error` flag which is checked in session callback - -#### 2.2.6 Session Callback - -**Location**: Lines 182-202 - -**Purpose**: Transforms JWT token into session object for client-side use - -**Flow**: -```typescript -async session({ session, token }) { - 1. Check for refresh errors: - if (token.error) throw new Error(token.error) - - 2. Build session.user object: - - id: token.sub (Keycloak user ID) - - email: token.email - - name: token.name - - image: null - - username: token.username - - first_name: token.first_name - - last_name: token.last_name - - role: token.role (array) - - nextcloudInitialized: false (default) - - 3. Add accessToken to session: - session.accessToken = token.accessToken - - 4. Return session -} -``` - -**Important**: The `accessToken` (Keycloak OAuth token) is exposed in the session object, making it available client-side via `useSession()` hook. - -#### 2.2.7 Custom Pages - -**Location**: Lines 204-207 - -```typescript -pages: { - signIn: '/signin', - error: '/signin', -} -``` - -**Custom Routes**: -- Sign-in page: `/signin` (instead of default `/api/auth/signin`) -- Error page: `/signin` (redirects to sign-in on errors) - ---- - -## 3. Authentication Flow Step-by-Step - -### 3.1 Initial Sign-In Flow - -``` -┌─────────────┐ -│ Browser │ -└──────┬──────┘ - │ - │ 1. GET /signin - ▼ -┌─────────────────────┐ -│ /app/signin/page.tsx │ -│ - Auto-triggers │ -│ signIn("keycloak") │ -└──────┬──────────────┘ - │ - │ 2. Redirect to NextAuth - ▼ -┌──────────────────────────────┐ -│ /api/auth/signin/keycloak │ -│ - Generates OAuth state │ -│ - Redirects to Keycloak │ -└──────┬───────────────────────┘ - │ - │ 3. GET /realms/{realm}/protocol/openid-connect/auth - │ ?client_id=... - │ &redirect_uri=... - │ &response_type=code - │ &scope=openid profile email roles - │ &state=... - ▼ -┌─────────────────────┐ -│ Keycloak Server │ -│ - Login page │ -│ - User credentials │ -└──────┬──────────────┘ - │ - │ 4. User authenticates - │ - │ 5. POST /realms/{realm}/protocol/openid-connect/token - │ (Authorization code exchange) - │ - │ 6. Keycloak returns: - │ - access_token - │ - refresh_token - │ - id_token - │ - expires_in - ▼ -┌──────────────────────────────┐ -│ /api/auth/callback/keycloak │ -│ - Receives authorization code│ -│ - Exchanges for tokens │ -│ - Fetches user profile │ -└──────┬───────────────────────┘ - │ - │ 7. JWT Callback - │ - Stores tokens in JWT - │ - Extracts user info - │ - Cleans roles - │ - │ 8. Session Callback - │ - Builds session object - │ - │ 9. Sets NextAuth cookies: - │ - next-auth.session-token (encrypted JWT) - │ - next-auth.csrf-token - ▼ -┌─────────────────────┐ -│ Browser (Client) │ -│ - Cookies set │ -│ - Redirect to / │ -└──────┬──────────────┘ - │ - │ 10. GET / (home page) - │ - getServerSession() validates JWT - │ - Session available - ▼ -┌─────────────────────┐ -│ Dashboard Loaded │ -└─────────────────────┘ -``` - -### 3.2 Subsequent Request Flow (Authenticated) - -``` -┌─────────────┐ -│ Browser │ -└──────┬──────┘ - │ - │ 1. GET /any-page - │ Cookie: next-auth.session-token=... - ▼ -┌──────────────────────────────┐ -│ Next.js Server │ -│ getServerSession(authOptions)│ -└──────┬───────────────────────┘ - │ - │ 2. Decrypt JWT from cookie - │ - │ 3. Check token expiration - │ - │ 4a. If valid: - │ - Extract user info - │ - Return session - │ - │ 4b. If expired: - │ - Call refreshAccessToken() - │ - POST to Keycloak /token - │ - Update JWT with new tokens - │ - Return session - ▼ -┌─────────────────────┐ -│ Page Component │ -│ - session available│ -└─────────────────────┘ -``` - -### 3.3 Token Refresh Flow - -``` -┌─────────────────────┐ -│ JWT Callback │ -│ (Token expired) │ -└──────┬──────────────┘ - │ - │ 1. Call refreshAccessToken() - │ - │ 2. POST ${KEYCLOAK_ISSUER}/protocol/openid-connect/token - │ Body: - │ - client_id - │ - client_secret - │ - grant_type: refresh_token - │ - refresh_token: - ▼ -┌─────────────────────┐ -│ Keycloak Server │ -│ - Validates refresh│ -│ token │ -│ - Issues new tokens│ -└──────┬──────────────┘ - │ - │ 3. Returns: - │ - access_token (new) - │ - refresh_token (new, optional) - │ - expires_in - ▼ -┌─────────────────────┐ -│ Update JWT Token │ -│ - New accessToken │ -│ - New refreshToken │ -│ - New expires time │ -└──────┬──────────────┘ - │ - │ 4. Return updated token - │ - │ 5. Session callback builds session - ▼ -┌─────────────────────┐ -│ Session Available │ -└─────────────────────┘ -``` - ---- - -## 4. Iframe SSO Architecture - -### 4.1 Overview - -The dashboard embeds multiple applications in iframes. These applications rely on **cookie-based SSO** to authenticate users automatically using the Keycloak session established in the parent dashboard. - -### 4.2 Iframe Application Pages - -**Pattern**: All iframe pages follow the same structure: - -```typescript -// Example: app/parole/page.tsx -export default async function Page() { - const session = await getServerSession(authOptions); - - if (!session) { - redirect("/signin"); - } - - return ( - - ); -} -``` - -**Iframe Applications Identified**: -1. **Parole** (`/parole`) - `NEXT_PUBLIC_IFRAME_PAROLE_URL` -2. **Agilite** (`/agilite`) - `NEXT_PUBLIC_IFRAME_AGILITY_URL` -3. **Alma** (`/alma`) - `NEXT_PUBLIC_IFRAME_AI_ASSISTANT_URL` -4. **Vision** (`/vision`) - `NEXT_PUBLIC_IFRAME_CONFERENCE_URL` -5. **The Message** (`/the-message`) - `NEXT_PUBLIC_IFRAME_THEMESSAGE_URL` -6. **WP Admin** (`/wp-admin`) - `NEXT_PUBLIC_IFRAME_MISSIONVIEW_URL` -7. **Mediation** (`/mediation`) - `NEXT_PUBLIC_IFRAME_MEDIATIONS_URL` -8. **Apprendre** (`/apprendre`) - `NEXT_PUBLIC_IFRAME_LEARN_URL` -9. **Gite** (`/gite`) - `NEXT_PUBLIC_IFRAME_GITE_URL` -10. **Artlab** (`/artlab`) - `NEXT_PUBLIC_IFRAME_ARTLAB_URL` -11. **Calcul** (`/calcul`) - `NEXT_PUBLIC_IFRAME_CALCULATION_URL` -12. **Chapitre** (`/chapitre`) - `NEXT_PUBLIC_IFRAME_CHAPTER_URL` -13. **Dossiers** (`/dossiers`) - `NEXT_PUBLIC_IFRAME_DRIVE_URL` -14. **CRM** (`/crm`) - `NEXT_PUBLIC_IFRAME_MEDIATIONS_URL` -15. **Livres** (`/livres`) - `NEXT_PUBLIC_IFRAME_LIVRE_URL` -16. **Showcase** (`/showcase`) - `NEXT_PUBLIC_IFRAME_SHOWCASE_URL` -17. **Radio** (`/radio`) - `NEXT_PUBLIC_IFRAME_RADIO_URL` -18. **Press** (`/press`) - `NEXT_PUBLIC_IFRAME_SHOWCASE_URL` -19. **Observatory** - `NEXT_PUBLIC_IFRAME_OBSERVATORY_URL` -20. **Time Tracker** - `NEXT_PUBLIC_IFRAME_TIMETRACKER_URL` -21. **Missions Board** - `NEXT_PUBLIC_IFRAME_MISSIONSBOARD_URL` -22. **Carnet** - `NEXT_PUBLIC_IFRAME_CARNET_URL` - -### 4.3 SSO Cookie Mechanism - -**How It Works**: - -1. **Parent Dashboard Authentication**: - - User authenticates via Keycloak in the dashboard - - Keycloak sets authentication cookies (domain: Keycloak domain) - - NextAuth sets session cookies (domain: dashboard domain) - -2. **Iframe Cookie Sharing**: - - When iframe loads, browser sends cookies for the iframe's domain - - If iframe application is on **same domain** or **subdomain** of Keycloak: - - Keycloak cookies are automatically sent - - Application can read Keycloak session cookies - - SSO works automatically - -3. **Cross-Domain Considerations**: - - If iframe apps are on different domains, they need: - - Same Keycloak realm configuration - - Proper CORS settings - - Cookie domain configuration in Keycloak - - `SameSite=None; Secure` cookie attributes for cross-site - -### 4.4 ResponsiveIframe Component - -**Location**: `app/components/responsive-iframe.tsx` - -**Features**: -- Auto-resizing based on viewport -- Hash synchronization (URL fragments) -- Full-screen support - -**Important**: This component does **NOT** handle authentication - it's purely presentational. SSO relies on browser cookie behavior. - ---- - -## 5. Sign-Out Flow - -### 5.1 Sign-Out Page - -**Location**: `app/signout/page.tsx` - -**Implementation**: -```typescript -export default function SignOut() { - return ( -
- -

Déconnexion en cours...

-
- ); -} -``` - -### 5.2 Sign-Out Handler - -**Location**: `components/auth/signout-handler.tsx` - -**Flow**: -```typescript -1. clearAuthCookies() - Clears NextAuth cookies client-side -2. signOut({ callbackUrl: "/signin", redirect: true }) - - Calls NextAuth signout endpoint - - Invalidates session - - Redirects to /signin -``` - -### 5.3 Cookie Clearing - -**Location**: `lib/session.ts` - `clearAuthCookies()` - -**Implementation**: -```typescript -export function clearAuthCookies() { - const cookies = document.cookie.split(';'); - for (const cookie of cookies) { - const [name] = cookie.split('='); - if (name.trim().startsWith('next-auth.') || - name.trim().startsWith('__Secure-next-auth.') || - name.trim().startsWith('__Host-next-auth.')) { - document.cookie = `${name.trim()}=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;`; - } - } -} -``` - -**Note**: This only clears NextAuth cookies. Keycloak cookies remain unless: -- User manually logs out of Keycloak -- Keycloak session expires -- Application calls Keycloak logout endpoint - -### 5.4 Service Token Invalidation - -**Location**: `lib/session.ts` - `invalidateServiceTokens()` - -**Purpose**: Logs out from integrated services (RocketChat, Leantime, etc.) - -**Services Handled**: -- RocketChat: `/api/v1/logout` -- Leantime: JSON-RPC logout method - -**Note**: This function exists but may not be called during standard sign-out flow. - ---- - -## 6. Server-Side Session Access - -### 6.1 getServerSession() - -**Usage Pattern** (seen in all iframe pages): -```typescript -import { getServerSession } from "next-auth/next"; -import { authOptions } from "@/app/api/auth/options"; - -const session = await getServerSession(authOptions); -``` - -**How It Works**: -1. Reads `next-auth.session-token` cookie from request -2. Decrypts JWT using `NEXTAUTH_SECRET` -3. Validates token signature and expiration -4. If expired, triggers refresh (via JWT callback) -5. Returns session object - -**Location**: Used in: -- All iframe page components -- Root layout (`app/layout.tsx`) -- Any server component needing authentication - -### 6.2 Client-Side Session Access - -**Usage Pattern**: -```typescript -import { useSession } from "next-auth/react"; - -const { data: session, status } = useSession(); -``` - -**How It Works**: -1. `useSession()` hook calls `/api/auth/session` -2. Server decrypts JWT and returns session -3. Client receives session object -4. Automatically refetches when token refreshes - -**Location**: Used in: -- `app/signin/page.tsx` -- `components/auth/auth-check.tsx` -- Any client component needing authentication - ---- - -## 7. Keycloak Admin Client - -### 7.1 Purpose - -**Location**: `lib/keycloak.ts` - -The Keycloak Admin Client is used for **server-side user management**, not for user authentication. It's a separate administrative interface. - -### 7.2 Authentication Methods - -**Two Methods Supported**: - -1. **Client Credentials** (Preferred): - ```typescript - grant_type: 'client_credentials' - client_id: KEYCLOAK_CLIENT_ID - client_secret: KEYCLOAK_CLIENT_SECRET - ``` - -2. **Password Grant** (Fallback): - ```typescript - grant_type: 'password' - client_id: KEYCLOAK_CLIENT_ID - username: KEYCLOAK_ADMIN_USERNAME - password: KEYCLOAK_ADMIN_PASSWORD - ``` - -### 7.3 Caching - -**Token Caching**: 5 minutes -- Validates cached token before reuse -- Creates new client if token invalid/expired - -### 7.4 Functions - -- `getKeycloakAdminClient()` - Get authenticated admin client -- `getUserById(userId)` - Get user by Keycloak ID -- `getUserByEmail(email)` - Get user by email -- `getAllRoles()` - Get all realm roles -- `getUserRoles(userId)` - Get user's role mappings - ---- - -## 8. Security Considerations - -### 8.1 Cookie Security - -**NextAuth Cookie Configuration** (implicit): -- **HttpOnly**: Yes (prevents XSS access) -- **Secure**: Yes (if `NEXTAUTH_URL` starts with `https://`) -- **SameSite**: Lax (default) -- **Path**: `/` -- **Domain**: Dashboard domain - -**Keycloak Cookie Configuration** (Keycloak-controlled): -- Set by Keycloak server -- Typically `SameSite=Lax` or `SameSite=None` (for cross-site) -- Domain: Keycloak domain or configured domain - -### 8.2 Token Storage - -- **Access Token**: Stored in encrypted JWT (server-side only accessible) -- **Refresh Token**: Stored in encrypted JWT -- **Session Token**: Encrypted JWT in HTTP-only cookie - -**Client-Side Access**: -- `session.accessToken` is exposed to client via `useSession()` -- This is the Keycloak OAuth access token -- Can be used for API calls to Keycloak-protected resources - -### 8.3 CORS & CSP - -**Content Security Policy** (`next.config.mjs`): -```typescript -'Content-Security-Policy': "frame-ancestors 'self' https://espace.slm-lab.net https://connect.slm-lab.net" -``` - -**Allows framing from**: -- Same origin (`'self'`) -- `https://espace.slm-lab.net` -- `https://connect.slm-lab.net` - -### 8.4 Role-Based Access Control - -**Role Extraction**: -- Roles come from Keycloak `realm_access.roles` -- Cleaned: `ROLE_` prefix removed, lowercased -- Stored in session: `session.user.role[]` - -**Usage**: Roles are available but not actively enforced in the codebase audit. Applications should implement RBAC checks. - ---- - -## 9. Environment Variables - -### 9.1 Required for Authentication - -```bash -# Keycloak OAuth Configuration -KEYCLOAK_CLIENT_ID=neah-dashboard -KEYCLOAK_CLIENT_SECRET= -KEYCLOAK_ISSUER=https://keycloak.example.com/realms/neah -KEYCLOAK_REALM=neah - -# NextAuth Configuration -NEXTAUTH_URL=https://dashboard.example.com -NEXTAUTH_SECRET= - -# Keycloak Admin (optional, for user management) -KEYCLOAK_ADMIN_USERNAME=admin -KEYCLOAK_ADMIN_PASSWORD= -KEYCLOAK_BASE_URL=https://keycloak.example.com -``` - -### 9.2 Iframe Application URLs - -All iframe applications require `NEXT_PUBLIC_IFRAME_*` environment variables: -- `NEXT_PUBLIC_IFRAME_PAROLE_URL` -- `NEXT_PUBLIC_IFRAME_AGILITY_URL` -- `NEXT_PUBLIC_IFRAME_AI_ASSISTANT_URL` -- `NEXT_PUBLIC_IFRAME_CONFERENCE_URL` -- ... (see section 4.2 for complete list) - ---- - -## 10. Potential Issues & Recommendations - -### 10.1 Token Expiration Bug - -**Location**: `app/api/auth/options.ts:176` - -```typescript -if (Date.now() < (token.accessTokenExpires as number) * 1000) { -``` - -**Issue**: `accessTokenExpires` from Keycloak `account.expires_at` is already in seconds since epoch. Multiplying by 1000 assumes it's in milliseconds, which may cause incorrect expiration checks. - -**Recommendation**: Verify Keycloak's `expires_at` format. If it's in seconds, remove the `* 1000`. If it's in milliseconds, keep it. - -### 10.2 Cookie SameSite for Cross-Domain Iframes - -**Issue**: If iframe applications are on different domains, Keycloak cookies may not be sent due to `SameSite` restrictions. - -**Recommendation**: -- Configure Keycloak cookies with `SameSite=None; Secure` -- Ensure all domains use HTTPS -- Consider using a shared parent domain for cookies - -### 10.3 Access Token Exposure - -**Issue**: `session.accessToken` (Keycloak OAuth token) is exposed client-side. - -**Recommendation**: -- Only expose if needed for client-side API calls -- Consider using proxy endpoints instead -- Implement token rotation if exposed - -### 10.4 No Explicit Cookie Configuration - -**Issue**: NextAuth cookie settings are implicit (defaults). - -**Recommendation**: Explicitly configure cookies in `authOptions`: -```typescript -cookies: { - sessionToken: { - name: `next-auth.session-token`, - options: { - httpOnly: true, - sameSite: 'lax', - path: '/', - secure: process.env.NEXTAUTH_URL?.startsWith('https://') ?? false, - } - } -} -``` - -### 10.5 Storage Initialization - -**Issue**: Storage initialization happens client-side after authentication, which may cause race conditions. - -**Recommendation**: Move storage initialization to server-side or use a more robust initialization pattern. - -### 10.6 Service Token Invalidation Not Called - -**Issue**: `invalidateServiceTokens()` exists but may not be called during sign-out. - -**Recommendation**: Integrate service token invalidation into the sign-out flow. - ---- - -## 11. Flow Diagrams - -### 11.1 Complete Authentication Flow - -``` -User → /signin - → signIn("keycloak") - → /api/auth/signin/keycloak - → Keycloak Authorization Endpoint - → User Login (Keycloak) - → Keycloak Token Endpoint - → /api/auth/callback/keycloak - → JWT Callback (store tokens) - → Session Callback (build session) - → Set Cookies - → Redirect to / - → Storage Init - → Dashboard Loaded -``` - -### 11.2 Iframe SSO Flow - -``` -Dashboard (authenticated) - → User clicks iframe app link - → Server checks session (getServerSession) - → If authenticated: Load iframe - → Browser sends cookies to iframe domain - → Iframe app reads Keycloak cookies - → Iframe app validates session - → Iframe app loads authenticated -``` - -### 11.3 Token Refresh Flow - -``` -Request with expired token - → getServerSession() - → Decrypt JWT - → Check expiration - → If expired: JWT Callback - → refreshAccessToken() - → POST to Keycloak /token - → Get new tokens - → Update JWT - → Return session -``` - ---- - -## 12. File Reference Map - -### Core Authentication Files - -| File | Purpose | -|------|---------| -| `app/api/auth/[...nextauth]/route.ts` | NextAuth route handler | -| `app/api/auth/options.ts` | **Main auth configuration** | -| `app/signin/page.tsx` | Sign-in page | -| `app/signout/page.tsx` | Sign-out page | -| `components/auth/signout-handler.tsx` | Sign-out logic | -| `components/auth/auth-check.tsx` | Client-side auth guard | -| `lib/keycloak.ts` | Keycloak admin client | -| `lib/session.ts` | Session utilities | -| `types/next-auth.d.ts` | TypeScript definitions | - -### Iframe Application Files - -All in `app/*/page.tsx`: -- `app/parole/page.tsx` -- `app/agilite/page.tsx` -- `app/alma/page.tsx` -- `app/vision/page.tsx` -- ... (see section 4.2) - -### Supporting Files - -| File | Purpose | -|------|---------| -| `app/components/responsive-iframe.tsx` | Iframe component | -| `app/layout.tsx` | Root layout (session check) | -| `components/providers.tsx` | SessionProvider wrapper | -| `components/layout/layout-wrapper.tsx` | Layout wrapper with auth | - ---- - -## 13. Testing Checklist - -### Authentication Flow -- [ ] Sign-in redirects to Keycloak -- [ ] Keycloak login works -- [ ] Callback receives tokens -- [ ] Session is created -- [ ] Cookies are set -- [ ] User redirected to dashboard -- [ ] Storage initializes - -### Session Management -- [ ] Session persists across page reloads -- [ ] Token refresh works when expired -- [ ] Session expires after 30 days -- [ ] Invalid tokens are rejected - -### Sign-Out -- [ ] Sign-out clears NextAuth cookies -- [ ] User redirected to sign-in -- [ ] Session invalidated - -### Iframe SSO -- [ ] Iframe apps receive Keycloak cookies -- [ ] Iframe apps authenticate automatically -- [ ] Cross-domain cookies work (if applicable) -- [ ] Unauthenticated users redirected - -### Security -- [ ] HttpOnly cookies enforced -- [ ] Secure cookies on HTTPS -- [ ] CSRF protection active -- [ ] Token encryption working - ---- - -## 14. Conclusion - -The authentication architecture uses a standard NextAuth + Keycloak OAuth 2.0 flow with JWT-based sessions. The system supports SSO for iframe applications via cookie sharing, assuming proper domain configuration. - -**Key Strengths**: -- Standard OAuth 2.0/OpenID Connect implementation -- Stateless JWT sessions (scalable) -- Automatic token refresh -- Role-based user information - -**Areas for Improvement**: -- Explicit cookie configuration -- Token expiration bug fix -- Service token invalidation integration -- Cross-domain cookie configuration verification -- Storage initialization robustness - ---- - -**Document Version**: 1.0 -**Last Updated**: 2024 -**Audited By**: AI Assistant -**Next Review**: After implementing recommendations - diff --git a/CAPROVER_NGINX_FIX.md b/CAPROVER_NGINX_FIX.md deleted file mode 100644 index 76d2b8e4..00000000 --- a/CAPROVER_NGINX_FIX.md +++ /dev/null @@ -1,163 +0,0 @@ -# Fix Nginx CapRover - Erreur "upstream sent too big header" - -## 🔍 Problème - -Erreur 502 avec message Nginx : -``` -upstream sent too big header while reading response header from upstream -``` - -**Cause** : Le cookie de session NextAuth (JWT avec tokens Keycloak) dépasse 4KB, la limite par défaut de Nginx. - -## ✅ Solution : Modifier la configuration CapRover - -### Option 1 : Via CapRover Dashboard (RECOMMANDÉ) - -1. **Aller dans CapRover Dashboard** -2. **Sélectionner votre app** (hub.slm-lab.net) -3. **Aller dans "HTTP Settings"** -4. **Cliquer sur "Edit Nginx Configuration"** (si disponible) -5. **OU aller dans "App Configs" → "nginx"** - -### Option 2 : Modifier le template Nginx directement - -Si vous avez accès au serveur CapRover, modifier le template dans : -- `/captain/templates/nginx.conf` (template principal) -- OU créer un override dans votre app - -## 📝 Configuration à Ajouter - -**Dans le bloc `location /`**, ajouter ces directives **AVANT** `proxy_pass` : - -```nginx -location / { - # ============================================ - # FIX: Augmenter la limite des headers pour NextAuth - # ============================================ - proxy_buffer_size 16k; - proxy_buffers 8 16k; - proxy_busy_buffers_size 32k; - large_client_header_buffers 4 32k; - - # Timeouts (pour éviter les timeouts) - proxy_connect_timeout 60s; - proxy_send_timeout 60s; - proxy_read_timeout 60s; - - # Configuration proxy existante - proxy_pass $upstream; - proxy_set_header Host $host; - proxy_set_header X-Real-IP $remote_addr; - proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; - proxy_set_header X-Forwarded-Proto $scheme; - - # WebSocket support (si activé) - proxy_set_header Upgrade $http_upgrade; - proxy_set_header Connection "upgrade"; - proxy_http_version 1.1; -} -``` - -## 🔧 Configuration Complète Modifiée - -Voici le bloc `location /` complet avec les corrections : - -```nginx -location / { - # FIX: Headers trop grands pour NextAuth - proxy_buffer_size 16k; - proxy_buffers 8 16k; - proxy_busy_buffers_size 32k; - large_client_header_buffers 4 32k; - - # Timeouts - proxy_connect_timeout 60s; - proxy_send_timeout 60s; - proxy_read_timeout 60s; - - # Proxy configuration - proxy_pass $upstream; - proxy_set_header Host $host; - proxy_set_header X-Real-IP $remote_addr; - proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; - proxy_set_header X-Forwarded-Proto $scheme; - - # WebSocket (si activé) - proxy_set_header Upgrade $http_upgrade; - proxy_set_header Connection "upgrade"; - proxy_http_version 1.1; -} -``` - -## 🎯 Méthode via CapRover Dashboard - -### Si CapRover permet l'édition Nginx : - -1. **Dashboard CapRover** → Votre app -2. **"App Configs"** → **"nginx"** -3. **Ajouter dans "Custom Nginx Configuration"** : - -```nginx -location / { - proxy_buffer_size 16k; - proxy_buffers 8 16k; - proxy_busy_buffers_size 32k; - large_client_header_buffers 4 32k; -} -``` - -4. **Sauvegarder** → CapRover regénère la config - -### Si CapRover ne permet pas l'édition : - -**Option A** : Modifier le template CapRover (avancé) -- Accéder au serveur CapRover -- Modifier `/captain/templates/nginx.conf` -- Redémarrer CapRover - -**Option B** : Créer un fichier de configuration personnalisé -- Créer un fichier dans votre app -- L'inclure dans la config Nginx - -## 🔄 Après Modification - -1. **Vérifier la config Nginx** : -```bash -sudo nginx -t -``` - -2. **Recharger Nginx** : -```bash -sudo systemctl reload nginx -# OU si CapRover gère Nginx -docker exec captain-nginx nginx -s reload -``` - -3. **Tester la connexion** : -- Se connecter via Keycloak -- Vérifier que l'erreur 502 ne se produit plus - -## 📊 Explication - -**Avant** : -- Limite par défaut Nginx : 4KB pour les headers -- Cookie NextAuth : ~4-7KB (JWT avec tokens Keycloak) -- Résultat : ❌ Erreur 502 - -**Après** : -- Limite augmentée : 32KB pour les headers -- Cookie NextAuth : ~4-7KB -- Résultat : ✅ Fonctionne - -## ⚠️ Note Importante - -Si vous modifiez le template CapRover directement, **vos modifications seront écrasées** lors d'une mise à jour de CapRover. - -**Recommandation** : Utiliser la méthode "Custom Nginx Configuration" dans CapRover si disponible, ou documenter vos modifications pour les réappliquer après mise à jour. - ---- - -**Document créé le** : $(date) -**Priorité** : HAUTE - Résout l'erreur 502 - - diff --git a/CHANGELOG_LOGIN_IMPROVEMENTS.md b/CHANGELOG_LOGIN_IMPROVEMENTS.md deleted file mode 100644 index 88071ce1..00000000 --- a/CHANGELOG_LOGIN_IMPROVEMENTS.md +++ /dev/null @@ -1,199 +0,0 @@ -# Changelog - Améliorations Flow de Connexion - -## Date : $(date) - -### ✅ Modifications Effectuées - -#### 1. **Suppression de `prompt=login` par défaut** ⭐ -**Fichier** : `app/api/auth/options.ts` -- **Avant** : `prompt: "login"` toujours actif → empêchait SSO naturel -- **Après** : `prompt` supprimé → SSO fonctionne naturellement pour utilisateurs légitimes -- **Impact** : ✅ Meilleure UX - SSO fonctionne pour les utilisateurs légitimes - -#### 2. **Création route API `/api/auth/mark-logout`** ⭐ -**Nouveau fichier** : `app/api/auth/mark-logout/route.ts` -- Crée un cookie HttpOnly `force_login_prompt=true` (5 minutes) -- Utilisé pour marquer qu'un logout a eu lieu -- **Impact** : ✅ Mécanisme robuste pour détecter logout côté serveur - -#### 3. **Modification signout-handler.tsx** ⭐ -**Fichier** : `components/auth/signout-handler.tsx` -- Ajout appel à `/api/auth/mark-logout` avant logout -- **Impact** : ✅ Flag serveur créé pour forcer login après logout - -#### 4. **Modification main-nav.tsx** ⭐ -**Fichier** : `components/main-nav.tsx` -- Ajout appel à `/api/auth/mark-logout` dans le handler logout -- **Impact** : ✅ Logout depuis navigation marque correctement le logout - -#### 5. **Modification layout-wrapper.tsx** ⭐ -**Fichier** : `components/layout/layout-wrapper.tsx` -- Ajout appel à `/api/auth/mark-logout` lors de logout depuis iframe -- **Impact** : ✅ Logout depuis iframe marque correctement le logout - -#### 6. **Simplification signin/page.tsx** ⭐ -**Fichier** : `app/signin/page.tsx` -- **Avant** : Logique complexe avec vérifications multiples (cookies, sessionStorage, URL params) -- **Après** : Logique simplifiée basée sur cookie serveur `force_login_prompt` -- Suppression de la détection complexe de session invalide -- **Impact** : ✅ Code plus simple et maintenable, moins de race conditions - -#### 7. **Configuration explicite des cookies NextAuth** ⭐ -**Fichier** : `app/api/auth/options.ts` -- Ajout configuration explicite pour : - - `sessionToken` - - `callbackUrl` - - `csrfToken` - - `state` -- **Impact** : ✅ Meilleur contrôle et sécurité des cookies - -#### 8. **Amélioration gestion erreur refresh token** ⭐ -**Fichier** : `app/api/auth/options.ts` -- Détection améliorée des différents types d'erreurs : - - `SessionNotActive` (session invalidée) - - `RefreshTokenExpired` (token expiré) -- Suppression explicite des tokens lors d'erreurs -- Clear des erreurs précédentes lors de refresh réussi -- **Impact** : ✅ Meilleure détection et gestion des sessions invalides - ---- - -## 🔄 Comportement Avant/Après - -### Avant les modifications - -**Login première visite** : -- ❌ Toujours demander credentials (même si SSO existe) -- ❌ Mauvaise UX - -**Login après logout** : -- ⚠️ Peut auto-login si session SSO Keycloak existe -- ⚠️ Logique complexe de détection - -**Logout** : -- ✅ Fonctionne mais peut laisser session SSO active - -**Code** : -- ❌ Logique complexe dans signin/page.tsx -- ❌ Race conditions possibles - -### Après les modifications - -**Login première visite** : -- ✅ SSO fonctionne naturellement (pas de prompt si session existe) -- ✅ Meilleure UX - -**Login après logout** : -- ✅ Session SSO terminée via `end-sso-session` (Admin API) -- ✅ Cookie `force_login_prompt` marque le logout -- ⚠️ Note: `prompt=login` n'est pas encore ajouté dynamiquement (limitation NextAuth) -- ✅ Mais session SSO est terminée donc credentials seront demandés - -**Logout** : -- ✅ Appelle `mark-logout` pour créer cookie serveur -- ✅ Termine session SSO via Admin API -- ✅ Supprime cookies NextAuth -- ✅ Redirige vers Keycloak logout - -**Code** : -- ✅ Logique simplifiée dans signin/page.tsx -- ✅ Moins de race conditions -- ✅ Configuration cookies explicite - ---- - -## 📝 Notes Importantes - -### Limitation Actuelle - -**`prompt=login` dynamique** : -- NextAuth v4 ne permet pas facilement d'ajouter `prompt=login` dynamiquement -- Solution actuelle : Terminer la session SSO via Admin API (`end-sso-session`) -- **Impact** : Si session SSO est bien terminée, Keycloak demandera credentials de toute façon -- **Amélioration future possible** : Middleware Next.js pour intercepter et modifier l'URL Keycloak - -### Solution de Contournement - -Au lieu d'ajouter `prompt=login` dynamiquement, nous : -1. Terminons la session SSO Keycloak via Admin API (`end-sso-logout`) -2. Créons un cookie serveur (`force_login_prompt`) pour tracking -3. Laissons Keycloak gérer naturellement (sans session SSO, il demandera credentials) - ---- - -## 🧪 Tests à Effectuer - -### Test 1 : Login première visite -1. Ouvrir navigateur en navigation privée -2. Aller sur `/signin` -3. **Attendu** : Redirection vers Keycloak, SSO fonctionne si session existe -4. **Résultat** : ✅ SSO fonctionne (pas de prompt forcé) - -### Test 2 : Login après logout -1. Se connecter -2. Se déconnecter -3. Cliquer "Se connecter" -4. **Attendu** : Keycloak demande credentials (session SSO terminée) -5. **Résultat** : ✅ Credentials demandés (session SSO terminée) - -### Test 3 : Logout depuis dashboard -1. Se connecter -2. Cliquer "Déconnexion" dans navigation -3. **Attendu** : Redirection vers `/signin?logout=true` -4. **Résultat** : ✅ Logout fonctionne - -### Test 4 : Logout depuis iframe -1. Se connecter -2. Ouvrir une application en iframe -3. Se déconnecter depuis l'iframe -4. **Attendu** : Dashboard se déconnecte aussi -5. **Résultat** : ✅ Logout synchronisé - -### Test 5 : Expiration session -1. Se connecter -2. Attendre expiration (ou invalider session Keycloak) -3. **Attendu** : Redirection vers `/signin` avec message approprié -4. **Résultat** : ✅ Détection et redirection fonctionnent - ---- - -## 🔧 Prochaines Améliorations Possibles - -### Option 1 : Middleware pour ajouter `prompt=login` -Créer un middleware Next.js qui intercepte `/api/auth/signin/keycloak` et modifie l'URL Keycloak pour ajouter `prompt=login` si cookie `force_login_prompt` existe. - -### Option 2 : Route custom signin -Créer une route custom qui gère complètement le flow OAuth avec `prompt=login` et utilise NextAuth pour valider le callback. - -### Option 3 : Modifier Keycloak configuration -Configurer Keycloak pour toujours demander credentials après logout (configuration côté Keycloak). - ---- - -## 📊 Fichiers Modifiés - -1. ✅ `app/api/auth/options.ts` - Configuration NextAuth -2. ✅ `app/api/auth/mark-logout/route.ts` - Nouveau fichier -3. ✅ `components/auth/signout-handler.tsx` - Handler logout -4. ✅ `components/main-nav.tsx` - Navigation logout -5. ✅ `components/layout/layout-wrapper.tsx` - Layout logout iframe -6. ✅ `app/signin/page.tsx` - Page signin simplifiée - ---- - -## ✅ Résumé - -**8 modifications majeures effectuées** : -- ✅ SSO naturel fonctionne -- ✅ Logout marqué côté serveur -- ✅ Code simplifié -- ✅ Meilleure gestion erreurs -- ✅ Configuration cookies explicite -- ⚠️ `prompt=login` dynamique non implémenté (limitation NextAuth, mais contourné via end-sso-session) - -**Impact global** : ✅ Flow de connexion amélioré, code plus maintenable, meilleure UX - ---- - -**Document créé le** : $(date) - diff --git a/COMPREHENSIVE_NOTIFICATION_ANALYSIS.md b/COMPREHENSIVE_NOTIFICATION_ANALYSIS.md deleted file mode 100644 index 51e29418..00000000 --- a/COMPREHENSIVE_NOTIFICATION_ANALYSIS.md +++ /dev/null @@ -1,789 +0,0 @@ -# Comprehensive Notification System Analysis & Improvement Recommendations - -**Date**: 2026-01-06 -**Purpose**: Complete step-by-step trace of notification system with improvement recommendations - ---- - -## 📋 **Table of Contents** - -1. [Architecture Overview](#architecture-overview) -2. [Complete Flow Traces](#complete-flow-traces) -3. [Current Issues Identified](#current-issues-identified) -4. [Improvement Recommendations](#improvement-recommendations) -5. [Performance Optimizations](#performance-optimizations) -6. [Reliability Improvements](#reliability-improvements) -7. [User Experience Enhancements](#user-experience-enhancements) - ---- - -## 🏗️ **Architecture Overview** - -### **Components**: - -``` -┌─────────────────────────────────────────────────────────────┐ -│ UI Layer (React) │ -│ ┌─────────────────────────────────────────────────────┐ │ -│ │ NotificationBadge Component │ │ -│ │ - Displays notification count badge │ │ -│ │ - Dropdown with notification list │ │ -│ │ - Mark as read / Mark all as read buttons │ │ -│ └─────────────────────────────────────────────────────┘ │ -│ ↓ │ -│ ┌─────────────────────────────────────────────────────┐ │ -│ │ useNotifications Hook │ │ -│ │ - State management (notifications, count, loading) │ │ -│ │ - Polling (60s interval) │ │ -│ │ - Optimistic updates │ │ -│ │ - Rate limiting (5s minimum between fetches) │ │ -│ └─────────────────────────────────────────────────────┘ │ -└─────────────────────────────────────────────────────────────┘ - ↓ -┌─────────────────────────────────────────────────────────────┐ -│ API Routes (Next.js) │ -│ ┌──────────────┐ ┌──────────────┐ ┌──────────────┐ │ -│ │ GET /count │ │ GET /list │ │ POST /read │ │ -│ │ │ │ │ │ POST /read-all│ │ -│ └──────────────┘ └──────────────┘ └──────────────┘ │ -└─────────────────────────────────────────────────────────────┘ - ↓ -┌─────────────────────────────────────────────────────────────┐ -│ Service Layer (NotificationService) │ -│ - Singleton pattern │ -│ - Adapter pattern (LeantimeAdapter, future adapters) │ -│ - Redis caching (count: 30s, list: 5min) │ -│ - Cache invalidation │ -│ - Background refresh scheduling │ -└─────────────────────────────────────────────────────────────┘ - ↓ -┌─────────────────────────────────────────────────────────────┐ -│ Adapter Layer (LeantimeAdapter) │ -│ - User ID caching (1 hour TTL) │ -│ - Retry logic (3 attempts, exponential backoff) │ -│ - Direct API calls to Leantime │ -│ - Notification transformation │ -└─────────────────────────────────────────────────────────────┘ - ↓ -┌─────────────────────────────────────────────────────────────┐ -│ External API (Leantime) │ -│ - JSON-RPC API │ -│ - getAllNotifications, markNotificationRead, etc. │ -└─────────────────────────────────────────────────────────────┘ -``` - ---- - -## 🔄 **Complete Flow Traces** - -### **Flow 1: Initial Page Load & Count Display** - -#### **Step-by-Step**: - -1. **Component Mount** (`notification-badge.tsx`) - ``` - - Component renders - - useNotifications() hook initializes - - useEffect triggers when status === 'authenticated' - ``` - -2. **Hook Initialization** (`use-notifications.ts`) - ``` - - Sets isMountedRef.current = true - - Calls fetchNotificationCount(true) - force refresh - - Calls fetchNotifications(1, 20) - - Starts polling: setInterval every 60 seconds - ``` - -3. **Count Fetch** (`use-notifications.ts` → `/api/notifications/count`) - ``` - - Checks: session exists, isMounted, rate limit (5s) - - Makes GET request: /api/notifications/count?_t=${Date.now()} - - Cache-busting parameter added - ``` - -4. **API Route** (`app/api/notifications/count/route.ts`) - ``` - - Authenticates user via getServerSession() - - Gets userId from session - - Calls NotificationService.getNotificationCount(userId) - ``` - -5. **Service Layer** (`notification-service.ts`) - ``` - - Checks Redis cache: notifications:count:${userId} - - If cached: Returns cached data (30s TTL) - - If not cached: Fetches from adapters - ``` - -6. **Adapter Layer** (`leantime-adapter.ts`) - ``` - - getNotificationCount() called - - Gets user email from session - - Gets Leantime user ID (checks cache first, then API with retry) - - Fetches up to 1000 notifications directly from API - - Counts unread: filter(n => n.read === 0) - - Returns count object - ``` - -7. **Cache Storage** (`notification-service.ts`) - ``` - - Stores count in Redis: notifications:count:${userId} - - TTL: 30 seconds - - Returns to API route - ``` - -8. **Response** (`app/api/notifications/count/route.ts`) - ``` - - Returns JSON with count - - Sets Cache-Control: private, max-age=10 - ``` - -9. **Hook Update** (`use-notifications.ts`) - ``` - - Receives count data - - Updates state: setNotificationCount(data) - ``` - -10. **UI Update** (`notification-badge.tsx`) - ``` - - Badge displays notificationCount.unread - - Shows "60" if 60 unread notifications - ``` - ---- - -### **Flow 2: Mark All Notifications as Read** - -#### **Step-by-Step**: - -1. **User Action** (`notification-badge.tsx`) - ``` - - User clicks "Mark all read" button - - Calls handleMarkAllAsRead() - - Calls markAllAsRead() from hook - ``` - -2. **Optimistic Update** (`use-notifications.ts`) - ``` - - Immediately updates state: - * All notifications: isRead = true - * Count: unread = 0 - - Provides instant UI feedback - ``` - -3. **API Call** (`use-notifications.ts`) - ``` - - Makes POST to /api/notifications/read-all - - Waits for response - ``` - -4. **API Route** (`app/api/notifications/read-all/route.ts`) - ``` - - Authenticates user - - Calls NotificationService.markAllAsRead(userId) - - Logs duration - ``` - -5. **Service Layer** (`notification-service.ts`) - ``` - - Loops through all adapters - - For each adapter: - * Checks if configured - * Calls adapter.markAllAsRead(userId) - - Collects results - - Always invalidates cache (even on failure) - ``` - -6. **Adapter Layer** (`leantime-adapter.ts`) - ``` - - Gets user email from session - - Gets Leantime user ID (cached or fetched with retry) - - Fetches all notifications from API (up to 1000) - - Filters unread: filter(n => n.read === 0) - - Marks each individually using Promise.all() - - Returns success if any were marked - ``` - -7. **Cache Invalidation** (`notification-service.ts`) - ``` - - Deletes count cache: notifications:count:${userId} - - Deletes all list caches: notifications:list:${userId}:* - - Uses SCAN to avoid blocking Redis - ``` - -8. **Count Refresh** (`use-notifications.ts`) - ``` - - After 200ms delay, calls fetchNotificationCount(true) - - Fetches fresh count from API - - Updates state with new count - ``` - ---- - -### **Flow 3: Polling for Updates** - -#### **Step-by-Step**: - -1. **Polling Setup** (`use-notifications.ts`) - ``` - - setInterval created: 60 seconds - - Calls debouncedFetchCount() on each interval - ``` - -2. **Debounced Fetch** (`use-notifications.ts`) - ``` - - Debounce delay: 300ms - - Prevents rapid successive calls - - Calls fetchNotificationCount(false) - ``` - -3. **Rate Limiting** (`use-notifications.ts`) - ``` - - Checks: now - lastFetchTime < 5 seconds - - If too soon, skips fetch - ``` - -4. **Count Fetch** (same as Flow 1, steps 3-10) - ``` - - Fetches from API - - Updates count if changed - ``` - ---- - -## 🐛 **Current Issues Identified** - -### **Issue #1: Multiple Fetching Mechanisms** - -**Problem**: -- `useNotifications` has its own polling (60s) -- `NotificationService` has background refresh -- `NotificationBadge` has manual fetch on open -- No coordination between them - -**Impact**: -- Redundant API calls -- Inconsistent refresh timing -- Potential race conditions - ---- - -### **Issue #2: Mark All As Read - Sequential Processing** - -**Problem**: -- Marks all notifications in parallel using `Promise.all()` -- No batching or rate limiting -- Can overwhelm Leantime API -- Connection resets on large batches (60+ notifications) - -**Impact**: -- Partial failures (some marked, some not) -- Network timeouts -- Poor user experience - ---- - -### **Issue #3: Cache TTL Mismatch** - -**Problem**: -- Count cache: 30 seconds -- List cache: 5 minutes -- Client cache: 10 seconds (count), 30 seconds (list) -- Background refresh: 1 minute cooldown - -**Impact**: -- Stale data inconsistencies -- Count and list can be out of sync -- Confusing UX - ---- - -### **Issue #4: No Progress Feedback** - -**Problem**: -- Mark all as read shows no progress -- User doesn't know how many are being marked -- No indication if operation is still running - -**Impact**: -- Poor UX -- User might click multiple times -- No way to cancel operation - ---- - -### **Issue #5: Optimistic Updates Can Be Wrong** - -**Problem**: -- Hook optimistically sets count to 0 -- But operation might fail or be partial -- Count refresh after 200ms might show different value -- Count jumps: 60 → 0 → 40 (confusing) - -**Impact**: -- Confusing UX -- User thinks operation failed when it partially succeeded - ---- - -### **Issue #6: No Retry for Mark All As Read** - -**Problem**: -- If connection resets during marking, operation fails -- No automatic retry for failed notifications -- User must manually retry - -**Impact**: -- Partial success requires manual intervention -- Poor reliability - ---- - -### **Issue #7: Session Lookup on Every Call** - -**Problem**: -- `getUserEmail()` calls `getServerSession()` every time -- `getLeantimeUserId()` is cached, but email lookup is not -- Multiple session lookups per request - -**Impact**: -- Performance overhead -- Potential session inconsistencies - ---- - -### **Issue #8: No Connection Pooling** - -**Problem**: -- Each API call creates new fetch request -- No connection reuse -- No request queuing - -**Impact**: -- Slower performance -- Higher connection overhead -- Potential connection exhaustion - ---- - -### **Issue #9: Background Refresh Uses setTimeout** - -**Problem**: -- `scheduleBackgroundRefresh()` uses `setTimeout(0)` -- Not reliable in serverless environments -- Can be lost if server restarts - -**Impact**: -- Background refresh might not happen -- Cache might become stale - ---- - -### **Issue #10: No Unified Refresh Integration** - -**Problem**: -- `useNotifications` has its own polling -- `RefreshManager` exists but not used -- `useUnifiedRefresh` hook exists but not integrated - -**Impact**: -- Duplicate refresh logic -- Inconsistent refresh intervals -- Not using centralized refresh system - ---- - -## 💡 **Improvement Recommendations** - -### **Priority 1: Integrate Unified Refresh System** - -**Current State**: -- `useNotifications` has custom polling (60s) -- `RefreshManager` exists but not used -- `useUnifiedRefresh` hook exists but not integrated - -**Recommendation**: -- Replace custom polling with `useUnifiedRefresh` -- Use `REFRESH_INTERVALS.NOTIFICATIONS_COUNT` (30s) -- Remove duplicate polling logic -- Centralize all refresh management - -**Benefits**: -- ✅ Consistent refresh intervals -- ✅ Reduced code duplication -- ✅ Better coordination with other widgets -- ✅ Easier to manage globally - ---- - -### **Priority 2: Batch Mark All As Read** - -**Current State**: -- Marks all notifications in parallel -- No batching or rate limiting -- Can overwhelm API - -**Recommendation**: -- Process in batches of 10-20 notifications -- Add delay between batches (100-200ms) -- Show progress indicator -- Retry failed batches automatically - -**Implementation**: -```typescript -// Pseudo-code -async markAllAsRead(userId: string): Promise { - const BATCH_SIZE = 10; - const BATCH_DELAY = 200; - - const batches = chunk(unreadNotifications, BATCH_SIZE); - - for (const batch of batches) { - await Promise.all(batch.map(n => markAsRead(n.id))); - await delay(BATCH_DELAY); - // Update progress - } -} -``` - -**Benefits**: -- ✅ Prevents API overload -- ✅ Better error recovery -- ✅ Progress feedback -- ✅ More reliable - ---- - -### **Priority 3: Fix Cache TTL Consistency** - -**Current State**: -- Count cache: 30s -- List cache: 5min -- Client cache: 10s/30s -- Background refresh: 1min - -**Recommendation**: -- Align all cache TTLs -- Count cache: 30s (matches refresh interval) -- List cache: 30s (same as count) -- Client cache: 0s (rely on server cache) -- Background refresh: 30s (matches TTL) - -**Benefits**: -- ✅ Consistent data -- ✅ Count and list always in sync -- ✅ Predictable behavior - ---- - -### **Priority 4: Add Progress Feedback** - -**Current State**: -- No progress indication -- User doesn't know operation status - -**Recommendation**: -- Show progress bar: "Marking X of Y..." -- Update in real-time as batches complete -- Show success/failure count -- Allow cancellation - -**Benefits**: -- ✅ Better UX -- ✅ User knows what's happening -- ✅ Prevents multiple clicks - ---- - -### **Priority 5: Improve Optimistic Updates** - -**Current State**: -- Optimistically sets count to 0 -- Might be wrong if operation fails -- Count jumps confusingly - -**Recommendation**: -- Only show optimistic update if confident -- Show loading state instead of immediate 0 -- Poll until count matches expected value -- Or: Show "Marking..." state instead of 0 - -**Benefits**: -- ✅ More accurate UI -- ✅ Less confusing -- ✅ Better error handling - ---- - -### **Priority 6: Add Automatic Retry** - -**Current State**: -- No retry for failed notifications -- User must manually retry - -**Recommendation**: -- Track which notifications failed -- Automatically retry failed ones -- Exponential backoff -- Max 3 retries per notification - -**Benefits**: -- ✅ Better reliability -- ✅ Automatic recovery -- ✅ Less manual intervention - ---- - -### **Priority 7: Cache User Email** - -**Current State**: -- `getUserEmail()` calls session every time -- Not cached - -**Recommendation**: -- Cache user email in Redis (same TTL as user ID) -- Invalidate on session change -- Reduce session lookups - -**Benefits**: -- ✅ Better performance -- ✅ Fewer session calls -- ✅ More consistent - ---- - -### **Priority 8: Add Connection Pooling** - -**Current State**: -- Each API call creates new fetch -- No connection reuse - -**Recommendation**: -- Use HTTP agent with connection pooling -- Reuse connections -- Queue requests if needed - -**Benefits**: -- ✅ Better performance -- ✅ Lower overhead -- ✅ More reliable connections - ---- - -### **Priority 9: Replace setTimeout with Proper Scheduling** - -**Current State**: -- Background refresh uses `setTimeout(0)` -- Not reliable in serverless - -**Recommendation**: -- Use proper job queue (Bull, Agenda, etc.) -- Or: Use Next.js API route for background jobs -- Or: Use cron job for scheduled refreshes - -**Benefits**: -- ✅ More reliable -- ✅ Works in serverless -- ✅ Better error handling - ---- - -### **Priority 10: Add Request Deduplication** - -**Current State**: -- Multiple components can trigger same fetch -- No deduplication - -**Recommendation**: -- Use `requestDeduplicator` utility (already exists) -- Deduplicate identical requests within short window -- Share results between callers - -**Benefits**: -- ✅ Fewer API calls -- ✅ Better performance -- ✅ Reduced server load - ---- - -## ⚡ **Performance Optimizations** - -### **1. Reduce API Calls** - -**Current**: -- Polling every 60s -- Background refresh every 1min -- Manual fetch on dropdown open -- Count refresh after marking - -**Optimization**: -- Use unified refresh (30s) -- Deduplicate requests -- Share cache between components -- Reduce redundant fetches - -**Expected Improvement**: 50-70% reduction in API calls - ---- - -### **2. Optimize Mark All As Read** - -**Current**: -- All notifications in parallel -- No batching -- Can timeout - -**Optimization**: -- Batch processing (10-20 at a time) -- Delay between batches -- Progress tracking -- Automatic retry - -**Expected Improvement**: 80-90% success rate (vs current 60-70%) - ---- - -### **3. Improve Cache Strategy** - -**Current**: -- Inconsistent TTLs -- Separate caches -- No coordination - -**Optimization**: -- Unified TTLs -- Coordinated invalidation -- Cache versioning -- Smart refresh - -**Expected Improvement**: 30-40% faster response times - ---- - -## 🛡️ **Reliability Improvements** - -### **1. Better Error Handling** - -**Current**: -- Basic try/catch -- Returns false on error -- No retry logic - -**Improvement**: -- Retry with exponential backoff -- Circuit breaker pattern -- Graceful degradation -- Better error messages - ---- - -### **2. Connection Resilience** - -**Current**: -- Fails on connection reset -- No recovery - -**Improvement**: -- Automatic retry -- Connection pooling -- Health checks -- Fallback mechanisms - ---- - -### **3. Partial Failure Handling** - -**Current**: -- All-or-nothing approach -- No tracking of partial success - -**Improvement**: -- Track which notifications succeeded -- Retry only failed ones -- Report partial success -- Allow resume - ---- - -## 🎨 **User Experience Enhancements** - -### **1. Progress Indicators** - -- Show "Marking X of Y..." during mark all -- Progress bar -- Success/failure count -- Estimated time remaining - ---- - -### **2. Better Loading States** - -- Skeleton loaders -- Optimistic updates with loading overlay -- Smooth transitions -- No jarring count jumps - ---- - -### **3. Error Messages** - -- User-friendly error messages -- Actionable suggestions -- Retry buttons -- Help text - ---- - -### **4. Real-time Updates** - -- WebSocket/SSE for real-time updates -- Instant count updates -- No polling needed -- Better UX - ---- - -## 📊 **Summary of Improvements** - -### **High Priority** (Implement First): -1. ✅ Integrate unified refresh system -2. ✅ Batch mark all as read -3. ✅ Fix cache TTL consistency -4. ✅ Add progress feedback - -### **Medium Priority**: -5. ✅ Improve optimistic updates -6. ✅ Add automatic retry -7. ✅ Cache user email -8. ✅ Add request deduplication - -### **Low Priority** (Nice to Have): -9. ✅ Connection pooling -10. ✅ Replace setTimeout with proper scheduling -11. ✅ WebSocket/SSE for real-time updates - ---- - -## 🎯 **Expected Results After Improvements** - -### **Performance**: -- 50-70% reduction in API calls -- 30-40% faster response times -- 80-90% success rate for mark all - -### **Reliability**: -- Automatic retry for failures -- Better error recovery -- More consistent behavior - -### **User Experience**: -- Progress indicators -- Better loading states -- Clearer error messages -- Smoother interactions - ---- - -**Status**: Analysis complete. Ready for implementation prioritization. - diff --git a/COURRIER_USER_MANAGEMENT.md b/COURRIER_USER_MANAGEMENT.md deleted file mode 100644 index 5ee71813..00000000 --- a/COURRIER_USER_MANAGEMENT.md +++ /dev/null @@ -1,304 +0,0 @@ -# Courrier User Management with Prisma - -## Overview - -**Important**: Courrier (the email system) does **NOT** create User records in Prisma. It only manages email account credentials (`MailCredentials`) for users that already exist in the database. - -## User Creation Flow - -### 1. User Creation in Keycloak (Primary Source) - -Users are created in **Keycloak** first, which is the primary authentication system: - -**Location**: `app/api/users/route.ts` (POST method) - -**Process**: -1. User is created in Keycloak via Admin API -2. Roles are assigned to the user -3. User may be created in external systems: - - **Leantime** (project management tool) - - **Dolibarr** (if user has "Mediation" or "Expression" roles) - -**Key Code**: -```typescript -// Create user in Keycloak -const createResponse = await fetch( - `${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users`, - { - method: "POST", - headers: { - Authorization: `Bearer ${token}`, - "Content-Type": "application/json", - }, - body: JSON.stringify({ - username: data.username, - enabled: true, - emailVerified: true, - firstName: data.firstName, - lastName: data.lastName, - email: data.email, - credentials: [{ type: "password", value: data.password, temporary: false }], - }), - } -); -``` - -### 2. User Sync to Prisma Database - -After creation in Keycloak, users need to be synced to the Prisma database. This happens via: - -**Option A: Manual Sync Script** -- `scripts/sync-users.ts` or `scripts/sync-users.js` -- Fetches users from Keycloak API -- Creates/updates User records in Prisma - -**Option B: API Endpoint** -- `app/api/sync-users/route.ts` (GET method) -- Can be called to sync users programmatically - -**Prisma User Creation**: -```typescript -await prisma.user.create({ - data: { - id: user.id, // Use the Keycloak ID as primary ID - email: user.email, - password: tempPassword, // Temporary password (not used for auth) - createdAt: new Date(), - updatedAt: new Date(), - }, -}); -``` - -**Important Notes**: -- The Prisma User `id` field uses the **Keycloak user ID** (UUID) -- The `password` field in Prisma is not used for authentication (Keycloak handles that) -- Users must exist in Prisma before they can use Courrier - -### 3. Prisma Schema - -**User Model** (`prisma/schema.prisma`): -```prisma -model User { - id String @id @default(uuid()) - email String @unique - password String - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt - mailCredentials MailCredentials[] // One-to-many relationship - // ... other relations -} -``` - -**MailCredentials Model**: -```prisma -model MailCredentials { - id String @id @default(uuid()) - userId String - email String - password String? // Optional (for OAuth accounts) - host String - port Int - secure Boolean @default(true) - use_oauth Boolean @default(false) - refresh_token String? - access_token String? - token_expiry DateTime? - smtp_host String? - smtp_port Int? - smtp_secure Boolean? @default(false) - display_name String? - color String? @default("#0082c9") - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt - user User @relation(fields: [userId], references: [id], onDelete: Cascade) - - @@unique([userId, email]) // One email account per user - @@index([userId]) -} -``` - -## Courrier's Role: Adding Email Accounts - -### How Courrier Adds Email Accounts - -**Location**: `app/api/courrier/account/route.ts` (POST method) - -**Process**: -1. **Authentication Check**: Verifies user session exists -2. **User Existence Check**: Verifies user exists in Prisma database -3. **Connection Test**: Tests IMAP connection before saving -4. **Save Credentials**: Creates/updates `MailCredentials` record - -**Key Code Flow**: - -```typescript -// 1. Check if user exists in database -const userExistsInDB = await userExists(session.user.id); -if (!userExistsInDB) { - return NextResponse.json({ - error: 'User not found in database', - details: `The user ID from your session (${session.user.id}) doesn't exist in the database.` - }, { status: 400 }); -} - -// 2. Test connection -const testResult = await testEmailConnection(credentials); -if (!testResult.imap) { - return NextResponse.json({ - error: `Connection test failed: ${testResult.error}` - }, { status: 400 }); -} - -// 3. Save credentials -await saveUserEmailCredentials(session.user.id, email, credentials); -``` - -### Saving Email Credentials - -**Location**: `lib/services/email-service.ts` → `saveUserEmailCredentials()` - -**Process**: -1. Prepares database credentials object (excluding OAuth tokens) -2. Uses `upsert` to create or update `MailCredentials` -3. Caches full credentials (including OAuth tokens) in Redis - -**Key Code**: -```typescript -// Save to database using upsert -await prisma.mailCredentials.upsert({ - where: { - // Finds existing record by userId + email - userId_email: { - userId: userId, - email: credentials.email - } - }, - update: dbCredentials, - create: { - userId, - ...dbCredentials - } -}); - -// Cache full credentials (including OAuth) in Redis -await cacheEmailCredentials(userId, accountId, fullCreds); -``` - -**Important Notes**: -- OAuth tokens (access_token, refresh_token) are stored in **Redis only**, not in Prisma -- The Prisma `MailCredentials` table stores IMAP/SMTP settings -- The `password` field is optional (for OAuth accounts like Microsoft) - -### Microsoft OAuth Flow - -**Location**: `app/api/courrier/microsoft/callback/route.ts` - -For Microsoft accounts, the flow is: -1. User authorizes via Microsoft OAuth -2. Access token and refresh token are obtained -3. Credentials are saved with `use_oauth: true` -4. OAuth tokens are cached in Redis (not in Prisma) - -## Data Flow Diagram - -``` -┌─────────────┐ -│ Keycloak │ ← Primary user creation -└──────┬──────┘ - │ - │ Sync - ↓ -┌─────────────┐ -│ Prisma │ ← User record created -│ User │ -└──────┬──────┘ - │ - │ User adds email account - ↓ -┌─────────────┐ -│ Prisma │ ← MailCredentials created -│MailCredentials│ -└──────┬──────┘ - │ - │ OAuth tokens (if applicable) - ↓ -┌─────────────┐ -│ Redis │ ← OAuth tokens cached -└─────────────┘ -``` - -## Key Files Reference - -### User Creation -- `app/api/users/route.ts` - Creates users in Keycloak -- `scripts/sync-users.ts` - Syncs users from Keycloak to Prisma -- `app/api/sync-users/route.ts` - API endpoint for syncing users - -### Courrier Email Management -- `app/api/courrier/account/route.ts` - Add/update/delete email accounts -- `lib/services/email-service.ts` - Core email service functions - - `saveUserEmailCredentials()` - Saves email credentials to Prisma - - `getUserEmailCredentials()` - Retrieves credentials from Prisma - - `testEmailConnection()` - Tests IMAP/SMTP connection - -### Database Schema -- `prisma/schema.prisma` - Prisma schema definitions -- `lib/prisma.ts` - Prisma client instance - -### Authentication -- `app/api/auth/options.ts` - NextAuth configuration -- `lib/auth.ts` - Authentication helpers - -## Auto-Creation of Users - -**As of recent updates**, Courrier now automatically creates User records in Prisma if they don't exist when: -- Adding an email account (`/api/courrier/account` POST) -- Checking session status (`/api/courrier/session` GET) - -This handles cases where: -- The database was reset/lost but users still exist in Keycloak -- Users were created in Keycloak but never synced to Prisma - -The auto-creation uses session data from Keycloak to populate: -- `id`: Keycloak user ID (UUID) -- `email`: User's email from session -- `password`: Temporary random password (not used for auth, Keycloak handles authentication) - -## Common Issues & Solutions - -### Issue: "User not found in database" when adding email account - -**Cause**: User exists in Keycloak but not in Prisma database - -**Solution**: -- **Automatic**: The system now auto-creates users when needed -- **Manual**: Run the sync script to create users in Prisma: -```bash -npm run sync-users -# or -node scripts/sync-users.js -``` - -### Issue: Email credentials not saving - -**Check**: -1. User exists in Prisma: `prisma.user.findUnique({ where: { id: userId } })` -2. Connection test passes before saving -3. Unique constraint `[userId, email]` is not violated - -### Issue: OAuth tokens not persisting - -**Note**: OAuth tokens are stored in Redis, not Prisma. Check: -- Redis connection and TTL settings -- Redis cache functions in `lib/redis.ts` - -## Summary - -1. **Users are created in Keycloak first** (via `app/api/users/route.ts`) -2. **Users are synced to Prisma** (via sync scripts or API) -3. **Courrier adds email accounts** by creating `MailCredentials` records linked to existing Users -4. **OAuth tokens are cached in Redis**, not stored in Prisma -5. **Users must exist in Prisma** before they can add email accounts via Courrier - -Courrier is a **credentials management system** for existing users, not a user creation system. - diff --git a/CRITICAL_FIXES_QUICK_REFERENCE.md b/CRITICAL_FIXES_QUICK_REFERENCE.md deleted file mode 100644 index 6008c0b0..00000000 --- a/CRITICAL_FIXES_QUICK_REFERENCE.md +++ /dev/null @@ -1,307 +0,0 @@ -# Critical Fixes - Quick Reference Guide - -## 🚨 Top 5 Critical Fixes (Do These First) - -### 1. Fix useNotifications Memory Leak ⚠️ CRITICAL - -**File**: `hooks/use-notifications.ts` -**Line**: 239-255 - -**Problem**: Cleanup function not properly placed, causing memory leaks - -**Quick Fix**: -```typescript -useEffect(() => { - if (status !== 'authenticated' || !session?.user) return; - - isMountedRef.current = true; - - // Initial fetch - fetchNotificationCount(true); - fetchNotifications(); - - // Start polling with proper cleanup - const intervalId = setInterval(() => { - if (isMountedRef.current) { - debouncedFetchCount(); - } - }, POLLING_INTERVAL); - - // ✅ Proper cleanup - return () => { - isMountedRef.current = false; - clearInterval(intervalId); - }; -}, [status, session?.user?.id]); // ✅ Only primitive dependencies -``` - ---- - -### 2. Fix Notification Badge Double Fetching ⚠️ CRITICAL - -**File**: `components/notification-badge.tsx` -**Lines**: 65-70, 82-87, 92-99 - -**Problem**: Three different places trigger the same fetch simultaneously - -**Quick Fix**: -```typescript -// Add at top of component -const fetchInProgressRef = useRef(false); -const lastFetchRef = useRef(0); -const FETCH_COOLDOWN = 1000; // 1 second cooldown - -const manualFetch = async () => { - const now = Date.now(); - - // Prevent duplicate fetches - if (fetchInProgressRef.current) { - console.log('[NOTIFICATION_BADGE] Fetch already in progress'); - return; - } - - // Cooldown check - if (now - lastFetchRef.current < FETCH_COOLDOWN) { - console.log('[NOTIFICATION_BADGE] Too soon since last fetch'); - return; - } - - fetchInProgressRef.current = true; - lastFetchRef.current = now; - - try { - await fetchNotifications(1, 10); - } finally { - fetchInProgressRef.current = false; - } -}; - -// Remove duplicate useEffect hooks, keep only one: -useEffect(() => { - if (isOpen && status === 'authenticated') { - manualFetch(); - } -}, [isOpen, status]); // Only this one -``` - ---- - -### 3. Fix Redis KEYS Performance Issue ⚠️ CRITICAL - -**File**: `lib/services/notifications/notification-service.ts` -**Line**: 293 - -**Problem**: `redis.keys()` blocks Redis and is O(N) - -**Quick Fix**: -```typescript -// BEFORE (Line 293) -const listKeys = await redis.keys(listKeysPattern); -if (listKeys.length > 0) { - await redis.del(...listKeys); -} - -// AFTER (Use SCAN) -const listKeys: string[] = []; -let cursor = '0'; -do { - const [nextCursor, keys] = await redis.scan( - cursor, - 'MATCH', - listKeysPattern, - 'COUNT', - 100 - ); - cursor = nextCursor; - if (keys.length > 0) { - listKeys.push(...keys); - } -} while (cursor !== '0'); - -if (listKeys.length > 0) { - await redis.del(...listKeys); -} -``` - ---- - -### 4. Fix Widget Interval Cleanup ⚠️ HIGH - -**Files**: -- `components/calendar.tsx` (line 70) -- `components/parole.tsx` (line 83) -- `components/calendar/calendar-widget.tsx` (line 110) - -**Problem**: Intervals may not be cleaned up properly - -**Quick Fix Pattern**: -```typescript -// BEFORE -useEffect(() => { - fetchEvents(); - const intervalId = setInterval(fetchEvents, 300000); - return () => clearInterval(intervalId); -}, []); // ❌ Missing dependencies - -// AFTER -useEffect(() => { - if (status !== 'authenticated') return; - - const fetchEvents = async () => { - // ... fetch logic - }; - - fetchEvents(); // Initial fetch - - const intervalId = setInterval(fetchEvents, 300000); - - return () => { - clearInterval(intervalId); - }; -}, [status]); // ✅ Proper dependencies -``` - ---- - -### 5. Fix useEffect Infinite Loop Risk ⚠️ HIGH - -**File**: `hooks/use-notifications.ts` -**Line**: 255 - -**Problem**: Function dependencies cause infinite re-renders - -**Quick Fix**: -```typescript -// Remove function dependencies, use refs for stable references -const fetchNotificationCountRef = useRef(fetchNotificationCount); -const fetchNotificationsRef = useRef(fetchNotifications); - -useEffect(() => { - fetchNotificationCountRef.current = fetchNotificationCount; - fetchNotificationsRef.current = fetchNotifications; -}); - -useEffect(() => { - if (status !== 'authenticated' || !session?.user) return; - - isMountedRef.current = true; - - fetchNotificationCountRef.current(true); - fetchNotificationsRef.current(); - - const intervalId = setInterval(() => { - if (isMountedRef.current) { - fetchNotificationCountRef.current(); - } - }, POLLING_INTERVAL); - - return () => { - isMountedRef.current = false; - clearInterval(intervalId); - }; -}, [status, session?.user?.id]); // ✅ Only primitive values -``` - ---- - -## 🔧 Additional Quick Wins - -### 6. Add Request Deduplication Utility - -**Create**: `lib/utils/request-deduplication.ts` - -```typescript -const pendingRequests = new Map>(); - -export function deduplicateRequest( - key: string, - requestFn: () => Promise -): Promise { - if (pendingRequests.has(key)) { - return pendingRequests.get(key)!; - } - - const promise = requestFn().finally(() => { - pendingRequests.delete(key); - }); - - pendingRequests.set(key, promise); - return promise; -} -``` - -**Usage**: -```typescript -const data = await deduplicateRequest( - `notifications-${userId}`, - () => fetch('/api/notifications').then(r => r.json()) -); -``` - ---- - -### 7. Extract Magic Numbers to Constants - -**Create**: `lib/constants/intervals.ts` - -```typescript -export const INTERVALS = { - NOTIFICATION_POLLING: 60000, // 1 minute - CALENDAR_REFRESH: 300000, // 5 minutes - PAROLE_POLLING: 30000, // 30 seconds - MIN_FETCH_INTERVAL: 5000, // 5 seconds - FETCH_COOLDOWN: 1000, // 1 second -} as const; -``` - ---- - -### 8. Add Error Retry Logic - -**Create**: `lib/utils/retry.ts` - -```typescript -export async function retry( - fn: () => Promise, - maxAttempts = 3, - delay = 1000 -): Promise { - for (let attempt = 1; attempt <= maxAttempts; attempt++) { - try { - return await fn(); - } catch (error) { - if (attempt === maxAttempts) throw error; - await new Promise(resolve => setTimeout(resolve, delay * attempt)); - } - } - throw new Error('Max retry attempts reached'); -} -``` - ---- - -## 📋 Testing Checklist - -After applying fixes, test: - -- [ ] No memory leaks (check browser DevTools Memory tab) -- [ ] No duplicate API calls (check Network tab) -- [ ] Intervals are cleaned up (check console for errors) -- [ ] No infinite loops (check React DevTools Profiler) -- [ ] Redis performance (check response times) -- [ ] Error handling works (test with network offline) - ---- - -## 🎯 Priority Order - -1. **Fix 1** (Memory Leak) - Do immediately -2. **Fix 2** (Double Fetching) - Do immediately -3. **Fix 3** (Redis KEYS) - Do immediately -4. **Fix 4** (Widget Cleanup) - Do within 24 hours -5. **Fix 5** (Infinite Loop) - Do within 24 hours -6. **Quick Wins** - Do within 1 week - ---- - -*Last Updated: Critical fixes quick reference* diff --git a/DEBUG_502_CALLBACK.md b/DEBUG_502_CALLBACK.md deleted file mode 100644 index 109a0523..00000000 --- a/DEBUG_502_CALLBACK.md +++ /dev/null @@ -1,224 +0,0 @@ -# Debug Erreur 502 - Callback Keycloak - -## 🔍 Situation Actuelle - -**URL** : `https://hub.slm-lab.net/api/auth/callback/keycloak?...` - -**Logs observés** : -- ✅ Profile callback : OK -- ✅ JWT callback : OK (rôles extraits depuis access token) -- ❌ Session callback : **PAS DE LOGS** (ne s'exécute pas ou échoue silencieusement) -- ❌ Erreur 502 Nginx - -## 🎯 Hypothèses - -### Hypothèse 1 : Session callback échoue silencieusement -Le session callback pourrait échouer avant d'atteindre les logs, causant une exception non gérée. - -### Hypothèse 2 : Problème avec token.email ou token.name -Si `token.email` ou `token.name` sont `undefined` et que le code s'attend à des valeurs, cela pourrait causer une erreur. - -### Hypothèse 3 : Timeout ou problème de mémoire -Le callback pourrait prendre trop de temps ou consommer trop de mémoire. - -### Hypothèse 4 : Problème avec NEXTAUTH_URL ou NEXTAUTH_SECRET -Configuration manquante ou incorrecte. - -## ✅ Corrections Appliquées - -### 1. Logs détaillés dans session callback -- Logs au début et à la fin -- Logs de chaque étape -- Logs des valeurs de token - -### 2. Try-catch complet -- Capture toutes les erreurs -- Logs détaillés de l'erreur -- Stack trace complète - -### 3. Validation des champs requis -- Vérification de `token.sub` (user ID) -- Gestion des valeurs `undefined` - -### 4. Events NextAuth -- `signIn` event pour tracker l'authentification -- `error` event pour capturer les erreurs NextAuth -- `signOut` event pour tracking - -## 🔍 Prochaines Étapes d'Investigation - -### Étape 1 : Vérifier les nouveaux logs - -Après redémarrage du serveur, vous devriez voir : - -``` -=== SESSION CALLBACK START === -Token error: undefined -Has accessToken: true -Has refreshToken: true -Token role: [...] -Token sub: ... -... -=== SESSION CALLBACK END === -``` - -**Si vous ne voyez PAS ces logs** : -- Le session callback ne s'exécute pas du tout -- Il y a une erreur avant d'atteindre le callback -- Problème dans NextAuth lui-même - -**Si vous voyez une erreur** : -- Les logs détaillés indiqueront exactement où ça échoue - -### Étape 2 : Vérifier les events NextAuth - -Vous devriez voir : -``` -=== NEXTAUTH SIGNIN EVENT === -User: ... ... -Account: keycloak -Profile: ... -``` - -**Si vous voyez `=== NEXTAUTH ERROR EVENT ===`** : -- L'erreur sera loggée avec détails - -### Étape 3 : Vérifier les variables d'environnement - -**Vérifier dans `.env` ou `.env.local`** : -```bash -NEXTAUTH_URL=https://hub.slm-lab.net -NEXTAUTH_SECRET=... (doit être défini) -KEYCLOAK_ISSUER=https://connect.slm-lab.net/realms/cercle -KEYCLOAK_CLIENT_ID=... -KEYCLOAK_CLIENT_SECRET=... -``` - -**Commandes pour vérifier** : -```bash -# Vérifier que les variables sont chargées -node -e "console.log(process.env.NEXTAUTH_URL)" -node -e "console.log(process.env.NEXTAUTH_SECRET ? 'SET' : 'MISSING')" -``` - -### Étape 4 : Vérifier les logs Nginx - -**Si Nginx est devant Next.js**, vérifier les logs Nginx : -```bash -# Logs d'erreur Nginx -sudo tail -f /var/log/nginx/error.log - -# Logs d'accès Nginx -sudo tail -f /var/log/nginx/access.log -``` - -**Chercher** : -- Timeout errors -- Connection refused -- Upstream errors - -### Étape 5 : Vérifier les logs système - -**Vérifier si Next.js crash** : -```bash -# Logs système -journalctl -u nextjs -f - -# Ou si PM2 -pm2 logs - -# Ou si systemd -systemctl status nextjs -``` - -## 🛠️ Actions Immédiates - -### 1. Redémarrer le serveur Next.js -```bash -# Arrêter -pm2 stop neah -# Ou -systemctl stop nextjs - -# Redémarrer -pm2 start neah -# Ou -systemctl start nextjs -``` - -### 2. Tester à nouveau la connexion - -1. Aller sur `/signin` -2. Se connecter avec Keycloak -3. Observer les logs dans le terminal - -### 3. Partager les logs complets - -**Ce qu'il faut partager** : -- Tous les logs depuis le début de la connexion -- Les logs jusqu'à l'erreur 502 -- Les logs Nginx (si disponibles) -- Les logs système (si disponibles) - -## 🔧 Solutions Possibles - -### Solution 1 : Problème avec token.email ou token.name - -**Si les logs montrent** : -``` -Token email: undefined -Token name: undefined -``` - -**Correction** : Le JWT callback doit extraire email et name depuis le profil ou le token d'accès. - -### Solution 2 : Problème avec NEXTAUTH_URL - -**Si NEXTAUTH_URL est incorrect** : -- NextAuth ne peut pas construire les URLs de callback -- Correction : Vérifier que `NEXTAUTH_URL` correspond à l'URL publique - -### Solution 3 : Problème avec NEXTAUTH_SECRET - -**Si NEXTAUTH_SECRET est manquant** : -- NextAuth ne peut pas signer les JWT -- Correction : Générer un secret et l'ajouter - -### Solution 4 : Timeout - -**Si le callback prend trop de temps** : -- Augmenter les timeouts Nginx -- Optimiser le code du callback - -## 📊 Checklist de Debugging - -- [ ] Serveur Next.js redémarré -- [ ] Logs `=== SESSION CALLBACK START ===` visibles -- [ ] Logs `=== SESSION CALLBACK END ===` visibles -- [ ] Pas d'erreur dans les logs -- [ ] Variables d'environnement vérifiées -- [ ] Logs Nginx vérifiés (si applicable) -- [ ] Logs système vérifiés (si applicable) - -## 🎯 Ce qu'on cherche - -**Dans les prochains logs, on cherche** : - -1. **Si on voit `=== SESSION CALLBACK START ===`** : - - ✅ Le callback s'exécute - - Chercher l'erreur dans les logs suivants - -2. **Si on NE voit PAS `=== SESSION CALLBACK START ===`** : - - ❌ Le callback ne s'exécute pas - - Problème dans NextAuth avant le callback - - Vérifier les events NextAuth - -3. **Si on voit `=== NEXTAUTH ERROR EVENT ===`** : - - ✅ NextAuth a capturé une erreur - - L'erreur sera loggée avec détails - ---- - -**Document créé le** : $(date) -**Statut** : En attente des nouveaux logs après redémarrage - diff --git a/DEPRECATED_FUNCTIONS.md b/DEPRECATED_FUNCTIONS.md deleted file mode 100644 index ef244117..00000000 --- a/DEPRECATED_FUNCTIONS.md +++ /dev/null @@ -1,145 +0,0 @@ -# Deprecated Functions and Files - -This document lists functions and files that have been deprecated and should not be used in new code. - -## Deprecated Files - -### 1. `lib/email-formatter.ts` (REMOVED) -- **Status**: Removed -- **Replacement**: Use `lib/utils/email-utils.ts` instead -- **Reason**: Consolidated email formatting to a single source of truth - -### 2. `lib/mail-parser-wrapper.ts` (REMOVED) -- **Status**: Removed -- **Replacement**: Use functions from `lib/utils/email-utils.ts` instead -- **Reason**: Consolidated email formatting and sanitization to a single source of truth - -### 3. `lib/email-parser.ts` (REMOVED) -- **Status**: Removed -- **Replacement**: Use `lib/server/email-parser.ts` for parsing and `lib/utils/email-utils.ts` for sanitization -- **Reason**: Consolidated email parsing and formatting to dedicated files - -### 4. `lib/compose-mime-decoder.ts` (REMOVED) -- **Status**: Removed -- **Replacement**: Use `decodeComposeContent` and `encodeComposeContent` functions from `lib/utils/email-utils.ts` -- **Reason**: Consolidated MIME handling into the centralized formatter - -## Deprecated Functions - -### 1. `formatEmailForReplyOrForward` in `lib/services/email-service.ts` (REMOVED) -- **Status**: Removed -- **Replacement**: Use `formatEmailForReplyOrForward` from `lib/utils/email-utils.ts` -- **Reason**: Consolidated email formatting to a single source of truth - -### 2. `formatSubject` in `lib/services/email-service.ts` (REMOVED) -- **Status**: Removed -- **Replacement**: None specific, handled by centralized formatter -- **Reason**: Internal function of the email formatter - -### 3. `createQuoteHeader` in `lib/services/email-service.ts` (REMOVED) -- **Status**: Removed -- **Replacement**: None specific, handled by centralized formatter -- **Reason**: Internal function of the email formatter - -## Centralized Email Formatting - -All email formatting is now handled by the centralized formatter in `lib/utils/email-utils.ts`. This file contains: - -1. `formatForwardedEmail`: Format emails for forwarding -2. `formatReplyEmail`: Format emails for replying or replying to all -3. `formatEmailForReplyOrForward`: Compatibility function that maps to the above two -4. `sanitizeHtml`: Safely sanitize HTML content while preserving direction attributes - -Use these functions for all email formatting needs. - -## Email Parsing and Processing Functions - -### 1. `splitEmailHeadersAndBody` (REMOVED) -- **Location**: Removed -- **Reason**: Email parsing has been centralized in `lib/server/email-parser.ts` and the API endpoint. -- **Replacement**: Use the `parseEmail` function from `lib/server/email-parser.ts` which provides a comprehensive parsing solution. - -### 2. `getReplyBody` -- **Location**: `app/courrier/page.tsx` -- **Reason**: Should use the `ReplyContent` component directly. -- **Replacement**: Use `` directly. -- **Status**: Currently marked with `@deprecated` comment, no direct usages found. - -### 3. `generateEmailPreview` -- **Location**: `app/courrier/page.tsx` -- **Reason**: Should use the `EmailPreview` component directly. -- **Replacement**: Use `` directly. -- **Status**: Currently marked with `@deprecated` comment, no usages found. - -### 4. `cleanHtml` (REMOVED) -- **Location**: Removed from `lib/server/email-parser.ts` -- **Reason**: HTML sanitization has been consolidated in `lib/utils/email-utils.ts`. -- **Replacement**: Use `sanitizeHtml` from `lib/utils/email-utils.ts`. - -### 5. `processHtml` (REMOVED) -- **Location**: Removed from `app/api/parse-email/route.ts` -- **Reason**: HTML processing has been consolidated in `lib/utils/email-utils.ts`. -- **Replacement**: Use `sanitizeHtml` from `lib/utils/email-utils.ts`. - -## Deprecated API Routes - -### 1. `app/api/mail/[id]/route.ts` (REMOVED) -- **Status**: Removed -- **Replacement**: Use `app/api/courrier/[id]/route.ts` instead. - -### 2. `app/api/mail/route.ts` (REMOVED) -- **Status**: Removed -- **Replacement**: Use `app/api/courrier/route.ts` instead. - -### 3. `app/api/mail/send/route.ts` (REMOVED) -- **Status**: Removed -- **Replacement**: Use `app/api/courrier/send/route.ts` instead. - -### 4. `DELETE /api/users/[userId]` (DEPRECATED) -- **Status**: Deprecated but maintained for backward compatibility -- **Replacement**: Use `DELETE /api/users?id=[userId]&email=[userEmail]` instead -- **Reason**: The new endpoint format supports deletion across all integrated systems (Keycloak, Leantime, and Dolibarr) -- **Notes**: The deprecated endpoint now forwards requests to the new endpoint but developers should update their code to use the new format directly - -## Deprecated Components - -### ComposeEmail (components/ComposeEmail.tsx) (REMOVED) - -**Status:** Removed -**Replacement:** Use `components/email/ComposeEmail.tsx` instead - -This component has been removed in favor of the more modular and better structured version in the email directory. The newer version has the following improvements: - -- Better separation between user message and quoted content in replies/forwards -- Improved styling and visual hierarchy -- Support for RTL/LTR text direction toggling -- More modern UI using Card components instead of a modal -- Better state management for email composition - -A compatibility layer has been added to the new component to ensure backward compatibility with existing code that uses the old component. This allows for a smooth transition without breaking changes. - -## Migration Plan - -### Phase 1: Deprecation (Completed) -- Mark all deprecated functions with `@deprecated` comments -- Add console warnings to deprecated functions -- Document alternatives - -### Phase 2: Removal (Completed) -- Remove deprecated files: `lib/email-parser.ts` and `lib/mail-parser-wrapper.ts` -- Consolidate all email formatting in `lib/utils/email-utils.ts` -- All email parsing now in `lib/server/email-parser.ts` -- Update documentation to point to the centralized utilities - -## Server-Client Code Separation - -### Server-side imports in client components -- **Status**: Fixed in November 2023 -- **Issue**: Server-only modules like ImapFlow were being imported directly in client components, causing build errors with messages like "Module not found: Can't resolve 'tls'" -- **Fix**: - 1. Added 'use server' directive to server-only modules - 2. Created client-safe interfaces in client components - 3. Added server actions for email operations that need server capabilities - 4. Refactored ComposeEmail component to avoid direct server imports - -This architecture ensures a clean separation between server and client code, which is essential for Next.js applications, particularly with the App Router. It prevents Node.js-specific modules from being bundled into client-side JavaScript. \ No newline at end of file diff --git a/ELECTRON.md b/ELECTRON.md deleted file mode 100644 index c33674ac..00000000 --- a/ELECTRON.md +++ /dev/null @@ -1,64 +0,0 @@ -# Neah Desktop Application - -This is the desktop version of Neah, built with Electron and Next.js. - -## Development - -To run the application in development mode: - -```bash -npm run electron:dev -``` - -This will start the Next.js development server and launch the Electron application that connects to it. - -## Building - -To build installers for your current platform: - -```bash -npm run electron:build -``` - -This will create installers in the `dist` directory. The build script handles: -- Building the Next.js application in static export mode -- Code signing (placeholder for macOS) -- Building installers for your current platform - -## Running from Production Build - -If you have already built the Next.js app and want to run the Electron app without building installers: - -```bash -npm run electron:start -``` - -## Platform-specific Notes - -### macOS - -- The app is configured with a placeholder code signing certificate -- For distribution, you will need to replace this with a real certificate from Apple - -### Windows - -- Windows builds are configured to use NSIS for creating installers - -### Linux - -- Linux builds create AppImage and Debian packages - -## Configuration - -The Electron configuration is in: -- `electron/main.js` - The main Electron process -- `electron/preload.js` - Preload script exposing APIs to the renderer -- `package.json` - Build configuration in the `build` section - -The window controls integration is in: -- `components/electron/WindowControls.tsx` - -## Known Issues - -- API routes: Since this is a static export, any server-side API routes will not work in the Electron app. You'll need to modify API calls to use external services or implement them in Electron. -- Authentication: If using server-side authentication, you may need to implement a custom flow for Electron. \ No newline at end of file diff --git a/IFRAME_LOGOUT_AUTO_LOGIN_ANALYSIS.md b/IFRAME_LOGOUT_AUTO_LOGIN_ANALYSIS.md deleted file mode 100644 index 98e30f87..00000000 --- a/IFRAME_LOGOUT_AUTO_LOGIN_ANALYSIS.md +++ /dev/null @@ -1,259 +0,0 @@ -# Iframe Logout Auto-Login Issue Analysis - -## Problem - -When you log out from an iframe application, you are automatically logged back into the dashboard without being prompted for credentials. - -## Flow Trace - -### Scenario: User Logs Out from Iframe Application - -#### Step 1: Iframe Application Logout -``` -Location: Iframe application (e.g., /parole, /gite, etc.) -Action: User clicks logout in iframe - -What happens: -- Iframe app may call Keycloak logout endpoint directly -- OR: Iframe app sends postMessage to parent: { type: 'KEYCLOAK_LOGOUT' } -- OR: Iframe app clears its own session cookies -``` - -#### Step 2A: If Iframe Sends PostMessage (Expected Flow) -``` -Location: components/layout/layout-wrapper.tsx (line 26-106) -OR: app/components/responsive-iframe.tsx (line 110-153) - -Action: Dashboard receives logout message - -What happens: -1. Sets sessionStorage.setItem('just_logged_out', 'true') -2. Sets document.cookie = 'logout_in_progress=true; path=/; max-age=60' -3. Calls /api/auth/end-sso-session (Admin API) -4. Calls signOut() from NextAuth -5. Redirects to Keycloak logout endpoint -6. Keycloak redirects back to /signin?logout=true -``` - -#### Step 2B: If Iframe Calls Keycloak Logout Directly (Actual Flow - Problem) -``` -Location: Iframe application - -Action: Iframe calls Keycloak logout endpoint directly - -What happens: -1. Iframe redirects to: ${KEYCLOAK_ISSUER}/protocol/openid-connect/logout -2. Keycloak clears session cookies -3. Keycloak may redirect iframe back to its own logout page -4. Dashboard doesn't know about this logout -5. Dashboard still has NextAuth session (valid for 30 days) -``` - -#### Step 3: Dashboard Detects Session Invalidation -``` -Location: app/api/auth/options.ts (refreshAccessToken function) - -When: NextAuth tries to refresh the access token - -What happens: -1. Dashboard calls Keycloak token refresh endpoint -2. Keycloak returns: { error: 'invalid_grant', error_description: 'Session not active' } -3. refreshAccessToken detects this error (line 100-108) -4. Returns token with error: "SessionNotActive" -5. JWT callback clears tokens (line 248-256) -6. Session callback returns null (line 272-276) -7. NextAuth treats user as unauthenticated -8. Status becomes "unauthenticated" -``` - -#### Step 4: Sign-In Page Auto-Login (THE PROBLEM) -``` -Location: app/signin/page.tsx (line 47-79) - -When: User is redirected to /signin (or status becomes "unauthenticated") - -What happens: -1. Component mounts -2. First useEffect (line 16-45) checks for logout flag - - If logout=true in URL, sets isLogoutRedirect.current = true - - Removes 'just_logged_out' from sessionStorage -3. Second useEffect (line 47-79) checks authentication status - - If status === "authenticated" → redirects to home ✅ - - If status === "unauthenticated" → triggers auto-login ❌ - -THE PROBLEM: -- When iframe logs out directly (not via postMessage), dashboard doesn't set logout flags -- Status becomes "unauthenticated" (because Keycloak session was cleared) -- Sign-in page sees status === "unauthenticated" -- Auto-login logic triggers after 1 second (line 69) -- signIn("keycloak") is called -- Keycloak still has SSO session cookie (if it wasn't fully cleared) -- User is auto-authenticated without credentials ❌ -``` - -## Root Cause Analysis - -### Problem 1: Missing Logout Flags - -**When iframe logs out directly (not via postMessage):** -- Dashboard doesn't know about the logout -- `just_logged_out` is NOT set in sessionStorage -- `logout_in_progress` cookie is NOT set -- Sign-in page doesn't know this is a logout scenario - -**Result**: Sign-in page treats it as a normal "unauthenticated" state and triggers auto-login. - -### Problem 2: Auto-Login Logic Timing - -**Sign-in page auto-login logic** (`app/signin/page.tsx:66-78`): -```typescript -if (status === "unauthenticated") { - hasAttemptedLogin.current = true; - const timer = setTimeout(() => { - if (!isLogoutRedirect.current) { - signIn("keycloak", { callbackUrl: "/" }); - } - }, 1000); -} -``` - -**The Issue**: -- `isLogoutRedirect.current` is set in the first useEffect (line 16-45) -- But it only checks for `logout=true` in URL or `just_logged_out` in sessionStorage -- If iframe logs out directly, neither of these is set -- After 1 second, auto-login triggers -- `isLogoutRedirect.current` is still `false` (because logout flags weren't set) -- `signIn("keycloak")` is called -- User is auto-authenticated - -### Problem 3: SSO Session Cookie Persistence - -**Even if logout flags are set correctly:** -- Keycloak SSO session cookie (`KEYCLOAK_SESSION`) may still exist -- When `signIn("keycloak")` is called, Keycloak checks for SSO session cookie -- If cookie exists, Keycloak auto-authenticates without credentials -- This happens even with `prompt=login` parameter (if SSO session is still valid) - -## Why This Happens - -### Flow 1: Iframe Logs Out via PostMessage (Works Correctly) -``` -1. Iframe sends postMessage → Dashboard receives it -2. Dashboard sets logout flags ✅ -3. Dashboard calls logout endpoints ✅ -4. Redirects to /signin?logout=true ✅ -5. Sign-in page sees logout=true ✅ -6. Auto-login is prevented ✅ -7. User must click "Se connecter" manually ✅ -``` - -### Flow 2: Iframe Logs Out Directly (THE PROBLEM) -``` -1. Iframe calls Keycloak logout directly -2. Keycloak clears session cookies -3. Dashboard doesn't know about logout ❌ -4. NextAuth tries to refresh token -5. Keycloak returns "Session not active" -6. NextAuth marks user as unauthenticated -7. User is redirected to /signin (no logout=true) ❌ -8. Sign-in page sees status="unauthenticated" ❌ -9. Auto-login triggers after 1 second ❌ -10. Keycloak still has SSO session cookie ❌ -11. User is auto-authenticated ❌ -``` - -## The Real Issue - -**The sign-in page auto-login logic is too aggressive:** - -1. It triggers auto-login for ANY "unauthenticated" state -2. It doesn't distinguish between: - - User never logged in (should auto-login) ✅ - - User logged out (should NOT auto-login) ❌ - - Session expired (should NOT auto-login) ❌ - - Keycloak session invalidated (should NOT auto-login) ❌ - -3. The logout detection only works if: - - `logout=true` is in URL (from Keycloak redirect) - - `just_logged_out` is in sessionStorage (from dashboard logout) - - But NOT if iframe logs out directly - -## Solution Requirements - -To fix this issue, you need to: - -1. **Detect Keycloak Session Invalidation**: - - When NextAuth detects "SessionNotActive" error - - Set a flag to prevent auto-login - - Mark this as a logout scenario, not a new login - -2. **Improve Logout Detection**: - - Check for Keycloak session cookie existence - - If session was invalidated (not just expired), prevent auto-login - - Store logout reason in sessionStorage - -3. **Modify Auto-Login Logic**: - - Only auto-login if: - - User is truly unauthenticated (never logged in) - - AND no logout flags are set - - AND no session invalidation detected - - Don't auto-login if: - - Logout flags are set - - Session was invalidated - - User came from a logout flow - -4. **Handle Iframe Direct Logout**: - - Detect when Keycloak session is invalidated - - Set logout flags automatically - - Prevent auto-login - -## Current Code Issues - -### Issue 1: Auto-Login Logic (`app/signin/page.tsx:66-78`) -```typescript -if (status === "unauthenticated") { - // This triggers for ANY unauthenticated state - // Doesn't check if session was invalidated - signIn("keycloak", { callbackUrl: "/" }); -} -``` - -### Issue 2: Logout Detection (`app/signin/page.tsx:16-45`) -```typescript -// Only checks for explicit logout flags -// Doesn't detect session invalidation -const logoutParam = searchParams.get('logout'); -const fromLogout = sessionStorage.getItem('just_logged_out'); -``` - -### Issue 3: Session Invalidation Detection (`app/api/auth/options.ts:248-256`) -```typescript -// Detects session invalidation -// But doesn't set logout flags -// Sign-in page doesn't know session was invalidated -if (refreshedToken.error === "SessionNotActive") { - return { - ...refreshedToken, - accessToken: undefined, - // Should set a flag here to prevent auto-login - }; -} -``` - -## Summary - -**Why you're auto-logged in after iframe logout:** - -1. Iframe logs out directly (not via postMessage) -2. Keycloak session is cleared -3. Dashboard detects session invalidation -4. User becomes "unauthenticated" -5. Sign-in page auto-login logic triggers (after 1 second) -6. Keycloak still has SSO session cookie -7. User is auto-authenticated without credentials - -**The fix requires:** -- Detecting session invalidation and setting logout flags -- Preventing auto-login when session was invalidated -- Only auto-login for truly new users (never logged in) - diff --git a/IFRAME_LOGOUT_FIX.md b/IFRAME_LOGOUT_FIX.md deleted file mode 100644 index 049da544..00000000 --- a/IFRAME_LOGOUT_FIX.md +++ /dev/null @@ -1,89 +0,0 @@ -# Iframe Logout Session Invalidation Fix - -## Problem - -When a user logs out from an application inside an iframe: -1. The iframe application calls Keycloak logout endpoint -2. Keycloak session is invalidated -3. NextAuth dashboard still has a valid JWT token -4. When NextAuth tries to refresh the token, Keycloak returns: `{ error: 'invalid_grant', error_description: 'Session not active' }` -5. This causes a `JWT_SESSION_ERROR` and the user sees errors but isn't automatically signed out - -## Root Cause - -The `refreshAccessToken` function was catching all errors generically and setting `error: "RefreshAccessTokenError"`. When the session callback received this error, it would throw, causing a JWT_SESSION_ERROR but not properly signing the user out. - -## Solution - -### 1. Detect Session Invalidation - -In `refreshAccessToken`, we now specifically detect when Keycloak returns `invalid_grant` with "Session not active": - -```typescript -if (refreshedTokens.error === 'invalid_grant' || - refreshedTokens.error_description?.includes('Session not active') || - refreshedTokens.error_description?.includes('Token is not active')) { - return { - ...token, - error: "SessionNotActive", - }; -} -``` - -### 2. Clear Tokens in JWT Callback - -When we detect `SessionNotActive`, we clear the tokens in the JWT callback: - -```typescript -if (refreshedToken.error === "SessionNotActive") { - return { - ...refreshedToken, - accessToken: undefined, - refreshToken: undefined, - idToken: undefined, - }; -} -``` - -### 3. Return Null in Session Callback - -When tokens are missing or session is invalidated, the session callback returns `null`, which makes NextAuth treat the user as unauthenticated: - -```typescript -if (token.error === "SessionNotActive" || !token.accessToken) { - return null as any; // NextAuth will treat user as unauthenticated -} -``` - -## Result - -Now when a user logs out from an iframe application: -1. Keycloak session is invalidated -2. NextAuth detects the invalid session on next token refresh -3. Tokens are cleared -4. Session callback returns null -5. User is automatically treated as unauthenticated -6. NextAuth redirects to sign-in page (via AuthCheck component) - -## Files Modified - -- `app/api/auth/options.ts`: - - Enhanced `refreshAccessToken` to detect `invalid_grant` errors - - Clear tokens when session is invalidated - - Return null from session callback when session is invalid - -## Testing - -To test this fix: -1. Log in to the dashboard -2. Open an iframe application -3. Log out from the iframe application -4. Wait for NextAuth to try to refresh the token (or trigger a page refresh) -5. User should be automatically signed out and redirected to sign-in - ---- - -**Date**: 2024 -**Status**: ✅ Fixed -**Version**: 1.0 - diff --git a/IMPLEMENTATION_CHECKLIST.md b/IMPLEMENTATION_CHECKLIST.md deleted file mode 100644 index 345950f9..00000000 --- a/IMPLEMENTATION_CHECKLIST.md +++ /dev/null @@ -1,286 +0,0 @@ -# Implementation Checklist: Unified Refresh System - -## 📋 Step-by-Step Implementation Guide - -### Phase 1: Foundation (Day 1) ⚡ CRITICAL - -#### ✅ Step 1.1: Create Refresh Manager -- [ ] Create `lib/services/refresh-manager.ts` -- [ ] Test singleton pattern -- [ ] Test register/unregister -- [ ] Test start/stop -- [ ] Test deduplication logic - -**Estimated Time**: 2-3 hours - ---- - -#### ✅ Step 1.2: Create Request Deduplication -- [ ] Create `lib/utils/request-deduplication.ts` -- [ ] Test deduplication with same key -- [ ] Test TTL expiration -- [ ] Test cleanup - -**Estimated Time**: 1 hour - ---- - -#### ✅ Step 1.3: Create Constants -- [ ] Create `lib/constants/refresh-intervals.ts` -- [ ] Define all intervals -- [ ] Export helper function - -**Estimated Time**: 30 minutes - ---- - -#### ✅ Step 1.4: Create Unified Hook -- [ ] Create `hooks/use-unified-refresh.ts` -- [ ] Test registration on mount -- [ ] Test cleanup on unmount -- [ ] Test manual refresh - -**Estimated Time**: 1-2 hours - ---- - -### Phase 2: Fix Critical Issues (Day 1-2) 🔴 URGENT - -#### ✅ Step 2.1: Fix Redis KEYS → SCAN -- [ ] Update `lib/services/notifications/notification-service.ts` line 293 -- [ ] Replace `redis.keys()` with `redis.scan()` -- [ ] Test with large key sets - -**Estimated Time**: 30 minutes - ---- - -#### ✅ Step 2.2: Fix Notification Hook Memory Leak -- [ ] Fix `hooks/use-notifications.ts` useEffect cleanup -- [ ] Remove function dependencies -- [ ] Test cleanup on unmount - -**Estimated Time**: 1 hour - ---- - -#### ✅ Step 2.3: Fix Notification Badge Double Fetch -- [ ] Update `components/notification-badge.tsx` -- [ ] Remove duplicate useEffect hooks -- [ ] Add request deduplication -- [ ] Test single fetch per action - -**Estimated Time**: 1 hour - ---- - -### Phase 3: Refactor Notifications (Day 2) 🟡 HIGH PRIORITY - -#### ✅ Step 3.1: Refactor useNotifications Hook -- [ ] Integrate unified refresh -- [ ] Add request deduplication -- [ ] Remove manual polling -- [ ] Test all functionality - -**Estimated Time**: 2-3 hours - ---- - -#### ✅ Step 3.2: Update Notification Badge -- [ ] Remove manual fetch logic -- [ ] Use hook's refresh function -- [ ] Test UI interactions - -**Estimated Time**: 1 hour - ---- - -### Phase 4: Refactor Widgets (Day 3-4) 🟢 MEDIUM PRIORITY - -#### ✅ Step 4.1: Refactor Calendar Widget -- [ ] Update `components/calendar.tsx` -- [ ] Use unified refresh hook -- [ ] Add request deduplication -- [ ] Test refresh functionality - -**Estimated Time**: 1 hour - ---- - -#### ✅ Step 4.2: Refactor Parole Widget -- [ ] Update `components/parole.tsx` -- [ ] Use unified refresh hook -- [ ] Remove manual interval -- [ ] Test chat updates - -**Estimated Time**: 1 hour - ---- - -#### ✅ Step 4.3: Refactor News Widget -- [ ] Update `components/news.tsx` -- [ ] Use unified refresh hook -- [ ] Add auto-refresh (was manual only) -- [ ] Test news updates - -**Estimated Time**: 1 hour - ---- - -#### ✅ Step 4.4: Refactor Email Widget -- [ ] Update `components/email.tsx` -- [ ] Use unified refresh hook -- [ ] Add auto-refresh (was manual only) -- [ ] Test email updates - -**Estimated Time**: 1 hour - ---- - -#### ✅ Step 4.5: Refactor Duties Widget -- [ ] Update `components/flow.tsx` -- [ ] Use unified refresh hook -- [ ] Add auto-refresh (was manual only) -- [ ] Test task updates - -**Estimated Time**: 1 hour - ---- - -#### ✅ Step 4.6: Refactor Navigation Bar Time -- [ ] Create `components/main-nav-time.tsx` -- [ ] Update `components/main-nav.tsx` to use new component -- [ ] Use unified refresh hook (1 second interval) -- [ ] Test time updates correctly -- [ ] Verify cleanup on unmount - -**Estimated Time**: 30 minutes - ---- - -### Phase 5: Testing & Validation (Day 5) ✅ FINAL - -#### ✅ Step 5.1: Memory Leak Testing -- [ ] Open DevTools Memory tab -- [ ] Monitor memory over 10 minutes -- [ ] Verify no memory leaks -- [ ] Check interval cleanup - -**Estimated Time**: 1 hour - ---- - -#### ✅ Step 5.2: API Call Reduction Testing -- [ ] Open DevTools Network tab -- [ ] Monitor API calls for 5 minutes -- [ ] Verify deduplication works -- [ ] Count total calls (should be ~60% less) - -**Estimated Time**: 1 hour - ---- - -#### ✅ Step 5.3: Performance Testing -- [ ] Test page load time -- [ ] Test widget refresh times -- [ ] Test with multiple tabs open -- [ ] Verify no performance degradation - -**Estimated Time**: 1 hour - ---- - -#### ✅ Step 5.4: User Experience Testing -- [ ] Test all widgets refresh correctly -- [ ] Test manual refresh buttons -- [ ] Test notification updates -- [ ] Verify smooth UX - -**Estimated Time**: 1 hour - ---- - -## 🎯 Daily Progress Tracking - -### Day 1 Target: -- [x] Phase 1: Foundation (Steps 1.1-1.4) -- [x] Phase 2: Critical Fixes (Steps 2.1-2.3) - -**Status**: ⏳ In Progress - ---- - -### Day 2 Target: -- [ ] Phase 3: Notifications (Steps 3.1-3.2) - -**Status**: ⏸️ Pending - ---- - -### Day 3 Target: -- [ ] Phase 4: Widgets Part 1 (Steps 4.1-4.2) - -**Status**: ⏸️ Pending - ---- - -### Day 4 Target: -- [ ] Phase 4: Widgets Part 2 (Steps 4.3-4.5) - -**Status**: ⏸️ Pending - ---- - -### Day 5 Target: -- [ ] Phase 5: Testing (Steps 5.1-5.4) - -**Status**: ⏸️ Pending - ---- - -## 🐛 Known Issues to Watch For - -1. **Race Conditions**: Monitor for duplicate requests -2. **Memory Leaks**: Watch for uncleaned intervals -3. **Performance**: Monitor API call frequency -4. **User Experience**: Ensure smooth refresh transitions - ---- - -## 📊 Success Criteria - -### Must Have: -- ✅ No memory leaks -- ✅ 60%+ reduction in API calls -- ✅ All widgets refresh correctly -- ✅ No duplicate requests - -### Nice to Have: -- ✅ Configurable refresh intervals -- ✅ Pause/resume functionality -- ✅ Refresh status monitoring -- ✅ Error recovery - ---- - -## 🔄 Rollback Plan - -If issues arise: - -1. **Keep old code**: Don't delete old implementations immediately -2. **Feature flag**: Use environment variable to toggle new/old system -3. **Gradual migration**: Migrate one widget at a time -4. **Monitor**: Watch for errors in production - ---- - -## 📝 Notes - -- All new code should be backward compatible -- Test each phase before moving to next -- Document any deviations from plan -- Update this checklist as you progress - ---- - -*Last Updated: Implementation Checklist v1.0* diff --git a/IMPLEMENTATION_PLAN_UNIFIED_REFRESH.md b/IMPLEMENTATION_PLAN_UNIFIED_REFRESH.md deleted file mode 100644 index c0c1f4a6..00000000 --- a/IMPLEMENTATION_PLAN_UNIFIED_REFRESH.md +++ /dev/null @@ -1,888 +0,0 @@ -# Implementation Plan: Unified Refresh System - -## 🎯 Goals - -1. **Harmonize auto-refresh** across all widgets and notifications -2. **Reduce redundancy** and eliminate duplicate API calls -3. **Improve API efficiency** with request deduplication and caching -4. **Prevent memory leaks** with proper cleanup mechanisms -5. **Centralize refresh logic** for easier maintenance - ---- - -## 📋 Current State Analysis - -### Current Refresh Intervals: -- **Notifications**: 60 seconds (polling) -- **Calendar**: 5 minutes (300000ms) -- **Parole (Chat)**: 30 seconds (30000ms) -- **Navbar Time**: Static (not refreshing - needs fix) -- **News**: Manual only -- **Email**: Manual only -- **Duties (Tasks)**: Manual only - -### Current Problems: -1. ❌ No coordination between widgets -2. ❌ Duplicate API calls from multiple components -3. ❌ Memory leaks from uncleaned intervals -4. ❌ No request deduplication -5. ❌ Inconsistent refresh patterns - ---- - -## 🏗️ Architecture: Unified Refresh System - -### Phase 1: Core Infrastructure - -#### 1.1 Create Unified Refresh Manager - -**File**: `lib/services/refresh-manager.ts` - -```typescript -/** - * Unified Refresh Manager - * Centralizes all refresh logic, prevents duplicates, manages intervals - */ - -export type RefreshableResource = - | 'notifications' - | 'notifications-count' - | 'calendar' - | 'news' - | 'email' - | 'parole' - | 'duties'; - -export interface RefreshConfig { - resource: RefreshableResource; - interval: number; // milliseconds - enabled: boolean; - priority: 'high' | 'medium' | 'low'; - onRefresh: () => Promise; -} - -class RefreshManager { - private intervals: Map = new Map(); - private configs: Map = new Map(); - private pendingRequests: Map> = new Map(); - private lastRefresh: Map = new Map(); - private isActive = false; - - /** - * Register a refreshable resource - */ - register(config: RefreshConfig): void { - this.configs.set(config.resource, config); - - if (config.enabled && this.isActive) { - this.startRefresh(config.resource); - } - } - - /** - * Unregister a resource - */ - unregister(resource: RefreshableResource): void { - this.stopRefresh(resource); - this.configs.delete(resource); - this.lastRefresh.delete(resource); - } - - /** - * Start all refresh intervals - */ - start(): void { - if (this.isActive) return; - - this.isActive = true; - - // Start all enabled resources - this.configs.forEach((config, resource) => { - if (config.enabled) { - this.startRefresh(resource); - } - }); - } - - /** - * Stop all refresh intervals - */ - stop(): void { - this.isActive = false; - - // Clear all intervals - this.intervals.forEach((interval) => { - clearInterval(interval); - }); - - this.intervals.clear(); - } - - /** - * Start refresh for a specific resource - */ - private startRefresh(resource: RefreshableResource): void { - // Stop existing interval if any - this.stopRefresh(resource); - - const config = this.configs.get(resource); - if (!config || !config.enabled) return; - - // Initial refresh - this.executeRefresh(resource); - - // Set up interval - const interval = setInterval(() => { - this.executeRefresh(resource); - }, config.interval); - - this.intervals.set(resource, interval); - } - - /** - * Stop refresh for a specific resource - */ - private stopRefresh(resource: RefreshableResource): void { - const interval = this.intervals.get(resource); - if (interval) { - clearInterval(interval); - this.intervals.delete(resource); - } - } - - /** - * Execute refresh with deduplication - */ - private async executeRefresh(resource: RefreshableResource): Promise { - const config = this.configs.get(resource); - if (!config) return; - - const requestKey = `${resource}-${Date.now()}`; - const now = Date.now(); - const lastRefreshTime = this.lastRefresh.get(resource) || 0; - - // Prevent too frequent refreshes (minimum 1 second between same resource) - if (now - lastRefreshTime < 1000) { - console.log(`[RefreshManager] Skipping ${resource} - too soon`); - return; - } - - // Check if there's already a pending request for this resource - const pendingKey = `${resource}-pending`; - if (this.pendingRequests.has(pendingKey)) { - console.log(`[RefreshManager] Deduplicating ${resource} request`); - return; - } - - // Create and track the request - const refreshPromise = config.onRefresh() - .then(() => { - this.lastRefresh.set(resource, Date.now()); - }) - .catch((error) => { - console.error(`[RefreshManager] Error refreshing ${resource}:`, error); - }) - .finally(() => { - this.pendingRequests.delete(pendingKey); - }); - - this.pendingRequests.set(pendingKey, refreshPromise); - - try { - await refreshPromise; - } catch (error) { - // Error already logged above - } - } - - /** - * Manually trigger refresh for a resource - */ - async refresh(resource: RefreshableResource, force = false): Promise { - const config = this.configs.get(resource); - if (!config) { - throw new Error(`Resource ${resource} not registered`); - } - - if (force) { - // Force refresh: clear last refresh time - this.lastRefresh.delete(resource); - } - - await this.executeRefresh(resource); - } - - /** - * Get refresh status - */ - getStatus(): { - active: boolean; - resources: Array<{ - resource: RefreshableResource; - enabled: boolean; - lastRefresh: number | null; - interval: number; - }>; - } { - const resources = Array.from(this.configs.entries()).map(([resource, config]) => ({ - resource, - enabled: config.enabled, - lastRefresh: this.lastRefresh.get(resource) || null, - interval: config.interval, - })); - - return { - active: this.isActive, - resources, - }; - } -} - -// Singleton instance -export const refreshManager = new RefreshManager(); -``` - ---- - -#### 1.2 Create Request Deduplication Utility - -**File**: `lib/utils/request-deduplication.ts` - -```typescript -/** - * Request Deduplication Utility - * Prevents duplicate API calls for the same resource - */ - -interface PendingRequest { - promise: Promise; - timestamp: number; -} - -class RequestDeduplicator { - private pendingRequests = new Map>(); - private readonly DEFAULT_TTL = 5000; // 5 seconds - - /** - * Execute a request with deduplication - */ - async execute( - key: string, - requestFn: () => Promise, - ttl: number = this.DEFAULT_TTL - ): Promise { - // Check if there's a pending request - const pending = this.pendingRequests.get(key); - - if (pending) { - const age = Date.now() - pending.timestamp; - - // If request is still fresh, reuse it - if (age < ttl) { - console.log(`[RequestDeduplicator] Reusing pending request: ${key}`); - return pending.promise; - } else { - // Request is stale, remove it - this.pendingRequests.delete(key); - } - } - - // Create new request - const promise = requestFn() - .finally(() => { - // Clean up after request completes - this.pendingRequests.delete(key); - }); - - this.pendingRequests.set(key, { - promise, - timestamp: Date.now(), - }); - - return promise; - } - - /** - * Cancel a pending request - */ - cancel(key: string): void { - this.pendingRequests.delete(key); - } - - /** - * Clear all pending requests - */ - clear(): void { - this.pendingRequests.clear(); - } - - /** - * Get pending requests count - */ - getPendingCount(): number { - return this.pendingRequests.size; - } -} - -export const requestDeduplicator = new RequestDeduplicator(); -``` - ---- - -#### 1.3 Create Unified Refresh Hook - -**File**: `hooks/use-unified-refresh.ts` - -```typescript -/** - * Unified Refresh Hook - * Provides consistent refresh functionality for all widgets - */ - -import { useEffect, useCallback, useRef } from 'react'; -import { useSession } from 'next-auth/react'; -import { refreshManager, RefreshableResource } from '@/lib/services/refresh-manager'; - -interface UseUnifiedRefreshOptions { - resource: RefreshableResource; - interval: number; - enabled?: boolean; - onRefresh: () => Promise; - priority?: 'high' | 'medium' | 'low'; -} - -export function useUnifiedRefresh({ - resource, - interval, - enabled = true, - onRefresh, - priority = 'medium', -}: UseUnifiedRefreshOptions) { - const { status } = useSession(); - const onRefreshRef = useRef(onRefresh); - const isMountedRef = useRef(true); - - // Update callback ref when it changes - useEffect(() => { - onRefreshRef.current = onRefresh; - }, [onRefresh]); - - // Register/unregister with refresh manager - useEffect(() => { - if (status !== 'authenticated' || !enabled) { - return; - } - - isMountedRef.current = true; - - // Register with refresh manager - refreshManager.register({ - resource, - interval, - enabled: true, - priority, - onRefresh: async () => { - if (isMountedRef.current) { - await onRefreshRef.current(); - } - }, - }); - - // Start refresh manager if not already started - refreshManager.start(); - - // Cleanup - return () => { - isMountedRef.current = false; - refreshManager.unregister(resource); - }; - }, [resource, interval, enabled, priority, status]); - - // Manual refresh function - const refresh = useCallback( - async (force = false) => { - if (status !== 'authenticated') return; - await refreshManager.refresh(resource, force); - }, - [resource, status] - ); - - return { - refresh, - isActive: refreshManager.getStatus().active, - }; -} -``` - ---- - -### Phase 2: Harmonized Refresh Intervals - -#### 2.1 Define Standard Intervals - -**File**: `lib/constants/refresh-intervals.ts` - -```typescript -/** - * Standard Refresh Intervals - * All intervals in milliseconds - */ - -export const REFRESH_INTERVALS = { - // High priority - real-time updates - NOTIFICATIONS: 30000, // 30 seconds (was 60s) - NOTIFICATIONS_COUNT: 30000, // 30 seconds (same as notifications) - PAROLE: 30000, // 30 seconds (unchanged) - NAVBAR_TIME: 1000, // 1 second (navigation bar time - real-time) - - // Medium priority - frequent but not real-time - EMAIL: 60000, // 1 minute (was manual only) - DUTIES: 120000, // 2 minutes (was manual only) - - // Low priority - less frequent updates - CALENDAR: 300000, // 5 minutes (unchanged) - NEWS: 600000, // 10 minutes (was manual only) - - // Minimum interval between refreshes (prevents spam) - MIN_INTERVAL: 1000, // 1 second -} as const; - -/** - * Get refresh interval for a resource - */ -export function getRefreshInterval(resource: string): number { - switch (resource) { - case 'notifications': - return REFRESH_INTERVALS.NOTIFICATIONS; - case 'notifications-count': - return REFRESH_INTERVALS.NOTIFICATIONS_COUNT; - case 'parole': - return REFRESH_INTERVALS.PAROLE; - case 'email': - return REFRESH_INTERVALS.EMAIL; - case 'duties': - return REFRESH_INTERVALS.DUTIES; - case 'calendar': - return REFRESH_INTERVALS.CALENDAR; - case 'news': - return REFRESH_INTERVALS.NEWS; - default: - return 60000; // Default: 1 minute - } -} -``` - ---- - -### Phase 3: Refactor Widgets - -#### 3.1 Refactor Notification Hook - -**File**: `hooks/use-notifications.ts` (Refactored) - -```typescript -import { useState, useEffect, useCallback, useRef } from 'react'; -import { useSession } from 'next-auth/react'; -import { Notification, NotificationCount } from '@/lib/types/notification'; -import { useUnifiedRefresh } from './use-unified-refresh'; -import { REFRESH_INTERVALS } from '@/lib/constants/refresh-intervals'; -import { requestDeduplicator } from '@/lib/utils/request-deduplication'; - -const defaultNotificationCount: NotificationCount = { - total: 0, - unread: 0, - sources: {}, -}; - -export function useNotifications() { - const { data: session, status } = useSession(); - const [notifications, setNotifications] = useState([]); - const [notificationCount, setNotificationCount] = useState(defaultNotificationCount); - const [loading, setLoading] = useState(false); - const [error, setError] = useState(null); - const isMountedRef = useRef(true); - - // Fetch notification count - const fetchNotificationCount = useCallback(async () => { - if (!session?.user || !isMountedRef.current) return; - - try { - setError(null); - - const data = await requestDeduplicator.execute( - `notifications-count-${session.user.id}`, - async () => { - const response = await fetch('/api/notifications/count', { - credentials: 'include', - }); - - if (!response.ok) { - throw new Error('Failed to fetch notification count'); - } - - return response.json(); - } - ); - - if (isMountedRef.current) { - setNotificationCount(data); - } - } catch (err) { - console.error('Error fetching notification count:', err); - if (isMountedRef.current) { - setError('Failed to fetch notification count'); - } - } - }, [session?.user]); - - // Fetch notifications - const fetchNotifications = useCallback(async (page = 1, limit = 20) => { - if (!session?.user || !isMountedRef.current) return; - - setLoading(true); - setError(null); - - try { - const data = await requestDeduplicator.execute( - `notifications-${session.user.id}-${page}-${limit}`, - async () => { - const response = await fetch(`/api/notifications?page=${page}&limit=${limit}`, { - credentials: 'include', - }); - - if (!response.ok) { - throw new Error('Failed to fetch notifications'); - } - - return response.json(); - } - ); - - if (isMountedRef.current) { - setNotifications(data.notifications); - } - } catch (err) { - console.error('Error fetching notifications:', err); - if (isMountedRef.current) { - setError('Failed to fetch notifications'); - } - } finally { - if (isMountedRef.current) { - setLoading(false); - } - } - }, [session?.user]); - - // Use unified refresh for notification count - useUnifiedRefresh({ - resource: 'notifications-count', - interval: REFRESH_INTERVALS.NOTIFICATIONS_COUNT, - enabled: status === 'authenticated', - onRefresh: fetchNotificationCount, - priority: 'high', - }); - - // Initial fetch - useEffect(() => { - isMountedRef.current = true; - - if (status === 'authenticated' && session?.user) { - fetchNotificationCount(); - fetchNotifications(); - } - - return () => { - isMountedRef.current = false; - }; - }, [status, session?.user, fetchNotificationCount, fetchNotifications]); - - // Mark as read - const markAsRead = useCallback(async (notificationId: string) => { - if (!session?.user) return false; - - try { - const response = await fetch(`/api/notifications/${notificationId}/read`, { - method: 'POST', - headers: { 'Content-Type': 'application/json' }, - credentials: 'include', - }); - - if (!response.ok) return false; - - setNotifications(prev => - prev.map(n => n.id === notificationId ? { ...n, isRead: true } : n) - ); - - await fetchNotificationCount(); - return true; - } catch (err) { - console.error('Error marking notification as read:', err); - return false; - } - }, [session?.user, fetchNotificationCount]); - - // Mark all as read - const markAllAsRead = useCallback(async () => { - if (!session?.user) return false; - - try { - const response = await fetch('/api/notifications/read-all', { - method: 'POST', - headers: { 'Content-Type': 'application/json' }, - credentials: 'include', - }); - - if (!response.ok) return false; - - setNotifications(prev => prev.map(n => ({ ...n, isRead: true }))); - await fetchNotificationCount(); - return true; - } catch (err) { - console.error('Error marking all notifications as read:', err); - return false; - } - }, [session?.user, fetchNotificationCount]); - - return { - notifications, - notificationCount, - loading, - error, - fetchNotifications, - fetchNotificationCount, - markAsRead, - markAllAsRead, - }; -} -``` - ---- - -#### 3.2 Refactor Widget Components - -**Example: Calendar Widget** - -**File**: `components/calendar.tsx` (Refactored) - -```typescript -"use client"; - -import { useEffect, useState } from "react"; -import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card"; -import { Button } from "@/components/ui/button"; -import { RefreshCw, Calendar as CalendarIcon } from "lucide-react"; -import { useUnifiedRefresh } from '@/hooks/use-unified-refresh'; -import { REFRESH_INTERVALS } from '@/lib/constants/refresh-intervals'; -import { requestDeduplicator } from '@/lib/utils/request-deduplication'; -import { useSession } from 'next-auth/react'; - -interface Event { - id: string; - title: string; - start: string; - end: string; - allDay: boolean; - calendar: string; - calendarColor: string; -} - -export function Calendar() { - const { status } = useSession(); - const [events, setEvents] = useState([]); - const [loading, setLoading] = useState(true); - const [error, setError] = useState(null); - - const fetchEvents = async () => { - if (status !== 'authenticated') return; - - setLoading(true); - setError(null); - - try { - const calendarsData = await requestDeduplicator.execute( - 'calendar-events', - async () => { - const response = await fetch('/api/calendars?refresh=true'); - if (!response.ok) { - throw new Error('Failed to fetch events'); - } - return response.json(); - } - ); - - const now = new Date(); - now.setHours(0, 0, 0, 0); - - const allEvents = calendarsData.flatMap((calendar: any) => - (calendar.events || []).map((event: any) => ({ - id: event.id, - title: event.title, - start: event.start, - end: event.end, - allDay: event.isAllDay, - calendar: calendar.name, - calendarColor: calendar.color - })) - ); - - const upcomingEvents = allEvents - .filter((event: any) => new Date(event.start) >= now) - .sort((a: any, b: any) => new Date(a.start).getTime() - new Date(b.start).getTime()) - .slice(0, 7); - - setEvents(upcomingEvents); - } catch (err) { - console.error('Error fetching events:', err); - setError('Failed to load events'); - } finally { - setLoading(false); - } - }; - - // Use unified refresh - const { refresh } = useUnifiedRefresh({ - resource: 'calendar', - interval: REFRESH_INTERVALS.CALENDAR, - enabled: status === 'authenticated', - onRefresh: fetchEvents, - priority: 'low', - }); - - // Initial fetch - useEffect(() => { - if (status === 'authenticated') { - fetchEvents(); - } - }, [status]); - - // ... rest of component (formatDate, formatTime, render) - - return ( - - - Agenda - - - {/* ... */} - - ); -} -``` - ---- - -### Phase 4: Implementation Steps - -#### Step 1: Create Core Infrastructure (Day 1) - -1. ✅ Create `lib/services/refresh-manager.ts` -2. ✅ Create `lib/utils/request-deduplication.ts` -3. ✅ Create `lib/constants/refresh-intervals.ts` -4. ✅ Create `hooks/use-unified-refresh.ts` - -**Testing**: Unit tests for each module - ---- - -#### Step 2: Fix Memory Leaks (Day 1-2) - -1. ✅ Fix `useNotifications` hook cleanup -2. ✅ Fix notification badge double fetching -3. ✅ Fix widget interval cleanup -4. ✅ Fix Redis KEYS → SCAN - -**Testing**: Memory leak detection in DevTools - ---- - -#### Step 3: Refactor Notifications (Day 2) - -1. ✅ Refactor `hooks/use-notifications.ts` -2. ✅ Update `components/notification-badge.tsx` -3. ✅ Remove duplicate fetch logic - -**Testing**: Verify no duplicate API calls - ---- - -#### Step 4: Refactor Widgets (Day 3-4) - -1. ✅ Refactor `components/calendar.tsx` -2. ✅ Refactor `components/parole.tsx` -3. ✅ Refactor `components/news.tsx` -4. ✅ Refactor `components/email.tsx` -5. ✅ Refactor `components/flow.tsx` (Duties) -6. ✅ Refactor `components/main-nav.tsx` (Time display) - -**Testing**: Verify all widgets refresh correctly - ---- - -#### Step 5: Testing & Optimization (Day 5) - -1. ✅ Performance testing -2. ✅ Memory leak verification -3. ✅ API call reduction verification -4. ✅ User experience testing - ---- - -## 📊 Expected Improvements - -### Before: -- **API Calls**: ~120-150 calls/minute (with duplicates) -- **Memory Leaks**: Yes (intervals not cleaned up) -- **Refresh Coordination**: None -- **Request Deduplication**: None - -### After: -- **API Calls**: ~40-50 calls/minute (60-70% reduction) -- **Memory Leaks**: None (proper cleanup) -- **Refresh Coordination**: Centralized -- **Request Deduplication**: Full coverage - ---- - -## 🎯 Success Metrics - -1. **API Call Reduction**: 60%+ reduction in duplicate calls -2. **Memory Usage**: No memory leaks detected -3. **Performance**: Faster page loads, smoother UX -4. **Maintainability**: Single source of truth for refresh logic - ---- - -## 🚀 Quick Start Implementation - -### Priority Order: - -1. **Critical** (Do First): - - Fix memory leaks - - Create refresh manager - - Create request deduplication - -2. **High** (Do Second): - - Refactor notifications - - Refactor high-frequency widgets (parole, notifications) - -3. **Medium** (Do Third): - - Refactor medium-frequency widgets (email, duties) - -4. **Low** (Do Last): - - Refactor low-frequency widgets (calendar, news) - ---- - -## 📝 Notes - -- All intervals are configurable via constants -- Refresh manager can be paused/resumed globally -- Request deduplication prevents duplicate calls within 5 seconds -- All cleanup is handled automatically -- Compatible with existing code (gradual migration) - ---- - -*Implementation Plan v1.0* diff --git a/IMPLEMENTATION_SUMMARY.md b/IMPLEMENTATION_SUMMARY.md deleted file mode 100644 index 8058f5f6..00000000 --- a/IMPLEMENTATION_SUMMARY.md +++ /dev/null @@ -1,142 +0,0 @@ -# Implementation Summary - Logging Improvements - -**Date**: 2026-01-01 -**Status**: ✅ Completed - ---- - -## 📋 What Was Done - -### 1. Impact Analysis Completed ✅ - -**File Created**: `SESSION_CALLBACK_LOGGING_IMPACT_ANALYSIS.md` - -**Key Findings**: -- ✅ **No functional impact** on Keycloak, MinIO, or external services -- ✅ **Logging was added for debugging**, not functionality -- ✅ **Error logging will be preserved** (critical for troubleshooting) -- ✅ **Conditional logging recommended** (DEBUG_SESSION flag) - -**Recommendation**: -- Proceed with conditional logging using `DEBUG_SESSION` environment variable -- Keep error logging always enabled -- Make success logging conditional - ---- - -### 2. Mark-as-Read Logging Added ✅ - -**Files Modified**: -1. `app/api/notifications/[id]/read/route.ts` -2. `app/api/notifications/read-all/route.ts` - -**Logging Added**: -- ✅ Entry logging (when endpoint is called) -- ✅ Authentication status logging -- ✅ User ID and notification ID logging -- ✅ Success/failure logging with duration -- ✅ Error logging with stack traces -- ✅ Timestamp logging - -**Log Format**: -``` -[NOTIFICATION_API] Mark as read endpoint called -[NOTIFICATION_API] Mark as read - Processing { userId, notificationId, timestamp } -[NOTIFICATION_API] Mark as read - Success { userId, notificationId, duration } -``` - ---- - -## 🔍 What to Look For in Logs - -### When Mark-as-Read is Called - -**Expected Logs**: -``` -[NOTIFICATION_API] Mark as read endpoint called -[NOTIFICATION_API] Mark as read - Processing { userId: "...", notificationId: "...", timestamp: "..." } -[NOTIFICATION_API] Mark as read - Success { userId: "...", notificationId: "...", duration: "Xms" } -``` - -**If Authentication Fails**: -``` -[NOTIFICATION_API] Mark as read - Authentication failed -``` - -**If Operation Fails**: -``` -[NOTIFICATION_API] Mark as read - Failed { userId: "...", notificationId: "...", duration: "Xms" } -``` - -**If Error Occurs**: -``` -[NOTIFICATION_API] Mark as read - Error { error: "...", stack: "...", duration: "Xms" } -``` - ---- - -## 📊 Next Steps - -### Immediate (Ready to Test) - -1. **Test Mark-as-Read Functionality** - - Mark a single notification as read - - Mark all notifications as read - - Check logs for the new logging statements - - Verify notification count updates correctly - -2. **Monitor Logs** - - Watch for `[NOTIFICATION_API]` log entries - - Verify timing information - - Check for any errors - -### Future (When Ready) - -3. **Implement Conditional Session Callback Logging** - - Add `DEBUG_SESSION` environment variable support - - Update `app/api/auth/options.ts` - - Test in development and production - - Document in README - ---- - -## 🎯 Testing Checklist - -- [ ] Mark single notification as read → Check logs -- [ ] Mark all notifications as read → Check logs -- [ ] Verify notification count updates -- [ ] Check for any errors in logs -- [ ] Verify performance (duration logging) -- [ ] Test with invalid notification ID -- [ ] Test without authentication - ---- - -## 📝 Files Changed - -1. ✅ `app/api/notifications/[id]/read/route.ts` - Added comprehensive logging -2. ✅ `app/api/notifications/read-all/route.ts` - Added comprehensive logging -3. ✅ `SESSION_CALLBACK_LOGGING_IMPACT_ANALYSIS.md` - Created impact analysis -4. ✅ `IMPLEMENTATION_SUMMARY.md` - This file - ---- - -## 🔧 Environment Variables - -**No new environment variables required** for mark-as-read logging. - -**Future**: `DEBUG_SESSION` will be needed for conditional session callback logging (not implemented yet). - ---- - -## ✅ Status - -**Mark-as-Read Logging**: ✅ **COMPLETE** -**Session Callback Impact Analysis**: ✅ **COMPLETE** -**Session Callback Conditional Logging**: ⏳ **PENDING** (awaiting approval) - ---- - -**Generated**: 2026-01-01 -**Ready for Testing**: ✅ Yes - diff --git a/IMPROVEMENTS_LOGIN_FLOW.md b/IMPROVEMENTS_LOGIN_FLOW.md deleted file mode 100644 index 24d8d0b7..00000000 --- a/IMPROVEMENTS_LOGIN_FLOW.md +++ /dev/null @@ -1,684 +0,0 @@ -# Améliorations du Flow de Connexion - Recommandations - -## 📋 Vue d'ensemble - -Ce document propose des améliorations concrètes pour corriger et optimiser le flow de connexion/déconnexion du dashboard Next.js avec NextAuth et Keycloak. - ---- - -## 🎯 Problèmes Identifiés et Solutions - -### Problème 1 : `prompt=login` toujours actif - Empêche SSO naturel - -**Situation actuelle** : -```typescript -// app/api/auth/options.ts ligne 154 -authorization: { - params: { - scope: "openid profile email roles", - prompt: "login" // ⚠️ TOUJOURS actif - } -} -``` - -**Impact** : -- ❌ L'utilisateur doit **toujours** saisir ses identifiants, même lors de la première visite -- ❌ Empêche l'expérience SSO naturelle -- ❌ Mauvaise UX pour les utilisateurs légitimes - -**Solution recommandée** : Gérer `prompt=login` conditionnellement - -```typescript -// app/api/auth/options.ts -authorization: { - params: { - scope: "openid profile email roles", - // Ne pas forcer prompt=login par défaut - // prompt: "login" // ❌ À SUPPRIMER - } -} -``` - -**ET** : Ajouter `prompt=login` uniquement après un logout explicite - -```typescript -// Dans signIn() après logout -signIn("keycloak", { - callbackUrl: "/", - // Ajouter prompt=login uniquement si logout récent - ...(shouldForceLogin ? { prompt: "login" } : {}) -}); -``` - ---- - -### Problème 2 : Détection session invalide trop complexe et fragile - -**Situation actuelle** : -- Logique complexe dans `app/signin/page.tsx` (lignes 17-67) -- Vérification multiple de cookies, sessionStorage, URL params -- Race conditions possibles -- Auto-login peut se déclencher incorrectement - -**Solution recommandée** : Simplifier avec un flag serveur - -#### Option A : Utiliser un cookie serveur pour marquer le logout - -```typescript -// app/api/auth/end-sso-session/route.ts -export async function POST(request: NextRequest) { - // ... code existant ... - - // Après logout réussi, créer un cookie de flag - const response = NextResponse.json({ success: true }); - response.cookies.set('force_login_prompt', 'true', { - httpOnly: true, - secure: process.env.NODE_ENV === 'production', - sameSite: 'lax', - path: '/', - maxAge: 300 // 5 minutes - }); - - return response; -} -``` - -#### Option B : Utiliser un paramètre d'état dans l'URL Keycloak - -```typescript -// Dans signout-handler.tsx, ajouter un paramètre state -const keycloakLogoutUrl = new URL( - `${keycloakIssuer}/protocol/openid-connect/logout` -); -keycloakLogoutUrl.searchParams.append('state', 'force_login'); -// Keycloak renverra ce state dans le redirect -``` - -**Simplification de signin/page.tsx** : - -```typescript -// Vérifier le cookie serveur au lieu de logique complexe -const forceLoginCookie = document.cookie - .split(';') - .find(c => c.trim().startsWith('force_login_prompt=')); - -if (forceLoginCookie) { - // Supprimer le cookie - document.cookie = 'force_login_prompt=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;'; - // Forcer prompt=login - signIn("keycloak", { - callbackUrl: "/", - // Ajouter prompt via custom params si possible - }); -} -``` - ---- - -### Problème 3 : Cookies Keycloak non supprimables (domaine différent) - -**Situation actuelle** : -- `clearKeycloakCookies()` échoue si Keycloak est sur un domaine différent -- Les cookies Keycloak persistent après logout -- Session SSO peut persister - -**Solutions recommandées** : - -#### Solution 1 : Utiliser Keycloak logout endpoint avec tous les paramètres - -```typescript -// Améliorer le logout URL dans signout-handler.tsx -const keycloakLogoutUrl = new URL( - `${keycloakIssuer}/protocol/openid-connect/logout` -); - -// Paramètres essentiels -keycloakLogoutUrl.searchParams.append('post_logout_redirect_uri', - window.location.origin + '/signin?logout=true'); -keycloakLogoutUrl.searchParams.append('id_token_hint', idToken); - -// ✅ AJOUTER ces paramètres pour forcer la suppression SSO -keycloakLogoutUrl.searchParams.append('kc_action', 'LOGOUT'); // Déjà présent -keycloakLogoutUrl.searchParams.append('logout_hint', 'true'); // Nouveau - -// Si possible, utiliser le client_id pour forcer logout client -if (process.env.NEXT_PUBLIC_KEYCLOAK_CLIENT_ID) { - keycloakLogoutUrl.searchParams.append('client_id', - process.env.NEXT_PUBLIC_KEYcloak_CLIENT_ID); -} -``` - -#### Solution 2 : Utiliser Admin API pour terminer TOUTES les sessions - -**Améliorer** `app/api/auth/end-sso-session/route.ts` : - -```typescript -// Au lieu de logout({ id: userId }), utiliser logoutAllSessions -try { - // Option 1 : Logout toutes les sessions de l'utilisateur - await adminClient.users.logout({ id: userId }); - - // Option 2 : Si disponible, utiliser une méthode plus agressive - // Note: Vérifier la version de Keycloak Admin Client - // Certaines versions supportent logoutAllSessions - - // Option 3 : Invalider les refresh tokens - const userSessions = await adminClient.users.listSessions({ id: userId }); - for (const session of userSessions) { - await adminClient.users.logoutSession({ - id: userId, - sessionId: session.id - }); - } -} catch (error) { - // ... gestion erreur -} -``` - -#### Solution 3 : Configurer Keycloak pour SameSite=None (si cross-domain) - -**Configuration Keycloak** (à faire côté Keycloak) : -``` -Cookie SameSite: None -Cookie Secure: true -Cookie Domain: .example.com (domaine parent partagé) -``` - -**Puis** améliorer `clearKeycloakCookies()` : - -```typescript -// lib/session.ts -export function clearKeycloakCookies() { - const keycloakIssuer = process.env.NEXT_PUBLIC_KEYCLOAK_ISSUER; - if (!keycloakIssuer) return; - - try { - const keycloakUrl = new URL(keycloakIssuer); - const keycloakDomain = keycloakUrl.hostname; - - // Extraire le domaine parent si possible - const domainParts = keycloakDomain.split('.'); - const parentDomain = domainParts.length > 2 - ? '.' + domainParts.slice(-2).join('.') - : keycloakDomain; - - const keycloakCookieNames = [ - 'KEYCLOAK_SESSION', - 'KEYCLOAK_SESSION_LEGACY', - 'KEYCLOAK_IDENTITY', - 'KEYCLOAK_IDENTITY_LEGACY', - 'AUTH_SESSION_ID', - 'KC_RESTART', - 'KC_RESTART_LEGACY' - ]; - - // Essayer avec domaine parent (pour SameSite=None) - keycloakCookieNames.forEach(cookieName => { - // Avec domaine parent - document.cookie = `${cookieName}=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/; domain=${parentDomain}; SameSite=None; Secure;`; - // Sans domaine (same-origin) - document.cookie = `${cookieName}=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/; SameSite=None; Secure;`; - // Avec path spécifique - document.cookie = `${cookieName}=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/realms/; domain=${parentDomain}; SameSite=None; Secure;`; - }); - } catch (error) { - console.error('Error clearing Keycloak cookies:', error); - } -} -``` - ---- - -### Problème 4 : Race condition entre logout et auto-login - -**Situation actuelle** : -- Auto-login avec délai de 1 seconde -- Peut se déclencher pendant le flow de logout -- Flags `logout_in_progress` et `session_invalidated` peuvent être perdus - -**Solution recommandée** : Utiliser un mécanisme plus robuste - -#### Option A : Utiliser un cookie HttpOnly pour le flag - -```typescript -// Créer une route API pour marquer le logout -// app/api/auth/mark-logout/route.ts -export async function POST() { - const response = NextResponse.json({ success: true }); - response.cookies.set('logout_completed', 'true', { - httpOnly: true, - secure: process.env.NODE_ENV === 'production', - sameSite: 'lax', - path: '/', - maxAge: 300 // 5 minutes - }); - return response; -} - -// Dans signout-handler.tsx -await fetch('/api/auth/mark-logout', { method: 'POST' }); - -// Dans signin/page.tsx -const logoutCompleted = document.cookie - .split(';') - .some(c => c.trim().startsWith('logout_completed=')); - -if (logoutCompleted) { - // Supprimer le cookie - document.cookie = 'logout_completed=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;'; - // Ne pas auto-login - return; -} -``` - -#### Option B : Utiliser un état dans l'URL Keycloak callback - -```typescript -// Lors du logout, ajouter un paramètre state -const logoutState = btoa(JSON.stringify({ - logout: true, - timestamp: Date.now() -})); - -// Keycloak renverra ce state dans le redirect -// Dans signin/page.tsx, vérifier le state -const urlParams = new URLSearchParams(window.location.search); -const state = urlParams.get('state'); -if (state) { - try { - const stateData = JSON.parse(atob(state)); - if (stateData.logout) { - // Ne pas auto-login - return; - } - } catch (e) { - // State invalide, ignorer - } -} -``` - ---- - -### Problème 5 : Configuration cookies NextAuth non explicite - -**Situation actuelle** : -- Pas de configuration explicite des cookies NextAuth -- Utilise les valeurs par défaut -- Pas de contrôle sur SameSite, Secure, etc. - -**Solution recommandée** : Configurer explicitement les cookies - -```typescript -// app/api/auth/options.ts -export const authOptions: NextAuthOptions = { - // ... providers ... - - // ✅ AJOUTER configuration explicite des cookies - cookies: { - sessionToken: { - name: `next-auth.session-token`, - options: { - httpOnly: true, - sameSite: 'lax', - path: '/', - secure: process.env.NEXTAUTH_URL?.startsWith('https://') ?? false, - // Domaine explicite si nécessaire - // domain: process.env.COOKIE_DOMAIN, - }, - }, - callbackUrl: { - name: `next-auth.callback-url`, - options: { - httpOnly: true, - sameSite: 'lax', - path: '/', - secure: process.env.NEXTAUTH_URL?.startsWith('https://') ?? false, - }, - }, - csrfToken: { - name: `next-auth.csrf-token`, - options: { - httpOnly: true, - sameSite: 'lax', - path: '/', - secure: process.env.NEXTAUTH_URL?.startsWith('https://') ?? false, - }, - }, - state: { - name: `next-auth.state`, - options: { - httpOnly: true, - sameSite: 'lax', - path: '/', - secure: process.env.NEXTAUTH_URL?.startsWith('https://') ?? false, - }, - }, - }, - - // ... reste de la config ... -}; -``` - -**Avantages** : -- Contrôle total sur les cookies -- Peut ajuster SameSite pour cross-domain si nécessaire -- Meilleure sécurité - ---- - -### Problème 6 : Détection session invalide côté client uniquement - -**Situation actuelle** : -- Détection session invalide uniquement côté client -- Vérification de cookies via `document.cookie` -- Peut être contourné ou mal interprété - -**Solution recommandée** : Détection serveur + client - -#### Créer une route API pour vérifier l'état de session - -```typescript -// app/api/auth/session-status/route.ts -import { NextRequest, NextResponse } from 'next/server'; -import { getServerSession } from 'next-auth/next'; -import { authOptions } from '../options'; - -export async function GET(request: NextRequest) { - const session = await getServerSession(authOptions); - - // Vérifier si la session existe mais est invalide - const hasSessionCookie = request.cookies.has('next-auth.session-token') || - request.cookies.has('__Secure-next-auth.session-token') || - request.cookies.has('__Host-next-auth.session-token'); - - return NextResponse.json({ - hasSession: !!session, - hasSessionCookie, - isInvalid: hasSessionCookie && !session, // Cookie existe mais session invalide - shouldForceLogin: request.cookies.get('force_login_prompt')?.value === 'true', - }); -} -``` - -#### Utiliser cette route dans signin/page.tsx - -```typescript -// app/signin/page.tsx -useEffect(() => { - const checkSessionStatus = async () => { - const response = await fetch('/api/auth/session-status'); - const status = await response.json(); - - if (status.isInvalid) { - // Session invalidée, ne pas auto-login - sessionStorage.setItem('session_invalidated', 'true'); - return; - } - - if (status.shouldForceLogin) { - // Forcer prompt=login - signIn("keycloak", { - callbackUrl: "/", - // prompt: "login" via custom params si possible - }); - return; - } - - // Nouvel utilisateur, auto-login OK - if (status === "unauthenticated" && !status.hasSessionCookie) { - signIn("keycloak", { callbackUrl: "/" }); - } - }; - - checkSessionStatus(); -}, []); -``` - ---- - -### Problème 7 : Gestion d'erreur token refresh insuffisante - -**Situation actuelle** : -- `refreshAccessToken()` détecte `SessionNotActive` -- Retourne `error: "SessionNotActive"` -- Mais la gestion peut être améliorée - -**Solution recommandée** : Améliorer la gestion d'erreur - -```typescript -// app/api/auth/options.ts -async function refreshAccessToken(token: ExtendedJWT) { - try { - const response = await fetch(`${process.env.KEYCLOAK_ISSUER}/protocol/openid-connect/token`, { - headers: { "Content-Type": "application/x-www-form-urlencoded" }, - body: new URLSearchParams({ - client_id: process.env.KEYCLOAK_CLIENT_ID!, - client_secret: process.env.KEYCLOAK_CLIENT_SECRET!, - grant_type: "refresh_token", - refresh_token: token.refreshToken || '', - }), - method: "POST", - }); - - const refreshedTokens = await response.json(); - - if (!response.ok) { - // ✅ AMÉLIORATION : Détecter différents types d'erreurs - const errorType = refreshedTokens.error; - const errorDescription = refreshedTokens.error_description || ''; - - // Session invalide (logout depuis iframe ou Keycloak) - if (errorType === 'invalid_grant' || - errorDescription.includes('Session not active') || - errorDescription.includes('Token is not active') || - errorDescription.includes('Session expired')) { - console.log("Keycloak session invalidated, marking for removal"); - return { - ...token, - error: "SessionNotActive", - // ✅ Supprimer tous les tokens - accessToken: undefined, - refreshToken: undefined, - idToken: undefined, - }; - } - - // Refresh token expiré (inactivité prolongée) - if (errorType === 'invalid_grant' && - errorDescription.includes('Refresh token expired')) { - console.log("Refresh token expired, user needs to re-authenticate"); - return { - ...token, - error: "RefreshTokenExpired", - accessToken: undefined, - refreshToken: undefined, - idToken: undefined, - }; - } - - // Autre erreur - throw refreshedTokens; - } - - return { - ...token, - accessToken: refreshedTokens.access_token, - refreshToken: refreshedTokens.refresh_token ?? token.refreshToken, - idToken: token.idToken, // Keycloak ne renvoie pas de nouvel ID token - accessTokenExpires: Date.now() + refreshedTokens.expires_in * 1000, - error: undefined, // ✅ Clear any previous errors - }; - } catch (error: any) { - console.error("Error refreshing access token:", error); - - // ✅ AMÉLIORATION : Gestion d'erreur plus robuste - if (error?.error === 'invalid_grant' || - error?.error_description?.includes('Session not active') || - error?.error_description?.includes('Token is not active')) { - return { - ...token, - error: "SessionNotActive", - accessToken: undefined, - refreshToken: undefined, - idToken: undefined, - }; - } - - return { - ...token, - error: "RefreshAccessTokenError", - accessToken: undefined, - refreshToken: undefined, - idToken: undefined, - }; - } -} -``` - ---- - -## 🚀 Plan d'Implémentation Recommandé - -### Phase 1 : Corrections Critiques (Priorité Haute) - -1. **Supprimer `prompt=login` par défaut** - - Fichier : `app/api/auth/options.ts` - - Impact : Améliore l'UX pour les utilisateurs légitimes - -2. **Améliorer la gestion d'erreur token refresh** - - Fichier : `app/api/auth/options.ts` - - Impact : Meilleure détection des sessions invalides - -3. **Configurer explicitement les cookies NextAuth** - - Fichier : `app/api/auth/options.ts` - - Impact : Meilleur contrôle et sécurité - -### Phase 2 : Améliorations Logout (Priorité Moyenne) - -4. **Créer route API pour marquer logout** - - Nouveau fichier : `app/api/auth/mark-logout/route.ts` - - Modifier : `components/auth/signout-handler.tsx` - - Modifier : `app/signin/page.tsx` - - Impact : Élimine les race conditions - -5. **Améliorer Keycloak logout URL** - - Modifier : `components/auth/signout-handler.tsx` - - Modifier : `components/main-nav.tsx` - - Modifier : `components/layout/layout-wrapper.tsx` - - Impact : Meilleure suppression des sessions SSO - -6. **Améliorer `clearKeycloakCookies()`** - - Modifier : `lib/session.ts` - - Impact : Meilleure tentative de suppression cookies cross-domain - -### Phase 3 : Optimisations (Priorité Basse) - -7. **Créer route API pour vérifier statut session** - - Nouveau fichier : `app/api/auth/session-status/route.ts` - - Modifier : `app/signin/page.tsx` - - Impact : Détection plus fiable - -8. **Simplifier logique signin/page.tsx** - - Modifier : `app/signin/page.tsx` - - Impact : Code plus maintenable - ---- - -## 📝 Checklist d'Implémentation - -### Étape 1 : Configuration NextAuth -- [ ] Supprimer `prompt: "login"` par défaut dans `options.ts` -- [ ] Ajouter configuration explicite des cookies -- [ ] Améliorer gestion d'erreur `refreshAccessToken()` - -### Étape 2 : Amélioration Logout -- [ ] Créer route `/api/auth/mark-logout` -- [ ] Modifier `signout-handler.tsx` pour utiliser la route -- [ ] Améliorer Keycloak logout URL avec tous les paramètres -- [ ] Améliorer `clearKeycloakCookies()` pour cross-domain - -### Étape 3 : Amélioration Signin -- [ ] Créer route `/api/auth/session-status` -- [ ] Simplifier logique dans `signin/page.tsx` -- [ ] Utiliser cookie serveur pour détecter logout -- [ ] Ajouter `prompt=login` conditionnel après logout - -### Étape 4 : Tests -- [ ] Tester login normal (première visite) -- [ ] Tester login après logout (doit demander credentials) -- [ ] Tester logout depuis dashboard -- [ ] Tester logout depuis iframe -- [ ] Tester expiration session -- [ ] Tester refresh token expiré - ---- - -## 🔧 Configuration Keycloak Recommandée - -### Paramètres Client OAuth - -1. **Valid Redirect URIs** : - ``` - http://localhost:3000/api/auth/callback/keycloak - https://your-domain.com/api/auth/callback/keycloak - ``` - -2. **Web Origins** : - ``` - http://localhost:3000 - https://your-domain.com - ``` - -3. **Post Logout Redirect URIs** : - ``` - http://localhost:3000/signin?logout=true - https://your-domain.com/signin?logout=true - ``` - -4. **Access Token Lifespan** : 5 minutes (recommandé) -5. **SSO Session Idle** : 30 minutes -6. **SSO Session Max** : 10 heures - -### Configuration Realm - -1. **Cookies** : - - SameSite: `None` (si cross-domain) - - Secure: `true` (si HTTPS) - - Domain: Domaine parent partagé (si cross-domain) - -2. **Session Management** : - - Enable SSO Session Idle: `true` - - SSO Session Idle Timeout: 30 minutes - - SSO Session Max Lifespan: 10 heures - ---- - -## 📊 Métriques de Succès - -Après implémentation, vérifier : - -1. ✅ **Login première visite** : SSO fonctionne (pas de prompt si session Keycloak existe) -2. ✅ **Login après logout** : Prompt credentials demandé -3. ✅ **Logout dashboard** : Toutes les sessions supprimées -4. ✅ **Logout iframe** : Dashboard se déconnecte automatiquement -5. ✅ **Expiration session** : Redirection vers signin propre -6. ✅ **Refresh token expiré** : Redirection vers signin avec message approprié - ---- - -## 🎯 Résumé des Améliorations - -| Problème | Solution | Priorité | Fichiers Impactés | -|----------|----------|----------|-------------------| -| `prompt=login` toujours actif | Gérer conditionnellement | Haute | `options.ts`, `signin/page.tsx` | -| Détection session invalide complexe | Cookie serveur + route API | Haute | `signin/page.tsx`, `end-sso-session/route.ts` | -| Cookies Keycloak non supprimables | Améliorer logout URL + Admin API | Moyenne | `signout-handler.tsx`, `session.ts` | -| Race condition logout/login | Cookie HttpOnly pour flag | Moyenne | `signout-handler.tsx`, `signin/page.tsx` | -| Config cookies non explicite | Configuration explicite | Haute | `options.ts` | -| Gestion erreur refresh | Améliorer détection erreurs | Haute | `options.ts` | - ---- - -**Document créé le** : $(date) -**Dernière mise à jour** : Recommandations d'amélioration du flow de connexion - diff --git a/INACTIVITY_AND_LOGOUT_ANALYSIS.md b/INACTIVITY_AND_LOGOUT_ANALYSIS.md deleted file mode 100644 index 7b73402b..00000000 --- a/INACTIVITY_AND_LOGOUT_ANALYSIS.md +++ /dev/null @@ -1,232 +0,0 @@ -# Inactivity Timeout and Logout Analysis - -## Issue 1: Dashboard Should Disconnect After 30 Minutes of Inactivity - -### Current State - -**Session Configuration** (`app/api/auth/options.ts:190`): -```typescript -session: { - strategy: "jwt", - maxAge: 30 * 24 * 60 * 60, // 30 days -} -``` - -**SessionProvider Configuration** (`components/providers.tsx`): -```typescript - - {children} - -``` - -### Problem Analysis - -1. **No Inactivity Detection**: - - NextAuth session is set to 30 days maximum - - No client-side inactivity timeout logic exists - - No activity tracking (mouse movements, clicks, keyboard input) - - SessionProvider doesn't have `refetchInterval` configured - -2. **How NextAuth Sessions Work**: - - NextAuth sessions are JWT-based (stateless) - - Session validity is checked on each request to `/api/auth/session` - - No automatic expiration based on inactivity - - Session expires only when `maxAge` is reached (30 days in your case) - -3. **What's Missing**: - - Client-side activity monitoring - - Automatic session invalidation after inactivity period - - Session refresh based on activity (not just time) - -### Root Cause - -**NextAuth doesn't track user activity** - it only tracks session age. The session will remain valid for 30 days regardless of whether the user is active or not. - -### Solution Requirements - -To implement 30-minute inactivity timeout, you need: - -1. **Client-Side Activity Tracking**: - - Monitor user activity (mouse, keyboard, clicks) - - Track last activity timestamp - - Store in `sessionStorage` or `localStorage` - -2. **Session Invalidation Logic**: - - Check inactivity period on each page interaction - - Call `signOut()` if inactivity exceeds 30 minutes - - Clear NextAuth session and Keycloak session - -3. **Activity Reset on User Actions**: - - Reset inactivity timer on any user interaction - - Update last activity timestamp - -4. **SessionProvider Configuration**: - - Optionally configure `refetchInterval` to check session periodically - - But this won't help with inactivity - it only refreshes the session - -### Implementation Approach - -The inactivity timeout must be implemented **client-side** because: -- NextAuth sessions are stateless (JWT) -- Server doesn't know about user activity -- Activity tracking requires browser events - -**Recommended Implementation**: -1. Create an `InactivityHandler` component -2. Monitor user activity events (mousemove, keydown, click, scroll) -3. Store last activity time in `sessionStorage` -4. Check inactivity every minute (or on page focus) -5. If inactivity > 30 minutes, trigger logout - ---- - -## Issue 2: Applications Outside Dashboard Still Connected After Logout - -### Current Implementation - -**Logout Flow** (`components/main-nav.tsx`, `components/auth/signout-handler.tsx`): -1. Clear NextAuth cookies -2. Clear Keycloak cookies (client-side attempt) -3. Call `/api/auth/end-sso-session` (NEW) - - Uses Keycloak Admin API: `adminClient.users.logout({ id: userId })` -4. Sign out from NextAuth -5. Redirect to Keycloak logout endpoint with `id_token_hint` - -### Problem Analysis - -**Why Applications Are Still Connected:** - -1. **Keycloak Admin API `users.logout()` Behavior**: - - The method `adminClient.users.logout({ id: userId })` **logs out the user from all client sessions** - - However, it may **NOT clear the SSO session cookie** (`KEYCLOAK_SESSION`) - - The SSO session cookie is what allows applications to auto-authenticate - -2. **SSO Session vs Client Sessions**: - - **Client Sessions**: Per OAuth client (dashboard, app1, app2, etc.) - - **SSO Session**: Realm-wide, shared across all clients - - `users.logout()` clears client sessions but may leave SSO session active - - Applications check for SSO session cookie, not client sessions - -3. **Cookie Domain/Path Issues**: - - Keycloak cookies are set on Keycloak's domain - - Client-side `clearKeycloakCookies()` may not work if: - - Cookies are `HttpOnly` (can't be cleared from JavaScript) - - Cookies are on different domain (cross-domain restrictions) - - Cookies have different path/domain settings - -4. **Logout Endpoint Behavior**: - - Keycloak logout endpoint (`/protocol/openid-connect/logout`) with `id_token_hint`: - - Clears the **client session** for that specific OAuth client - - May clear SSO session **only if it's the last client session** - - If other applications have active sessions, SSO session persists - -### Root Cause - -**The SSO session cookie persists** because: -1. `users.logout()` Admin API method clears client sessions but may not clear SSO session cookie -2. Keycloak logout endpoint only clears SSO session if it's the last client session -3. If other applications have active sessions, the SSO session remains valid -4. Applications check for SSO session cookie, not client sessions - -### Why This Happens - -**Keycloak's SSO Design**: -- SSO session is designed to persist across client logouts -- This allows users to stay logged in across multiple applications -- Logging out from one application shouldn't log out from all applications -- This is **by design** for SSO functionality - -**However**, when you want **global logout**, you need to: -1. Clear the SSO session cookie explicitly -2. Or ensure all client sessions are logged out first -3. Or use Keycloak's Single Logout (SLO) feature - -### Solution Requirements - -To ensure applications are logged out: - -1. **Keycloak Configuration** (Server-Side): - - Enable **Front-Channel Logout** for all clients - - Configure **Back-Channel Logout URLs** for each client - - This allows Keycloak to notify all applications when logout occurs - -2. **Admin API Limitations**: - - `users.logout()` may not clear SSO session cookie - - Need to use Keycloak's logout endpoint with proper parameters - - Or use Keycloak Admin API to end SSO session directly (if available) - -3. **Alternative Approach**: - - Use Keycloak's **Single Logout (SLO)** feature - - Configure all clients to participate in SLO - - When one client logs out, all clients are notified - -### What's Actually Happening - -When you call `/api/auth/end-sso-session`: -1. ✅ Admin API `users.logout()` is called -2. ✅ All client sessions are logged out -3. ❌ SSO session cookie may still exist -4. ❌ Applications check SSO session cookie → still authenticated - -When you redirect to Keycloak logout endpoint: -1. ✅ Dashboard client session is cleared -2. ✅ If it's the last client session, SSO session is cleared -3. ❌ If other applications have active sessions, SSO session persists -4. ❌ Applications can still authenticate using SSO session cookie - -### Verification Steps - -To verify why applications are still connected: - -1. **Check if Admin API call succeeds**: - - Look for console logs: "Successfully ended SSO session for user: {userId}" - - Check for errors in `/api/auth/end-sso-session` endpoint - -2. **Check Keycloak session cookies**: - - After logout, check browser cookies for: - - `KEYCLOAK_SESSION` - - `KEYCLOAK_SESSION_LEGACY` - - `KEYCLOAK_IDENTITY` - - If these cookies still exist, SSO session is still active - -3. **Check if other applications have active sessions**: - - If other applications are open in other tabs/windows - - They may have active client sessions - - This prevents SSO session from being cleared - -4. **Check Keycloak Admin Console**: - - Navigate to: Users → [User] → Sessions - - Check if sessions are actually cleared - - Verify SSO session status - -### Recommended Solutions - -**Option 1: Keycloak Configuration (Recommended)** -- Enable Front-Channel Logout for all clients -- Configure Back-Channel Logout URLs -- This ensures all applications are notified of logout - -**Option 2: Clear SSO Session Cookie Explicitly** -- After Admin API logout, redirect to Keycloak logout endpoint -- Use `kc_action=LOGOUT` parameter (already implemented) -- Ensure all client sessions are logged out first - -**Option 3: Use Keycloak Single Logout (SLO)** -- Configure all clients to participate in SLO -- When dashboard logs out, all clients are automatically logged out -- Requires Keycloak configuration changes - ---- - -## Summary - -### Issue 1: 30-Minute Inactivity Timeout -- **Status**: Not implemented -- **Reason**: NextAuth doesn't track activity, only session age -- **Solution**: Client-side activity tracking + automatic logout - -### Issue 2: Applications Still Connected -- **Status**: Partially working -- **Reason**: SSO session cookie persists even after client sessions are cleared -- **Solution**: Keycloak configuration (Front-Channel Logout) or SLO - diff --git a/INVESTIGATION_502_ERROR.md b/INVESTIGATION_502_ERROR.md deleted file mode 100644 index 5f341c80..00000000 --- a/INVESTIGATION_502_ERROR.md +++ /dev/null @@ -1,224 +0,0 @@ -# Investigation Erreur 502 - Redirection depuis Keycloak - -## 🔍 Problème Identifié - -**Symptôme** : Erreur 502 après redirection depuis Keycloak après authentification - -**Logs observés** : -``` -Keycloak profile callback: { - rawProfile: { ... }, - rawRoles: undefined, - realmAccess: undefined, // ⚠️ PROBLÈME ICI - groups: [ ... ] // ✅ Groups présents -} -Profile callback raw roles: [] -Profile callback cleaned roles: [] -``` - -## 🎯 Cause Racine - -Le profil Keycloak (ID token) **ne contient pas** `realm_access.roles`, mais contient `groups`. Les rôles sont probablement dans le **token d'accès**, pas dans le token ID. - -## ✅ Correction Appliquée - -**Fichier** : `app/api/auth/options.ts` - Callback JWT - -**Changements** : -1. ✅ Extraction des rôles depuis le **token d'accès** (pas seulement le profil) -2. ✅ Fallback sur `groups` si pas de rôles -3. ✅ Logs améliorés pour debugging - -**Code modifié** : -```typescript -// Avant : Seulement depuis profile -const roles = keycloakProfile.realm_access?.roles || []; - -// Après : Multi-sources -1. Essayer depuis profile (ID token) -2. Si vide, décoder access token -3. Si toujours vide, utiliser groups comme fallback -``` - -## 🔍 Points d'Investigation pour l'Erreur 502 - -### 1. Vérifier les logs serveur Next.js complets - -**Où chercher** : -- Terminal Next.js (erreurs complètes) -- Logs de production (si déployé) -- Console navigateur (erreurs client) - -**Commandes utiles** : -```bash -# Voir tous les logs -npm run dev 2>&1 | tee logs.txt - -# Filtrer les erreurs -npm run dev 2>&1 | grep -i "error\|502\|exception" -``` - -### 2. Vérifier le callback NextAuth - -**Fichier** : `app/api/auth/callback/keycloak` (géré par NextAuth) - -**Points à vérifier** : -- ✅ Le callback reçoit bien le code OAuth -- ✅ L'échange code → tokens fonctionne -- ✅ Le callback JWT s'exécute sans erreur -- ✅ Le callback session s'exécute sans erreur - -**Ajouter des logs** : -```typescript -// Dans options.ts, callback jwt -console.log('JWT callback - account:', !!account, 'profile:', !!profile); -console.log('JWT callback - accessToken length:', account?.access_token?.length); -console.log('JWT callback - roles extracted:', cleanRoles); -``` - -### 3. Vérifier l'initialisation storage - -**Fichier** : `app/api/storage/init/route.ts` - -**Problème potentiel** : -- Si `createUserFolderStructure()` échoue, ça peut causer une 502 -- Si la session n'est pas encore complètement créée - -**Vérifications** : -```typescript -// Vérifier si l'erreur vient de là -console.log('Storage init - session:', session); -console.log('Storage init - user id:', session?.user?.id); -``` - -### 4. Vérifier la configuration Keycloak - -**Problème potentiel** : -- Les rôles ne sont pas mappés correctement dans Keycloak -- Le scope "roles" n'est pas configuré correctement -- Les mappers de token ne sont pas configurés - -**À vérifier dans Keycloak** : -1. **Client Configuration** : - - Scope "roles" est-il activé ? - - Mapper "realm roles" est-il configuré ? - -2. **Token Mappers** : - - `realm_access.roles` mapper existe-t-il ? - - Est-il ajouté au token d'accès ? - -3. **User Roles** : - - L'utilisateur a-t-il des rôles assignés dans le realm ? - -### 5. Vérifier les erreurs dans le callback session - -**Fichier** : `app/api/auth/options.ts` - Callback session - -**Problème potentiel** : -- Si `token.role` est undefined et qu'un code s'attend à un array -- Si `session.user` est mal formé - -**Vérifications** : -```typescript -// Dans callback session -console.log('Session callback - token.role:', token.role); -console.log('Session callback - token.error:', token.error); -console.log('Session callback - hasAccessToken:', !!token.accessToken); -``` - -### 6. Vérifier les timeouts - -**Problème potentiel** : -- Timeout lors de l'appel à Keycloak pour échanger le code -- Timeout lors de l'initialisation storage -- Timeout lors de la création de la session - -**Solutions** : -- Augmenter les timeouts si nécessaire -- Vérifier la latence réseau vers Keycloak - -## 🛠️ Actions Immédiates - -### Action 1 : Ajouter plus de logs - -**Fichier** : `app/api/auth/options.ts` - -```typescript -// Dans callback jwt, après extraction des rôles -console.log('=== JWT CALLBACK DEBUG ==='); -console.log('Has account:', !!account); -console.log('Has profile:', !!profile); -console.log('Access token present:', !!account?.access_token); -console.log('Roles from profile:', keycloakProfile.realm_access?.roles); -console.log('Roles from access token:', roles); -console.log('Final roles:', cleanRoles); -console.log('=========================='); -``` - -### Action 2 : Vérifier l'erreur exacte - -**Dans le terminal Next.js**, chercher : -- Stack trace complète -- Message d'erreur exact -- Ligne de code qui cause l'erreur - -### Action 3 : Tester avec un utilisateur simple - -**Tester avec** : -- Un utilisateur avec des rôles -- Un utilisateur sans rôles -- Vérifier si l'erreur est liée aux rôles - -### Action 4 : Vérifier la configuration Keycloak - -**Dans Keycloak Admin Console** : - -1. **Client → Mappers** : - - Vérifier qu'il y a un mapper "realm roles" - - Vérifier qu'il est ajouté au token d'accès - - Vérifier qu'il ajoute `realm_access.roles` - -2. **Client → Settings** : - - Vérifier que "Full Scope Allowed" est activé - - Vérifier les "Default Client Scopes" incluent "roles" - -3. **Realm → Roles** : - - Vérifier que l'utilisateur a des rôles assignés - -## 📊 Checklist de Debugging - -- [ ] Logs serveur Next.js complets vérifiés -- [ ] Erreur exacte identifiée (stack trace) -- [ ] Callback JWT s'exécute sans erreur -- [ ] Callback session s'exécute sans erreur -- [ ] Rôles extraits correctement (depuis access token) -- [ ] Storage init fonctionne -- [ ] Configuration Keycloak vérifiée -- [ ] Mappers Keycloak vérifiés -- [ ] Timeouts vérifiés - -## 🔧 Solution Temporaire (si nécessaire) - -Si l'erreur persiste, on peut temporairement utiliser les `groups` comme rôles : - -```typescript -// Dans profile callback -const roles = keycloakProfile.realm_access?.roles || - keycloakProfile.groups || - []; -``` - -**Note** : Ce n'est qu'une solution temporaire. Il faut corriger la configuration Keycloak pour avoir les rôles correctement mappés. - -## 📝 Prochaines Étapes - -1. ✅ **Correction appliquée** : Extraction rôles depuis access token -2. ⏳ **À faire** : Vérifier les logs serveur pour l'erreur exacte -3. ⏳ **À faire** : Vérifier configuration Keycloak -4. ⏳ **À faire** : Tester après correction - ---- - -**Document créé le** : $(date) -**Statut** : Correction appliquée, investigation en cours - diff --git a/KEYCLOAK_SESSION_SYNC.md b/KEYCLOAK_SESSION_SYNC.md deleted file mode 100644 index 64bfa32d..00000000 --- a/KEYCLOAK_SESSION_SYNC.md +++ /dev/null @@ -1,123 +0,0 @@ -# Keycloak Session Synchronization Fix - -## Problem - -When a user is still logged into the NextAuth dashboard (session valid for 30 days), but Keycloak session cookies have expired (typically 30 minutes to a few hours), iframe applications can't authenticate because they rely on Keycloak cookies for SSO. - -**Symptoms**: -- User is logged into dashboard -- Iframe applications ask for Keycloak login again -- NextAuth session is still valid, but Keycloak cookies expired - -## Root Cause - -**Session Mismatch**: -- **NextAuth Session**: 30 days (JWT-based, stored in encrypted cookie) -- **Keycloak Session Cookies**: Typically 30 minutes to a few hours (set by Keycloak server) -- **Iframe Applications**: Rely on Keycloak session cookies for SSO, not NextAuth tokens - -When Keycloak session cookies expire, iframe applications can't authenticate even though NextAuth session is still valid. - -## Solution Implemented - -### 1. Session Refresh API Endpoint - -Created `/api/auth/refresh-keycloak-session` that: -- Uses the refresh token to get new Keycloak tokens -- Ensures tokens are fresh before loading iframes -- Helps maintain token synchronization - -### 2. Automatic Session Refresh Before Iframe Load - -Updated `ResponsiveIframe` component to: -- Automatically refresh the session before loading iframe applications -- Show a loading indicator during refresh -- Ensure tokens are fresh when iframes load - -### 3. Exposed Refresh Token in Session - -- Added `refreshToken` to session object -- Allows API endpoints to refresh tokens when needed - -## Files Modified - -1. **`app/api/auth/refresh-keycloak-session/route.ts`** (NEW) - - API endpoint to refresh Keycloak tokens - - Uses refresh token to get new access tokens - -2. **`app/components/responsive-iframe.tsx`** - - Automatically refreshes session before loading iframe - - Shows loading indicator during refresh - -3. **`app/api/auth/options.ts`** - - Exposes `refreshToken` in session object - -4. **`types/next-auth.d.ts`** - - Added `refreshToken` to Session interface - -## Limitations - -**Important**: This solution refreshes OAuth tokens, but **Keycloak session cookies are separate** and are set by Keycloak when the user authenticates via the browser. Refreshing OAuth tokens doesn't automatically refresh Keycloak session cookies. - -### Why This Happens - -Keycloak maintains two separate sessions: -1. **OAuth Token Session**: Managed via refresh tokens (what we refresh) -2. **Browser Session Cookies**: Set by Keycloak during login, expire based on Keycloak's session timeout settings - -### Recommended Solutions - -#### Option 1: Configure Keycloak Session Timeout (Recommended) - -Increase Keycloak's SSO session timeout to match or exceed NextAuth's 30-day session: - -1. Go to Keycloak Admin Console -2. Navigate to: Realm Settings → Sessions -3. Set **SSO Session Idle** to match your needs (e.g., 30 days) -4. Set **SSO Session Max** to match (e.g., 30 days) - -This ensures Keycloak cookies don't expire before NextAuth session. - -#### Option 2: Pass Access Token to Iframe Applications - -If iframe applications support token-based authentication: -- Pass `accessToken` via URL parameter: `?token=${accessToken}` -- Or use `postMessage` to send token to iframe -- Iframe applications can then use the token for authentication - -#### Option 3: Periodic Session Refresh - -Implement a periodic refresh mechanism that: -- Checks session validity every 15-20 minutes -- Refreshes tokens proactively -- May help keep Keycloak session active - -## Testing - -1. Log in to dashboard -2. Wait for Keycloak session to expire (or manually clear Keycloak cookies) -3. Navigate to an iframe application -4. Session should be refreshed automatically -5. Iframe should load without requiring login - -## Environment Variables Required - -```bash -NEXT_PUBLIC_KEYCLOAK_ISSUER=https://keycloak.example.com/realms/neah -KEYCLOAK_CLIENT_ID=neah-dashboard -KEYCLOAK_CLIENT_SECRET= -``` - -## Future Improvements - -1. **Implement invisible iframe to Keycloak**: Use Keycloak's check-session-iframe to refresh cookies -2. **Token passing**: Pass access tokens to iframe applications if they support it -3. **Proactive refresh**: Implement periodic token refresh to prevent expiration -4. **Session monitoring**: Monitor Keycloak session status and refresh proactively - ---- - -**Date**: 2024 -**Status**: ✅ Implemented (with limitations) -**Version**: 1.0 - diff --git a/LEANTIME_API_FIXES.md b/LEANTIME_API_FIXES.md deleted file mode 100644 index 73cb379f..00000000 --- a/LEANTIME_API_FIXES.md +++ /dev/null @@ -1,157 +0,0 @@ -# Leantime API Fixes - Mark Notifications as Read - -**Date**: 2026-01-01 -**Issue**: Mark all as read failing due to incorrect API method names -**Status**: ✅ Fixed - ---- - -## 🔍 Issues Found - -### Issue 1: Incorrect Method Name for Single Notification - -**Current Code** (WRONG): -```typescript -method: 'leantime.rpc.Notifications.Notifications.markNotificationAsRead' -params: { - userId: leantimeUserId, - notificationId: parseInt(sourceId) // Wrong parameter name -} -``` - -**Leantime Documentation** (CORRECT): -```typescript -method: 'leantime.rpc.Notifications.Notifications.markNotificationRead' // No "As" in method name -params: { - id: parseInt(sourceId), // Parameter is "id", not "notificationId" - userId: leantimeUserId -} -``` - -**Fix Applied**: ✅ Changed method name and parameter names to match Leantime API - ---- - -### Issue 2: No "Mark All" Method Exists - -**Problem**: -- Leantime API does NOT have a `markAllNotificationsAsRead` method -- Current code tries to call a non-existent method - -**Solution**: -- Fetch all unread notifications -- Mark each one individually using `markNotificationRead` -- Process in parallel for better performance - -**Fix Applied**: ✅ Implemented loop-based approach to mark all notifications individually - ---- - -## ✅ Changes Made - -### 1. Fixed `markAsRead` Method - -**File**: `lib/services/notifications/leantime-adapter.ts` - -**Changes**: -- ✅ Method name: `markNotificationAsRead` → `markNotificationRead` -- ✅ Parameter: `notificationId` → `id` -- ✅ Parameter order: `id` first, then `userId` (matching Leantime docs) -- ✅ Added request logging - ---- - -### 2. Fixed `markAllAsRead` Method - -**File**: `lib/services/notifications/leantime-adapter.ts` - -**New Implementation**: -1. Fetch all unread notifications (up to 1000) -2. Filter to get only unread ones -3. Mark each notification individually using `markNotificationRead` -4. Process in parallel using `Promise.all()` -5. Return success if majority succeed - -**Benefits**: -- ✅ Works with actual Leantime API -- ✅ Handles partial failures gracefully -- ✅ Parallel processing for better performance -- ✅ Detailed logging for each notification - ---- - -## 📊 Expected Behavior After Fix - -### Mark Single Notification as Read - -**Before**: ❌ Failed (wrong method name) -**After**: ✅ Should work correctly - -**Logs**: -``` -[LEANTIME_ADAPTER] markAsRead - Request body: {"method":"markNotificationRead",...} -[LEANTIME_ADAPTER] markAsRead - Success: true -``` - ---- - -### Mark All Notifications as Read - -**Before**: ❌ Failed (method doesn't exist) -**After**: ✅ Should work (marks each individually) - -**Logs**: -``` -[LEANTIME_ADAPTER] markAllAsRead - Fetching all unread notifications -[LEANTIME_ADAPTER] markAllAsRead - Found 66 unread notifications to mark -[LEANTIME_ADAPTER] markAllAsRead - Results: 66 succeeded, 0 failed out of 66 total -[LEANTIME_ADAPTER] markAllAsRead - Overall success: true -``` - ---- - -## 🎯 Count vs Display Issue - -**Current Situation**: -- Count: 66 unread (from first 100 notifications) -- Display: 10 notifications shown (pagination) - -**Why**: -- `getNotificationCount()` fetches first 100 notifications and counts unread -- `getNotifications()` with default limit=20 shows first 10-20 -- This is expected behavior but can be confusing - -**Options**: -1. **Accept limitation**: Document that count is based on first 100 -2. **Fetch all for count**: More accurate but slower -3. **Use dedicated count API**: If Leantime provides one -4. **Show "66+ unread"**: If count reaches 100, indicate there may be more - -**Recommendation**: Keep current behavior but add a note in UI if count = 100 (may have more) - ---- - -## 🚀 Next Steps - -1. ✅ **Test Mark Single as Read**: Should work now with correct method name -2. ✅ **Test Mark All as Read**: Should work by marking each individually -3. ⏳ **Verify Count Updates**: After marking, count should decrease -4. ⏳ **Monitor Performance**: Marking 66 notifications individually may take a few seconds - ---- - -## 📝 Summary - -**Fixes Applied**: -1. ✅ Fixed `markAsRead` method name and parameters -2. ✅ Implemented `markAllAsRead` using individual marking approach -3. ✅ Added comprehensive logging - -**Status**: Ready for testing after `rm -rf .next && npm run build` - -**Expected Result**: Mark all as read should now work correctly - ---- - -**Generated**: 2026-01-01 - diff --git a/LOGIN_LOGOUT_FILES_AUDIT.md b/LOGIN_LOGOUT_FILES_AUDIT.md deleted file mode 100644 index 646ebdad..00000000 --- a/LOGIN_LOGOUT_FILES_AUDIT.md +++ /dev/null @@ -1,646 +0,0 @@ -# Audit Complet des Fichiers Login/Logout - Analyse des Cookies - -## 📋 Vue d'ensemble - -Ce document liste **TOUS** les fichiers de code impliqués dans le processus de **login** et **logout** du dashboard Next.js avec NextAuth et Keycloak, avec une analyse approfondie de la gestion des cookies. - ---- - -## 🔐 FICHIERS CORE - Configuration NextAuth - -### 1. **`app/api/auth/[...nextauth]/route.ts`** -**Rôle** : Route handler NextAuth pour tous les endpoints d'authentification -**Cookies gérés** : -- `next-auth.session-token` (ou variantes sécurisées) - Cookie principal de session NextAuth -- `next-auth.csrf-token` - Token CSRF pour la sécurité -- `next-auth.state` - État OAuth pour le flow Keycloak -- `next-auth.callback-url` - URL de callback après authentification - -**Fonctions** : -- Gère `GET/POST /api/auth/signin` → Redirige vers Keycloak -- Gère `GET/POST /api/auth/signout` → Déconnecte et nettoie les cookies -- Gère `GET /api/auth/session` → Lit le cookie de session -- Gère `GET /api/auth/callback/keycloak` → Reçoit le code OAuth de Keycloak -- Gère `GET /api/auth/csrf` → Génère le token CSRF -- Gère `GET /api/auth/providers` → Liste les providers disponibles - -**Cookies créés/supprimés** : -- **Login** : Crée `next-auth.session-token` (HttpOnly, Secure, SameSite=Lax) -- **Logout** : Supprime `next-auth.session-token` via `signOut()` - ---- - -### 2. **`app/api/auth/options.ts`** ⭐ **FICHIER CRITIQUE** -**Rôle** : Configuration principale de NextAuth avec Keycloak -**Cookies gérés** : -- Tous les cookies NextAuth (via configuration implicite) -- Les tokens Keycloak sont stockés dans le JWT (pas de cookies séparés) - -**Fonctions clés** : - -#### `refreshAccessToken(token)` (lignes 83-139) -- **Cookies utilisés** : Aucun directement, mais utilise `refreshToken` du JWT -- **Comportement** : - - Appelle Keycloak `/token` endpoint pour rafraîchir - - Détecte si la session Keycloak est invalide (erreur `invalid_grant`) - - Retourne `error: "SessionNotActive"` si session Keycloak expirée - -#### `jwt` callback (lignes 196-282) -- **Cookies utilisés** : Lit `next-auth.session-token` (décrypté par NextAuth) -- **Comportement** : - - **Initial login** : Stocke `accessToken`, `refreshToken`, `idToken` dans le JWT - - **Subsequent requests** : Vérifie expiration, rafraîchit si nécessaire - - **Token expired** : Appelle `refreshAccessToken()` - - **Session invalidated** : Retourne token avec `error: "SessionNotActive"` - -#### `session` callback (lignes 283-324) -- **Cookies utilisés** : Lit le JWT depuis `next-auth.session-token` -- **Comportement** : - - Si `token.error === "SessionNotActive"` → Retourne `null` (force logout) - - Sinon, construit la session avec les données utilisateur - -**Configuration cookies** : -```typescript -session: { - strategy: "jwt", - maxAge: 4 * 60 * 60, // 4 heures -} -// Les cookies sont gérés automatiquement par NextAuth -// Pas de configuration explicite des cookies dans ce fichier -``` - -**Paramètres OAuth** : -```typescript -authorization: { - params: { - scope: "openid profile email roles", - prompt: "login" // Force le prompt de login même si SSO existe - } -} -``` - ---- - -## 🚪 FICHIERS PAGES - Interface Utilisateur - -### 3. **`app/signin/page.tsx`** ⭐ **FICHIER CRITIQUE** -**Rôle** : Page de connexion avec logique complexe de détection de logout -**Cookies analysés** : -- `next-auth.session-token` (ou variantes) - Vérifie si cookie existe mais invalide -- `logout_in_progress` - Cookie temporaire (60s) pour marquer logout en cours -- Cookies Keycloak (via `document.cookie`) - -**Fonctions clés** : - -#### Détection de logout/session invalide (lignes 17-67) -```typescript -// Vérifie les cookies NextAuth -const hasInvalidSessionCookie = document.cookie - .split(';') - .some(c => c.trim().startsWith('next-auth.session-token=') || - c.trim().startsWith('__Secure-next-auth.session-token=') || - c.trim().startsWith('__Host-next-auth.session-token=')); - -// Si cookie existe mais status = unauthenticated → Session invalidée -if (status === 'unauthenticated' && hasInvalidSessionCookie) { - sessionStorage.setItem('session_invalidated', 'true'); - // Empêche auto-login -} -``` - -#### Auto-login (lignes 69-124) -- **Condition** : Seulement si **PAS** de cookie de session existant -- **Comportement** : Appelle `signIn("keycloak")` après 1 seconde -- **Protection** : Ne s'exécute pas si `logout_in_progress` ou `session_invalidated` - -#### Initialisation storage (lignes 126-158) -- Appelle `/api/storage/init` après authentification réussie -- Force reload pour mettre à jour la session - -**Cookies créés/supprimés** : -- **Aucun cookie créé directement** (NextAuth gère ça) -- **Supprime** : `sessionStorage` items (`just_logged_out`, `session_invalidated`) - ---- - -### 4. **`app/signout/page.tsx`** -**Rôle** : Page de déconnexion (simple wrapper) -**Cookies** : Aucune manipulation directe, délègue à `SignOutHandler` - ---- - -## 🔧 FICHIERS COMPOSANTS - Logique Métier - -### 5. **`components/auth/signout-handler.tsx`** ⭐ **FICHIER CRITIQUE** -**Rôle** : Gère la déconnexion complète (NextAuth + Keycloak) -**Cookies manipulés** : - -#### Cookies NextAuth (ligne 23) -```typescript -clearAuthCookies(); // Supprime next-auth.session-token -``` - -#### Cookies Keycloak (ligne 25) -```typescript -clearKeycloakCookies(); // Tente de supprimer KEYCLOAK_SESSION, etc. -``` - -#### Cookie de flag (ligne 16) -```typescript -document.cookie = 'logout_in_progress=true; path=/; max-age=60'; -``` - -**Flow de logout** : -1. Marque logout en cours (`sessionStorage` + cookie) -2. **Supprime cookies NextAuth** (`clearAuthCookies()`) -3. **Tente de supprimer cookies Keycloak** (`clearKeycloakCookies()`) -4. Appelle `/api/auth/end-sso-session` (Admin API Keycloak) -5. Appelle `signOut()` NextAuth (supprime cookie serveur) -6. Redirige vers Keycloak logout endpoint avec `id_token_hint` -7. Keycloak redirige vers `/signin?logout=true` - -**Cookies supprimés** : -- `next-auth.session-token` (et variantes) -- `KEYCLOAK_SESSION`, `KEYCLOAK_IDENTITY`, `AUTH_SESSION_ID` (si même domaine) -- `logout_in_progress` (expire après 60s) - ---- - -### 6. **`components/main-nav.tsx`** (lignes 364-446) -**Rôle** : Bouton de déconnexion dans la navigation -**Cookies** : Même logique que `signout-handler.tsx` -- Appelle `clearAuthCookies()` et `clearKeycloakCookies()` -- Crée cookie `logout_in_progress` -- Même flow que `SignOutHandler` - ---- - -### 7. **`components/layout/layout-wrapper.tsx`** ⭐ **FICHIER CRITIQUE** -**Rôle** : Écoute les messages de logout depuis les iframes -**Cookies manipulés** : -- Même pattern que `signout-handler.tsx` -- Gère les logout déclenchés par les iframes via `postMessage` - -**Fonction clé** (lignes 23-112) : -```typescript -const handleMessage = async (event: MessageEvent) => { - if (event.data.type === 'KEYCLOAK_LOGOUT' || event.data.type === 'LOGOUT') { - // Même flow que signout-handler.tsx - clearAuthCookies(); - clearKeycloakCookies(); - // ... logout complet - } -}; -``` - -**Cookies** : Identique à `signout-handler.tsx` - ---- - -### 8. **`components/auth/auth-check.tsx`** -**Rôle** : Guard d'authentification côté client -**Cookies** : Aucune manipulation directe -- Utilise `useSession()` qui lit `next-auth.session-token` -- Redirige vers `/signin` si `status === "unauthenticated"` - ---- - -### 9. **`components/providers.tsx`** -**Rôle** : Wrapper `SessionProvider` pour NextAuth -**Cookies** : Aucune manipulation, fournit le contexte de session - ---- - -## 🛠️ FICHIERS UTILITAIRES - Gestion Sessions/Cookies - -### 10. **`lib/session.ts`** ⭐ **FICHIER CRITIQUE** -**Rôle** : Utilitaires pour gérer les cookies et sessions - -#### `clearAuthCookies()` (lignes 93-108) -**Cookies supprimés** : -```typescript -// Supprime SEULEMENT les cookies de session, PAS les cookies OAuth -if (cookieName.startsWith('next-auth.session-token') || - cookieName.startsWith('__Secure-next-auth.session-token') || - cookieName.startsWith('__Host-next-auth.session-token')) { - document.cookie = `${cookieName}=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;`; -} -``` - -**Important** : Ne supprime **PAS** : -- `next-auth.csrf-token` (nécessaire pour OAuth) -- `next-auth.state` (nécessaire pour OAuth flow) -- `next-auth.callback-url` - -#### `clearKeycloakCookies()` (lignes 115-154) -**Cookies Keycloak tentés de supprimer** : -```typescript -const keycloakCookieNames = [ - 'KEYCLOAK_SESSION', - 'KEYCLOAK_SESSION_LEGACY', - 'KEYCLOAK_IDENTITY', - 'KEYCLOAK_IDENTITY_LEGACY', - 'AUTH_SESSION_ID', - 'KC_RESTART', - 'KC_RESTART_LEGACY' -]; -``` - -**Limitation** : Ces cookies sont sur le domaine Keycloak, donc **ne peuvent pas être supprimés** depuis le domaine du dashboard (même origine). Cette fonction tente plusieurs combinaisons domain/path mais échouera si Keycloak est sur un domaine différent. - -#### `invalidateServiceTokens()` (lignes 53-91) -**Cookies** : Aucun, invalide les tokens de services externes (RocketChat, Leantime) - ---- - -### 11. **`lib/keycloak.ts`** -**Rôle** : Client Admin Keycloak pour gestion serveur -**Cookies** : Aucune manipulation directe -- Utilisé par `/api/auth/end-sso-session` pour terminer la session SSO - ---- - -## 🌐 FICHIERS API - Endpoints Serveur - -### 12. **`app/api/auth/end-sso-session/route.ts`** ⭐ **FICHIER CRITIQUE** -**Rôle** : Termine la session SSO Keycloak via Admin API -**Cookies** : Aucune manipulation directe -- Utilise Keycloak Admin API pour logout utilisateur -- **Important** : Termine la session **realm-wide**, pas seulement client - -**Flow** : -1. Lit `next-auth.session-token` via `getServerSession()` -2. Extrait `idToken` de la session -3. Décode `idToken` pour obtenir `userId` -4. Appelle `adminClient.users.logout({ id: userId })` -5. Keycloak supprime **toutes** les sessions de l'utilisateur - -**Impact cookies** : -- Keycloak supprime ses cookies côté serveur -- Les cookies Keycloak deviennent invalides (mais restent dans le navigateur jusqu'à expiration) - ---- - -### 13. **`app/api/auth/refresh-keycloak-session/route.ts`** -**Rôle** : Rafraîchit la session Keycloak (si existe) -**Cookies** : Lit `next-auth.session-token` via `getServerSession()` - ---- - -## 📄 FICHIERS LAYOUT - Structure Application - -### 14. **`app/layout.tsx`** -**Rôle** : Layout racine avec vérification de session serveur -**Cookies** : Lit `next-auth.session-token` via `getServerSession(authOptions)` -- Passe `isAuthenticated` à `LayoutWrapper` -- Détermine si c'est la page signin - ---- - -### 15. **`app/components/responsive-iframe.tsx`** (lignes 109-153) -**Rôle** : Composant iframe avec écoute de messages logout -**Cookies** : Aucune manipulation directe -- Écoute `postMessage` depuis iframes -- Déclenche logout si message `KEYCLOAK_LOGOUT` reçu -- **Note** : Logique similaire à `layout-wrapper.tsx` mais dans le composant iframe - ---- - -## 📝 FICHIERS TYPES - Définitions TypeScript - -### 16. **`types/next-auth.d.ts`** -**Rôle** : Extensions TypeScript pour NextAuth -**Cookies** : Aucune manipulation, définit les types de session/JWT - ---- - -## 🔍 ANALYSE DÉTAILLÉE DES COOKIES - -### Cookies NextAuth - -#### 1. **`next-auth.session-token`** (ou variantes sécurisées) -- **Domaine** : Domaine du dashboard -- **Path** : `/` -- **HttpOnly** : `true` (sécurité) -- **Secure** : `true` (si HTTPS) -- **SameSite** : `Lax` (par défaut) -- **Contenu** : JWT encrypté contenant : - - `accessToken` (Keycloak) - - `refreshToken` (Keycloak) - - `idToken` (Keycloak) - - Données utilisateur (id, email, roles, etc.) -- **Durée** : 4 heures (configuré dans `options.ts`) -- **Créé** : Lors de `signIn()` réussi -- **Supprimé** : Lors de `signOut()` ou expiration -- **Variantes** : - - `__Secure-next-auth.session-token` (si HTTPS) - - `__Host-next-auth.session-token` (si domaine racine) - -#### 2. **`next-auth.csrf-token`** -- **Domaine** : Domaine du dashboard -- **Path** : `/` -- **HttpOnly** : `true` -- **Secure** : `true` (si HTTPS) -- **SameSite** : `Lax` -- **Contenu** : Token CSRF pour protection OAuth -- **Durée** : Session (supprimé à la fermeture du navigateur) -- **Créé** : Lors de la première requête OAuth -- **Supprimé** : À la fermeture du navigateur -- **Important** : **N'EST PAS supprimé** par `clearAuthCookies()` (nécessaire pour OAuth) - -#### 3. **`next-auth.state`** -- **Domaine** : Domaine du dashboard -- **Path** : `/` -- **HttpOnly** : `true` -- **Secure** : `true` (si HTTPS) -- **SameSite** : `Lax` -- **Contenu** : État OAuth pour validation du callback -- **Durée** : Court (pendant le flow OAuth) -- **Créé** : Lors de `signIn()` (début flow OAuth) -- **Supprimé** : Après validation du callback OAuth -- **Important** : **N'EST PAS supprimé** par `clearAuthCookies()` (nécessaire pour OAuth) - -#### 4. **`next-auth.callback-url`** -- **Domaine** : Domaine du dashboard -- **Path** : `/` -- **HttpOnly** : `true` -- **Secure** : `true` (si HTTPS) -- **SameSite** : `Lax` -- **Contenu** : URL de redirection après authentification -- **Durée** : Court (pendant le flow OAuth) -- **Créé** : Lors de `signIn()` avec `callbackUrl` -- **Supprimé** : Après redirection - -### Cookies Keycloak - -#### 1. **`KEYCLOAK_SESSION`** -- **Domaine** : Domaine Keycloak (peut être différent du dashboard) -- **Path** : `/` ou `/realms/{realm}` -- **HttpOnly** : `true` -- **Secure** : `true` (si HTTPS) -- **SameSite** : `Lax` ou `None` (pour cross-site) -- **Contenu** : Identifiant de session SSO Keycloak -- **Durée** : Configuré dans Keycloak (typiquement 30 min - quelques heures) -- **Créé** : Lors de l'authentification Keycloak -- **Supprimé** : Lors de logout Keycloak ou expiration -- **Problème** : **Ne peut pas être supprimé** depuis le dashboard si domaine différent - -#### 2. **`KEYCLOAK_IDENTITY`** -- **Domaine** : Domaine Keycloak -- **Path** : `/` ou `/realms/{realm}` -- **HttpOnly** : `true` -- **Secure** : `true` -- **SameSite** : `Lax` ou `None` -- **Contenu** : Identité utilisateur Keycloak -- **Durée** : Même que `KEYCLOAK_SESSION` -- **Créé** : Lors de l'authentification Keycloak -- **Supprimé** : Lors de logout Keycloak ou expiration - -#### 3. **`AUTH_SESSION_ID`** -- **Domaine** : Domaine Keycloak -- **Path** : `/` ou `/realms/{realm}` -- **HttpOnly** : `true` -- **Secure** : `true` -- **SameSite** : `Lax` ou `None` -- **Contenu** : ID de session d'authentification -- **Durée** : Court (pendant le flow d'authentification) -- **Créé** : Lors du début du flow d'authentification -- **Supprimé** : Après authentification réussie ou échec - -### Cookies Custom - -#### 1. **`logout_in_progress`** -- **Domaine** : Domaine du dashboard -- **Path** : `/` -- **HttpOnly** : `false` (accessible via JavaScript) -- **Secure** : `false` -- **SameSite** : Non défini -- **Contenu** : `"true"` -- **Durée** : 60 secondes (`max-age=60`) -- **Créé** : Lors de `signOut()` (dans `signout-handler.tsx`, `main-nav.tsx`, `layout-wrapper.tsx`) -- **Supprimé** : Expire après 60s ou manuellement -- **Usage** : Empêche l'auto-login après logout - ---- - -## 🔄 FLOW COMPLET DE LOGIN - -### Étape 1 : Utilisateur accède à `/signin` -**Fichier** : `app/signin/page.tsx` -**Cookies** : -- Vérifie si `next-auth.session-token` existe -- Si existe mais `status === "unauthenticated"` → Session invalidée -- Si n'existe pas → Nouvel utilisateur, déclenche auto-login - -### Étape 2 : Auto-login déclenché -**Fichier** : `app/signin/page.tsx` (ligne 118) -**Action** : `signIn("keycloak", { callbackUrl: "/" })` -**Cookies créés** : -- `next-auth.csrf-token` (par NextAuth) -- `next-auth.state` (par NextAuth) -- `next-auth.callback-url` (par NextAuth) - -### Étape 3 : Redirection vers Keycloak -**Fichier** : `app/api/auth/[...nextauth]/route.ts` → NextAuth interne -**URL** : `${KEYCLOAK_ISSUER}/protocol/openid-connect/auth?...&prompt=login` -**Cookies Keycloak créés** : -- `AUTH_SESSION_ID` (par Keycloak) - -### Étape 4 : Authentification Keycloak -**Fichier** : Keycloak serveur -**Cookies Keycloak créés** : -- `KEYCLOAK_SESSION` (session SSO) -- `KEYCLOAK_IDENTITY` (identité utilisateur) - -### Étape 5 : Callback OAuth -**Fichier** : `app/api/auth/callback/keycloak` (géré par NextAuth) -**Cookies** : -- `next-auth.state` vérifié et supprimé -- `next-auth.callback-url` lu et utilisé - -### Étape 6 : JWT Callback -**Fichier** : `app/api/auth/options.ts` → `jwt` callback (ligne 196) -**Cookies** : -- Lit `next-auth.session-token` (décrypté) -- Stocke tokens Keycloak dans le JWT -- **Crée** `next-auth.session-token` (nouveau JWT avec tokens) - -### Étape 7 : Session Callback -**Fichier** : `app/api/auth/options.ts` → `session` callback (ligne 283) -**Cookies** : Lit `next-auth.session-token` pour construire la session - -### Étape 8 : Redirection vers `/` -**Fichier** : `app/signin/page.tsx` (ligne 72) -**Cookies** : `next-auth.session-token` maintenant présent - -### Étape 9 : Initialisation Storage -**Fichier** : `app/signin/page.tsx` (lignes 126-158) -**Action** : Appelle `/api/storage/init` -**Cookies** : Utilise `next-auth.session-token` (via `getServerSession()`) - ---- - -## 🔄 FLOW COMPLET DE LOGOUT - -### Étape 1 : Utilisateur clique "Déconnexion" -**Fichiers** : -- `components/main-nav.tsx` (ligne 364) -- OU `components/auth/signout-handler.tsx` (ligne 11) -- OU `components/layout/layout-wrapper.tsx` (ligne 32) si message iframe - -**Cookies créés** : -- `logout_in_progress=true; path=/; max-age=60` (ligne 16/369/38) -- `sessionStorage.setItem('just_logged_out', 'true')` (ligne 14/367/37) - -### Étape 2 : Suppression cookies NextAuth -**Fichier** : `lib/session.ts` → `clearAuthCookies()` (ligne 93) -**Cookies supprimés** : -- `next-auth.session-token` (et variantes) -- **PAS** `next-auth.csrf-token` (nécessaire pour OAuth) -- **PAS** `next-auth.state` (nécessaire pour OAuth) - -### Étape 3 : Tentative suppression cookies Keycloak -**Fichier** : `lib/session.ts` → `clearKeycloakCookies()` (ligne 115) -**Cookies tentés de supprimer** : -- `KEYCLOAK_SESSION`, `KEYCLOAK_IDENTITY`, etc. -- **Limitation** : Échoue si Keycloak sur domaine différent - -### Étape 4 : Fin de session SSO via Admin API -**Fichier** : `app/api/auth/end-sso-session/route.ts` (ligne 15) -**Action** : `adminClient.users.logout({ id: userId })` -**Cookies** : -- Keycloak supprime **toutes** les sessions côté serveur -- Les cookies Keycloak deviennent invalides (mais restent dans le navigateur) - -### Étape 5 : SignOut NextAuth -**Fichier** : `components/auth/signout-handler.tsx` (ligne 52) -**Action** : `signOut({ callbackUrl: "/signin?logout=true", redirect: false })` -**Cookies supprimés** : -- `next-auth.session-token` (supprimé côté serveur) - -### Étape 6 : Redirection vers Keycloak Logout -**Fichier** : `components/auth/signout-handler.tsx` (ligne 58) -**URL** : `${KEYCLOAK_ISSUER}/protocol/openid-connect/logout?...&id_token_hint=...&kc_action=LOGOUT` -**Cookies Keycloak** : -- Keycloak supprime ses cookies (si même domaine ou cross-domain configuré) - -### Étape 7 : Redirection vers `/signin?logout=true` -**Fichier** : Keycloak → `app/signin/page.tsx` -**Cookies** : -- `next-auth.session-token` : Supprimé -- `KEYCLOAK_SESSION` : Peut encore exister (si domaine différent) -- `logout_in_progress` : Existe encore (60s) - -### Étape 8 : Détection logout dans signin -**Fichier** : `app/signin/page.tsx` (lignes 17-67) -**Cookies vérifiés** : -- `logout_in_progress` (ligne 19) -- `next-auth.session-token` (ligne 25-29) -- `sessionStorage.getItem('just_logged_out')` (ligne 20) - -**Comportement** : -- Si `logout=true` dans URL → Affiche message "Vous avez été déconnecté" -- Si cookie session existe mais invalide → Empêche auto-login -- Si pas de cookie session → Auto-login après 1s (nouvel utilisateur) - ---- - -## ⚠️ PROBLÈMES IDENTIFIÉS - -### Problème 1 : Cookies Keycloak non supprimables -**Fichier** : `lib/session.ts` → `clearKeycloakCookies()` -**Cause** : Cookies Keycloak sur domaine différent -**Impact** : Les cookies Keycloak persistent après logout dashboard -**Solution actuelle** : Appel à Keycloak logout endpoint avec `id_token_hint` - -### Problème 2 : Session SSO Keycloak peut persister -**Fichier** : `app/api/auth/options.ts` (ligne 154) -**Cause** : `prompt=login` force le prompt, mais si SSO session existe, Keycloak peut auto-authentifier -**Impact** : Utilisateur peut être reconnecté automatiquement sans credentials -**Solution actuelle** : `prompt=login` + appel Admin API pour terminer session SSO - -### Problème 3 : Détection session invalide complexe -**Fichier** : `app/signin/page.tsx` (lignes 17-67) -**Cause** : Logique complexe pour détecter si session invalidée vs nouvel utilisateur -**Impact** : Auto-login peut se déclencher incorrectement -**Solution actuelle** : Vérification multiple (cookies, sessionStorage, URL params) - -### Problème 4 : Race condition logout/login -**Fichier** : `app/signin/page.tsx` (lignes 69-124) -**Cause** : Auto-login avec délai de 1s peut se déclencher pendant logout -**Impact** : Utilisateur peut être reconnecté immédiatement après logout -**Solution actuelle** : Flags `logout_in_progress` et `session_invalidated` - ---- - -## 📊 RÉSUMÉ DES FICHIERS PAR CATÉGORIE - -### Configuration Core (2 fichiers) -1. `app/api/auth/[...nextauth]/route.ts` -2. `app/api/auth/options.ts` ⭐ - -### Pages (2 fichiers) -3. `app/signin/page.tsx` ⭐ -4. `app/signout/page.tsx` - -### Composants Auth (4 fichiers) -5. `components/auth/signout-handler.tsx` ⭐ -6. `components/auth/auth-check.tsx` -7. `components/auth/signin-form.tsx` (si existe) -8. `components/auth/login-card.tsx` (si existe) - -### Composants Layout (2 fichiers) -9. `components/layout/layout-wrapper.tsx` ⭐ -10. `components/providers.tsx` - -### Navigation (1 fichier) -11. `components/main-nav.tsx` ⭐ - -### Utilitaires (2 fichiers) -12. `lib/session.ts` ⭐ -13. `lib/keycloak.ts` - -### API Routes (2 fichiers) -14. `app/api/auth/end-sso-session/route.ts` ⭐ -15. `app/api/auth/refresh-keycloak-session/route.ts` - -### Layout Root (1 fichier) -16. `app/layout.tsx` - -### Iframe (1 fichier) -17. `app/components/responsive-iframe.tsx` - -### Types (1 fichier) -18. `types/next-auth.d.ts` - -**Total : 18 fichiers principaux** - ---- - -## 🎯 FICHIERS CRITIQUES (⭐) - -Les fichiers marqués ⭐ sont **critiques** pour le flow login/logout : - -1. **`app/api/auth/options.ts`** - Configuration NextAuth, callbacks JWT/session -2. **`app/signin/page.tsx`** - Logique complexe de détection logout/auto-login -3. **`components/auth/signout-handler.tsx`** - Flow complet de logout -4. **`lib/session.ts`** - Gestion des cookies (suppression) -5. **`components/layout/layout-wrapper.tsx`** - Écoute logout depuis iframes -6. **`components/main-nav.tsx`** - Bouton logout -7. **`app/api/auth/end-sso-session/route.ts`** - Termine session SSO Keycloak - ---- - -## 📝 NOTES IMPORTANTES - -1. **Cookies NextAuth** : Gérés automatiquement par NextAuth, pas besoin de manipulation manuelle sauf pour suppression -2. **Cookies Keycloak** : Ne peuvent pas être supprimés depuis le dashboard si domaine différent (limitation navigateur) -3. **Session SSO** : Doit être terminée via Admin API Keycloak pour être complètement supprimée -4. **Auto-login** : Logique complexe pour distinguer nouvel utilisateur vs session invalidée -5. **Iframe logout** : Communication via `postMessage` pour synchroniser logout - ---- - -**Document créé le** : $(date) -**Dernière mise à jour** : Analyse complète du workflow login/logout avec focus sur les cookies - diff --git a/LOGOUT_LOGIN_FLOW_TRACE.md b/LOGOUT_LOGIN_FLOW_TRACE.md deleted file mode 100644 index 42a74033..00000000 --- a/LOGOUT_LOGIN_FLOW_TRACE.md +++ /dev/null @@ -1,305 +0,0 @@ -# Full Logout/Login Flow Trace - -## Issue 1: No Credentials Asked After Logout/Login - -### Current Flow Trace - -#### Step 1: User Clicks Logout -``` -Location: components/main-nav.tsx (line 364) -Action: onClick handler triggered - -1. sessionStorage.setItem('just_logged_out', 'true') -2. document.cookie = 'logout_in_progress=true; path=/; max-age=60' -3. clearAuthCookies() - Clears NextAuth cookies client-side -4. signOut({ callbackUrl: '/signin?logout=true', redirect: false }) - → Calls NextAuth /api/auth/signout endpoint - → Clears NextAuth session cookie server-side -5. window.location.replace(keycloakLogoutUrl) - → Redirects to: ${KEYCLOAK_ISSUER}/protocol/openid-connect/logout - → Parameters: - - post_logout_redirect_uri: /signin?logout=true - - id_token_hint: -``` - -#### Step 2: Keycloak Logout Endpoint -``` -Location: Keycloak Server -URL: ${KEYCLOAK_ISSUER}/protocol/openid-connect/logout - -Expected Behavior: -- Keycloak should invalidate the session -- Keycloak should clear session cookies: - - KEYCLOAK_SESSION (main session cookie) - - KEYCLOAK_SESSION_LEGACY - - KEYCLOAK_IDENTITY (identity cookie) - - KEYCLOAK_IDENTITY_LEGACY - - AUTH_SESSION_ID - - KC_RESTART (if exists) - -ACTUAL BEHAVIOR (PROBLEM): -- Keycloak logout endpoint with id_token_hint SHOULD clear cookies -- BUT: Keycloak might have SSO session that persists across clients -- OR: Cookies might not be cleared if domain/path mismatch -- OR: Keycloak might set new cookies during redirect -``` - -#### Step 3: Redirect Back to Signin -``` -Location: app/signin/page.tsx -URL: /signin?logout=true - -1. Component mounts -2. useEffect checks for logout flag (line 16-45) - - Sets isLogoutRedirect.current = true - - Removes 'just_logged_out' from sessionStorage - - Clears OAuth params from URL -3. Shows logout message with "Se connecter" button -4. User clicks "Se connecter" button (line 143-148) - - Calls: signIn("keycloak", { callbackUrl: "/" }) -``` - -#### Step 4: Keycloak Authorization Request -``` -Location: NextAuth → Keycloak -URL: ${KEYCLOAK_ISSUER}/protocol/openid-connect/auth - -Parameters sent: -- client_id: KEYCLOAK_CLIENT_ID -- redirect_uri: ${NEXTAUTH_URL}/api/auth/callback/keycloak -- response_type: code -- scope: openid profile email roles -- state: -- code_challenge: (if PKCE enabled) - -KEYCLOAK BEHAVIOR: -1. Keycloak receives authorization request -2. Keycloak checks for existing session cookies -3. IF Keycloak session cookies still exist: - → Keycloak finds valid SSO session - → Keycloak auto-authenticates user (no login prompt) - → Keycloak redirects back with authorization code -4. IF Keycloak session cookies are cleared: - → Keycloak shows login page - → User enters credentials - → Keycloak creates new session - → Keycloak redirects back with authorization code - -PROBLEM IDENTIFIED: -- Keycloak logout endpoint might not be clearing ALL session cookies -- OR: Keycloak has SSO session that persists (separate from client session) -- OR: Keycloak sets new cookies during the logout redirect process -- OR: Browser is preserving cookies due to SameSite/domain issues -``` - -### Root Cause Analysis - -**Problem**: Keycloak SSO Session Persistence - -Keycloak maintains two types of sessions: -1. **Client Session** (per OAuth client) - Cleared by logout endpoint -2. **SSO Session** (realm-wide) - May persist even after client logout - -When you call: -``` -GET /protocol/openid-connect/logout?id_token_hint=...&post_logout_redirect_uri=... -``` - -Keycloak behavior: -- ✅ Clears the **client session** for that specific OAuth client -- ✅ Invalidates tokens for that client -- ❌ **MIGHT NOT** clear the **SSO session** (realm-wide session) -- ❌ **MIGHT NOT** clear all session cookies if cookies are set with different domain/path - -**Why SSO Session Persists:** -- Keycloak SSO session is realm-wide, not client-specific -- Multiple clients can share the same SSO session -- Logging out from one client doesn't necessarily log out from the realm -- The SSO session cookie (KEYCLOAK_SESSION) might persist - -**When User Clicks "Se connecter":** -1. Redirects to Keycloak authorization endpoint -2. Keycloak checks for SSO session cookie -3. If SSO session cookie exists → Auto-authenticates (no credentials asked) -4. If SSO session cookie cleared → Shows login page - ---- - -## Issue 2: Cannot Logout from Iframe Application - -### Current Flow Trace - -#### Step 1: User in Iframe Application -``` -Location: Iframe application (e.g., /parole, /gite, etc.) -State: -- Dashboard has NextAuth session -- Keycloak session cookies exist -- Iframe app authenticated via Keycloak cookies -``` - -#### Step 2: User Clicks Logout in Iframe -``` -Location: Iframe application's logout button -Action: Iframe app's logout handler - -Possible Scenarios: - -Scenario A: Iframe calls its own logout endpoint -- Iframe app might call: POST /api/logout (iframe app's endpoint) -- This might clear iframe app's session -- BUT: Keycloak session cookies might still exist -- Result: Iframe app logs out, but Keycloak session persists - -Scenario B: Iframe calls Keycloak logout -- Iframe app might call: GET ${KEYCLOAK_ISSUER}/protocol/openid-connect/logout -- Keycloak clears session cookies -- BUT: NextAuth dashboard session still exists -- Result: Keycloak session cleared, but dashboard still logged in - -Scenario C: Iframe doesn't have logout -- Iframe app might not have logout functionality -- User stuck in iframe with no way to logout -``` - -#### Step 3: What Happens After Iframe Logout -``` -If iframe calls Keycloak logout: -1. Keycloak invalidates session -2. Keycloak clears session cookies -3. NextAuth dashboard still has valid JWT (30-day expiration) -4. NextAuth doesn't know Keycloak session was cleared -5. Dashboard widgets still show (using NextAuth session) -6. Iframe apps can't authenticate (Keycloak cookies cleared) -``` - -### Root Cause Analysis - -**Problem**: No Communication Between Iframe and Dashboard - -When iframe app logs out: -1. **Iframe app** calls Keycloak logout → Clears Keycloak cookies -2. **Dashboard** doesn't know about this → NextAuth session still valid -3. **Dashboard widgets** continue to work (using NextAuth session) -4. **Iframe apps** can't authenticate (Keycloak cookies gone) - -**Why Dashboard Doesn't Know:** -- NextAuth session is independent of Keycloak session cookies -- NextAuth JWT has 30-day expiration -- No mechanism to detect Keycloak session invalidation from iframe -- Dashboard only detects invalidation when trying to refresh tokens - -**Why Iframe Can't Logout Properly:** -- Iframe apps rely on Keycloak cookies for SSO -- If iframe calls Keycloak logout, it clears cookies -- But dashboard session persists -- If iframe doesn't call logout, user can't logout from iframe -- No way for iframe to trigger dashboard logout - ---- - -## Key Findings - -### Finding 1: Keycloak SSO Session Persistence -- **Issue**: Keycloak logout endpoint might not clear SSO session -- **Evidence**: User auto-authenticates without credentials after logout -- **Root Cause**: SSO session cookie persists after client logout -- **Impact**: Security issue - user should be asked for credentials - -### Finding 2: Missing `prompt=login` Parameter -- **Issue**: When calling `signIn("keycloak")`, no `prompt=login` parameter is sent -- **Evidence**: Keycloak auto-authenticates if SSO session exists -- **Root Cause**: NextAuth Keycloak provider doesn't force login prompt -- **Impact**: User bypasses credential check - -### Finding 3: Iframe Logout Isolation -- **Issue**: Iframe logout doesn't affect dashboard session -- **Evidence**: Dashboard widgets still show after iframe logout -- **Root Cause**: No communication mechanism between iframe and dashboard -- **Impact**: Inconsistent logout state - -### Finding 4: No Cross-Origin Logout Communication -- **Issue**: Iframe can't trigger dashboard logout -- **Evidence**: User stuck in iframe after logout -- **Root Cause**: No postMessage or other communication mechanism -- **Impact**: Poor user experience - ---- - -## Flow Diagram - -### Current Logout Flow (Dashboard) -``` -User clicks logout - ↓ -Clear NextAuth cookies - ↓ -Call NextAuth signOut() - ↓ -Redirect to Keycloak logout - ↓ -Keycloak clears client session - ↓ -Keycloak MAY clear SSO session (not guaranteed) - ↓ -Redirect to /signin?logout=true - ↓ -User clicks "Se connecter" - ↓ -Redirect to Keycloak auth - ↓ -Keycloak checks SSO session - ↓ -IF SSO session exists → Auto-authenticate (NO CREDENTIALS) -IF SSO session cleared → Show login page -``` - -### Current Iframe Logout Flow -``` -User in iframe clicks logout - ↓ -Iframe app calls logout (varies by app) - ↓ -IF calls Keycloak logout: - → Keycloak clears session cookies - → Dashboard session still valid - → Widgets still show - → Iframe can't authenticate -IF doesn't call logout: - → User stuck in iframe - → No way to logout -``` - ---- - -## Recommendations - -### For Issue 1 (No Credentials After Logout) -1. **Add `prompt=login` parameter** to Keycloak authorization request - - Forces Keycloak to show login page even if SSO session exists - - Location: `app/api/auth/options.ts` - KeycloakProvider authorization params - -2. **Clear Keycloak SSO session explicitly** - - Add `kc_action=LOGOUT` parameter to logout URL - - Or call Keycloak admin API to end SSO session - -3. **Clear Keycloak cookies client-side** - - After Keycloak logout redirect, clear any remaining Keycloak cookies - - Check for cookies with Keycloak domain - -### For Issue 2 (Iframe Logout) -1. **Implement postMessage communication** - - Iframe sends logout message to parent - - Dashboard listens for logout messages - - Dashboard triggers logout when iframe logs out - -2. **Detect Keycloak session invalidation** - - Poll Keycloak session status - - Detect when Keycloak cookies are cleared - - Automatically logout dashboard - -3. **Unified logout endpoint** - - Create API endpoint that logs out both dashboard and Keycloak - - Iframe apps call this endpoint - - Ensures synchronized logout - diff --git a/LOG_ANALYSIS_FEEDBACK.md b/LOG_ANALYSIS_FEEDBACK.md deleted file mode 100644 index cdf1348c..00000000 --- a/LOG_ANALYSIS_FEEDBACK.md +++ /dev/null @@ -1,239 +0,0 @@ -# Log Analysis & Feedback Report - -**Date**: 2026-01-01 -**Log File**: `log` -**Analysis Scope**: Application startup, notifications, session management, API calls - ---- - -## 🔴 Critical Issues - -### 1. Excessive Session Callback Logging (HIGH PRIORITY) - -**Problem**: -- **10+ session callbacks** triggered in a short period -- Each `getServerSession()` call triggers verbose logging -- Logs show `=== SESSION CALLBACK START ===` and `=== SESSION CALLBACK END ===` repeatedly - -**Root Cause**: -- Every API route calls `getServerSession(authOptions)` -- Root layout (`app/layout.tsx`) also calls it -- Session callback has extensive logging (lines 407-415 in `app/api/auth/options.ts`) - -**Impact**: -- ⚠️ **Performance**: Unnecessary logging overhead on every request -- ⚠️ **Log Noise**: Makes it hard to find actual issues -- ⚠️ **Debugging**: Difficult to identify real problems - -**Recommendation**: -```typescript -// In app/api/auth/options.ts, line 405-415 -async session({ session, token }) { - try { - // Only log in development or when there's an error - if (process.env.NODE_ENV === 'development' || token.error) { - console.log('=== SESSION CALLBACK START ==='); - console.log('Token error:', token.error); - // ... rest of logging - } - - // Or use a debug flag - const DEBUG_SESSION = process.env.DEBUG_SESSION === 'true'; - if (DEBUG_SESSION) { - console.log('=== SESSION CALLBACK START ==='); - // ... logging - } - - // ... rest of callback - } -} -``` - -**Priority**: 🔴 **HIGH** - Should be fixed immediately - ---- - -### 2. Missing markAsRead/markAllAsRead Logs - -**Problem**: -- No API calls to `/api/notifications/[id]/read` or `/api/notifications/read-all` in the log -- User reported notification count not updating after marking as read - -**Possible Causes**: -1. User didn't actually mark notifications as read during this log session -2. API calls are failing silently (network errors, CORS, etc.) -3. Client-side code isn't calling the API correctly -4. API routes aren't logging their calls - -**Investigation Steps**: -1. Add logging to mark-as-read API routes: - ```typescript - // In app/api/notifications/[id]/read/route.ts - export async function POST(request: Request, context: { params: Promise<{ id: string }> }) { - console.log('[NOTIFICATION_API] Mark as read called', { id: context.params?.id }); - // ... rest of code - } - ``` - -2. Check browser console for client-side errors -3. Verify network tab shows the API calls being made -4. Test the mark-as-read functionality while monitoring logs - -**Priority**: 🟡 **MEDIUM** - Needs investigation - ---- - -## ✅ Positive Observations - -### 1. Notification Service Working Correctly -- ✅ Service initialized properly -- ✅ Count fetched: **100 total, 66 unread** -- ✅ List fetched: **20 notifications** -- ✅ Caching working: `Cached notification counts for user` -- ✅ Leantime adapter functioning correctly - -### 2. Infrastructure Healthy -- ✅ Redis connection successful -- ✅ Database queries working (Prisma) -- ✅ IMAP connection successful (633ms) -- ✅ External APIs responding (News, Leantime, Rocket.Chat) - -### 3. Data Flow -- ✅ Session creation successful -- ✅ User authentication working -- ✅ Token refresh logic functioning -- ✅ OAuth token management working - ---- - -## 📊 Performance Metrics from Log - -| Metric | Value | Status | -|--------|-------|--------| -| Redis Connection | ✅ Success | Good | -| IMAP Connection Time | 633ms | Acceptable | -| Notification Count Fetch | ✅ Success | Good | -| Notification List Fetch | ✅ Success | Good | -| Session Callbacks | 10+ in short period | ⚠️ Too many | -| Database Queries | ✅ Working | Good | - ---- - -## 🔧 Recommended Actions - -### Immediate (This Week) - -1. **Reduce Session Callback Logging** - - Add environment-based conditional logging - - Only log errors or use `DEBUG_SESSION` flag - - **File**: `app/api/auth/options.ts` - -2. **Add Logging to Mark-as-Read Endpoints** - - Add console.log to track when mark-as-read is called - - Log success/failure - - **Files**: - - `app/api/notifications/[id]/read/route.ts` - - `app/api/notifications/read-all/route.ts` - -3. **Test Notification Mark-as-Read Flow** - - Monitor logs while marking notifications as read - - Verify API calls are being made - - Check if cache invalidation is working - -### Short Term (Next Sprint) - -4. **Implement Request Deduplication** - - Use the `request-deduplication` utility - - Prevent duplicate API calls - - **Already planned in unified refresh system** - -5. **Add Performance Monitoring** - - Track API call frequency - - Monitor session callback frequency - - Alert on excessive calls - -6. **Optimize Session Access** - - Consider caching session data - - Reduce redundant `getServerSession()` calls - - Use session context where possible - ---- - -## 🐛 Potential Issues Not Visible in Log - -### 1. Client-Side Errors -- Browser console errors not captured in server logs -- Network request failures -- React component errors - -### 2. Cache Invalidation -- No logs showing cache invalidation after mark-as-read -- May need to verify `invalidateCache()` is being called - -### 3. Race Conditions -- Multiple simultaneous API calls -- State update conflicts -- Not visible in single-threaded log - ---- - -## 📝 Log Patterns Analysis - -### Session Callback Pattern -``` -=== SESSION CALLBACK START === -Token error: undefined -Has accessToken: true -Has refreshToken: true -Token role: [...] -Token sub: 203cbc91-61ab-47a2-95d2-b5e1159327d7 -Token email: a.tmiri@clm.foundation -... -✅ Session created successfully -=== SESSION CALLBACK END === -``` - -**Frequency**: Every API call that uses `getServerSession()` - -**Recommendation**: Reduce to error-only logging - ---- - -### Notification Service Pattern -``` -[NOTIFICATION_SERVICE] getNotificationCount called for user ... -[LEANTIME_ADAPTER] getNotificationCount called for userId: ... -[LEANTIME_ADAPTER] Notification counts: { total: 100, unread: 66 } -[NOTIFICATION_SERVICE] Cached notification counts for user ... -``` - -**Status**: ✅ Working correctly - ---- - -## 🎯 Next Steps - -1. **Immediate**: Fix session callback logging (5 minutes) -2. **Today**: Add logging to mark-as-read endpoints (10 minutes) -3. **This Week**: Test notification mark-as-read flow end-to-end -4. **Next Sprint**: Implement unified refresh system (already planned) - ---- - -## 📌 Summary - -**Overall Assessment**: ✅ **System is functioning correctly** - -**Main Concerns**: -1. Excessive logging causing performance overhead -2. Missing visibility into mark-as-read operations -3. Need to verify notification count update flow - -**Confidence Level**: 🟢 **HIGH** - Core functionality working, minor optimizations needed - ---- - -**Generated**: 2026-01-01 -**Analyst**: AI Code Assistant -**Next Review**: After implementing fixes - diff --git a/LOG_FLOW_ANALYSIS.md b/LOG_FLOW_ANALYSIS.md deleted file mode 100644 index 18f93d46..00000000 --- a/LOG_FLOW_ANALYSIS.md +++ /dev/null @@ -1,342 +0,0 @@ -# Log Flow Analysis - Application Startup & Runtime - -**Date**: 2026-01-01 -**Log Source**: Application startup and initial page load -**Analysis Focus**: Flow patterns, errors, and system behavior - ---- - -## 🔍 Executive Summary - -**Overall Status**: 🟡 **MOSTLY HEALTHY** with one non-critical error - -**Key Findings**: -1. ⚠️ **Syntax Error**: Non-critical error during startup (doesn't block execution) -2. ✅ **Session Management**: Working correctly (5 session callbacks during startup) -3. ✅ **Notification Service**: Initialized and functioning (100 total, 66 unread) -4. ✅ **External Services**: All connecting successfully -5. ⚠️ **No Mark-as-Read Activity**: No API calls to mark notifications as read - ---- - -## 📊 Flow Breakdown - -### Phase 1: Application Startup (Lines 1-33) - -``` -1. Next.js starts (1313ms) -2. Redis connection established ✅ -3. Microsoft OAuth configuration loaded ✅ -4. ⚠️ SyntaxError: Unexpected identifier 'http' (line 29) -5. Redis connection warmed up ✅ -``` - -**Observations**: -- ✅ Startup is fast (1.3 seconds) -- ✅ Redis connection successful -- ⚠️ **Syntax Error** appears but doesn't block execution -- Error occurs between Redis warmup calls - -**Syntax Error Details**: -``` -⨯ SyntaxError: Unexpected identifier 'http' - at Object.Function [as get] () { - digest: '2421336728' - } -``` - -**Analysis**: -- Error is in a route handler (Function.get) -- Likely related to a route file with syntax issue -- Doesn't crash the application -- May be related to dynamic route generation - -**Recommendation**: Investigate route files for syntax errors, especially those using `http` in identifiers. - ---- - -### Phase 2: Initial Session Creation (Lines 34-71) - -``` -1. Session callback triggered -2. Token validation ✅ -3. User roles extracted ✅ -4. Session created successfully ✅ -``` - -**Session Details**: -- User ID: `203cbc91-61ab-47a2-95d2-b5e1159327d7` -- Email: `a.tmiri@clm.foundation` -- Roles: `['expression', 'entrepreneurship', 'admin', 'dataintelligence', 'mediation', 'mentors']` -- Tokens: Access token ✅, Refresh token ✅ - -**Status**: ✅ **HEALTHY** - ---- - -### Phase 3: Rocket.Chat Integration (Lines 72-91) - -``` -1. Rocket.Chat base URL: https://parole.slm-lab.net ✅ -2. Users list fetched (13 users) ✅ -3. User found: aminetmiri ✅ -4. Subscriptions filtered (1 room) ✅ -5. Messages fetched (5 messages) ✅ -6. Messages cached ✅ -7. ⚠️ "No valid session or email found" (line 92) -``` - -**Observations**: -- ✅ Rocket.Chat integration working -- ✅ User authentication successful -- ✅ Messages retrieved and cached -- ⚠️ Warning message at line 92 (may be from another service) - -**Status**: ✅ **HEALTHY** (warning is likely from a different service) - ---- - -### Phase 4: Additional Session Callbacks (Lines 93-169) - -``` -1. Session callback #2 (lines 93-130) -2. Session callback #3 (lines 132-169) -``` - -**Pattern**: Multiple session callbacks during initial page load - -**Frequency**: 3 session callbacks in ~40 lines of log - -**Analysis**: -- Normal behavior for Next.js with multiple API routes -- Each `getServerSession()` call triggers session callback -- All callbacks successful ✅ - -**Status**: ✅ **NORMAL** (but verbose logging as discussed) - ---- - -### Phase 5: Notification Service Initialization (Lines 170-246) - -``` -1. Notification service instance created ✅ -2. Leantime adapter initialized ✅ -3. Adapter registered ✅ -4. getNotificationCount called ✅ -5. Leantime API called ✅ -6. Response received (200) ✅ -7. Notifications parsed ✅ -8. Count calculated: 100 total, 66 unread ✅ -9. Counts cached ✅ -``` - -**Notification Details**: -- **Total**: 100 notifications -- **Unread**: 66 notifications -- **Source**: Leantime -- **Status**: ✅ **WORKING CORRECTLY** - -**Flow**: -``` -[NOTIFICATION_SERVICE] → [LEANTIME_ADAPTER] → Leantime API → Parse → Cache -``` - -**Status**: ✅ **HEALTHY** - ---- - -### Phase 6: Additional Operations (Lines 247-289) - -``` -1. IMAP pool status logged -2. Session callback #4 (lines 248-285) -3. Cached messages used -4. IMAP pool status logged again -``` - -**Observations**: -- ✅ IMAP connection pool healthy (0 active, max 20) -- ✅ Session callbacks continuing (normal) -- ✅ Caching working (messages from cache) - -**Status**: ✅ **HEALTHY** - ---- - -## 🔴 Issues Identified - -### 1. Syntax Error (Line 29) ⚠️ - -**Error**: -``` -⨯ SyntaxError: Unexpected identifier 'http' - at Object.Function [as get] () -``` - -**Impact**: -- ⚠️ **Low**: Doesn't crash application -- ⚠️ **Unknown**: May affect specific route -- ⚠️ **Non-blocking**: Application continues normally - -**Possible Causes**: -1. Route file with syntax error -2. Dynamic route generation issue -3. Template literal or string interpolation problem -4. Environment variable parsing issue - -**Investigation Steps**: -1. Search codebase for routes using `http` as identifier -2. Check dynamic route files -3. Review route handlers for syntax errors -4. Check Next.js route generation - -**Priority**: 🟡 **MEDIUM** - Should be fixed but not blocking - ---- - -### 2. "No valid session or email found" (Line 92) ⚠️ - -**Message**: `No valid session or email found` - -**Context**: Appears after Rocket.Chat operations - -**Analysis**: -- May be from a different service/route -- Doesn't affect Rocket.Chat functionality -- Could be from email service or another API route - -**Investigation**: Check which service logs this message - -**Priority**: 🟡 **LOW** - Appears to be a warning, not an error - ---- - -### 3. No Mark-as-Read Activity ⚠️ - -**Observation**: No `[NOTIFICATION_API]` log entries - -**Expected**: Should see logs when user marks notifications as read - -**Possible Reasons**: -1. User hasn't tested mark-as-read yet -2. API calls not reaching server -3. Client-side errors preventing API calls - -**Status**: ⏳ **PENDING TESTING** - -**Action**: Test mark-as-read functionality and check for new log entries - ---- - -## ✅ Positive Observations - -### 1. Fast Startup -- ✅ Application ready in 1.3 seconds -- ✅ All services initialized quickly - -### 2. Session Management -- ✅ All session callbacks successful -- ✅ Token validation working -- ✅ User roles extracted correctly - -### 3. Notification Service -- ✅ Service initialized correctly -- ✅ Leantime adapter working -- ✅ API calls successful -- ✅ Caching functioning - -### 4. External Services -- ✅ Redis connected -- ✅ Rocket.Chat connected -- ✅ Leantime API responding -- ✅ IMAP pool healthy - ---- - -## 📈 Performance Metrics - -| Metric | Value | Status | -|--------|-------|--------| -| Startup Time | 1313ms | ✅ Good | -| Redis Connection | ✅ Success | ✅ Good | -| Session Callbacks | 5 during startup | ✅ Normal | -| Notification Count | 100 total, 66 unread | ✅ Working | -| Rocket.Chat | ✅ Connected | ✅ Good | -| IMAP Pool | 0/20 active | ✅ Healthy | - ---- - -## 🔄 Flow Patterns - -### Session Callback Pattern -``` -Every getServerSession() call → Session callback → Token validation → Session created -``` - -**Frequency**: 5 times during startup (normal for multi-route page) - -**Recommendation**: Conditional logging (as discussed in impact analysis) - ---- - -### Notification Service Pattern -``` -Service init → Adapter registration → API call → Parse → Cache -``` - -**Status**: ✅ Working correctly - ---- - -## 🎯 Recommendations - -### Immediate Actions - -1. **Investigate Syntax Error** 🔴 - - Search for route files with `http` identifier - - Check dynamic routes - - Fix syntax error - -2. **Test Mark-as-Read** 🟡 - - Mark a notification as read - - Check logs for `[NOTIFICATION_API]` entries - - Verify notification count updates - -3. **Identify "No valid session" Source** 🟡 - - Find which service logs this message - - Determine if it's an error or warning - - Fix if necessary - -### Future Improvements - -4. **Implement Conditional Session Logging** (as planned) - - Add `DEBUG_SESSION` flag - - Reduce production logging - - Keep error logging - -5. **Add Error Monitoring** - - Track syntax errors - - Monitor route handler failures - - Alert on critical errors - ---- - -## 📝 Summary - -**Overall Assessment**: 🟢 **HEALTHY** with minor issues - -**Critical Issues**: 0 -**Warnings**: 2 (syntax error, "no valid session" message) -**Working Correctly**: ✅ All core functionality - -**Next Steps**: -1. Fix syntax error (investigate route files) -2. Test mark-as-read functionality -3. Identify source of "no valid session" message -4. Proceed with conditional session logging (when ready) - ---- - -**Generated**: 2026-01-01 -**Status**: Ready for action items - diff --git a/LOG_SEARCH_INSTRUCTIONS.md b/LOG_SEARCH_INSTRUCTIONS.md deleted file mode 100644 index 56d946d8..00000000 --- a/LOG_SEARCH_INSTRUCTIONS.md +++ /dev/null @@ -1,96 +0,0 @@ -# Log Search Instructions - Mark All As Read - -**Purpose**: Find the exact error causing mark-all-as-read to fail - ---- - -## 🔍 What to Do - -After you do `rm -rf .next && npm run build && npm start` and test "mark all as read", please: - -### Option 1: Search for Specific Markers - -In your log output, search for these exact strings: - -```bash -# Search for the adapter start marker -grep "===== markAllAsRead START =====" log - -# Search for all notification service logs -grep "NOTIFICATION_SERVICE.*markAllAsRead" log - -# Search for all leantime adapter logs -grep "LEANTIME_ADAPTER.*markAllAsRead" log - -# Search for API logs -grep "NOTIFICATION_API.*Mark all as read" log -``` - -### Option 2: Provide Complete Log Snippet - -When you test "mark all as read", copy the **COMPLETE** log output from: -- **Before**: 5-10 lines before `[NOTIFICATION_API] Mark all as read endpoint called` -- **After**: 50-100 lines after the failure - -This will show us the full flow. - ---- - -## 🎯 What We're Looking For - -### Expected Log Sequence - -``` -[NOTIFICATION_API] Mark all as read endpoint called -[NOTIFICATION_API] Mark all as read - Processing { userId: '...', timestamp: '...' } -[NOTIFICATION_SERVICE] markAllAsRead called for user ... -[NOTIFICATION_SERVICE] Available adapters: leantime -[NOTIFICATION_SERVICE] Processing adapter: leantime -[NOTIFICATION_SERVICE] Adapter leantime is configured: true -[NOTIFICATION_SERVICE] Calling markAllAsRead on adapter leantime -[LEANTIME_ADAPTER] ===== markAllAsRead START ===== ← MUST APPEAR -[LEANTIME_ADAPTER] markAllAsRead called for userId: ... -[LEANTIME_ADAPTER] API URL: ... -[LEANTIME_ADAPTER] Has API Token: true -[LEANTIME_ADAPTER] markAllAsRead - User email: ... -[LEANTIME_ADAPTER] markAllAsRead - Leantime user ID: ... -[LEANTIME_ADAPTER] markAllAsRead - Request body: {...} -[LEANTIME_ADAPTER] markAllAsRead - Response status: XXX -[LEANTIME_ADAPTER] markAllAsRead - Response body: {...} -[LEANTIME_ADAPTER] markAllAsRead - API Error: {...} ← This will show the actual error -[NOTIFICATION_SERVICE] Adapter leantime markAllAsRead result: false -``` - ---- - -## ❓ Questions - -1. **Do you see `[NOTIFICATION_SERVICE] markAllAsRead called for user`?** - - If NO → Service layer not being called - - If YES → Continue to next question - -2. **Do you see `[NOTIFICATION_SERVICE] Calling markAllAsRead on adapter leantime`?** - - If NO → Adapter not being called - - If YES → Continue to next question - -3. **Do you see `===== markAllAsRead START =====`?** - - If NO → Adapter method not executing (very strange!) - - If YES → We'll see the Leantime API error - ---- - -## 🔧 Quick Test - -After restart, run this command to see if the marker appears: - -```bash -# Test mark all as read, then immediately: -tail -n 200 log | grep -A 50 "Mark all as read" -``` - -This will show the last 200 lines of the log, filtered for mark-all-as-read operations, with 50 lines of context after each match. - ---- - -**Status**: Enhanced logging with multiple output methods. Awaiting complete log output to identify the exact failure point. - diff --git a/MARK_ALL_READ_CACHE_ISSUE.md b/MARK_ALL_READ_CACHE_ISSUE.md deleted file mode 100644 index 499d9061..00000000 --- a/MARK_ALL_READ_CACHE_ISSUE.md +++ /dev/null @@ -1,140 +0,0 @@ -# Mark All As Read - Cache Issue Analysis - -**Date**: 2026-01-01 -**Issue**: After marking all as read, list is empty but count still shows 66 - ---- - -## 🔍 Problem Analysis - -### Current Flow - -1. **User clicks "Mark all as read"** -2. **`markAllAsRead()` is called** -3. **Fetches notifications**: `this.getNotifications(userId, 1, 1000)` - - ⚠️ **PROBLEM**: This goes through `NotificationService.getNotifications()` - - ⚠️ **PROBLEM**: Which uses **CACHED** data if available - - ⚠️ **PROBLEM**: Cached notifications still have `isRead: false` -4. **Filters unread**: Gets 66 unread from cached data -5. **Marks each as read**: Calls Leantime API for each -6. **Invalidates cache**: After marking completes -7. **Count is fetched**: But might use stale cache or be fetched before invalidation - -### The Issue - -**Cache Race Condition**: -- `markAllAsRead` uses cached notifications (which are stale) -- Marks them as read in Leantime -- Invalidates cache -- But count might be fetched from cache **before** invalidation completes -- Or count cache might not be properly invalidated - -**Why List is Empty**: -- After marking, all notifications are read -- List might filter to show only unread -- So list is empty (correct behavior) -- But count still shows 66 (stale cache) - ---- - -## 🔧 Root Causes - -### 1. Using Cached Data in `markAllAsRead` - -**Current Code**: -```typescript -// In markAllAsRead -const allNotifications = await this.getNotifications(userId, 1, 1000); -``` - -**Problem**: `getNotifications()` uses cache, so we're working with stale data. - -**Solution**: Fetch directly from Leantime API, bypassing cache. - ---- - -### 2. Cache Invalidation Timing - -**Current Flow**: -1. Mark all as read (uses cached data) -2. Invalidate cache -3. Count is fetched (might use stale cache if fetched too soon) - -**Problem**: Race condition between invalidation and count fetch. - -**Solution**: -- Invalidate cache **before** marking (or fetch fresh data) -- Force immediate count refresh after marking -- Add delay before count fetch to ensure cache is cleared - ---- - -### 3. Count Cache Not Properly Invalidated - -**Current Code**: -```typescript -if (success) { - await this.invalidateCache(userId); -} -``` - -**Problem**: If `markAllAsRead` fails partially, cache might not be invalidated. - -**Solution**: Always invalidate cache, even on partial success. - ---- - -## ✅ Recommended Fixes - -### Fix 1: Bypass Cache in `markAllAsRead` - -**Change**: Fetch notifications directly from Leantime API, not through cached service. - -**Implementation**: -- Add a method to fetch notifications directly from adapter (bypassing cache) -- Or add a `forceRefresh` parameter to `getNotifications` -- Or fetch directly in `markAllAsRead` using Leantime API - -### Fix 2: Always Invalidate Cache - -**Change**: Invalidate cache even if some notifications fail to mark. - -**Implementation**: -- Invalidate cache if **any** notifications were successfully marked -- Not just if **all** succeeded - -### Fix 3: Force Fresh Count After Marking - -**Change**: After marking, force an immediate fresh count fetch. - -**Implementation**: -- After `markAllAsRead` completes, immediately call `getNotificationCount()` with cache bypass -- Or add a delay before count fetch to ensure cache is cleared - ---- - -## 📊 Expected Behavior After Fixes - -### After Mark All As Read - -**Before**: -- List: Empty (all read) ✅ -- Count: 66 (stale cache) ❌ - -**After**: -- List: Empty (all read) ✅ -- Count: 0 (fresh data) ✅ - ---- - -## 🎯 Next Steps - -1. **Fix cache usage in `markAllAsRead`**: Fetch fresh data, not cached -2. **Improve cache invalidation**: Always invalidate, even on partial success -3. **Force count refresh**: Immediately fetch fresh count after marking -4. **Test**: Verify count updates correctly after marking - ---- - -**Status**: Analysis complete. Ready to implement fixes. - diff --git a/MARK_ALL_READ_DIAGNOSTIC.md b/MARK_ALL_READ_DIAGNOSTIC.md deleted file mode 100644 index 4ea7e250..00000000 --- a/MARK_ALL_READ_DIAGNOSTIC.md +++ /dev/null @@ -1,132 +0,0 @@ -# Mark All As Read - Diagnostic Guide - -**Issue**: Adapter returns `false` but no detailed logs appear - ---- - -## 🔍 Current Situation - -**What We See**: -``` -[NOTIFICATION_SERVICE] Adapter leantime markAllAsRead result: false -[NOTIFICATION_SERVICE] markAllAsRead results: [ false ] -[NOTIFICATION_SERVICE] markAllAsRead overall success: false -``` - -**What's Missing**: -- `[NOTIFICATION_SERVICE] markAllAsRead called for user ...` -- `[NOTIFICATION_SERVICE] Processing adapter: leantime` -- `[NOTIFICATION_SERVICE] Calling markAllAsRead on adapter leantime` -- `[LEANTIME_ADAPTER] ===== markAllAsRead START =====` ← **NEW: Very prominent marker** - ---- - -## 🚨 Possible Causes - -### 1. Server Not Fully Restarted -**Solution**: Do a **hard restart**: -```bash -# Stop completely -sudo npm stop -# Or kill the process -sudo pkill -f "next start" - -# Wait a few seconds -sleep 3 - -# Start fresh -sudo npm start -``` - -### 2. Next.js Build Cache -**Solution**: Clear cache and rebuild: -```bash -rm -rf .next -sudo npm start -``` - -### 3. Log Buffering/Filtering -**Solution**: Check if logs are being filtered. Look for ALL logs around the mark-all-as-read operation. - -### 4. Code Not Deployed -**Solution**: Verify the file was saved and the server picked it up. - ---- - -## ✅ What to Look For After Restart - -### Expected Complete Log Flow - -When you click "Mark all as read", you should see **ALL** of these logs: - -``` -[NOTIFICATION_API] Mark all as read endpoint called -[NOTIFICATION_API] Mark all as read - Processing { userId: '...', timestamp: '...' } -[NOTIFICATION_SERVICE] markAllAsRead called for user ... -[NOTIFICATION_SERVICE] Available adapters: leantime -[NOTIFICATION_SERVICE] Processing adapter: leantime -[NOTIFICATION_SERVICE] Adapter leantime is configured: true -[NOTIFICATION_SERVICE] Calling markAllAsRead on adapter leantime -[LEANTIME_ADAPTER] ===== markAllAsRead START ===== ← VERY PROMINENT -[LEANTIME_ADAPTER] markAllAsRead called for userId: ... -[LEANTIME_ADAPTER] API URL: ... -[LEANTIME_ADAPTER] Has API Token: true -[LEANTIME_ADAPTER] markAllAsRead - User email: ... -[LEANTIME_ADAPTER] markAllAsRead - Leantime user ID: ... -[LEANTIME_ADAPTER] markAllAsRead - Request body: {...} -[LEANTIME_ADAPTER] markAllAsRead - API URL: ... -[LEANTIME_ADAPTER] markAllAsRead - Response status: XXX -[LEANTIME_ADAPTER] markAllAsRead - Response body: {...} -[LEANTIME_ADAPTER] markAllAsRead - Parsed response: {...} -[LEANTIME_ADAPTER] markAllAsRead - API Error: {...} ← This will show the actual error -[LEANTIME_ADAPTER] ===== markAllAsRead END (success: false) ===== -[NOTIFICATION_SERVICE] Adapter leantime markAllAsRead result: false -[NOTIFICATION_SERVICE] markAllAsRead results: [ false ] -[NOTIFICATION_SERVICE] markAllAsRead overall success: false -[NOTIFICATION_SERVICE] Not invalidating caches - operation failed -[NOTIFICATION_API] Mark all as read - Failed { userId: '...', duration: '...ms' } -``` - ---- - -## 🎯 Critical Check - -**After restarting**, search your logs for: -``` -===== markAllAsRead START ===== -``` - -If you **DON'T** see this line, the adapter method is **NOT** being called, which means: -- Server not restarted properly -- Code not deployed -- Different code path being used - -If you **DO** see this line, we'll have all the details we need to fix the Leantime API call. - ---- - -## 📋 Action Items - -1. ✅ **Hard Restart Server** (stop completely, wait, start) -2. ✅ **Test Mark All As Read** -3. ✅ **Search logs for `===== markAllAsRead START =====`** -4. ✅ **Share ALL logs** from the mark-all-as-read operation -5. ✅ **Look for `API Error:`** in the logs (this will show what Leantime is returning) - ---- - -## 🔧 If Logs Still Don't Appear - -If after restart you still don't see the `===== markAllAsRead START =====` log: - -1. **Verify file was saved**: Check `lib/services/notifications/leantime-adapter.ts` line 220-224 -2. **Check for syntax errors**: Run `npm run build` or check for TypeScript errors -3. **Verify server is using the file**: Check if there are multiple versions or build artifacts -4. **Check log output**: Make sure you're looking at the right log file/stream - ---- - -**Status**: Enhanced logging with prominent markers added. Awaiting server restart and test. - -**Next**: After restart, the `===== markAllAsRead START =====` marker will confirm the method is being called, and we'll see the exact Leantime API error. - diff --git a/MICROSOFT_OAUTH_ANALYSIS.md b/MICROSOFT_OAUTH_ANALYSIS.md deleted file mode 100644 index e7e9417b..00000000 --- a/MICROSOFT_OAUTH_ANALYSIS.md +++ /dev/null @@ -1,153 +0,0 @@ -# Microsoft OAuth Token Management Analysis - -## Current Implementation - -### Token Storage Locations - -1. **Redis Cache** (Primary for OAuth tokens) - - **Location**: `lib/redis.ts` → `cacheEmailCredentials()` - - **TTL**: 24 hours (`TTL.CREDENTIALS = 60 * 60 * 24`) - - **Stored**: `accessToken`, `refreshToken`, `tokenExpiry`, `useOAuth` - - **Key Format**: `email:credentials:${userId}:${accountId}` - -2. **Prisma Database** (Schema has fields but NOT used for OAuth tokens) - - **Location**: `prisma/schema.prisma` → `MailCredentials` model - - **Fields Available**: `refresh_token`, `access_token`, `token_expiry`, `use_oauth` - - **Current Status**: ❌ **Tokens are NOT saved to Prisma** (only Redis) - - **Code Comment**: "OAuth fields don't exist" (but they DO exist in schema!) - -### Token Refresh Flow - -**Location**: `lib/services/token-refresh.ts` → `ensureFreshToken()` - -1. Checks Redis for credentials -2. Validates token expiry (5-minute buffer) -3. Refreshes token if needed via Microsoft API -4. **Updates Redis only** (not Prisma) -5. Returns new access token - -### Issues Identified - -#### 🔴 Critical Issue #1: Refresh Tokens Not Persisted to Database - -**Problem**: -- Refresh tokens are only stored in Redis with 24-hour TTL -- If Redis is cleared, restarted, or TTL expires, refresh tokens are **permanently lost** -- Microsoft refresh tokens can last up to **90 days** (or indefinitely with `offline_access` scope) -- Users would need to re-authenticate if Redis data is lost - -**Impact**: -- ❌ Not viable for long-term production use -- ❌ Data loss risk on Redis restarts -- ❌ No backup/recovery mechanism - -#### 🟡 Issue #2: Token Refresh Doesn't Update Database - -**Problem**: -- When tokens are refreshed, only Redis is updated -- Prisma database still has old/expired tokens (if any) -- Schema has the fields but they're never populated - -**Impact**: -- ⚠️ Inconsistency between Redis and Database -- ⚠️ Can't recover from Redis cache loss - -#### 🟡 Issue #3: Missing Refresh Token in Logs - -From your logs: -``` -hasRefreshToken: false -``` - -This suggests the refresh token might not be properly saved or retrieved. - -### Microsoft OAuth Token Lifespan - -- **Access Token**: ~1 hour (3600 seconds) -- **Refresh Token**: Up to 90 days (with `offline_access` scope) -- **Token Refresh**: Returns new access token, may return new refresh token - -### Required Scopes - -Current implementation uses: -```typescript -const REQUIRED_SCOPES = [ - 'offline_access', // ✅ Required for long-lived refresh tokens - 'https://outlook.office.com/IMAP.AccessAsUser.All', - 'https://outlook.office.com/SMTP.Send' -].join(' '); -``` - -✅ `offline_access` is included - this is correct for long-term use. - -## Recommendations - -### ✅ Fix #1: Persist Refresh Tokens to Prisma - -**Why**: Refresh tokens are critical for long-term access and should be persisted to database. - -**Implementation**: -1. Save `refresh_token` to Prisma `MailCredentials.refresh_token` field -2. Update `token_expiry` when tokens are refreshed -3. Keep access tokens in Redis (short-lived, can be regenerated) -4. Use Prisma as source of truth for refresh tokens - -### ✅ Fix #2: Update Database on Token Refresh - -**Why**: Keep database in sync with refreshed tokens. - -**Implementation**: -1. After refreshing tokens, update Prisma `MailCredentials` record -2. Update `access_token` and `token_expiry` fields -3. Update `refresh_token` if Microsoft returns a new one - -### ✅ Fix #3: Fallback to Database if Redis Missing - -**Why**: Recover from Redis cache loss. - -**Implementation**: -1. If Redis cache is empty, check Prisma for refresh token -2. Use Prisma refresh token to get new access token -3. Re-populate Redis cache - -## Long-Term Viability Assessment - -### Current State: ⚠️ **NOT VIABLE** for long-term production - -**Reasons**: -1. ❌ Refresh tokens only in volatile Redis cache -2. ❌ No persistence mechanism -3. ❌ Risk of data loss on Redis restart -4. ❌ No recovery mechanism - -### After Fixes: ✅ **VIABLE** for long-term production - -**With recommended fixes**: -1. ✅ Refresh tokens persisted to database -2. ✅ Redis used for fast access token retrieval -3. ✅ Database as source of truth -4. ✅ Recovery mechanism in place - -## Token Storage Strategy (Recommended) - -### Access Tokens -- **Storage**: Redis (fast, short-lived) -- **TTL**: 1 hour (matches Microsoft token expiry) -- **Purpose**: Fast IMAP/SMTP authentication - -### Refresh Tokens -- **Storage**: Prisma Database (persistent, long-term) -- **TTL**: None (stored indefinitely until revoked) -- **Purpose**: Long-term access, token renewal - -### Token Expiry -- **Storage**: Both Redis and Prisma -- **Purpose**: Know when to refresh tokens - -## Implementation Priority - -1. **HIGH**: Persist refresh tokens to Prisma -2. **HIGH**: Update Prisma on token refresh -3. **MEDIUM**: Add fallback to database if Redis missing -4. **LOW**: Add token encryption at rest (if required by compliance) - diff --git a/MICROSOFT_OAUTH_FIXES.md b/MICROSOFT_OAUTH_FIXES.md deleted file mode 100644 index 6f670254..00000000 --- a/MICROSOFT_OAUTH_FIXES.md +++ /dev/null @@ -1,130 +0,0 @@ -# Microsoft OAuth Token Management - Fixes Applied - -## Issues Fixed - -### ✅ Fix #1: Refresh Tokens Now Persisted to Prisma Database - -**Problem**: Refresh tokens were only stored in Redis (24-hour TTL), risking permanent loss. - -**Solution**: -- Refresh tokens are now saved to `MailCredentials.refresh_token` in Prisma -- Access tokens and expiry also persisted to database -- Database acts as source of truth for long-term token storage - -**Files Modified**: -- `lib/services/email-service.ts` - `saveUserEmailCredentials()` now saves OAuth tokens to Prisma - -### ✅ Fix #2: Database Updated on Token Refresh - -**Problem**: When tokens were refreshed, only Redis was updated, leaving database stale. - -**Solution**: -- Token refresh now updates both Redis AND Prisma -- New refresh tokens (if provided by Microsoft) are persisted -- Token expiry timestamp updated in database - -**Files Modified**: -- `lib/services/token-refresh.ts` - `ensureFreshToken()` now updates Prisma after refresh - -### ✅ Fix #3: Fallback to Database if Redis Missing - -**Problem**: If Redis cache was empty, system couldn't recover refresh tokens. - -**Solution**: -- If Redis cache miss, system checks Prisma database -- Retrieves refresh token from database -- Re-populates Redis cache for future use - -**Files Modified**: -- `lib/services/token-refresh.ts` - Added database fallback logic - -### ✅ Fix #4: OAuth Fields Retrieved from Database - -**Problem**: When loading credentials from database, OAuth fields were ignored. - -**Solution**: -- Database queries now include OAuth fields (`access_token`, `refresh_token`, `token_expiry`, `use_oauth`) -- Credentials object properly populated with OAuth data from database - -**Files Modified**: -- `lib/services/email-service.ts` - `getImapConnection()` now includes OAuth fields from database - -## Token Storage Strategy (Current) - -### Access Tokens -- **Primary**: Redis (fast access, 24-hour TTL) -- **Backup**: Prisma Database (persisted) -- **Lifespan**: ~1 hour (Microsoft default) - -### Refresh Tokens -- **Primary**: Prisma Database (persistent, long-term) -- **Cache**: Redis (24-hour TTL, for fast access) -- **Lifespan**: Up to 90 days (with `offline_access` scope) - -### Token Expiry -- **Storage**: Both Redis and Prisma -- **Purpose**: Determine when to refresh tokens - -## Long-Term Viability - -### ✅ NOW VIABLE for Production - -**Improvements**: -1. ✅ Refresh tokens persisted to database -2. ✅ Database updated on token refresh -3. ✅ Fallback mechanism if Redis fails -4. ✅ No data loss on Redis restart -5. ✅ Recovery mechanism in place - -## What Happens Now - -### When Adding Microsoft Account: -1. OAuth tokens saved to **both** Redis and Prisma -2. Refresh token stored in database for long-term access -3. Access token cached in Redis for fast retrieval - -### When Token Expires: -1. System checks Redis first (fast path) -2. If Redis miss, checks Prisma database (fallback) -3. Uses refresh token to get new access token -4. Updates **both** Redis and Prisma with new tokens -5. Continues normal operation - -### If Redis is Cleared: -1. System detects Redis cache miss -2. Retrieves refresh token from Prisma database -3. Gets new access token using refresh token -4. Re-populates Redis cache -5. **No user action required** ✅ - -## Testing Recommendations - -1. **Test Token Refresh**: - - Wait for access token to expire (~1 hour) - - Verify system automatically refreshes - - Check both Redis and Prisma are updated - -2. **Test Redis Failure**: - - Clear Redis cache - - Try to access email - - Verify system recovers from database - -3. **Test Long-Term Access**: - - Wait several days - - Verify refresh token still works - - Check no re-authentication required - -## Monitoring - -Watch for these log messages: -- ✅ `Token for ${email} persisted to Prisma database` - Token saved successfully -- ✅ `Recovered credentials from Prisma and cached in Redis` - Fallback working -- ⚠️ `Error persisting tokens to database` - Database update failed (check logs) - -## Next Steps - -1. **Monitor**: Watch logs for token refresh operations -2. **Verify**: Check Prisma database has `refresh_token` values -3. **Test**: Verify email access works after Redis restart -4. **Optional**: Consider encrypting tokens at rest (if compliance requires) - diff --git a/MISSIONS_CENTRALE_WORKFLOW_ANALYSIS.md b/MISSIONS_CENTRALE_WORKFLOW_ANALYSIS.md deleted file mode 100644 index 1955b506..00000000 --- a/MISSIONS_CENTRALE_WORKFLOW_ANALYSIS.md +++ /dev/null @@ -1,930 +0,0 @@ -# Analyse Complète : Pages Missions et Centrale - Workflow Complet - -## 📋 Table des Matières -1. [Vue d'ensemble](#vue-densemble) -2. [Architecture des Pages](#architecture-des-pages) -3. [Workflow de Navigation](#workflow-de-navigation) -4. [Workflow de Création de Mission](#workflow-de-création-de-mission) -5. [Workflow de Consultation](#workflow-de-consultation) -6. [API Routes](#api-routes) -7. [Base de Données](#base-de-données) -8. [Intégrations Externes](#intégrations-externes) -9. [Stockage de Fichiers](#stockage-de-fichiers) -10. [Composants Réutilisables](#composants-réutilisables) - ---- - -## 🎯 Vue d'ensemble - -### Page "Centrale" -- **Route**: `/missions` -- **Nom dans le menu**: "Centrale" -- **Accès**: Rôles `entrepreneurship` ou `admin` (défini dans `components/main-nav.tsx`) -- **Description**: Centre d'Administration et de Pilotage (CAP) - Interface principale pour gérer les missions - -### Page "Missions" -- **Route principale**: `/missions` -- **Sous-routes**: - - `/missions` - Liste des missions de l'utilisateur - - `/missions/new` - Création d'une nouvelle mission - - `/missions/[missionId]` - Détails d'une mission - - `/missions/[missionId]/edit` - Édition d'une mission - -### Page "Mission Tab" (Tableau des Missions) -- **Route**: `/mission-tab` -- **Description**: Vue publique de toutes les missions disponibles -- **Sous-routes**: - - `/mission-tab` - Liste de toutes les missions - - `/mission-tab/[missionId]` - Détails d'une mission (vue publique) - ---- - -## 🏗️ Architecture des Pages - -### 1. Layout Principal - Missions (`app/missions/layout.tsx`) - -**Structure**: -``` -┌─────────────────────────────────────────┐ -│ Sidebar (CAP) - Fond rose clair │ -│ ┌───────────────────────────────────┐ │ -│ │ CAP │ │ -│ │ Centre d'Administration et de │ │ -│ │ Pilotage │ │ -│ └───────────────────────────────────┘ │ -│ • Mes Missions (/missions) │ -│ • Nouvelle Mission (/missions/new) │ -└─────────────────────────────────────────┘ -│ Contenu Principal (children) │ -└─────────────────────────────────────────┘ -``` - -**Fonctionnalités**: -- Sidebar fixe avec navigation -- Fond rose clair (`bg-pink-50`) pour la sidebar -- Fond blanc pour le contenu principal -- Navigation active highlightée - -### 2. Page Liste des Missions (`app/missions/page.tsx`) - -**Fonctionnalités**: -- Affichage en grille (responsive: 1/2/3 colonnes) -- Recherche par nom, niveau, type, ODD scope -- Filtrage en temps réel -- Cartes de mission avec: - - Logo (ou initiales si pas de logo) - - Nom de la mission - - Badge niveau (A/B/C/S) avec couleurs - - Icône ODD (Objectifs de Développement Durable) - - Services associés - - Description (intention) tronquée - - Date de création - - Bouton "Voir détails" - -**API utilisée**: `GET /api/missions` -- Retourne uniquement les missions où l'utilisateur est: - - Créateur (`creatorId`) - - Ou membre (`missionUsers`) - -### 3. Page Création de Mission (`app/missions/new/page.tsx`) - -**Composant principal**: `MissionsAdminPanel` -- Formulaire multi-onglets (5 onglets) -- Navigation séquentielle avec boutons Précédent/Suivant - -### 4. Page Détails Mission (`app/missions/[missionId]/page.tsx`) - -**Fonctionnalités**: -- Affichage complet des informations -- Logo de la mission -- Grille d'informations (Type, Donneur d'ordre, Durée, Niveau, Participation, ODD) -- Description complète -- Liste des documents/attachments -- Profils recherchés -- Services -- Bouton de suppression (si créateur ou admin) - -**API utilisée**: `GET /api/missions/[missionId]` - -### 5. Page Mission Tab (`app/mission-tab/page.tsx`) - -**Différences avec `/missions`**: -- Affiche **TOUTES** les missions (pas de filtre utilisateur) -- API utilisée: `GET /api/missions/all` -- Vue publique pour découvrir toutes les missions disponibles - ---- - -## 🔄 Workflow de Navigation - -### Accès à la Centrale - -``` -1. Utilisateur connecté avec rôle "entrepreneurship" ou "admin" - ↓ -2. Menu déroulant utilisateur (MainNav) - ↓ -3. Clic sur "Centrale" (href: '/missions') - ↓ -4. Redirection vers /missions - ↓ -5. Layout Missions s'affiche avec sidebar CAP - ↓ -6. Page Liste des Missions (/missions/page.tsx) -``` - -### Navigation dans la Centrale - -``` -┌─────────────────────────────────────────┐ -│ Sidebar CAP │ -│ ├─ Mes Missions (/missions) │ -│ └─ Nouvelle Mission (/missions/new) │ -└─────────────────────────────────────────┘ - │ │ - │ │ - ▼ ▼ -┌─────────────────┐ ┌──────────────────┐ -│ Liste Missions │ │ Création Mission │ -│ │ │ │ -│ [Carte Mission] │ │ [Formulaire] │ -│ └─► Détails │ │ │ -└─────────────────┘ └──────────────────┘ - │ - ▼ -┌─────────────────┐ -│ Détails Mission │ -│ │ -│ [Éditer] │ -│ [Supprimer] │ -└─────────────────┘ -``` - ---- - -## 🚀 Workflow de Création de Mission - -### Étape 1: Accès au Formulaire -``` -User → /missions/new → MissionsAdminPanel -``` - -### Étape 2: Formulaire Multi-Onglets - -**Onglet 1: General** -- Nom de la mission (requis) -- Logo (upload) -- ODD scope (requis) - Sélection parmi 17 ODD -- Niveau (requis) - A/B/C/S -- Intention (requis) - Description avec éditeur de texte - -**Onglet 2: Details** -- Type de mission (requis) - Remote/Onsite/Hybrid -- Donneur d'ordre (requis) - Individu/ONG/Start-ups -- Projection (requis) - Short/Medium/Long term -- Services - Checkboxes (Gite, ArtLab, Calcul) -- Participation (requis) - Volontaire/Cooptation -- Profils - Checkboxes (DataIntelligence, Expression, Mediation, Investigation, Coding, Lean) - -**Onglet 3: Attachments** -- Upload de fichiers (PDF, DOC, DOCX, XLS, XLSX, JPG, JPEG, PNG) -- Liste des fichiers sélectionnés -- Upload immédiat vers Minio (bucket 'missions') - -**Onglet 4: Skills** -- Liste de compétences (non fonctionnel actuellement - placeholders) - -**Onglet 5: Membres** -- **Les Gardiens de l'Intention** (3 gardiens requis): - - Gardien du Temps - - Gardien de la Parole - - Gardien de la Mémoire -- **Volontaires** (optionnel) -- Recherche d'utilisateurs ou groupes -- Assignation de rôles - -### Étape 3: Validation et Soumission - -**Validation**: -```typescript -const requiredFields = { - name: !!missionData.name, - oddScope: Array.isArray(missionData.oddScope) && missionData.oddScope.length > 0, - niveau: !!missionData.niveau, - intention: !!missionData.intention, - missionType: !!missionData.missionType, - donneurDOrdre: !!missionData.donneurDOrdre, - projection: !!missionData.projection, - participation: !!missionData.participation, - gardiens: gardienDuTemps !== null && - gardienDeLaParole !== null && - gardienDeLaMemoire !== null -} -``` - -**Soumission**: -```typescript -POST /api/missions -Body: { - name, oddScope, niveau, intention, missionType, - donneurDOrdre, projection, services, profils, - participation, guardians, volunteers, logo, attachments -} -``` - -### Étape 4: Traitement Backend - -**Séquence d'exécution**: - -1. **Création de la mission en base de données** - ```typescript - prisma.mission.create({ - data: { name, oddScope, niveau, intention, ... } - }) - ``` - -2. **Création des MissionUsers (gardiens + volontaires)** - ```typescript - prisma.missionUser.createMany({ - data: [ - { missionId, userId, role: 'gardien-temps' }, - { missionId, userId, role: 'gardien-parole' }, - { missionId, userId, role: 'gardien-memoire' }, - { missionId, userId, role: 'volontaire' }, // pour chaque volontaire - ] - }) - ``` - -3. **Upload du logo vers Minio** - - Path: `missions/{missionId}/logo{extension}` - - Bucket: `missions` - - Mise à jour du champ `logo` dans la mission - -4. **Upload des attachments vers Minio** - - Path: `missions/{missionId}/attachments/{filename}` - - Création des enregistrements `Attachment` en base - -5. **Vérification des fichiers dans Minio** - - Vérifie que tous les fichiers sont bien présents avant de continuer - -6. **Déclenchement du workflow N8N** - ```typescript - n8nService.triggerMissionCreation({ - ...missionData, - creatorId, - logoPath, - config: { N8N_API_KEY, MISSION_API_URL } - }) - ``` - -7. **Intégrations externes (via N8N)**: - - Création projet Leantime (si applicable) - - Création collection Outline (si applicable) - - Création canal RocketChat (si applicable) - - Création repository Gitea (si applicable) - - Création projet Penpot (si applicable) - -8. **Retour succès/erreur** - - Si succès: Redirection vers `/missions` - - Si erreur: Nettoyage des fichiers uploadés + message d'erreur - ---- - -## 👀 Workflow de Consultation - -### Consultation Liste des Missions - -**Route**: `/missions` ou `/mission-tab` - -**Flux**: -``` -1. Chargement de la page - ↓ -2. useEffect → fetch('/api/missions') ou fetch('/api/missions/all') - ↓ -3. Affichage du loader - ↓ -4. Réception des données - ↓ -5. Transformation des données: - - Ajout des logoUrl (si logo existe) - - Formatage des dates - - Calcul des couleurs de badges - - Extraction des infos ODD - ↓ -6. Filtrage par terme de recherche (si présent) - ↓ -7. Affichage en grille -``` - -**Recherche**: -- Filtre en temps réel sur: nom, niveau, type, ODD scope -- Pas de requête API supplémentaire (filtrage côté client) - -### Consultation Détails Mission - -**Route**: `/missions/[missionId]` ou `/mission-tab/[missionId]` - -**Flux**: -``` -1. Chargement de la page - ↓ -2. Récupération du missionId depuis les params - ↓ -3. useEffect → fetch(`/api/missions/${missionId}`) - ↓ -4. Affichage du loader - ↓ -5. Réception des données complètes: - - Mission avec tous les champs - - Creator (id, email) - - MissionUsers (avec user details) - - Attachments (avec publicUrl) - ↓ -6. Transformation: - - Ajout des logoUrl - - Formatage des dates - - Labels pour les types/niveaux - - URLs publiques pour les attachments - ↓ -7. Affichage des sections: - - Header avec nom et logo - - Grille d'informations - - Description - - Documents - - Profils recherchés - - Services - - Actions (Éditer/Supprimer) -``` - ---- - -## 🔌 API Routes - -### 1. `GET /api/missions` -**Fichier**: `app/api/missions/route.ts` - -**Fonctionnalité**: Liste les missions de l'utilisateur connecté - -**Filtres**: -- `limit` (default: 10) -- `offset` (default: 0) -- `search` (recherche dans name et intention) -- `name` (filtre exact) - -**Where Clause**: -```typescript -{ - OR: [ - { creatorId: userId }, - { missionUsers: { some: { userId } } } - ] -} -``` - -**Retour**: -```json -{ - "missions": [ - { - "id": "...", - "name": "...", - "logo": "missions/{id}/logo.png", - "logoUrl": "/api/missions/image/missions/{id}/logo.png", - "oddScope": ["odd-3"], - "niveau": "a", - "missionType": "remote", - "projection": "short", - "services": ["Gite"], - "intention": "...", - "createdAt": "...", - "creator": { "id": "...", "email": "..." }, - "missionUsers": [...], - "attachments": [...] - } - ], - "pagination": { - "total": 10, - "offset": 0, - "limit": 10 - } -} -``` - -### 2. `POST /api/missions` -**Fichier**: `app/api/missions/route.ts` - -**Fonctionnalité**: Crée une nouvelle mission - -**Body**: -```typescript -{ - name: string; - oddScope: string[]; - niveau?: string; - intention?: string; - missionType?: string; - donneurDOrdre?: string; - projection?: string; - services?: string[]; - profils?: string[]; - participation?: string; - guardians?: { - "gardien-temps": string; - "gardien-parole": string; - "gardien-memoire": string; - }; - volunteers?: string[]; - logo?: { - data: string; // base64 - name?: string; - type?: string; - }; - attachments?: Array<{ - data: string; // base64 - name?: string; - type?: string; - }>; -} -``` - -**Retour**: -```json -{ - "success": true, - "mission": { ... }, - "message": "Mission created successfully with all integrations" -} -``` - -### 3. `GET /api/missions/[missionId]` -**Fichier**: `app/api/missions/[missionId]/route.ts` - -**Fonctionnalité**: Récupère les détails d'une mission - -**Contrôle d'accès**: -- Utilisateur doit être créateur OU membre de la mission - -**Retour**: Mission complète avec relations - -### 4. `PUT /api/missions/[missionId]` -**Fichier**: `app/api/missions/[missionId]/route.ts` - -**Fonctionnalité**: Met à jour une mission - -**Contrôle d'accès**: -- Créateur OU gardien-temps/gardien-parole - -**Body**: Même structure que POST (tous les champs optionnels) - -### 5. `DELETE /api/missions/[missionId]` -**Fichier**: `app/api/missions/[missionId]/route.ts` - -**Fonctionnalité**: Supprime une mission - -**Contrôle d'accès**: -- Créateur OU admin uniquement - -**Actions**: -- Suppression du logo dans Minio -- Suppression de la mission en base (cascade sur MissionUsers et Attachments) -- TODO: Rollback N8N (non implémenté) - -### 6. `GET /api/missions/all` -**Fichier**: `app/api/missions/all/route.ts` - -**Fonctionnalité**: Liste TOUTES les missions (pas de filtre utilisateur) - -**Différences avec `/api/missions`**: -- Pas de filtre par utilisateur -- Retourne toutes les missions publiques -- Utilisé par `/mission-tab` - -### 7. `GET /api/missions/image/[...path]` -**Fichier**: `app/api/missions/image/[...path]/route.ts` - -**Fonctionnalité**: Sert les images (logos et attachments) depuis Minio - -**Path**: `missions/{missionId}/logo.png` ou `missions/{missionId}/attachments/{filename}` - -### 8. `POST /api/missions/upload` -**Fichier**: `app/api/missions/upload/route.ts` - -**Fonctionnalité**: Upload de fichiers (logo ou attachments) - -### 9. `GET /api/missions/[missionId]/attachments` -**Fichier**: `app/api/missions/[missionId]/attachments/route.ts` - -**Fonctionnalité**: Liste les attachments d'une mission - -### 10. `POST /api/missions/[missionId]/attachments` -**Fichier**: `app/api/missions/[missionId]/attachments/route.ts` - -**Fonctionnalité**: Ajoute un attachment à une mission existante - -### 11. `DELETE /api/missions/[missionId]/attachments/[attachmentId]` -**Fichier**: `app/api/missions/[missionId]/attachments/[attachmentId]/route.ts` - -**Fonctionnalité**: Supprime un attachment - ---- - -## 🗄️ Base de Données - -### Modèle Mission (`prisma/schema.prisma`) - -```prisma -model Mission { - id String @id @default(uuid()) - name String - logo String? // Path dans Minio - oddScope String[] // Catégories ODD - niveau String // A/B/C/S - intention String // Description - missionType String // remote/onsite/hybrid - donneurDOrdre String // individual/group/organization - projection String // short/medium/long - services String[] // ["Gite", "ArtLab", "Calcul"] - participation String? // volontaire/cooptation - profils String[] // ["DataIntelligence", ...] - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt - creator User @relation(fields: [creatorId], references: [id]) - creatorId String - attachments Attachment[] - missionUsers MissionUser[] - - // Intégrations externes - leantimeProjectId String? - outlineCollectionId String? - rocketChatChannelId String? - giteaRepositoryUrl String? - penpotProjectId String? - - @@index([creatorId]) -} -``` - -### Modèle MissionUser - -```prisma -model MissionUser { - id String @id @default(uuid()) - role String // 'gardien-temps', 'gardien-parole', 'gardien-memoire', 'volontaire' - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt - mission Mission @relation(fields: [missionId], references: [id]) - missionId String - user User @relation(fields: [userId], references: [id]) - userId String - - @@unique([missionId, userId, role]) - @@index([missionId]) - @@index([userId]) -} -``` - -### Modèle Attachment - -```prisma -model Attachment { - id String @id @default(uuid()) - filename String - filePath String // Path dans Minio: missions/{missionId}/attachments/{filename} - fileType String // MIME type - fileSize Int - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt - mission Mission @relation(fields: [missionId], references: [id]) - missionId String - uploader User @relation(fields: [uploaderId], references: [id]) - uploaderId String - - @@index([missionId]) - @@index([uploaderId]) -} -``` - ---- - -## 🔗 Intégrations Externes - -### Service N8N (`lib/services/n8n-service.ts`) - -**Webhook URL**: `https://brain.slm-lab.net/webhook/mission-created` - -**Données envoyées**: -```typescript -{ - name, oddScope, niveau, intention, missionType, - donneurDOrdre, projection, services, participation, - profils, guardians, volunteers, creatorId, - config: { - N8N_API_KEY, - MISSION_API_URL - } -} -``` - -**Workflow N8N déclenche**: -1. Création projet Leantime (si applicable) -2. Création collection Outline (si applicable) -3. Création canal RocketChat (si applicable) -4. Création repository Gitea (si applicable) -5. Création projet Penpot (si applicable) - -**Retour**: -```typescript -{ - success: boolean; - results?: { - leantimeProjectId?: string; - outlineCollectionId?: string; - rocketChatChannelId?: string; - giteaRepositoryUrl?: string; - penpotProjectId?: string; - failedServices?: { - gitRepo?: boolean; - leantimeProject?: boolean; - docCollection?: boolean; - rocketChatChannel?: boolean; - } - }; - error?: string; -} -``` - -**Rollback** (non implémenté): -- Webhook: `https://brain.slm-lab.net/webhook/mission-rollback` -- Appelé lors de la suppression d'une mission - ---- - -## 📦 Stockage de Fichiers - -### Minio Configuration - -**Endpoint**: `https://dome-api.slm-lab.net` -**Bucket**: `missions` -**Credentials**: Hardcodés dans `lib/mission-uploads.ts` (⚠️ à sécuriser) - -### Structure des Chemins - -**Logo**: -``` -missions/{missionId}/logo{extension} -Exemple: missions/abc-123/logo.png -``` - -**Attachments**: -``` -missions/{missionId}/attachments/{filename} -Exemple: missions/abc-123/attachments/document.pdf -``` - -### URLs Publiques - -**Format**: `/api/missions/image/{path}` - -**Exemples**: -- Logo: `/api/missions/image/missions/{missionId}/logo.png` -- Attachment: `/api/missions/image/missions/{missionId}/attachments/document.pdf` - -### Fonctions Utilitaires (`lib/mission-uploads.ts`) - -- `getMissionLogoPath()` - Génère le chemin du logo -- `getMissionAttachmentPath()` - Génère le chemin d'un attachment -- `uploadMissionLogo()` - Upload un logo vers Minio -- `uploadMissionAttachment()` - Upload un attachment vers Minio -- `deleteMissionLogo()` - Supprime un logo (TODO) -- `deleteMissionAttachment()` - Supprime un attachment -- `getMissionFileUrl()` - Construit l'URL publique -- `ensureMissionsPrefix()` - Normalise le chemin - ---- - -## 🧩 Composants Réutilisables - -### 1. `MissionsAdminPanel` (`components/missions/missions-admin-panel.tsx`) - -**Fonctionnalités**: -- Formulaire multi-onglets -- Gestion des gardiens et volontaires -- Upload de fichiers -- Validation complète -- Soumission vers API - -**Props**: Aucune (composant autonome) - -**State**: -- `missionData` - Données de la mission -- `selectedServices` - Services sélectionnés -- `selectedProfils` - Profils sélectionnés -- `gardienDuTemps`, `gardienDeLaParole`, `gardienDeLaMemoire` - IDs des gardiens -- `volontaires` - Array d'IDs de volontaires -- `activeTab` - Onglet actif -- `isSubmitting` - État de soumission - -### 2. `FileUpload` (`components/missions/file-upload.tsx`) - -**Fonctionnalités**: -- Upload de logo ou attachment -- Conversion en base64 -- Preview pour les images - -**Props**: -- `type`: 'logo' | 'attachment' -- `isNewMission`: boolean -- `onFileSelect`: (fileData) => void - -### 3. `AttachmentsList` (`components/missions/attachments-list.tsx`) - -**Fonctionnalités**: -- Liste des attachments d'une mission -- Upload de nouveaux attachments -- Suppression d'attachments - -**Props**: -- `missionId`: string -- `allowUpload`: boolean -- `allowDelete`: boolean - -### 4. `MissionsFrame` (`components/missions/missions-frame.tsx`) - -**Fonctionnalités**: Wrapper iframe (non utilisé actuellement) - ---- - -## 🔐 Contrôles d'Accès - -### Page Centrale (`/missions`) -- **Rôles requis**: `entrepreneurship` ou `admin` -- Vérifié dans `components/main-nav.tsx` via `hasRole()` - -### API Routes -- **Authentification**: Session NextAuth requise -- **GET /api/missions**: Missions où user est créateur ou membre -- **GET /api/missions/all**: Toutes les missions (authentifié) -- **GET /api/missions/[id]**: Créateur ou membre -- **PUT /api/missions/[id]**: Créateur ou gardien-temps/gardien-parole -- **DELETE /api/missions/[id]**: Créateur ou admin uniquement - ---- - -## 📊 Flux de Données Complet - -### Création de Mission - -``` -[Frontend] - MissionsAdminPanel - ↓ (soumission) - POST /api/missions - ↓ -[Backend] - 1. Validation - 2. prisma.mission.create() - 3. prisma.missionUser.createMany() - 4. uploadMissionLogo() → Minio - 5. uploadMissionAttachment() → Minio (pour chaque attachment) - 6. prisma.attachment.create() (pour chaque attachment) - 7. verifyFileExists() (vérification Minio) - 8. n8nService.triggerMissionCreation() - ↓ -[N8N Workflow] - - Création Leantime - - Création Outline - - Création RocketChat - - Création Gitea - - Création Penpot - ↓ -[Backend] - 9. Retour succès/erreur - ↓ -[Frontend] - Redirection → /missions -``` - -### Consultation de Mission - -``` -[Frontend] - MissionsPage ou MissionDetailPage - ↓ - fetch('/api/missions') ou fetch('/api/missions/[id]') - ↓ -[Backend] - 1. Vérification session - 2. Query Prisma avec relations - 3. Transformation des paths en URLs publiques - 4. Retour JSON - ↓ -[Frontend] - Affichage des données -``` - -### Affichage d'Image - -``` -[Frontend] - - ↓ -[Backend] - GET /api/missions/image/[...path] - ↓ - Lecture depuis Minio - ↓ - Stream vers client -``` - ---- - -## 🎨 Styles et UI - -### Couleurs des Badges Niveau -- **A (Apprentissage)**: `bg-green-100 text-green-800` -- **B (Basique)**: `bg-blue-100 text-blue-800` -- **C (Complexe)**: `bg-purple-100 text-purple-800` -- **S (Spécial)**: `bg-amber-100 text-amber-800` - -### Layout Sidebar CAP -- **Fond**: `bg-pink-50` -- **Bordure**: `border-pink-100` -- **Largeur**: `234px` fixe - -### Grille de Missions -- **Mobile**: 1 colonne -- **Tablet**: 2 colonnes (`md:grid-cols-2`) -- **Desktop**: 3 colonnes (`lg:grid-cols-3`) - ---- - -## 🐛 Points d'Attention - -1. **Credentials Minio hardcodés** dans `lib/mission-uploads.ts` - À déplacer vers variables d'environnement -2. **Rollback N8N non implémenté** lors de la suppression -3. **Skills tab non fonctionnel** - Placeholders uniquement -4. **Presigned URLs non implémentées** - Upload direct uniquement -5. **Gestion d'erreurs N8N** - Partielle (continue même si certaines intégrations échouent) - ---- - -## 📝 Notes Techniques - -### Types TypeScript - -**Mission Interface** (utilisée dans les pages): -```typescript -interface Mission { - id: string; - name: string; - logo?: string; - logoUrl?: string; - oddScope: string[]; - niveau: string; - missionType: string; - projection: string; - participation?: string; - services?: string[]; - profils?: string[]; - intention?: string; - donneurDOrdre?: string; - createdAt: string; - creator: User; - missionUsers: MissionUser[]; - attachments?: Attachment[]; -} -``` - -### Validation - -**Côté Frontend**: Validation dans `MissionsAdminPanel.validateMission()` -**Côté Backend**: Validation minimale (name et oddScope requis) - -### Gestion d'Erreurs - -- **Frontend**: Toast notifications via `useToast()` -- **Backend**: Retour JSON avec `error` et `details` -- **N8N**: Retour avec `success` et `failedServices` pour erreurs partielles - ---- - -## 🔄 Évolutions Possibles - -1. **Pagination** côté client pour les listes -2. **Filtres avancés** (par niveau, type, ODD, etc.) -3. **Recherche full-text** dans l'intention -4. **Export** des missions (PDF, CSV) -5. **Notifications** lors de l'assignation à une mission -6. **Statistiques** des missions -7. **Timeline** des activités d'une mission -8. **Commentaires** sur les missions -9. **États** des missions (brouillon, publiée, terminée, etc.) -10. **Permissions granulaires** par rôle de gardien - ---- - -**Document généré le**: $(date) -**Version**: 1.0 -**Auteur**: Analyse automatique du codebase - diff --git a/MISSION_CREATION_WORKFLOW_DETAILED.md b/MISSION_CREATION_WORKFLOW_DETAILED.md deleted file mode 100644 index c7a91d55..00000000 --- a/MISSION_CREATION_WORKFLOW_DETAILED.md +++ /dev/null @@ -1,982 +0,0 @@ -# Workflow Détaillé : Création de Mission - Prisma, Minio et N8N - -## 📋 Vue d'Ensemble - -Ce document trace **chaque étape** du workflow de création de mission, depuis le formulaire frontend jusqu'aux intégrations externes via N8N, en passant par Prisma (base de données) et Minio (stockage de fichiers). - ---- - -## 🔄 Flux Complet - Vue d'Ensemble - -``` -┌─────────────────────────────────────────────────────────────┐ -│ 1. FRONTEND - MissionsAdminPanel │ -│ - Validation des champs │ -│ - Préparation des données (base64 pour fichiers) │ -│ - POST /api/missions │ -└─────────────────────────────────────────────────────────────┘ - ↓ -┌─────────────────────────────────────────────────────────────┐ -│ 2. BACKEND - POST /api/missions │ -│ ├─ Authentification (NextAuth) │ -│ ├─ Validation des champs requis │ -│ ├─ STEP 1: Prisma.mission.create() │ -│ ├─ STEP 2: Prisma.missionUser.createMany() │ -│ ├─ STEP 3: Upload Logo → Minio │ -│ ├─ STEP 4: Upload Attachments → Minio │ -│ ├─ STEP 5: Vérification fichiers Minio │ -│ └─ STEP 6: N8N Workflow Trigger │ -└─────────────────────────────────────────────────────────────┘ - ↓ -┌─────────────────────────────────────────────────────────────┐ -│ 3. N8N WORKFLOW │ -│ - Création Leantime Project │ -│ - Création Outline Collection │ -│ - Création RocketChat Channel │ -│ - Création Gitea Repository │ -│ - Création Penpot Project │ -└─────────────────────────────────────────────────────────────┘ -``` - ---- - -## 📝 ÉTAPE 1 : Frontend - Préparation des Données - -### Fichier : `components/missions/missions-admin-panel.tsx` - -### 1.1 Validation (`validateMission()`) - -**Lignes 369-397** - -```typescript -const validateMission = () => { - const requiredFields = { - name: !!missionData.name, - oddScope: Array.isArray(missionData.oddScope) && missionData.oddScope.length > 0, - niveau: !!missionData.niveau, - intention: !!missionData.intention, - missionType: !!missionData.missionType, - donneurDOrdre: !!missionData.donneurDOrdre, - projection: !!missionData.projection, - participation: !!missionData.participation, - gardiens: gardienDuTemps !== null && - gardienDeLaParole !== null && - gardienDeLaMemoire !== null - }; - - // Vérifie que tous les champs requis sont remplis - // Retourne false si un champ manque -} -``` - -**Champs requis** : -- ✅ `name` : Nom de la mission -- ✅ `oddScope` : Array avec au moins 1 ODD -- ✅ `niveau` : A/B/C/S -- ✅ `intention` : Description -- ✅ `missionType` : remote/onsite/hybrid -- ✅ `donneurDOrdre` : individual/group/organization -- ✅ `projection` : short/medium/long -- ✅ `participation` : volontaire/cooptation -- ✅ `gardiens` : Les 3 gardiens doivent être assignés - -### 1.2 Préparation des Données (`handleSubmitMission()`) - -**Lignes 400-460** - -```typescript -const handleSubmitMission = async () => { - // 1. Validation - if (!validateMission()) return; - - // 2. Préparation des gardiens - const guardians = { - "gardien-temps": gardienDuTemps, - "gardien-parole": gardienDeLaParole, - "gardien-memoire": gardienDeLaMemoire - }; - - // 3. Construction de l'objet de soumission - const missionSubmitData = { - ...missionData, // name, oddScope, niveau, intention, etc. - services: selectedServices, // ["Gite", "ArtLab", "Calcul"] - profils: selectedProfils, // ["DataIntelligence", "Expression", ...] - guardians, // { "gardien-temps": userId, ... } - volunteers: volontaires, // [userId1, userId2, ...] - logo: missionData.logo // { data: "data:image/png;base64,...", name, type } - }; - - // 4. Envoi à l'API - const response = await fetch('/api/missions', { - method: 'POST', - headers: { 'Content-Type': 'application/json' }, - body: JSON.stringify(missionSubmitData) - }); -} -``` - -**Format du logo** (si présent) : -```typescript -logo: { - data: "...", // Base64 avec préfixe - name: "logo.png", - type: "image/png" -} -``` - -**Format des attachments** (si présents) : -```typescript -attachments: [ - { - data: "data:application/pdf;base64,JVBERi0xLjQKJe...", - name: "document.pdf", - type: "application/pdf" - } -] -``` - ---- - -## 🗄️ ÉTAPE 2 : Backend - POST /api/missions - -### Fichier : `app/api/missions/route.ts` - -### 2.1 Authentification et Validation - -**Lignes 205-224** - -```typescript -export async function POST(request: Request) { - // 1. Vérification de l'authentification - const { authorized, userId } = await checkAuth(request); - if (!authorized || !userId) { - return NextResponse.json({ error: 'Unauthorized' }, { status: 401 }); - } - - // 2. Parsing du body - const body = await request.json(); - - // 3. Validation minimale (name et oddScope requis) - if (!body.name || !body.oddScope) { - return NextResponse.json({ - error: 'Missing required fields', - missingFields: ['name', 'oddScope'].filter(field => !body[field]) - }, { status: 400 }); - } -} -``` - -### 2.2 STEP 1 : Création de la Mission en Base de Données - -**Lignes 226-248** - -```typescript -// Préparation des données pour Prisma -const missionData = { - name: body.name, - oddScope: body.oddScope, // Array de strings - niveau: body.niveau, - intention: body.intention, - missionType: body.missionType, - donneurDOrdre: body.donneurDOrdre, - projection: body.projection, - services: body.services, // Array de strings - profils: body.profils, // Array de strings - participation: body.participation, - creatorId: userId, // ID de l'utilisateur connecté - logo: null, // Sera mis à jour après upload -}; - -// Création en base de données -const mission = await prisma.mission.create({ - data: missionData -}); - -// Résultat : mission.id est maintenant disponible -// Exemple : mission.id = "abc-123-def-456" -``` - -**Schéma Prisma** : -```prisma -model Mission { - id String @id @default(uuid()) - name String - logo String? // null pour l'instant - oddScope String[] - niveau String - intention String - missionType String - donneurDOrdre String - projection String - services String[] - participation String? - profils String[] - creatorId String - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt - // Relations - creator User @relation(...) - missionUsers MissionUser[] - attachments Attachment[] -} -``` - -**Points importants** : -- ✅ La mission est créée **AVANT** l'upload des fichiers -- ✅ Le `mission.id` est généré automatiquement (UUID) -- ✅ Le champ `logo` est `null` pour l'instant -- ✅ Tous les champs sont sauvegardés sauf les fichiers - -### 2.3 STEP 2 : Création des MissionUsers (Gardiens + Volontaires) - -**Lignes 250-283** - -```typescript -// Préparation du tableau de MissionUsers -const missionUsers = []; - -// 2.1 Ajout des gardiens -if (body.guardians) { - for (const [role, guardianId] of Object.entries(body.guardians)) { - if (guardianId) { - missionUsers.push({ - missionId: mission.id, // ID de la mission créée - userId: guardianId, // ID de l'utilisateur gardien - role: role // "gardien-temps", "gardien-parole", "gardien-memoire" - }); - } - } -} - -// 2.2 Ajout des volontaires -if (body.volunteers && body.volunteers.length > 0) { - for (const volunteerId of body.volunteers) { - missionUsers.push({ - missionId: mission.id, - userId: volunteerId, - role: 'volontaire' - }); - } -} - -// 2.3 Création en batch dans Prisma -if (missionUsers.length > 0) { - await prisma.missionUser.createMany({ - data: missionUsers - }); -} -``` - -**Schéma Prisma MissionUser** : -```prisma -model MissionUser { - id String @id @default(uuid()) - role String // 'gardien-temps', 'gardien-parole', 'gardien-memoire', 'volontaire' - missionId String - userId String - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt - - mission Mission @relation(...) - user User @relation(...) - - @@unique([missionId, userId, role]) // Un utilisateur ne peut avoir qu'un rôle par mission -} -``` - -**Exemple de données créées** : -```typescript -missionUsers = [ - { missionId: "abc-123", userId: "user-1", role: "gardien-temps" }, - { missionId: "abc-123", userId: "user-2", role: "gardien-parole" }, - { missionId: "abc-123", userId: "user-3", role: "gardien-memoire" }, - { missionId: "abc-123", userId: "user-4", role: "volontaire" }, - { missionId: "abc-123", userId: "user-5", role: "volontaire" } -] -``` - -**Points importants** : -- ✅ Utilisation de `createMany()` pour performance (1 requête au lieu de N) -- ✅ Contrainte unique : un utilisateur ne peut avoir qu'un rôle par mission -- ✅ Les gardiens sont obligatoires, les volontaires optionnels - -### 2.4 STEP 3 : Upload du Logo vers Minio - -**Lignes 285-310** - -```typescript -let logoPath = null; -if (body.logo?.data) { - try { - // 3.1 Conversion base64 → Buffer → File - const base64Data = body.logo.data.split(',')[1]; // Retire "data:image/png;base64," - const buffer = Buffer.from(base64Data, 'base64'); - const file = new File( - [buffer], - body.logo.name || 'logo.png', - { type: body.logo.type || 'image/png' } - ); - - // 3.2 Upload vers Minio - const { filePath } = await uploadMissionLogo(userId, mission.id, file); - logoPath = filePath; // Ex: "missions/abc-123/logo.png" - uploadedFiles.push({ type: 'logo', path: filePath }); - - // 3.3 Mise à jour de la mission avec le chemin du logo - await prisma.mission.update({ - where: { id: mission.id }, - data: { logo: filePath } - }); - } catch (uploadError) { - throw new Error('Failed to upload logo'); - } -} -``` - -**Fonction `uploadMissionLogo()` - `lib/mission-uploads.ts`** : - -**Lignes 96-145** - -```typescript -export async function uploadMissionLogo( - userId: string, - missionId: string, - file: File -): Promise<{ filePath: string }> { - - // 1. Génération du chemin - const fileExtension = file.name.substring(file.name.lastIndexOf('.')); - const filePath = getMissionLogoPath(userId, missionId, fileExtension); - // Résultat : "missions/{missionId}/logo.png" - - // 2. Conversion pour Minio (retire le préfixe "missions/") - const minioPath = filePath.replace(/^missions\//, ''); - // Résultat : "{missionId}/logo.png" - - // 3. Conversion File → Buffer - const arrayBuffer = await file.arrayBuffer(); - const buffer = Buffer.from(arrayBuffer); - - // 4. Upload vers Minio via S3 SDK - await s3Client.send(new PutObjectCommand({ - Bucket: 'missions', // Bucket Minio - Key: minioPath, // "{missionId}/logo.png" - Body: buffer, // Contenu du fichier - ContentType: file.type, // "image/png" - ACL: 'public-read' // Accès public en lecture - })); - - return { filePath }; // Retourne le chemin complet avec préfixe -} -``` - -**Configuration Minio** : -```typescript -// lib/mission-uploads.ts -const s3Client = new S3Client({ - region: 'us-east-1', - endpoint: 'https://dome-api.slm-lab.net', // Endpoint Minio - credentials: { - accessKeyId: '4aBT4CMb7JIMMyUtp4Pl', - secretAccessKey: 'HGn39XhCIlqOjmDVzRK9MED2Fci2rYvDDgbLFElg' - }, - forcePathStyle: true // Requis pour MinIO -}); -``` - -**Structure dans Minio** : -``` -Bucket: missions - └── {missionId}/ - └── logo.png -``` - -**Points importants** : -- ✅ Le logo est uploadé **APRÈS** la création de la mission (pour avoir le missionId) -- ✅ Le chemin est mis à jour dans Prisma après l'upload -- ✅ Le fichier est stocké avec ACL `public-read` pour accès public -- ✅ Le chemin stocké inclut le préfixe `missions/` pour cohérence - -### 2.5 STEP 4 : Upload des Attachments vers Minio - -**Lignes 312-343** - -```typescript -if (body.attachments && body.attachments.length > 0) { - try { - // 4.1 Traitement parallèle de tous les attachments - const attachmentPromises = body.attachments.map(async (attachment: any) => { - // Conversion base64 → Buffer → File - const base64Data = attachment.data.split(',')[1]; - const buffer = Buffer.from(base64Data, 'base64'); - const file = new File( - [buffer], - attachment.name || 'attachment', - { type: attachment.type || 'application/octet-stream' } - ); - - // 4.2 Upload vers Minio - const { filePath, filename, fileType, fileSize } = - await uploadMissionAttachment(userId, mission.id, file); - uploadedFiles.push({ type: 'attachment', path: filePath }); - - // 4.3 Création de l'enregistrement Attachment en base - return prisma.attachment.create({ - data: { - missionId: mission.id, - filename, // "document.pdf" - filePath, // "missions/abc-123/attachments/document.pdf" - fileType, // "application/pdf" - fileSize, // 123456 (bytes) - uploaderId: userId - } - }); - }); - - // 4.4 Attente de tous les uploads (parallèle) - await Promise.all(attachmentPromises); - } catch (attachmentError) { - throw new Error('Failed to upload attachments'); - } -} -``` - -**Fonction `uploadMissionAttachment()` - `lib/mission-uploads.ts`** : - -**Lignes 148-210** - -```typescript -export async function uploadMissionAttachment( - userId: string, - missionId: string, - file: File -): Promise<{ - filename: string; - filePath: string; - fileType: string; - fileSize: number; -}> { - - // 1. Génération du chemin - const filePath = getMissionAttachmentPath(userId, missionId, file.name); - // Résultat : "missions/{missionId}/attachments/{filename}" - - // 2. Conversion pour Minio - const minioPath = filePath.replace(/^missions\//, ''); - // Résultat : "{missionId}/attachments/{filename}" - - // 3. Conversion File → Buffer - const arrayBuffer = await file.arrayBuffer(); - const buffer = Buffer.from(arrayBuffer); - - // 4. Upload vers Minio - await s3Client.send(new PutObjectCommand({ - Bucket: 'missions', - Key: minioPath, - Body: buffer, - ContentType: file.type, - ACL: 'public-read' - })); - - return { - filename: file.name, - filePath, // Chemin complet avec préfixe - fileType: file.type, - fileSize: file.size - }; -} -``` - -**Schéma Prisma Attachment** : -```prisma -model Attachment { - id String @id @default(uuid()) - filename String // "document.pdf" - filePath String // "missions/{missionId}/attachments/{filename}" - fileType String // "application/pdf" - fileSize Int // 123456 - missionId String - uploaderId String - createdAt DateTime @default(now()) - updatedAt DateTime @updatedAt - - mission Mission @relation(...) - uploader User @relation(...) -} -``` - -**Structure dans Minio** : -``` -Bucket: missions - └── {missionId}/ - ├── logo.png - └── attachments/ - ├── document1.pdf - ├── document2.docx - └── image.jpg -``` - -**Points importants** : -- ✅ Uploads en **parallèle** avec `Promise.all()` pour performance -- ✅ Chaque attachment crée un enregistrement Prisma séparé -- ✅ Le `uploaderId` est l'utilisateur qui a créé la mission -- ✅ Les fichiers sont stockés avec ACL `public-read` - -### 2.6 STEP 5 : Vérification des Fichiers dans Minio - -**Lignes 345-365** - -```typescript -// 5.1 Vérification du logo -if (logoPath) { - const logoExists = await verifyFileExists(logoPath); - if (!logoExists) { - throw new Error('Logo file not found in Minio'); - } -} - -// 5.2 Vérification des attachments -if (body.attachments?.length > 0) { - const attachmentVerifications = uploadedFiles - .filter(f => f.type === 'attachment') - .map(f => verifyFileExists(f.path)); - - const attachmentResults = await Promise.all(attachmentVerifications); - if (attachmentResults.some(exists => !exists)) { - throw new Error('One or more attachment files not found in Minio'); - } -} -``` - -**Fonction `verifyFileExists()` - Lignes 191-202** - -```typescript -async function verifyFileExists(filePath: string): Promise { - try { - await s3Client.send(new HeadObjectCommand({ - Bucket: 'missions', - Key: filePath.replace('missions/', '') // Retire le préfixe - })); - return true; // Fichier existe - } catch (error) { - return false; // Fichier n'existe pas - } -} -``` - -**Points importants** : -- ✅ Vérification **AVANT** de déclencher N8N -- ✅ Utilise `HeadObjectCommand` (légère, ne télécharge pas le fichier) -- ✅ Si un fichier manque, le workflow s'arrête avec erreur - -### 2.7 STEP 6 : Déclenchement du Workflow N8N - -**Lignes 367-393** - -```typescript -// 6.1 Préparation des données pour N8N -const n8nData = { - ...body, // Toutes les données de la mission - creatorId: userId, // ID du créateur - logoPath: logoPath, // Chemin du logo (ou null) - config: { - N8N_API_KEY: process.env.N8N_API_KEY, - MISSION_API_URL: process.env.NEXT_PUBLIC_API_URL - } -}; - -// 6.2 Déclenchement du workflow -const n8nService = new N8nService(); -const workflowResult = await n8nService.triggerMissionCreation(n8nData); - -// 6.3 Vérification du résultat -if (!workflowResult.success) { - throw new Error(workflowResult.error || 'N8N workflow failed'); -} - -// 6.4 Retour succès -return NextResponse.json({ - success: true, - mission, - message: 'Mission created successfully with all integrations' -}); -``` - ---- - -## 🔗 ÉTAPE 3 : Service N8N - -### Fichier : `lib/services/n8n-service.ts` - -### 3.1 Configuration - -**Lignes 3-17** - -```typescript -export class N8nService { - private webhookUrl: string; - private rollbackWebhookUrl: string; - private apiKey: string; - - constructor() { - this.webhookUrl = process.env.N8N_WEBHOOK_URL || - 'https://brain.slm-lab.net/webhook/mission-created'; - this.rollbackWebhookUrl = process.env.N8N_ROLLBACK_WEBHOOK_URL || - 'https://brain.slm-lab.net/webhook/mission-rollback'; - this.apiKey = process.env.N8N_API_KEY || ''; - } -} -``` - -### 3.2 Nettoyage et Validation des Données - -**Lignes 19-49** - -```typescript -async triggerMissionCreation(data: any): Promise { - // Nettoyage des données - const cleanData = { - name: data.name, - oddScope: Array.isArray(data.oddScope) ? data.oddScope : [data.oddScope], - niveau: data.niveau || 'default', - intention: data.intention?.trim() || '', - missionType: data.missionType || 'default', - donneurDOrdre: data.donneurDOrdre || 'default', - projection: data.projection || 'default', - services: Array.isArray(data.services) ? data.services : [], - participation: data.participation || 'default', - profils: Array.isArray(data.profils) ? data.profils : [], - guardians: data.guardians || {}, - volunteers: Array.isArray(data.volunteers) ? data.volunteers : [], - creatorId: data.creatorId, - config: { - ...data.config, - N8N_API_KEY: this.apiKey, - MISSION_API_URL: process.env.NEXT_PUBLIC_API_URL || 'https://api.slm-lab.net/api' - } - }; -} -``` - -**Points importants** : -- ✅ Normalisation des arrays (assure qu'ils sont bien des arrays) -- ✅ Valeurs par défaut pour éviter les undefined -- ✅ Trim de l'intention pour retirer les espaces -- ✅ Conservation de la config avec les clés API - -### 3.3 Envoi au Webhook N8N - -**Lignes 73-96** - -```typescript -const response = await fetch(this.webhookUrl, { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - 'x-api-key': this.apiKey // Authentification - }, - body: JSON.stringify(cleanData) -}); - -if (!response.ok) { - const errorText = await response.text(); - throw new Error(`HTTP error! status: ${response.status}, body: ${errorText}`); -} -``` - -### 3.4 Traitement de la Réponse - -**Lignes 98-143** - -```typescript -const responseText = await response.text(); - -try { - const result = JSON.parse(responseText); - - // Détection d'erreurs partielles - if (result.error || result.message?.includes('failed')) { - const errorMessage = result.message || result.error; - const failedServices = { - gitRepo: errorMessage.includes('Git repository creation failed'), - leantimeProject: errorMessage.includes('Leantime project creation failed'), - docCollection: errorMessage.includes('Documentation collection creation failed'), - rocketChatChannel: errorMessage.includes('RocketChat channel creation failed') - }; - - // Retourne succès avec erreurs partielles - return { - success: true, - results: { ...result, failedServices } - }; - } - - return { success: true, results: result }; - -} catch (parseError) { - // Si la réponse n'est pas JSON, considère comme succès - return { - success: true, - results: { - logoUrl: null, - leantimeProjectId: null, - outlineCollectionId: null, - rocketChatChannelId: null, - giteaRepositoryUrl: null - } - }; -} -``` - -**Format de réponse attendu de N8N** : -```json -{ - "leantimeProjectId": "project-123", - "outlineCollectionId": "collection-456", - "rocketChatChannelId": "channel-789", - "giteaRepositoryUrl": "https://git.slm-lab.net/mission-abc", - "penpotProjectId": "penpot-xyz" -} -``` - -**Intégrations créées par N8N** : -1. **Leantime** : Projet de gestion de projet -2. **Outline** : Collection de documentation -3. **RocketChat** : Canal de communication -4. **Gitea** : Repository Git -5. **Penpot** : Projet de design - -**Points importants** : -- ✅ Gestion des erreurs partielles (certains services peuvent échouer) -- ✅ Si la réponse n'est pas JSON, considère comme succès (workflow déclenché) -- ✅ Les IDs retournés ne sont **PAS** sauvegardés en base (TODO) - ---- - -## 🧹 Gestion des Erreurs et Cleanup - -### Fichier : `app/api/missions/route.ts` - Lignes 398-418 - -```typescript -catch (error) { - console.error('Error in mission creation:', error); - - // Cleanup: Suppression de tous les fichiers uploadés - for (const file of uploadedFiles) { - try { - await s3Client.send(new DeleteObjectCommand({ - Bucket: 'missions', - Key: file.path.replace('missions/', '') - })); - console.log('Cleaned up file:', file.path); - } catch (cleanupError) { - console.error('Error cleaning up file:', file.path, cleanupError); - } - } - - return NextResponse.json({ - error: 'Failed to create mission', - details: error instanceof Error ? error.message : String(error) - }, { status: 500 }); -} -``` - -**Scénarios de cleanup** : -1. ✅ Erreur lors de l'upload du logo → Suppression du logo -2. ✅ Erreur lors de l'upload d'un attachment → Suppression de tous les fichiers -3. ✅ Erreur lors de la vérification Minio → Suppression de tous les fichiers -4. ✅ Erreur N8N → Suppression de tous les fichiers - -**Points importants** : -- ✅ La mission reste en base même en cas d'erreur (orphan) -- ✅ Les MissionUsers restent en base même en cas d'erreur -- ✅ Seuls les fichiers Minio sont nettoyés -- ⚠️ **TODO** : Rollback complet (suppression mission + users si erreur) - ---- - -## 📊 Résumé des Opérations Prisma - -### Requêtes Prisma dans l'ordre d'exécution : - -1. **`prisma.mission.create()`** - - Crée la mission avec tous les champs - - Génère un UUID pour `id` - - `logo` = `null` initialement - -2. **`prisma.missionUser.createMany()`** - - Crée tous les gardiens et volontaires en une requête - - Utilise `createMany()` pour performance - -3. **`prisma.mission.update()`** (si logo) - - Met à jour le champ `logo` avec le chemin Minio - -4. **`prisma.attachment.create()`** (pour chaque attachment) - - Créé en parallèle avec `Promise.all()` - - Un enregistrement par fichier - -**Total** : 1 + 1 + (0 ou 1) + N requêtes Prisma -- Minimum : 2 requêtes (mission + users) -- Avec logo : 3 requêtes -- Avec N attachments : 3 + N requêtes - ---- - -## 📦 Résumé des Opérations Minio - -### Uploads Minio dans l'ordre d'exécution : - -1. **Logo** (si présent) - - Bucket : `missions` - - Key : `{missionId}/logo.png` - - Path stocké : `missions/{missionId}/logo.png` - -2. **Attachments** (si présents, en parallèle) - - Bucket : `missions` - - Key : `{missionId}/attachments/{filename}` - - Path stocké : `missions/{missionId}/attachments/{filename}` - -3. **Vérifications** (après uploads) - - `HeadObjectCommand` pour chaque fichier - - Vérifie l'existence avant N8N - -**Total** : 1 + N uploads + (1 + N) vérifications - ---- - -## 🔄 Résumé du Workflow N8N - -### Données envoyées à N8N : - -```typescript -{ - name: "Mission Example", - oddScope: ["odd-3"], - niveau: "a", - intention: "Description...", - missionType: "remote", - donneurDOrdre: "individual", - projection: "short", - services: ["Gite", "ArtLab"], - participation: "volontaire", - profils: ["DataIntelligence", "Expression"], - guardians: { - "gardien-temps": "user-1", - "gardien-parole": "user-2", - "gardien-memoire": "user-3" - }, - volunteers: ["user-4", "user-5"], - creatorId: "user-creator", - logoPath: "missions/abc-123/logo.png", - config: { - N8N_API_KEY: "...", - MISSION_API_URL: "https://api.slm-lab.net/api" - } -} -``` - -### Actions N8N (workflow interne) : - -1. Création projet Leantime -2. Création collection Outline -3. Création canal RocketChat -4. Création repository Gitea -5. Création projet Penpot - -**Note** : Les IDs retournés ne sont **PAS** sauvegardés en base actuellement. - ---- - -## ⚠️ Points d'Attention et TODOs - -### 1. Sauvegarde des IDs N8N -**Problème** : Les IDs retournés par N8N (leantimeProjectId, etc.) ne sont pas sauvegardés en base. - -**Solution proposée** : -```typescript -// Après le workflow N8N -if (workflowResult.success && workflowResult.results) { - await prisma.mission.update({ - where: { id: mission.id }, - data: { - leantimeProjectId: workflowResult.results.leantimeProjectId, - outlineCollectionId: workflowResult.results.outlineCollectionId, - rocketChatChannelId: workflowResult.results.rocketChatChannelId, - giteaRepositoryUrl: workflowResult.results.giteaRepositoryUrl, - penpotProjectId: workflowResult.results.penpotProjectId - } - }); -} -``` - -### 2. Rollback Complet -**Problème** : En cas d'erreur, la mission reste en base (orphan). - -**Solution proposée** : -```typescript -catch (error) { - // Suppression de la mission et des relations - await prisma.mission.delete({ where: { id: mission.id } }); - // Les MissionUsers et Attachments seront supprimés en cascade - // Nettoyage Minio... -} -``` - -### 3. Credentials Minio Hardcodés -**Problème** : Les credentials Minio sont hardcodés dans `lib/mission-uploads.ts`. - -**Solution** : Déplacer vers variables d'environnement. - -### 4. Gestion des Erreurs N8N Partielles -**Problème** : Si certains services N8N échouent, on continue quand même. - -**Solution** : Décider si on continue ou on rollback selon la criticité. - ---- - -## 📈 Performance et Optimisations - -### Optimisations actuelles : -- ✅ `createMany()` pour MissionUsers (1 requête au lieu de N) -- ✅ `Promise.all()` pour les attachments (parallèle) -- ✅ `HeadObjectCommand` pour vérification (léger) - -### Optimisations possibles : -- 🔄 Transaction Prisma pour atomicité -- 🔄 Batch upload Minio (si supporté) -- 🔄 Retry logic pour N8N -- 🔄 Cache des vérifications Minio - ---- - -## 🔍 Debugging et Logs - -### Points de logging importants : - -1. **Début du workflow** : `=== Mission Creation Started ===` -2. **Création Prisma** : `Mission created successfully` -3. **Upload Minio** : `Logo upload successful` / `Attachment upload successful` -4. **Vérification** : `verifyFileExists()` logs -5. **N8N** : `=== Starting N8N Workflow ===` / `N8N Workflow Result` -6. **Erreurs** : Tous les catch blocks loggent les erreurs - -### Commandes de debug : - -```bash -# Voir les logs du serveur -npm run dev # Logs dans la console - -# Vérifier Minio -# Accéder à https://dome-api.slm-lab.net -# Bucket: missions - -# Vérifier Prisma -npx prisma studio # Interface graphique -``` - ---- - -**Document généré le** : $(date) -**Version** : 1.0 -**Auteur** : Analyse complète du codebase - diff --git a/MISSION_DELETION_IDS_EMPTY_ANALYSIS.md b/MISSION_DELETION_IDS_EMPTY_ANALYSIS.md deleted file mode 100644 index 6bf6b769..00000000 --- a/MISSION_DELETION_IDS_EMPTY_ANALYSIS.md +++ /dev/null @@ -1,377 +0,0 @@ -# Analyse : IDs Vides lors de la Suppression de Mission - -## 🔍 Problème - -Lors de la suppression d'une mission, N8N reçoit des IDs vides : - -```json -{ - "repoName": "", - "leantimeProjectId": 0, - "documentationCollectionId": "", - "rocketchatChannelId": "", - "giteaRepositoryUrl": null, - "outlineCollectionId": null, - "rocketChatChannelId": null -} -``` - -**Cela signifie que les IDs ne sont PAS sauvegardés en base lors de la création.** - ---- - -## 🔄 Flow de Création (Théorique) - -``` -1. POST /api/missions - ↓ -2. Crée mission en Prisma (sans IDs) - ↓ -3. Upload logo dans Minio - ↓ -4. POST N8N webhook (mission-created) - ↓ -5. N8N crée intégrations : - - Gitea → retourne html_url - - Leantime → retourne projectId - - Outline → retourne collectionId - - RocketChat → retourne channelId - ↓ -6. N8N → POST /mission-created (avec les IDs) - ↓ -7. Backend sauvegarde les IDs en base ✅ -``` - ---- - -## ❌ Problèmes Possibles - -### Problème 1: N8N n'appelle pas `/mission-created` - -**Symptôme** : Les IDs ne sont jamais sauvegardés - -**Vérification** : -- Vérifier les logs N8N : le node "Save Mission To API" est-il exécuté ? -- Vérifier les logs backend : y a-t-il des appels à `/mission-created` ? -- Vérifier les erreurs N8N : le workflow s'arrête-t-il avant "Save Mission To API" ? - -**Solution** : -- Vérifier que le node "Save Mission To API" est bien connecté dans le workflow -- Vérifier que l'URL est correcte : `{{ MISSION_API_URL }}/mission-created` -- Vérifier que l'API key est correcte dans les headers - ---- - -### Problème 2: N8N appelle `/mission-created` mais sans `missionId` - -**Symptôme** : L'endpoint ne trouve pas la mission - -**Vérification** : -- Vérifier les logs backend : - ``` - === Mission Created Webhook Received === - Looking up mission by ID: ... - Mission not found: ... - ``` -- Vérifier le body reçu par `/mission-created` : contient-il `missionId` ? - -**Solution** : -- Modifier le workflow N8N pour inclure `missionId` dans le body : - ```json - { - "missionId": "={{ $node['Process Mission Data'].json.missionId }}", - ... - } - ``` - ---- - -### Problème 3: N8N appelle `/mission-created` mais les IDs sont vides - -**Symptôme** : L'endpoint trouve la mission mais les IDs sont `null` ou vides - -**Vérification** : -- Vérifier les logs backend : - ``` - Received mission-created data: { - missionId: "...", - gitRepoUrl: null, // ❌ Vide - leantimeProjectId: null, // ❌ Vide - ... - } - ``` -- Vérifier les logs N8N : les nodes de création retournent-ils bien les IDs ? - -**Solution** : -- Vérifier que les nodes N8N (Create Git Repository, Create Leantime Project, etc.) retournent bien les IDs -- Vérifier que le node "Combine Results" combine correctement les IDs -- Vérifier que le node "Save Mission To API" utilise les bons chemins pour les IDs - ---- - -### Problème 4: Mapping incorrect des champs - -**Symptôme** : Les IDs sont envoyés mais avec des noms incorrects - -**Vérification** : -- Vérifier le body envoyé par N8N : - ```json - { - "gitRepoUrl": "...", // ✅ Correct - "leantimeProjectId": "...", // ✅ Correct - "documentationCollectionId": "...", // ✅ Correct - "rocketchatChannelId": "..." // ✅ Correct - } - ``` -- Vérifier le mapping dans `/mission-created` : - - `gitRepoUrl` → `giteaRepositoryUrl` ✅ - - `documentationCollectionId` → `outlineCollectionId` ✅ - - `rocketchatChannelId` → `rocketChatChannelId` ✅ - -**Solution** : -- Vérifier que les noms de champs correspondent exactement - ---- - -### Problème 5: API Key incorrecte - -**Symptôme** : L'endpoint retourne 401 Unauthorized - -**Vérification** : -- Vérifier les logs backend : - ``` - Invalid API key: { received: '...', expected: '...' } - ``` -- Vérifier que `N8N_API_KEY` est bien configuré dans l'environnement -- Vérifier que N8N envoie bien `x-api-key` dans les headers - -**Solution** : -- Vérifier la variable d'environnement `N8N_API_KEY` -- Vérifier que N8N utilise la bonne API key dans le header - ---- - -## 🔍 Points de Vérification - -### 1. Vérifier les Logs Backend - -**Lors de la création** : -``` -=== Starting N8N Workflow === -Sending to N8N: { missionId: "...", ... } -N8N Workflow Result: { success: true, ... } -``` - -**Lors de l'appel `/mission-created`** : -``` -=== Mission Created Webhook Received === -Received mission-created data: { ... } -Looking up mission by ID: ... -Found mission: { id: "...", ... } -Mission updated successfully: { ... } -``` - -**Si ces logs n'apparaissent pas** → N8N n'appelle pas `/mission-created` - ---- - -### 2. Vérifier les Logs N8N - -**Dans le workflow N8N** : -- Le node "Save Mission To API" est-il exécuté ? -- Y a-t-il des erreurs dans ce node ? -- Le body envoyé contient-il les IDs ? - -**Vérifier le body du node "Save Mission To API"** : -```json -{ - "missionId": "={{ $node['Process Mission Data'].json.missionId }}", - "gitRepoUrl": "={{ $node['Combine Results'].json.gitRepo?.html_url }}", - "leantimeProjectId": "={{ $node['Combine Results'].json.leantimeProject?.result?.[0] }}", - "documentationCollectionId": "={{ $node['Combine Results'].json.docCollection?.data?.id }}", - "rocketchatChannelId": "={{ $node['Combine Results'].json.rocketChatChannel?.channel?._id }}" -} -``` - ---- - -### 3. Vérifier la Base de Données - -**Requête SQL** : -```sql -SELECT - id, - name, - giteaRepositoryUrl, - leantimeProjectId, - outlineCollectionId, - rocketChatChannelId, - createdAt, - updatedAt -FROM Mission -WHERE name = 'Creation' -ORDER BY createdAt DESC -LIMIT 1; -``` - -**Résultat attendu** : -``` -id: cd0225cf-8dfd-4bf0-a20a-6aa9c04ebb42 -name: Creation -giteaRepositoryUrl: https://gite.slm-lab.net/alma/creation ✅ -leantimeProjectId: 123 ✅ -outlineCollectionId: collection-456 ✅ -rocketChatChannelId: channel-789 ✅ -``` - -**Si tous les IDs sont `null`** → Ils ne sont pas sauvegardés - ---- - -## 📋 Checklist de Diagnostic - -### Étape 1: Vérifier que N8N reçoit missionId - -- [ ] Les logs backend montrent `missionId` dans `n8nData` -- [ ] N8N reçoit bien `missionId` dans le webhook - -### Étape 2: Vérifier que N8N crée les intégrations - -- [ ] Les logs N8N montrent que les nodes de création sont exécutés -- [ ] Les nodes retournent bien les IDs (html_url, projectId, etc.) - -### Étape 3: Vérifier que N8N combine les résultats - -- [ ] Le node "Combine Results" contient les IDs -- [ ] Les IDs sont accessibles via les chemins corrects - -### Étape 4: Vérifier que N8N appelle `/mission-created` - -- [ ] Les logs backend montrent des appels à `/mission-created` -- [ ] Le node "Save Mission To API" est exécuté dans N8N -- [ ] Pas d'erreur 401 (API key) ou 404 (mission not found) - -### Étape 5: Vérifier que les IDs sont sauvegardés - -- [ ] Les logs backend montrent "Mission updated successfully" -- [ ] La base de données contient les IDs après création -- [ ] Les IDs sont correctement mappés - ---- - -## 🎯 Actions Recommandées (Sans Toucher au Code) - -### 1. Vérifier les Logs Backend - -```bash -# Chercher les appels à /mission-created -grep "Mission Created Webhook Received" logs.txt - -# Chercher les erreurs -grep "Mission not found" logs.txt -grep "Invalid API key" logs.txt -``` - -### 2. Vérifier le Workflow N8N - -1. Ouvrir le workflow `NeahMissionCreate` -2. Vérifier que le node "Save Mission To API" : - - Est bien connecté après "Combine Results" - - Contient `missionId` dans le body - - Utilise les bons chemins pour les IDs - - A l'URL correcte : `{{ MISSION_API_URL }}/mission-created` - - A l'API key correcte dans les headers - -### 3. Tester Manuellement - -**Appel manuel à `/mission-created`** : -```bash -curl -X POST https://hub.slm-lab.net/api/missions/mission-created \ - -H "Content-Type: application/json" \ - -H "x-api-key: YOUR_API_KEY" \ - -d '{ - "missionId": "cd0225cf-8dfd-4bf0-a20a-6aa9c04ebb42", - "name": "Creation", - "creatorId": "user-id", - "gitRepoUrl": "https://gite.slm-lab.net/alma/creation", - "leantimeProjectId": "123", - "documentationCollectionId": "collection-456", - "rocketchatChannelId": "channel-789" - }' -``` - -**Vérifier la réponse** : -- 200 OK → L'endpoint fonctionne -- 401 Unauthorized → Problème d'API key -- 404 Not Found → Problème de recherche de mission -- 400 Bad Request → Problème de validation - -### 4. Vérifier la Base de Données - -```sql --- Vérifier une mission spécifique -SELECT * FROM Mission WHERE id = 'cd0225cf-8dfd-4bf0-a20a-6aa9c04ebb42'; - --- Vérifier les missions récentes sans IDs -SELECT id, name, createdAt -FROM Mission -WHERE giteaRepositoryUrl IS NULL - AND leantimeProjectId IS NULL - AND outlineCollectionId IS NULL - AND rocketChatChannelId IS NULL -ORDER BY createdAt DESC -LIMIT 10; -``` - ---- - -## 🔧 Solutions Probables - -### Solution 1: Ajouter missionId dans N8N - -**Dans le node "Save Mission To API"** : -```json -{ - "missionId": "={{ $node['Process Mission Data'].json.missionId }}", - ... -} -``` - -### Solution 2: Vérifier les Chemins des IDs dans N8N - -**Vérifier que les chemins sont corrects** : -- `gitRepo.html_url` (pas `gitRepo.body.html_url`) -- `leantimeProject.result[0]` (array avec index 0) -- `docCollection.data.id` (pas `docCollection.id`) -- `rocketChatChannel.channel._id` (pas `rocketChatChannel._id`) - -### Solution 3: Vérifier l'API Key - -**Vérifier que** : -- `N8N_API_KEY` est configuré dans `.env` -- N8N utilise la même clé dans le header `x-api-key` - ---- - -## 📝 Résumé - -**Le problème est que les IDs ne sont pas sauvegardés en base.** - -**Causes possibles** : -1. ❌ N8N n'appelle pas `/mission-created` -2. ❌ N8N appelle mais sans `missionId` → mission non trouvée -3. ❌ N8N appelle mais les IDs sont vides dans le body -4. ❌ API key incorrecte → 401 Unauthorized -5. ❌ Mapping incorrect des champs - -**Action immédiate** : -1. Vérifier les logs backend pour voir si `/mission-created` est appelé -2. Vérifier le workflow N8N pour voir si `missionId` est inclus -3. Vérifier la base de données pour voir si les IDs sont sauvegardés - ---- - -**Date**: $(date) -**Status**: Analyse sans modification de code -**Action Requise**: Vérification des logs et du workflow N8N - diff --git a/MISSION_DELETION_WORKFLOW.md b/MISSION_DELETION_WORKFLOW.md deleted file mode 100644 index 14c88f35..00000000 --- a/MISSION_DELETION_WORKFLOW.md +++ /dev/null @@ -1,625 +0,0 @@ -# Workflow Détaillé : Suppression de Mission - Centrale (/missions) - -## 📋 Vue d'Ensemble - -Ce document trace **chaque étape** du workflow de suppression d'une mission depuis la page Centrale (`/missions/[missionId]`), incluant les vérifications de permissions, la suppression des fichiers Minio, et la suppression en base de données. - ---- - -## 🔄 Flux Complet - Vue d'Ensemble - -``` -┌─────────────────────────────────────────────────────────────┐ -│ 1. FRONTEND - MissionDetailPage │ -│ - Clic sur bouton "Supprimer" │ -│ - Confirmation utilisateur (confirm dialog) │ -│ - DELETE /api/missions/[missionId] │ -└─────────────────────────────────────────────────────────────┘ - ↓ -┌─────────────────────────────────────────────────────────────┐ -│ 2. BACKEND - DELETE /api/missions/[missionId] │ -│ ├─ Authentification (NextAuth) │ -│ ├─ Vérification mission existe │ -│ ├─ Vérification permissions (créateur ou admin) │ -│ ├─ STEP 1: Suppression logo Minio (si existe) │ -│ ├─ STEP 2: Rollback N8N (TODO - non implémenté) │ -│ └─ STEP 3: Suppression mission Prisma (CASCADE) │ -└─────────────────────────────────────────────────────────────┘ - ↓ -┌─────────────────────────────────────────────────────────────┐ -│ 3. PRISMA CASCADE │ -│ - Suppression automatique MissionUsers │ -│ - Suppression automatique Attachments │ -│ ⚠️ Fichiers Minio des attachments NON supprimés │ -└─────────────────────────────────────────────────────────────┘ -``` - ---- - -## 📝 ÉTAPE 1 : Frontend - Clic sur Supprimer - -### Fichier : `app/missions/[missionId]/page.tsx` - -### 1.1 Bouton Supprimer - -**Lignes 397-411** - -```typescript - -``` - -**Caractéristiques** : -- ✅ Bouton rouge avec icône Trash2 -- ✅ Désactivé pendant la suppression (`deleting` state) -- ✅ Affiche un spinner pendant le traitement - -### 1.2 Handler de Suppression - -**Lignes 143-176** - -```typescript -const handleDeleteMission = async () => { - // 1. Confirmation utilisateur - if (!confirm("Êtes-vous sûr de vouloir supprimer cette mission ? Cette action est irréversible.")) { - return; // Annulation si l'utilisateur refuse - } - - try { - setDeleting(true); // Active le spinner - - // 2. Appel API DELETE - const response = await fetch(`/api/missions/${missionId}`, { - method: 'DELETE', - }); - - // 3. Vérification de la réponse - if (!response.ok) { - throw new Error('Failed to delete mission'); - } - - // 4. Notification de succès - toast({ - title: "Mission supprimée", - description: "La mission a été supprimée avec succès", - }); - - // 5. Redirection vers la liste des missions - router.push('/missions'); - - } catch (error) { - console.error('Error deleting mission:', error); - toast({ - title: "Erreur", - description: "Impossible de supprimer la mission", - variant: "destructive", - }); - } finally { - setDeleting(false); // Désactive le spinner - } -}; -``` - -**Points importants** : -- ✅ **Double confirmation** : Dialog natif du navigateur -- ✅ **Gestion d'état** : `deleting` pour le spinner -- ✅ **Redirection automatique** vers `/missions` en cas de succès -- ✅ **Gestion d'erreurs** avec toast notification - ---- - -## 🗄️ ÉTAPE 2 : Backend - DELETE /api/missions/[missionId] - -### Fichier : `app/api/missions/[missionId]/route.ts` - -### 2.1 Authentification et Récupération de la Mission - -**Lignes 292-316** - -```typescript -export async function DELETE( - request: Request, - props: { params: Promise<{ missionId: string }> } -) { - const params = await props.params; - - try { - // 1. Vérification de l'authentification - const session = await getServerSession(authOptions); - if (!session?.user) { - return NextResponse.json({ error: 'Unauthorized' }, { status: 401 }); - } - - // 2. Récupération de la mission avec relations - const mission = await prisma.mission.findUnique({ - where: { id: params.missionId }, - include: { - missionUsers: { - include: { - user: true // Inclut les infos des utilisateurs - } - } - } - }); - - // 3. Vérification que la mission existe - if (!mission) { - return NextResponse.json({ error: 'Mission not found' }, { status: 404 }); - } -``` - -**Points importants** : -- ✅ Vérification de session NextAuth -- ✅ Récupération de la mission avec `missionUsers` (pour logging/info) -- ✅ Retour 404 si mission n'existe pas - -### 2.2 Vérification des Permissions - -**Lignes 318-324** - -```typescript -// Vérification : utilisateur doit être créateur OU admin -const isCreator = mission.creatorId === session.user.id; -const userRoles = Array.isArray(session.user.role) ? session.user.role : []; -const isAdmin = userRoles.includes('admin') || userRoles.includes('ADMIN'); - -if (!isCreator && !isAdmin) { - return NextResponse.json({ error: 'Forbidden' }, { status: 403 }); -} -``` - -**Règles de permissions** : -- ✅ **Créateur** : Peut supprimer sa propre mission -- ✅ **Admin** : Peut supprimer n'importe quelle mission -- ❌ **Autres utilisateurs** : Même gardiens/volontaires ne peuvent pas supprimer - -**Points importants** : -- ✅ Vérification stricte : seul le créateur ou un admin peut supprimer -- ✅ Les gardiens (même gardien-memoire) ne peuvent pas supprimer -- ✅ Retour 403 Forbidden si pas autorisé - -### 2.3 STEP 1 : Suppression du Logo dans Minio - -**Lignes 326-334** - -```typescript -// Suppression du logo si présent -if (mission.logo) { - try { - await deleteMissionLogo(params.missionId, mission.logo); - } catch (error) { - console.error('Error deleting mission logo:', error); - // Continue deletion even if logo deletion fails - } -} -``` - -**Fonction `deleteMissionLogo()` - `lib/mission-uploads.ts`** : - -**Lignes 42-61** - -```typescript -export async function deleteMissionLogo( - missionId: string, - logoPath: string -): Promise { - try { - const normalizedPath = ensureMissionsPrefix(logoPath); - // ⚠️ TODO: La fonction ne fait que logger, ne supprime pas vraiment ! - console.log('Deleting mission logo:', { - missionId, - originalPath: logoPath, - normalizedPath - }); - // TODO: Implémenter la suppression réelle avec DeleteObjectCommand - } catch (error) { - console.error('Error deleting mission logo:', error); - throw error; - } -} -``` - -**⚠️ PROBLÈME IDENTIFIÉ** : -- ❌ La fonction `deleteMissionLogo()` ne supprime **PAS** réellement le fichier -- ❌ Elle ne fait que logger les informations -- ⚠️ Le logo reste dans Minio après suppression de la mission - -**Solution proposée** : -```typescript -export async function deleteMissionLogo( - missionId: string, - logoPath: string -): Promise { - try { - const { DeleteObjectCommand } = await import('@aws-sdk/client-s3'); - const normalizedPath = ensureMissionsPrefix(logoPath); - const minioPath = normalizedPath.replace(/^missions\//, ''); - - await s3Client.send(new DeleteObjectCommand({ - Bucket: 'missions', - Key: minioPath - })); - - console.log('Mission logo deleted successfully:', minioPath); - } catch (error) { - console.error('Error deleting mission logo:', error); - throw error; - } -} -``` - -**Points importants** : -- ✅ Continue la suppression même si le logo échoue (try/catch) -- ⚠️ **BUG** : Le logo n'est pas réellement supprimé actuellement - -### 2.4 STEP 2 : Rollback N8N (TODO - Non Implémenté) - -**Lignes 336-344** - -```typescript -// Trigger n8n workflow for rollback -// TODO: Implement rollbackMission method in N8nService -// const n8nService = new N8nService(); -// try { -// await n8nService.rollbackMission(mission); -// } catch (error) { -// console.error('Error during mission rollback:', error); -// // Continue with mission deletion even if rollback fails -// } -``` - -**⚠️ NON IMPLÉMENTÉ** : -- ❌ Le rollback N8N n'est pas appelé -- ❌ Les intégrations externes (Leantime, Outline, RocketChat, etc.) ne sont **PAS** supprimées -- ⚠️ Les ressources externes restent orphelines - -**Méthode disponible mais non utilisée** : -- ✅ `N8nService.triggerMissionRollback()` existe dans `lib/services/n8n-service.ts` -- ✅ Webhook URL : `https://brain.slm-lab.net/webhook/mission-rollback` -- ❌ Mais n'est pas appelée dans le DELETE - -**Solution proposée** : -```typescript -// Rollback N8N -const n8nService = new N8nService(); -try { - await n8nService.triggerMissionRollback({ - missionId: mission.id, - leantimeProjectId: mission.leantimeProjectId, - outlineCollectionId: mission.outlineCollectionId, - rocketChatChannelId: mission.rocketChatChannelId, - giteaRepositoryUrl: mission.giteaRepositoryUrl, - penpotProjectId: mission.penpotProjectId - }); -} catch (error) { - console.error('Error during mission rollback:', error); - // Continue with mission deletion even if rollback fails -} -``` - -### 2.5 STEP 3 : Suppression de la Mission en Base de Données - -**Lignes 346-349** - -```typescript -// Suppression de la mission (CASCADE automatique) -await prisma.mission.delete({ - where: { id: params.missionId } -}); -``` - -**Schéma Prisma - Relations avec CASCADE** : - -```prisma -model Mission { - // ... - attachments Attachment[] - missionUsers MissionUser[] -} - -model Attachment { - mission Mission @relation(fields: [missionId], references: [id], onDelete: Cascade) - // ... -} - -model MissionUser { - mission Mission @relation(fields: [missionId], references: [id], onDelete: Cascade) - // ... -} -``` - -**CASCADE automatique** : -- ✅ **MissionUsers** : Supprimés automatiquement (`onDelete: Cascade`) -- ✅ **Attachments** : Supprimés automatiquement (`onDelete: Cascade`) -- ❌ **Fichiers Minio** : **NON supprimés automatiquement** (pas de trigger) - -**Points importants** : -- ✅ Une seule requête Prisma supprime la mission et toutes ses relations -- ✅ Atomicité : Si la suppression échoue, rien n'est supprimé -- ⚠️ **PROBLÈME** : Les fichiers Minio des attachments ne sont pas supprimés - -### 2.6 Retour de Succès - -**Lignes 351-358** - -```typescript -return NextResponse.json({ success: true }); - -} catch (error) { - console.error('Error deleting mission:', error); - return NextResponse.json( - { error: 'Failed to delete mission' }, - { status: 500 } - ); -} -``` - ---- - -## 🔄 ÉTAPE 3 : Cascade Prisma - -### Suppression Automatique des Relations - -Quand `prisma.mission.delete()` est exécuté, Prisma supprime automatiquement : - -1. **Tous les MissionUsers** associés - ```sql - DELETE FROM "MissionUser" WHERE "missionId" = 'abc-123'; - ``` - -2. **Tous les Attachments** associés - ```sql - DELETE FROM "Attachment" WHERE "missionId" = 'abc-123'; - ``` - -**⚠️ PROBLÈME MAJEUR** : -- ❌ Les fichiers Minio des attachments ne sont **PAS** supprimés -- ❌ Les fichiers restent dans Minio : `missions/{missionId}/attachments/*` -- ⚠️ **Orphelins** : Fichiers sans enregistrement en base - ---- - -## 🧹 Problèmes Identifiés et Solutions - -### Problème 1 : Logo non supprimé dans Minio - -**Symptôme** : Le logo reste dans Minio après suppression - -**Cause** : `deleteMissionLogo()` ne fait que logger, ne supprime pas - -**Solution** : -```typescript -export async function deleteMissionLogo( - missionId: string, - logoPath: string -): Promise { - const { DeleteObjectCommand } = await import('@aws-sdk/client-s3'); - const normalizedPath = ensureMissionsPrefix(logoPath); - const minioPath = normalizedPath.replace(/^missions\//, ''); - - await s3Client.send(new DeleteObjectCommand({ - Bucket: 'missions', - Key: minioPath - })); -} -``` - -### Problème 2 : Attachments non supprimés dans Minio - -**Symptôme** : Les fichiers attachments restent dans Minio - -**Cause** : Pas de suppression des fichiers avant la suppression Prisma - -**Solution** : -```typescript -// Avant la suppression Prisma -if (mission.attachments && mission.attachments.length > 0) { - // Récupérer les attachments - const attachments = await prisma.attachment.findMany({ - where: { missionId: params.missionId } - }); - - // Supprimer chaque fichier Minio - for (const attachment of attachments) { - try { - await deleteMissionAttachment(attachment.filePath); - } catch (error) { - console.error('Error deleting attachment file:', error); - // Continue même si un fichier échoue - } - } -} -``` - -### Problème 3 : Rollback N8N non implémenté - -**Symptôme** : Les intégrations externes restent orphelines - -**Cause** : Code commenté, non implémenté - -**Solution** : -```typescript -// Décommenter et implémenter -const n8nService = new N8nService(); -try { - await n8nService.triggerMissionRollback({ - missionId: mission.id, - leantimeProjectId: mission.leantimeProjectId, - outlineCollectionId: mission.outlineCollectionId, - rocketChatChannelId: mission.rocketChatChannelId, - giteaRepositoryUrl: mission.giteaRepositoryUrl, - penpotProjectId: mission.penpotProjectId - }); -} catch (error) { - console.error('Error during mission rollback:', error); - // Continue avec la suppression même si rollback échoue -} -``` - ---- - -## 📊 Résumé des Opérations - -### Opérations Effectuées ✅ - -1. ✅ **Vérification authentification** : Session NextAuth -2. ✅ **Vérification permissions** : Créateur ou Admin -3. ✅ **Suppression Prisma Mission** : Avec cascade automatique -4. ✅ **Suppression Prisma MissionUsers** : Cascade automatique -5. ✅ **Suppression Prisma Attachments** : Cascade automatique - -### Opérations NON Effectuées ❌ - -1. ❌ **Suppression logo Minio** : Fonction ne fait que logger -2. ❌ **Suppression attachments Minio** : Pas de code pour supprimer -3. ❌ **Rollback N8N** : Code commenté, non implémenté - ---- - -## 🔍 Workflow Complet Corrigé (Proposé) - -```typescript -export async function DELETE(...) { - // 1. Authentification - const session = await getServerSession(authOptions); - if (!session?.user) return 401; - - // 2. Récupération mission avec attachments - const mission = await prisma.mission.findUnique({ - where: { id: params.missionId }, - include: { - attachments: true, // Inclure les attachments - missionUsers: true - } - }); - - if (!mission) return 404; - - // 3. Vérification permissions - const isCreator = mission.creatorId === session.user.id; - const isAdmin = session.user.role?.includes('admin'); - if (!isCreator && !isAdmin) return 403; - - // 4. Suppression logo Minio - if (mission.logo) { - try { - await deleteMissionLogo(mission.id, mission.logo); // À implémenter - } catch (error) { - console.error('Error deleting logo:', error); - // Continue - } - } - - // 5. Suppression attachments Minio - if (mission.attachments && mission.attachments.length > 0) { - for (const attachment of mission.attachments) { - try { - await deleteMissionAttachment(attachment.filePath); - } catch (error) { - console.error('Error deleting attachment:', error); - // Continue - } - } - } - - // 6. Rollback N8N - const n8nService = new N8nService(); - try { - await n8nService.triggerMissionRollback({ - missionId: mission.id, - leantimeProjectId: mission.leantimeProjectId, - outlineCollectionId: mission.outlineCollectionId, - rocketChatChannelId: mission.rocketChatChannelId, - giteaRepositoryUrl: mission.giteaRepositoryUrl, - penpotProjectId: mission.penpotProjectId - }); - } catch (error) { - console.error('Error during N8N rollback:', error); - // Continue même si rollback échoue - } - - // 7. Suppression Prisma (CASCADE) - await prisma.mission.delete({ - where: { id: params.missionId } - }); - - return NextResponse.json({ success: true }); -} -``` - ---- - -## 📈 Ordre d'Exécution Recommandé - -1. **Vérifications** (authentification, permissions, existence) -2. **Suppression fichiers Minio** (logo + attachments) -3. **Rollback N8N** (intégrations externes) -4. **Suppression Prisma** (mission + cascade automatique) - -**Pourquoi cet ordre ?** -- ✅ Supprimer les fichiers avant la base pour éviter les orphelins -- ✅ Rollback N8N avant suppression Prisma pour avoir les IDs -- ✅ Suppression Prisma en dernier (point de non-retour) - ---- - -## ⚠️ Points d'Attention - -### 1. Atomicité -**Problème** : Si une étape échoue, les précédentes sont déjà faites - -**Solution** : Transaction Prisma + Rollback manuel si erreur - -### 2. Performance -**Problème** : Suppression séquentielle des fichiers Minio - -**Solution** : `Promise.all()` pour suppressions parallèles - -### 3. Gestion d'Erreurs -**Problème** : Continue même si certaines suppressions échouent - -**Solution** : Décider si on continue ou on rollback selon criticité - ---- - -## 🔍 Debugging - -### Logs à surveiller : - -1. **Début suppression** : `DELETE /api/missions/[id]` -2. **Permissions** : `isCreator` / `isAdmin` -3. **Suppression logo** : `Deleting mission logo` -4. **Suppression attachments** : `Deleting mission attachment` -5. **Rollback N8N** : `Triggering n8n rollback workflow` -6. **Suppression Prisma** : `prisma.mission.delete()` - -### Vérifications manuelles : - -```bash -# Vérifier Minio -# Accéder à https://dome-api.slm-lab.net -# Bucket: missions -# Vérifier que les dossiers {missionId} sont supprimés - -# Vérifier Prisma -npx prisma studio -# Vérifier que la mission et ses relations sont supprimées -``` - ---- - -**Document généré le** : $(date) -**Version** : 1.0 -**Auteur** : Analyse complète du codebase - diff --git a/MISSION_INTEGRATION_IDS_FIX.md b/MISSION_INTEGRATION_IDS_FIX.md deleted file mode 100644 index 0d26fb56..00000000 --- a/MISSION_INTEGRATION_IDS_FIX.md +++ /dev/null @@ -1,308 +0,0 @@ -# Fix : Sauvegarde des IDs d'Intégration N8N - -## 🔍 Problème Identifié - -Lors de la suppression d'une mission, le webhook N8N reçoit des IDs vides : - -```json -{ - "repoName": "", - "leantimeProjectId": 0, - "documentationCollectionId": "", - "rocketchatChannelId": "" -} -``` - -**Cause** : Les IDs retournés par N8N lors de la création des intégrations n'étaient **jamais sauvegardés en base**. - -### Workflow Problématique - -``` -1. Frontend → POST /api/missions - ↓ -2. Backend crée mission en Prisma - ↓ -3. Backend upload fichiers Minio - ↓ -4. Backend → POST N8N webhook (mission-created) - ↓ -5. N8N crée intégrations (Gitea, Leantime, Outline, RocketChat) - ↓ -6. N8N → POST /mission-created ❌ ENDPOINT N'EXISTAIT PAS - ↓ -7. IDs jamais sauvegardés ❌ -``` - -### Conséquence - -Lors de la suppression : -- Les IDs sont `null` en base -- On envoie des valeurs vides à N8N -- N8N ne peut pas supprimer/fermer les intégrations -- Les ressources externes restent orphelines - ---- - -## ✅ Solution Implémentée - -### 1. Endpoint `/mission-created` Créé - -**Fichier** : `app/api/missions/mission-created/route.ts` - -**Fonctionnalités** : -- ✅ Reçoit les IDs des intégrations créées par N8N -- ✅ Vérifie l'API key (`x-api-key` header) -- ✅ Trouve la mission par `name` + `creatorId` -- ✅ Met à jour la mission avec les IDs -- ✅ Mappe correctement les champs : - - `gitRepoUrl` → `giteaRepositoryUrl` - - `documentationCollectionId` → `outlineCollectionId` - - `rocketchatChannelId` → `rocketChatChannelId` - - `leantimeProjectId` → `leantimeProjectId` (converti en string) - -### 2. Format des Données - -**N8N envoie** : -```json -{ - "name": "Mission Example", - "creatorId": "user-id", - "gitRepoUrl": "https://gite.slm-lab.net/alma/mission-example", - "leantimeProjectId": "123", - "documentationCollectionId": "collection-456", - "rocketchatChannelId": "channel-789" -} -``` - -**Endpoint sauvegarde** : -```typescript -{ - giteaRepositoryUrl: "https://gite.slm-lab.net/alma/mission-example", - leantimeProjectId: "123", - outlineCollectionId: "collection-456", - rocketChatChannelId: "channel-789" -} -``` - -### 3. Sécurité - -- ✅ Vérification de l'API key (`x-api-key` header) -- ✅ Validation des champs requis (`name`, `creatorId`) -- ✅ Gestion d'erreurs complète -- ✅ Logging détaillé pour debugging - ---- - -## 🔄 Nouveau Workflow - -``` -1. Frontend → POST /api/missions - ↓ -2. Backend crée mission en Prisma - ↓ -3. Backend upload fichiers Minio - ↓ -4. Backend → POST N8N webhook (mission-created) - ↓ -5. N8N crée intégrations (Gitea, Leantime, Outline, RocketChat) - ↓ -6. N8N → POST /mission-created ✅ ENDPOINT EXISTE MAINTENANT - ↓ -7. Backend sauvegarde les IDs ✅ - ↓ -8. Mission complète avec tous les IDs ✅ -``` - -**Lors de la suppression** : -``` -1. Frontend → DELETE /api/missions/[id] - ↓ -2. Backend récupère mission (avec IDs sauvegardés) - ↓ -3. Backend extrait/mappe les données - ↓ -4. Backend → POST N8N webhook (mission-delete) - ↓ -5. N8N reçoit les IDs ✅ - ↓ -6. N8N supprime/ferme les intégrations ✅ -``` - ---- - -## 📋 Format de Requête - -### POST /api/missions/mission-created - -**Headers** : -``` -Content-Type: application/json -x-api-key: {N8N_API_KEY} -Authorization: Bearer {keycloak_token} (optionnel) -``` - -**Body** : -```json -{ - "name": "Mission Example", - "creatorId": "user-uuid", - "gitRepoUrl": "https://gite.slm-lab.net/alma/mission-example", - "leantimeProjectId": "123", - "documentationCollectionId": "collection-456", - "rocketchatChannelId": "channel-789", - "niveau": "default", - "intention": "...", - "description": "...", - "donneurDOrdre": "...", - "projection": "...", - "missionType": "remote" -} -``` - -**Réponse Succès** (200) : -```json -{ - "success": true, - "message": "Mission updated successfully", - "mission": { - "id": "mission-uuid", - "name": "Mission Example", - "giteaRepositoryUrl": "https://gite.slm-lab.net/alma/mission-example", - "leantimeProjectId": "123", - "outlineCollectionId": "collection-456", - "rocketChatChannelId": "channel-789" - } -} -``` - -**Réponse Erreur** (400/404/500) : -```json -{ - "error": "Error message", - "details": "Detailed error information" -} -``` - ---- - -## ⚠️ Missions Existantes - -**Problème** : Les missions créées avant cette correction n'ont pas leurs IDs sauvegardés. - -**Solutions possibles** : - -### Option 1 : Migration Manuelle -Pour chaque mission existante, récupérer les IDs depuis les services externes et les mettre à jour manuellement. - -### Option 2 : Script de Migration -Créer un script qui : -1. Liste toutes les missions sans IDs -2. Interroge les services externes (si possible) -3. Met à jour les missions - -### Option 3 : Re-création -Supprimer et recréer les missions (si acceptable). - -**Recommandation** : Option 1 pour les missions critiques, Option 2 pour un grand nombre. - ---- - -## 🧪 Tests - -### Test 1 : Création de Mission - -1. Créer une nouvelle mission via le frontend -2. Vérifier que N8N appelle `/mission-created` -3. Vérifier que la mission en base a les IDs sauvegardés : - ```sql - SELECT id, name, giteaRepositoryUrl, leantimeProjectId, - outlineCollectionId, rocketChatChannelId - FROM Mission - WHERE name = 'Mission Test'; - ``` - -### Test 2 : Suppression de Mission - -1. Supprimer une mission avec IDs sauvegardés -2. Vérifier que N8N reçoit les IDs : - ```json - { - "repoName": "mission-example", - "leantimeProjectId": 123, - "documentationCollectionId": "collection-456", - "rocketchatChannelId": "channel-789" - } - ``` -3. Vérifier que N8N supprime/ferme les intégrations - -### Test 3 : API Key - -1. Appeler `/mission-created` sans `x-api-key` → 401 -2. Appeler avec mauvais `x-api-key` → 401 -3. Appeler avec bon `x-api-key` → 200 - ---- - -## 📝 Logs à Surveiller - -### Création - -``` -=== Mission Created Webhook Received === -Received mission-created data: { ... } -Found mission: { id: "...", name: "..." } -Updating giteaRepositoryUrl: ... -Updating leantimeProjectId: ... -Mission updated successfully: { ... } -``` - -### Suppression - -``` -=== Starting N8N Deletion Workflow === -Extracted repo name from URL: { url: "...", repoName: "..." } -Sending deletion data to N8N: { ... } -N8N Deletion Workflow Result: { success: true, ... } -``` - ---- - -## 🔧 Configuration Requise - -### Variables d'Environnement - -```env -N8N_API_KEY=your-api-key-here -NEXT_PUBLIC_API_URL=https://hub.slm-lab.net -``` - -### N8N Workflow - -Le workflow N8N doit appeler : -- **URL** : `{{ MISSION_API_URL }}/mission-created` -- **Méthode** : POST -- **Headers** : - - `Content-Type: application/json` - - `x-api-key: {{ N8N_API_KEY }}` - - `Authorization: Bearer {{ Keycloak Token }}` (optionnel) - ---- - -## ✅ Checklist - -- [x] Endpoint `/mission-created` créé -- [x] Vérification API key implémentée -- [x] Mapping des champs correct -- [x] Gestion d'erreurs complète -- [x] Logging détaillé -- [ ] Tests manuels effectués -- [ ] Migration des missions existantes (si nécessaire) -- [ ] Documentation N8N mise à jour - ---- - -**Date de correction** : $(date) -**Version** : 1.0 -**Fichiers modifiés** : -- `app/api/missions/mission-created/route.ts` (nouveau) - diff --git a/MISSION_INTEGRATION_IDS_ISSUE_FIX.md b/MISSION_INTEGRATION_IDS_ISSUE_FIX.md deleted file mode 100644 index 4233dc4e..00000000 --- a/MISSION_INTEGRATION_IDS_ISSUE_FIX.md +++ /dev/null @@ -1,253 +0,0 @@ -# Fix: IDs d'Intégration Vides lors de la Suppression - -## 🔍 Problème Identifié - -Lors de la suppression d'une mission, N8N reçoit des IDs vides : - -```json -{ - "missionId": "cd0225cf-8dfd-4bf0-a20a-6aa9c04ebb42", - "name": "Creation", - "repoName": "", - "leantimeProjectId": 0, - "documentationCollectionId": "", - "rocketchatChannelId": "", - "giteaRepositoryUrl": null, - "outlineCollectionId": null, - "rocketChatChannelId": null -} -``` - -**Cause** : Les IDs retournés par N8N lors de la création ne sont **pas sauvegardés en base**. - ---- - -## 🔍 Analyse du Problème - -### Flow Actuel - -``` -1. POST /api/missions → Crée mission en Prisma -2. Upload logo dans Minio -3. POST N8N webhook → N8N crée intégrations -4. N8N → POST /mission-created (avec IDs) -5. ❌ Endpoint cherche mission par name + creatorId (peut échouer) -6. ❌ IDs jamais sauvegardés -7. ❌ Lors de suppression → IDs vides -``` - -### Problèmes Identifiés - -1. **Recherche de mission fragile** : L'endpoint `/mission-created` cherche par `name` + `creatorId`, ce qui peut échouer si : - - Plusieurs missions ont le même nom - - Le nom a changé - - Le creatorId ne correspond pas exactement - -2. **missionId non envoyé** : On n'envoie pas le `missionId` à N8N, donc N8N ne peut pas le renvoyer - -3. **N8N ne renvoie peut-être pas missionId** : Même si on l'envoie, N8N doit le renvoyer dans `/mission-created` - ---- - -## ✅ Solutions Implémentées - -### 1. Envoyer missionId à N8N - -**Fichier** : `app/api/missions/route.ts` - -```typescript -const n8nData = { - ...body, - missionId: mission.id, // ✅ Send missionId so N8N can return it - creatorId: userId, - logoPath: logoPath, - logoUrl: logoUrl, - config: { ... } -}; -``` - -**Avantage** : N8N peut maintenant renvoyer le `missionId` dans `/mission-created` - -### 2. Améliorer la Recherche de Mission - -**Fichier** : `app/api/missions/mission-created/route.ts` - -```typescript -// Prefer missionId if provided, otherwise use name + creatorId -let mission; - -if (body.missionId) { - // ✅ Use missionId if provided (more reliable) - mission = await prisma.mission.findUnique({ - where: { id: body.missionId } - }); -} else if (body.name && body.creatorId) { - // Fallback to name + creatorId (for backward compatibility) - mission = await prisma.mission.findFirst({ - where: { - name: body.name, - creatorId: body.creatorId - }, - orderBy: { createdAt: 'desc' } - }); -} -``` - -**Avantages** : -- ✅ Recherche par `missionId` (plus fiable) -- ✅ Fallback vers `name` + `creatorId` (rétrocompatibilité) -- ✅ Gestion d'erreurs améliorée - ---- - -## 📋 Format de Requête N8N → /mission-created - -### Format Recommandé (avec missionId) - -```json -{ - "missionId": "cd0225cf-8dfd-4bf0-a20a-6aa9c04ebb42", - "name": "Creation", - "creatorId": "user-id", - "gitRepoUrl": "https://gite.slm-lab.net/alma/creation", - "leantimeProjectId": "123", - "documentationCollectionId": "collection-456", - "rocketchatChannelId": "channel-789" -} -``` - -### Format de Fallback (sans missionId) - -```json -{ - "name": "Creation", - "creatorId": "user-id", - "gitRepoUrl": "https://gite.slm-lab.net/alma/creation", - "leantimeProjectId": "123", - "documentationCollectionId": "collection-456", - "rocketchatChannelId": "channel-789" -} -``` - ---- - -## 🔧 Action Requise dans N8N - -### Modifier le Node "Save Mission To API" - -Le node N8N doit inclure `missionId` dans le body : - -**Avant** : -```json -{ - "name": "{{ name }}", - "creatorId": "{{ creatorId }}", - "gitRepoUrl": "{{ gitRepo.html_url }}", - ... -} -``` - -**Après** : -```json -{ - "missionId": "{{ missionId }}", // ✅ Ajouter missionId - "name": "{{ name }}", - "creatorId": "{{ creatorId }}", - "gitRepoUrl": "{{ gitRepo.html_url }}", - "leantimeProjectId": "{{ leantimeProject.result[0] }}", - "documentationCollectionId": "{{ docCollection.data.id }}", - "rocketchatChannelId": "{{ rocketChatChannel.channel._id }}", - ... -} -``` - -**Où trouver missionId dans N8N** : -- Il est dans les données initiales : `{{ $node['Process Mission Data'].json.missionId }}` -- Ou dans le body original : `{{ $json.missionId }}` - ---- - -## 🧪 Tests - -### Test 1: Vérifier missionId est envoyé à N8N - -1. Créer une mission -2. Vérifier les logs : - ``` - Sending to N8N: { missionId: "...", ... } - ``` -3. ✅ `missionId` doit être présent - -### Test 2: Vérifier N8N renvoie missionId - -1. Vérifier les logs N8N -2. Vérifier que le node "Save Mission To API" inclut `missionId` -3. ✅ `missionId` doit être dans le body envoyé à `/mission-created` - -### Test 3: Vérifier IDs sont sauvegardés - -1. Créer une mission -2. Vérifier les logs : - ``` - === Mission Created Webhook Received === - Looking up mission by ID: ... - Mission updated successfully: { ... } - ``` -3. Vérifier en base : - ```sql - SELECT id, name, giteaRepositoryUrl, leantimeProjectId, - outlineCollectionId, rocketChatChannelId - FROM Mission - WHERE id = '...'; - ``` -4. ✅ Les IDs doivent être présents - -### Test 4: Vérifier Suppression - -1. Supprimer une mission avec IDs sauvegardés -2. Vérifier les logs : - ``` - Sending deletion data to N8N: { - repoName: "creation", - leantimeProjectId: 123, - ... - } - ``` -3. ✅ Les IDs doivent être présents (pas vides) - ---- - -## 📝 Checklist - -- [x] Envoyer `missionId` à N8N lors de la création -- [x] Améliorer recherche de mission dans `/mission-created` -- [ ] **Modifier N8N workflow pour inclure `missionId` dans `/mission-created`** -- [ ] Tester création avec `missionId` -- [ ] Tester sauvegarde des IDs -- [ ] Tester suppression avec IDs sauvegardés - ---- - -## ⚠️ Action Immédiate Requise - -**Modifier le workflow N8N** pour inclure `missionId` dans le node "Save Mission To API" : - -1. Ouvrir le workflow N8N `NeahMissionCreate` -2. Trouver le node "Save Mission To API" -3. Ajouter `missionId` dans le body : - ```json - { - "missionId": "={{ $node['Process Mission Data'].json.missionId }}", - ... - } - ``` -4. Sauvegarder et activer le workflow - ---- - -**Date**: $(date) -**Version**: 1.1 -**Fichiers Modifiés**: -- `app/api/missions/route.ts` (ajout missionId dans n8nData) -- `app/api/missions/mission-created/route.ts` (recherche par missionId) - diff --git a/N8N_COMPLETE_WORKFLOW_MAPPING.md b/N8N_COMPLETE_WORKFLOW_MAPPING.md deleted file mode 100644 index cfddacb6..00000000 --- a/N8N_COMPLETE_WORKFLOW_MAPPING.md +++ /dev/null @@ -1,673 +0,0 @@ -# Mapping Complet N8N - Création et Suppression de Mission - -## 📋 Vue d'Ensemble - -Ce document décrit le mapping complet entre notre API et les workflows N8N pour la création et la suppression de missions, basé sur les workflows réels partagés. - ---- - -## 🔄 Workflow de Création - NeahMissionCreate - -### Structure du Workflow - -``` -Webhook (mission-created) - ↓ -Process Mission Data - ↓ -Get Keycloak Token - ↓ -Process Token - ↓ -Debug Service Data - ↓ -Merge Paths - ↓ -IF Run Integrations - ├─ IF Needs Git Repository - │ ├─ Create Git Repository (si Gite ou Calcul) - │ ├─ Create Readme - │ └─ Git Wiki - ├─ Create Documentation Collection - ├─ Create Leantime Project - │ └─ Leantime Avatar - └─ Create RocketChat Channel - ↓ -Combine Results - ↓ -Save Mission To API (POST /mission-created) - ↓ -Process Results - ↓ -Respond To Webhook -``` - -### Données Envoyées par Notre API → N8N - -**Endpoint** : `POST https://brain.slm-lab.net/webhook/mission-created` - -**Format** : -```typescript -{ - name: string, - oddScope: string[], - niveau: string, - intention: string, - missionType: string, - donneurDOrdre: string, - projection: string, - services: string[], - participation: string, - profils: string[], - guardians: { - "gardien-temps": userId, - "gardien-parole": userId, - "gardien-memoire": userId - }, - volunteers: string[], - creatorId: string, - logo: { - data: "data:image/png;base64,...", - name: string, - type: string - }, - attachments: Array<{ - data: "data:...;base64,...", - name: string, - type: string - }>, - config: { - N8N_API_KEY: string, - MISSION_API_URL: string - } -} -``` - -### Traitement N8N - Process Mission Data - -Le node "Process Mission Data" transforme les données en : - -```javascript -{ - missionOriginal: { ... }, // Données originales - missionProcessed: { - name: "Mission Example", - sanitizedName: "mission-example", // Nom nettoyé pour URLs - intention: "...", - description: "...", - startDate: "2024-01-01", - endDate: "2024-01-31", - missionType: "remote", - guardians: { ... }, - volunteers: [ ... ], - profils: [ ... ], - services: ["Gite", "ArtLab"], // Détermine quelles intégrations créer - clientId: 2, - rocketChatUsernames: [userId1, userId2, ...], // Gardiens + volontaires - logo: { data: "...", name: "...", type: "..." }, - attachments: [ ... ] - }, - config: { - GITEA_API_URL: "https://gite.slm-lab.net/api/v1", - GITEA_API_TOKEN: "...", - GITEA_OWNER: "alma", - LEANTIME_API_URL: "https://agilite.slm-lab.net", - LEANTIME_API_TOKEN: "...", - ROCKETCHAT_API_URL: "https://parole.slm-lab.net/", - ROCKETCHAT_AUTH_TOKEN: "...", - ROCKETCHAT_USER_ID: "...", - OUTLINE_API_URL: "https://chapitre.slm-lab.net/api", - OUTLINE_API_TOKEN: "...", - MISSION_API_URL: "https://hub.slm-lab.net", - // ... autres configs - }, - creatorId: "user-id" -} -``` - -### Intégrations Créées par N8N - -#### 1. Gitea Repository (Conditionnel) - -**Condition** : `services.includes('Gite') || services.includes('Calcul')` - -**Node** : "Create Git Repository" -- **Méthode** : POST -- **URL** : `{{ GITEA_API_URL }}/user/repos` -- **Body** : - ```json - { - "name": "{{ sanitizedName }}", - "private": true, - "auto_init": true, - "avatar_url": "{{ logo.data }}" - } - ``` -- **Résultat** : `{ html_url: "https://gite.slm-lab.net/alma/mission-example" }` - -**Actions supplémentaires** : -- Create Readme : Crée un document README dans Outline -- Git Wiki : Configure le wiki externe du repo vers Outline - -#### 2. Leantime Project - -**Node** : "Create Leantime Project" -- **Méthode** : POST -- **URL** : `{{ LEANTIME_API_URL }}/api/jsonrpc` -- **Body** : - ```json - { - "method": "leantime.rpc.Projects.Projects.addProject", - "jsonrpc": "2.0", - "id": 1, - "params": { - "values": { - "name": "{{ name }}", - "clientId": {{ clientId }}, - "details": "{{ intention }}", - "type": "project", - "start": "{{ startDate }}", - "end": "{{ endDate }}", - "status": "open", - "psettings": "restricted", - "avatar": "{{ logo.data }}" - } - } - } - ``` -- **Résultat** : `{ result: [projectId] }` (array avec 1 élément) - -**Action supplémentaire** : -- Leantime Avatar : Met à jour l'avatar du projet - -#### 3. Outline Collection - -**Node** : "Create Documentation Collection" -- **Méthode** : POST -- **URL** : `{{ OUTLINE_API_URL }}/api/collections.create` -- **Body** : - ```json - { - "name": "{{ sanitizedName }}", - "description": "{{ description }}", - "permission": "read", - "private": true - } - ``` -- **Résultat** : `{ data: { id: "collection-id", url: "/collection/..." } }` - -#### 4. RocketChat Channel - -**Node** : "Create RocketChat Channel" -- **Méthode** : POST -- **URL** : `{{ ROCKETCHAT_API_URL }}/api/v1/channels.create` -- **Body** : - ```json - { - "name": "{{ sanitizedName }}", - "members": [{{ rocketChatUsernames }}], - "readOnly": false, - "avatarUrl": "{{ logo.data }}" - } - ``` -- **Résultat** : `{ channel: { _id: "channel-id", ... } }` - -### Save Mission To API - Retour vers Notre API - -**Node** : "Save Mission To API" -- **Méthode** : POST -- **URL** : `{{ MISSION_API_URL }}/mission-created` -- **Headers** : - - `Content-Type: application/json` - - `Authorization: Bearer {{ Keycloak Token }}` - - `x-api-key: {{ N8N_API_KEY }}` -- **Body** : - ```json - { - "name": "{{ name }}", - "niveau": "{{ niveau }}", - "intention": "{{ intention }}", - "description": "{{ description }}", - "gitRepoUrl": "{{ gitRepo.html_url }}", - "leantimeProjectId": "{{ leantimeProject.result[0] }}", - "documentationCollectionId": "{{ docCollection.data.id }}", - "rocketchatChannelId": "{{ rocketChatChannel.channel._id }}", - "donneurDOrdre": "{{ donneurDOrdre }}", - "projection": "{{ projection }}", - "missionType": "{{ missionType }}", - "creatorId": "{{ creatorId }}" - } - ``` - -**⚠️ IMPORTANT** : Cet endpoint `/mission-created` n'existe **PAS** actuellement dans notre codebase. Il devrait : -1. Recevoir les IDs des intégrations créées -2. Mettre à jour la mission en base avec ces IDs -3. Mapper les champs : - - `gitRepoUrl` → `giteaRepositoryUrl` - - `documentationCollectionId` → `outlineCollectionId` - - `rocketchatChannelId` → `rocketChatChannelId` - ---- - -## 🗑️ Workflow de Suppression - NeahMissionDelete_Pro - -### Structure du Workflow - -``` -Webhook Delete (mission-delete) - ↓ -Process Delete Data - ↓ -Get Keycloak Token - ↓ -[En parallèle] - ├─ Delete Gitea Repo - ├─ Close Leantime Project - ├─ Delete Outline Collection - └─ Close RocketChat Channel - ↓ -Combine Results - ↓ -Save Deletion To API (POST /mission-deleted) -``` - -### Données Envoyées par Notre API → N8N - -**Endpoint** : `POST https://brain.slm-lab.net/webhook-test/mission-delete` - -**Format** : -```typescript -{ - missionId: string, - name: string, - repoName: string, // ✅ Extrait de giteaRepositoryUrl - leantimeProjectId: number | 0, // ✅ Converti en number - documentationCollectionId: string, // ✅ Mappé depuis outlineCollectionId - rocketchatChannelId: string, // ✅ Mappé depuis rocketChatChannelId - // Champs originaux pour référence - giteaRepositoryUrl: string | null, - outlineCollectionId: string | null, - rocketChatChannelId: string | null, - penpotProjectId: string | null, - config: { - N8N_API_KEY: string, - MISSION_API_URL: string - } -} -``` - -### Traitement N8N - Process Delete Data - -Le node "Process Delete Data" transforme les données en : - -```javascript -{ - missionData: { - repoName: input.repoName || '', - leantimeId: input.leantimeProjectId || 0, - collectionId: input.documentationCollectionId || '', - rocketChatRoomId: input.rocketchatChannelId || '' - }, - config: { - GITEA_API_URL: "https://gite.slm-lab.net/api/v1", - GITEA_API_TOKEN: "...", - GITEA_OWNER: "alma", - LEANTIME_API_URL: "https://agilite.slm-lab.net", - LEANTIME_API_TOKEN: "...", - ROCKETCHAT_API_URL: "https://parole.slm-lab.net/", - ROCKETCHAT_AUTH_TOKEN: "...", - ROCKETCHAT_USER_ID: "...", - OUTLINE_API_URL: "https://chapitre.slm-lab.net/api", - OUTLINE_API_TOKEN: "...", - MISSION_API_URL: "https://hub.slm-lab.net", - KEYCLOAK_BASE_URL: "https://connect.slm-lab.net", - KEYCLOAK_REALM: "cercle", - KEYCLOAK_CLIENT_ID: "lab", - KEYCLOAK_CLIENT_SECRET: "..." - } -} -``` - -### Actions de Suppression N8N - -#### 1. Delete Gitea Repo - -**Node** : "Delete Gitea Repo" -- **Méthode** : DELETE -- **URL** : `{{ GITEA_API_URL }}/repos/{{ GITEA_OWNER }}/{{ repoName }}` -- **Headers** : `Authorization: token {{ GITEA_API_TOKEN }}` -- **ContinueOnFail** : `true` -- **Résultat attendu** : Status 204 = succès - -#### 2. Close Leantime Project - -**Node** : "Close Leantime Project" -- **Méthode** : POST -- **URL** : `{{ LEANTIME_API_URL }}/api/jsonrpc` -- **Body** : - ```json - { - "method": "leantime.rpc.Projects.Projects.patch", - "jsonrpc": "2.0", - "id": 1, - "params": { - "id": {{ leantimeId }}, - "params": { "status": "closed" } - } - } - ``` -- **ContinueOnFail** : `true` -- **Note** : Le projet est **fermé** (status: "closed"), pas supprimé - -#### 3. Delete Outline Collection - -**Node** : "Delete Outline Collection" -- **Méthode** : POST -- **URL** : `{{ OUTLINE_API_URL }}/api/collections.delete` -- **Body** : `{ "id": "{{ collectionId }}" }` -- **ContinueOnFail** : `true` -- **Résultat attendu** : Status 200 = succès - -#### 4. Close RocketChat Channel - -**Node** : "Close RocketChat Channel" -- **Méthode** : POST -- **URL** : `{{ ROCKETCHAT_API_URL }}/api/v1/channels.close` -- **Body** : `{ "roomId": "{{ rocketChatRoomId }}" }` -- **ContinueOnFail** : `true` -- **Note** : Le canal est **fermé**, pas supprimé - -### Combine Results - -Le node "Combine Results" combine les résultats : - -```javascript -{ - status: "deleted", - timestamp: "2024-01-01T12:00:00.000Z", - details: { - gitea: true || "already_deleted", - leantime: true || false, - outline: true || false, - rocketchat: true || false - } -} -``` - -### Save Deletion To API - Retour vers Notre API - -**Node** : "Save Deletion To API" -- **Méthode** : POST -- **URL** : `{{ MISSION_API_URL }}/mission-deleted` -- **Headers** : - - `Authorization: Bearer {{ Keycloak Token }}` -- **Body** : - ```json - { - "status": "archived", - "results": { - "gitea": true, - "leantime": true, - "outline": true, - "rocketchat": true - } - } - ``` - -**⚠️ IMPORTANT** : Cet endpoint `/mission-deleted` n'existe **PAS** actuellement dans notre codebase. Il pourrait servir à : -1. Confirmer la suppression -2. Logger les résultats -3. Nettoyer des données supplémentaires si nécessaire - ---- - -## 📊 Mapping Complet des Champs - -### Création (Notre API → N8N → Retour) - -| Notre Base | Envoyé à N8N | N8N Crée | Retour N8N | Stocké en Base | -|-----------|--------------|----------|------------|----------------| -| - | `name` | - | `name` | `name` | -| - | `services` | Détermine intégrations | - | `services` | -| - | `logo.data` | Avatar/Logo | - | `logo` (path) | -| - | - | Gitea Repo | `gitRepoUrl` | `giteaRepositoryUrl` | -| - | - | Leantime Project | `leantimeProjectId` | `leantimeProjectId` | -| - | - | Outline Collection | `documentationCollectionId` | `outlineCollectionId` | -| - | - | RocketChat Channel | `rocketchatChannelId` | `rocketChatChannelId` | - -### Suppression (Notre Base → N8N) - -| Notre Base | Extrait/Transformé | Envoyé à N8N | N8N Attend | -|-----------|-------------------|--------------|------------| -| `giteaRepositoryUrl` | Extraction nom | `repoName` | `repoName` | -| `leantimeProjectId` | Converti en number | `leantimeProjectId` | `leantimeId` | -| `outlineCollectionId` | Direct | `documentationCollectionId` | `collectionId` | -| `rocketChatChannelId` | Direct | `rocketchatChannelId` | `rocketChatRoomId` | - ---- - -## 🔧 Transformations Clés - -### 1. Extraction du Nom du Repository Gitea - -**Problème** : Notre base stocke l'URL complète, N8N attend le nom seul - -**Solution** : -```typescript -// Format: https://gite.slm-lab.net/alma/mission-example -// ou: https://gite.slm-lab.net/api/v1/repos/alma/mission-example - -let repoName = ''; -if (giteaRepositoryUrl) { - try { - const url = new URL(giteaRepositoryUrl); - const pathParts = url.pathname.split('/').filter(Boolean); - repoName = pathParts[pathParts.length - 1] || ''; - } catch (error) { - const match = giteaRepositoryUrl.match(/\/([^\/]+)\/?$/); - repoName = match ? match[1] : ''; - } -} -``` - -### 2. Mapping des Champs - -**Création** : -- N8N retourne `gitRepoUrl` → On stocke `giteaRepositoryUrl` -- N8N retourne `documentationCollectionId` → On stocke `outlineCollectionId` -- N8N retourne `rocketchatChannelId` → On stocke `rocketChatChannelId` - -**Suppression** : -- On stocke `giteaRepositoryUrl` → On envoie `repoName` (extrait) -- On stocke `outlineCollectionId` → On envoie `documentationCollectionId` -- On stocke `rocketChatChannelId` → On envoie `rocketchatChannelId` - -### 3. Conversion de Types - -**Leantime Project ID** : -- Stocké en base : `string | null` -- Envoyé à N8N : `number | 0` (converti) -- N8N attend : `number` (dans `leantimeId`) - ---- - -## ⚠️ Endpoints Manquants - -### 1. POST /mission-created - -**Rôle** : Recevoir les IDs des intégrations créées par N8N - -**Format attendu** : -```typescript -POST /mission-created -Headers: { - Authorization: "Bearer {keycloak_token}", - x-api-key: "{N8N_API_KEY}" -} -Body: { - name: string, - niveau: string, - intention: string, - description: string, - gitRepoUrl: string, // À mapper vers giteaRepositoryUrl - leantimeProjectId: string, // À mapper vers leantimeProjectId - documentationCollectionId: string, // À mapper vers outlineCollectionId - rocketchatChannelId: string, // À mapper vers rocketChatChannelId - donneurDOrdre: string, - projection: string, - missionType: string, - creatorId: string -} -``` - -**Action requise** : -1. Trouver la mission par `name` + `creatorId` -2. Mettre à jour avec les IDs retournés -3. Mapper les champs correctement - -### 2. POST /mission-deleted - -**Rôle** : Confirmer la suppression (optionnel) - -**Format attendu** : -```typescript -POST /mission-deleted -Headers: { - Authorization: "Bearer {keycloak_token}" -} -Body: { - status: "archived", - results: { - gitea: boolean, - leantime: boolean, - outline: boolean, - rocketchat: boolean - } -} -``` - -**Action requise** : -- Logger les résultats -- Potentiellement nettoyer des données supplémentaires - ---- - -## 🔄 Flow Complet - Vue d'Ensemble - -### Création - -``` -1. Frontend → POST /api/missions - ↓ -2. Backend crée mission en Prisma - ↓ -3. Backend upload fichiers Minio - ↓ -4. Backend → POST N8N webhook (mission-created) - ↓ -5. N8N crée intégrations (Gitea, Leantime, Outline, RocketChat) - ↓ -6. N8N → POST /mission-created (⚠️ endpoint manquant) - ↓ -7. Backend met à jour mission avec IDs (⚠️ non implémenté) -``` - -### Suppression - -``` -1. Frontend → DELETE /api/missions/[id] - ↓ -2. Backend récupère mission - ↓ -3. Backend extrait/mappe les données - ↓ -4. Backend → POST N8N webhook (mission-delete) - ↓ -5. N8N supprime/ferme intégrations - ↓ -6. N8N → POST /mission-deleted (⚠️ endpoint manquant) - ↓ -7. Backend supprime logo Minio - ↓ -8. Backend supprime attachments Minio - ↓ -9. Backend supprime mission Prisma (CASCADE) -``` - ---- - -## 📝 Notes Importantes - -### 1. Noms de Champs Incohérents - -- **Création** : N8N retourne `gitRepoUrl`, `documentationCollectionId`, `rocketchatChannelId` -- **Suppression** : N8N attend `repoName`, `documentationCollectionId`, `rocketchatChannelId` -- **Notre Base** : Stocke `giteaRepositoryUrl`, `outlineCollectionId`, `rocketChatChannelId` - -**Solution** : Mapping cohérent dans les deux sens - -### 2. Endpoint /mission-created Manquant - -Actuellement, les IDs retournés par N8N ne sont **PAS** sauvegardés en base. Il faudrait : -- Créer l'endpoint `/mission-created` -- Trouver la mission (par `name` + `creatorId` ou `missionId`) -- Mettre à jour avec les IDs - -### 3. Services Conditionnels - -- **Gitea** : Créé seulement si `services.includes('Gite') || services.includes('Calcul')` -- **Leantime** : Toujours créé -- **Outline** : Toujours créé -- **RocketChat** : Toujours créé - -### 4. Gestion d'Erreurs - -- Tous les nodes N8N ont `continueOnFail: true` -- Les erreurs sont loggées mais n'arrêtent pas le workflow -- Les résultats indiquent quelles intégrations ont réussi/échoué - ---- - -## 🔍 Points de Debugging - -### Création - -1. **Vérifier données envoyées à N8N** : - ``` - Sending to N8N: { ... } - ``` - -2. **Vérifier réponse N8N** : - ``` - N8N Workflow Result: { success: true, results: {...} } - ``` - -3. **Vérifier endpoint /mission-created** : - - Doit recevoir les IDs - - Doit mettre à jour la mission - -### Suppression - -1. **Vérifier extraction repoName** : - ``` - Extracted repo name from URL: { url: "...", repoName: "..." } - ``` - -2. **Vérifier données envoyées à N8N** : - ``` - Sending deletion data to N8N: { ... } - ``` - -3. **Vérifier réponse N8N** : - ``` - N8N Deletion Workflow Result: { success: true, results: {...} } - ``` - ---- - -**Document généré le** : $(date) -**Version** : 1.0 -**Workflows N8N** : -- NeahMissionCreate (création) -- NeahMissionDelete_Pro (suppression) - diff --git a/N8N_DELETION_WORKFLOW_MAPPING.md b/N8N_DELETION_WORKFLOW_MAPPING.md deleted file mode 100644 index f7c21f2d..00000000 --- a/N8N_DELETION_WORKFLOW_MAPPING.md +++ /dev/null @@ -1,342 +0,0 @@ -# Mapping N8N Workflow - Mission Deletion - -## 📋 Vue d'Ensemble - -Ce document décrit le mapping entre les données de notre API et le format attendu par le workflow N8N `NeahMissionDelete_Pro`. - ---- - -## 🔄 Workflow N8N - Structure - -### Nodes du Workflow - -1. **Webhook Delete** : Reçoit POST sur `/mission-delete` -2. **Process Delete Data** : Transforme les données d'entrée -3. **Get Keycloak Token** : Obtient un token d'authentification -4. **Delete Gitea Repo** : Supprime le repository Gitea (continueOnFail: true) -5. **Close Leantime Project** : Ferme le projet Leantime (continueOnFail: true) -6. **Delete Outline Collection** : Supprime la collection Outline (continueOnFail: true) -7. **Close RocketChat Channel** : Ferme le canal RocketChat (continueOnFail: true) -8. **Combine Results** : Combine les résultats de toutes les suppressions -9. **Save Deletion To API** : Envoie les résultats à l'API - ---- - -## 📊 Mapping des Données - -### Données Envoyées par Notre API - -```typescript -{ - missionId: string, - name: string, - repoName: string, // Extrait de giteaRepositoryUrl - leantimeProjectId: number | null, - documentationCollectionId: string, // Mappé depuis outlineCollectionId - rocketchatChannelId: string, // Mappé depuis rocketChatChannelId - // Champs originaux conservés pour référence - giteaRepositoryUrl: string | null, - outlineCollectionId: string | null, - rocketChatChannelId: string | null, - penpotProjectId: string | null, - config: { - N8N_API_KEY: string, - MISSION_API_URL: string - } -} -``` - -### Données Attendues par N8N (Process Delete Data) - -Le node "Process Delete Data" transforme les données en : - -```javascript -{ - missionData: { - repoName: input.repoName || '', - leantimeId: input.leantimeProjectId || 0, - collectionId: input.documentationCollectionId || '', - rocketChatRoomId: input.rocketchatChannelId || '' - }, - config: { - GITEA_API_URL: "https://gite.slm-lab.net/api/v1", - GITEA_API_TOKEN: "...", - GITEA_OWNER: "alma", - LEANTIME_API_URL: "https://agilite.slm-lab.net", - LEANTIME_API_TOKEN: "...", - ROCKETCHAT_API_URL: "https://parole.slm-lab.net/", - ROCKETCHAT_AUTH_TOKEN: "...", - ROCKETCHAT_USER_ID: "...", - OUTLINE_API_URL: "https://chapitre.slm-lab.net/api", - OUTLINE_API_TOKEN: "...", - MISSION_API_URL: "https://hub.slm-lab.net", - KEYCLOAK_BASE_URL: "https://connect.slm-lab.net", - KEYCLOAK_REALM: "cercle", - KEYCLOAK_CLIENT_ID: "lab", - KEYCLOAK_CLIENT_SECRET: "..." - } -} -``` - ---- - -## 🔧 Transformations Effectuées - -### 1. Extraction du Nom du Repository Gitea - -**Problème** : Notre base stocke `giteaRepositoryUrl` (URL complète), mais N8N attend `repoName` (nom seul) - -**Solution** : Extraction du nom depuis l'URL - -```typescript -// Format possible: -// - https://gite.slm-lab.net/alma/repo-name -// - https://gite.slm-lab.net/api/v1/repos/alma/repo-name - -let repoName = ''; -if (mission.giteaRepositoryUrl) { - try { - const url = new URL(mission.giteaRepositoryUrl); - const pathParts = url.pathname.split('/').filter(Boolean); - repoName = pathParts[pathParts.length - 1] || ''; - } catch (error) { - // Fallback: extraction regex - const match = mission.giteaRepositoryUrl.match(/\/([^\/]+)\/?$/); - repoName = match ? match[1] : ''; - } -} -``` - -**Exemples** : -- `https://gite.slm-lab.net/alma/mission-abc` → `mission-abc` -- `https://gite.slm-lab.net/api/v1/repos/alma/mission-xyz` → `mission-xyz` - -### 2. Mapping des Champs - -| Notre Base de Données | N8N Attend | Transformation | -|----------------------|------------|----------------| -| `giteaRepositoryUrl` | `repoName` | Extraction du nom depuis URL | -| `leantimeProjectId` | `leantimeProjectId` | Direct (converti en number) | -| `outlineCollectionId` | `documentationCollectionId` | Direct mapping | -| `rocketChatChannelId` | `rocketchatChannelId` | Direct mapping (lowercase 'c') | - ---- - -## 🎯 Actions N8N par Service - -### 1. Gitea Repository - -**Node** : "Delete Gitea Repo" -- **Méthode** : DELETE -- **URL** : `{{ GITEA_API_URL }}/repos/{{ GITEA_OWNER }}/{{ repoName }}` -- **Headers** : `Authorization: token {{ GITEA_API_TOKEN }}` -- **ContinueOnFail** : `true` (continue même si échoue) - -**Résultat attendu** : Status 204 (No Content) = succès - -### 2. Leantime Project - -**Node** : "Close Leantime Project" -- **Méthode** : POST -- **URL** : `{{ LEANTIME_API_URL }}/api/jsonrpc` -- **Headers** : `X-API-Key: {{ LEANTIME_API_TOKEN }}` -- **Body** : - ```json - { - "method": "leantime.rpc.Projects.Projects.patch", - "jsonrpc": "2.0", - "id": 1, - "params": { - "id": {{ leantimeId }}, - "params": { "status": "closed" } - } - } - ``` -- **ContinueOnFail** : `true` - -**Note** : Le projet est **fermé** (status: "closed"), pas supprimé - -### 3. Outline Collection - -**Node** : "Delete Outline Collection" -- **Méthode** : POST -- **URL** : `{{ OUTLINE_API_URL }}/api/collections.delete` -- **Headers** : `Authorization: Bearer {{ OUTLINE_API_TOKEN }}` -- **Body** : `{ "id": "{{ collectionId }}" }` -- **ContinueOnFail** : `true` - -**Résultat attendu** : Status 200 = succès - -### 4. RocketChat Channel - -**Node** : "Close RocketChat Channel" -- **Méthode** : POST -- **URL** : `{{ ROCKETCHAT_API_URL }}/api/v1/channels.close` -- **Headers** : - - `X-Auth-Token: {{ ROCKETCHAT_AUTH_TOKEN }}` - - `X-User-Id: {{ ROCKETCHAT_USER_ID }}` -- **Body** : `{ "roomId": "{{ rocketChatRoomId }}" }` -- **ContinueOnFail** : `true` - -**Note** : Le canal est **fermé**, pas supprimé - ---- - -## 📤 Réponse N8N - -### Format de Réponse (Combine Results) - -```javascript -{ - status: "deleted", - timestamp: "2024-01-01T12:00:00.000Z", - details: { - gitea: true || "already_deleted", - leantime: true || false, - outline: true || false, - rocketchat: true || false - } -} -``` - -### Envoi à l'API (Save Deletion To API) - -Le workflow envoie ensuite les résultats à : -- **URL** : `{{ MISSION_API_URL }}/mission-deleted` -- **Méthode** : POST -- **Headers** : `Authorization: Bearer {{ Keycloak Token }}` -- **Body** : - ```json - { - "status": "archived", - "results": { - "gitea": true, - "leantime": true, - "outline": true, - "rocketchat": true - } - } - ``` - ---- - -## ⚠️ Points d'Attention - -### 1. Gestion des Erreurs - -- Tous les nodes de suppression ont `continueOnFail: true` -- Si une suppression échoue, le workflow continue avec les autres -- Les résultats indiquent quelles suppressions ont réussi/échoué - -### 2. Différences de Comportement - -- **Gitea** : Suppression complète du repository -- **Leantime** : Fermeture (status: "closed"), pas suppression -- **Outline** : Suppression complète de la collection -- **RocketChat** : Fermeture du canal, pas suppression - -### 3. Extraction du Repo Name - -- L'extraction doit gérer différents formats d'URL -- Si l'extraction échoue, `repoName` sera vide -- Le workflow N8N gérera le cas où `repoName` est vide - -### 4. Mapping des Champs - -- **documentationCollectionId** : Mappé depuis `outlineCollectionId` -- **rocketchatChannelId** : Mappé depuis `rocketChatChannelId` (attention au 'c' minuscule) -- **leantimeProjectId** : Converti en number (0 si null) - ---- - -## 🔍 Debugging - -### Logs à Surveiller - -1. **Extraction repo name** : - ``` - Extracted repo name from URL: { url: "...", repoName: "..." } - ``` - -2. **Données envoyées à N8N** : - ``` - Sending deletion data to N8N: { ... } - ``` - -3. **Résultat N8N** : - ``` - N8N Deletion Workflow Result: { success: true, results: {...} } - ``` - -### Vérifications - -1. **Repo name extrait correctement** ? - - Vérifier les logs d'extraction - - Format attendu : nom simple sans URL - -2. **Mapping des champs correct** ? - - `documentationCollectionId` = `outlineCollectionId` - - `rocketchatChannelId` = `rocketChatChannelId` - -3. **N8N a reçu les données** ? - - Vérifier les logs N8N - - Vérifier le webhook a été appelé - ---- - -## 📝 Exemple Complet - -### Données en Base - -```typescript -{ - id: "abc-123", - name: "Mission Example", - giteaRepositoryUrl: "https://gite.slm-lab.net/alma/mission-example", - leantimeProjectId: "123", - outlineCollectionId: "collection-456", - rocketChatChannelId: "channel-789" -} -``` - -### Données Envoyées à N8N - -```typescript -{ - missionId: "abc-123", - name: "Mission Example", - repoName: "mission-example", // Extrait de l'URL - leantimeProjectId: 123, // Converti en number - documentationCollectionId: "collection-456", // Mappé - rocketchatChannelId: "channel-789", // Mappé (lowercase 'c') - giteaRepositoryUrl: "https://gite.slm-lab.net/alma/mission-example", - outlineCollectionId: "collection-456", - rocketChatChannelId: "channel-789", - config: { - N8N_API_KEY: "...", - MISSION_API_URL: "https://hub.slm-lab.net" - } -} -``` - -### Données Traitées par N8N - -```javascript -{ - missionData: { - repoName: "mission-example", - leantimeId: 123, - collectionId: "collection-456", - rocketChatRoomId: "channel-789" - }, - config: { ... } -} -``` - ---- - -**Document généré le** : $(date) -**Version** : 1.0 -**Workflow N8N** : NeahMissionDelete_Pro -**Webhook URL** : https://brain.slm-lab.net/webhook-test/mission-delete - diff --git a/NAVBAR_TIME_INTEGRATION.md b/NAVBAR_TIME_INTEGRATION.md deleted file mode 100644 index 3a224ee9..00000000 --- a/NAVBAR_TIME_INTEGRATION.md +++ /dev/null @@ -1,129 +0,0 @@ -# Navigation Bar Time Integration - -## 🎯 Overview - -The navigation bar (`components/main-nav.tsx`) currently displays a static time that doesn't refresh. This document outlines how to integrate it into the unified refresh system. - -## 🔍 Current Issue - -**File**: `components/main-nav.tsx` (lines 228-231) - -```typescript -// Current code - STATIC (doesn't refresh) -const now = new Date(); -const formattedDate = format(now, "d MMMM yyyy", { locale: fr }); -const formattedTime = format(now, "HH:mm"); -``` - -**Problem**: Time is calculated once when component renders and never updates. - -## ✅ Solution - -### Step 1: Create Time Component - -**File**: `components/main-nav-time.tsx` (✅ Already created) - -This component: -- Uses `useState` to track current time -- Uses `useUnifiedRefresh` hook for 1-second updates -- Properly cleans up on unmount -- No API calls needed (client-side only) - -### Step 2: Update MainNav Component - -**File**: `components/main-nav.tsx` - -**Changes needed**: - -1. **Import the new component**: -```typescript -import { MainNavTime } from './main-nav-time'; -``` - -2. **Remove static time code** (lines 228-231): -```typescript -// DELETE THESE LINES: -// Format current date and time -const now = new Date(); -const formattedDate = format(now, "d MMMM yyyy", { locale: fr }); -const formattedTime = format(now, "HH:mm"); -``` - -3. **Replace time display** (lines 294-298): -```typescript -// BEFORE: -{/* Center - Date and Time */} -
-
{formattedDate}
-
{formattedTime}
-
- -// AFTER: -{/* Center - Date and Time */} - -``` - -### Step 3: Verify Integration - -After changes: -- ✅ Time updates every second -- ✅ Uses unified refresh system -- ✅ Proper cleanup on unmount -- ✅ No memory leaks -- ✅ Consistent with other widgets - -## 📊 Benefits - -1. **Real-time clock**: Time updates every second -2. **Unified system**: Uses same refresh manager as widgets -3. **Memory safe**: Proper cleanup prevents leaks -4. **Consistent**: Same pattern as other components -5. **Maintainable**: Centralized refresh logic - -## 🔧 Technical Details - -### Refresh Configuration - -- **Resource**: `navbar-time` -- **Interval**: 1000ms (1 second) -- **Priority**: `high` (real-time display) -- **API Calls**: None (client-side only) -- **Cleanup**: Automatic via `useUnifiedRefresh` - -### Integration with Refresh Manager - -The time component registers with the refresh manager: - -```typescript -useUnifiedRefresh({ - resource: 'navbar-time', - interval: REFRESH_INTERVALS.NAVBAR_TIME, // 1000ms - enabled: true, // Always enabled - onRefresh: async () => { - setCurrentTime(new Date()); - }, - priority: 'high', -}); -``` - -## ✅ Implementation Checklist - -- [x] Create `components/main-nav-time.tsx` -- [x] Add `NAVBAR_TIME` to refresh intervals -- [x] Add `navbar-time` to refreshable resources -- [ ] Update `components/main-nav.tsx` to use new component -- [ ] Test time updates correctly -- [ ] Verify cleanup on unmount -- [ ] Test with multiple tabs - -## 🎯 Expected Result - -After implementation: -- Time updates smoothly every second -- No performance impact -- No memory leaks -- Consistent with unified refresh system - ---- - -*Last Updated: Navbar Time Integration Guide* diff --git a/NGINX_HEADER_SIZE_FIX.md b/NGINX_HEADER_SIZE_FIX.md deleted file mode 100644 index c66d9fe6..00000000 --- a/NGINX_HEADER_SIZE_FIX.md +++ /dev/null @@ -1,156 +0,0 @@ -# Fix Erreur 502 - Headers trop grands (Nginx) - -## 🔍 Problème Identifié - -**Erreur Nginx** : -``` -upstream sent too big header while reading response header from upstream -``` - -**Cause** : Le cookie de session NextAuth est trop grand (> 4KB par défaut dans Nginx). Le JWT contient : -- `accessToken` (Keycloak) - ~1-2KB -- `refreshToken` (Keycloak) - ~1-2KB -- `idToken` (Keycloak) - ~1-2KB -- Données utilisateur (roles, etc.) - ~500B-1KB -- **Total** : ~4-7KB, ce qui dépasse la limite Nginx par défaut - -## ✅ Solutions - -### Solution 1 : Augmenter la limite Nginx (RECOMMANDÉ) - -**Fichier** : Configuration Nginx (généralement `/etc/nginx/sites-available/hub.slm-lab.net` ou similaire) - -**Ajouter dans le bloc `server` ou `location`** : -```nginx -server { - # ... autres configs ... - - # Augmenter la taille maximale des headers - proxy_buffer_size 16k; - proxy_buffers 8 16k; - proxy_busy_buffers_size 32k; - large_client_header_buffers 4 32k; - - # Spécifiquement pour les headers de réponse - proxy_headers_hash_max_size 512; - proxy_headers_hash_bucket_size 128; - - # ... reste de la config ... -} -``` - -**OU** pour une solution plus simple, ajouter seulement : -```nginx -server { - # ... autres configs ... - - # Augmenter la limite des headers - large_client_header_buffers 4 32k; - - # ... reste de la config ... -} -``` - -**Puis redémarrer Nginx** : -```bash -sudo nginx -t # Vérifier la config -sudo systemctl reload nginx # Ou sudo service nginx reload -``` - -### Solution 2 : Réduire la taille du JWT (ALTERNATIVE) - -Si on ne peut pas modifier Nginx, on peut réduire la taille du JWT en ne stockant pas tous les tokens. - -**Modification** : `app/api/auth/options.ts` - -**Option A** : Ne pas stocker `idToken` dans le JWT (si pas nécessaire) -```typescript -// Dans JWT callback -token.idToken = account.id_token ?? ''; // ❌ Supprimer cette ligne -``` - -**Option B** : Stocker seulement les tokens nécessaires -```typescript -// Stocker seulement accessToken et refreshToken -// idToken peut être récupéré depuis Keycloak si nécessaire -``` - -**Note** : Cette solution réduit la fonctionnalité. La Solution 1 est préférable. - -## 🔧 Configuration Nginx Complète Recommandée - -```nginx -server { - listen 443 ssl http2; - server_name hub.slm-lab.net; - - # ... SSL config ... - - # Augmenter les limites pour les gros headers NextAuth - proxy_buffer_size 16k; - proxy_buffers 8 16k; - proxy_busy_buffers_size 32k; - large_client_header_buffers 4 32k; - - # Timeouts - proxy_connect_timeout 60s; - proxy_send_timeout 60s; - proxy_read_timeout 60s; - - location / { - proxy_pass http://172.16.0.102:3000; - proxy_http_version 1.1; - proxy_set_header Upgrade $http_upgrade; - proxy_set_header Connection 'upgrade'; - proxy_set_header Host $host; - proxy_set_header X-Real-IP $remote_addr; - proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; - proxy_set_header X-Forwarded-Proto $scheme; - proxy_cache_bypass $http_upgrade; - - # Headers pour NextAuth - proxy_set_header Cookie $http_cookie; - } -} -``` - -## 📊 Vérification - -**Après modification Nginx** : - -1. **Tester la config** : -```bash -sudo nginx -t -``` - -2. **Recharger Nginx** : -```bash -sudo systemctl reload nginx -``` - -3. **Tester la connexion** : -- Se connecter via Keycloak -- Vérifier que l'erreur 502 ne se produit plus -- Vérifier les logs Nginx pour confirmer - -## 🎯 Cause Technique - -NextAuth crée un cookie JWT qui contient : -- Le JWT encrypté avec `NEXTAUTH_SECRET` -- Le JWT contient tous les tokens Keycloak -- La taille totale peut dépasser 4KB - -Nginx a une limite par défaut de 4KB pour les headers de réponse. Quand Next.js essaie de renvoyer un cookie > 4KB, Nginx rejette avec "upstream sent too big header". - -## ✅ Solution Immédiate - -**Action** : Modifier la configuration Nginx pour augmenter `large_client_header_buffers` à au moins `4 32k` ou `8 16k`. - -**Impact** : Résout immédiatement l'erreur 502. - ---- - -**Document créé le** : $(date) -**Priorité** : HAUTE - C'est la cause de l'erreur 502 - - diff --git a/NOTIFICATION_AND_WIDGET_ANALYSIS.md b/NOTIFICATION_AND_WIDGET_ANALYSIS.md deleted file mode 100644 index 86aeefeb..00000000 --- a/NOTIFICATION_AND_WIDGET_ANALYSIS.md +++ /dev/null @@ -1,548 +0,0 @@ -# Notification and Widget Update System - Complete File & Route Analysis - -## 📋 Table of Contents -1. [Notification System](#notification-system) -2. [Widget Update System](#widget-update-system) -3. [API Routes](#api-routes) -4. [Components](#components) -5. [Services & Libraries](#services--libraries) -6. [Hooks](#hooks) -7. [Types](#types) - ---- - -## 🔔 Notification System - -### API Routes - -#### 1. **GET `/api/notifications`** -- **File**: `app/api/notifications/route.ts` -- **Purpose**: Fetch paginated notifications for authenticated user -- **Query Parameters**: - - `page` (default: 1) - - `limit` (default: 20, max: 100) -- **Response**: - ```json - { - "notifications": Notification[], - "page": number, - "limit": number, - "total": number - } - ``` -- **Cache**: 30 seconds client-side cache -- **Authentication**: Required (session-based) - -#### 2. **GET `/api/notifications/count`** -- **File**: `app/api/notifications/count/route.ts` -- **Purpose**: Get notification count (total and unread) for authenticated user -- **Response**: - ```json - { - "total": number, - "unread": number, - "sources": { - [source]: { - "total": number, - "unread": number - } - } - } - ``` -- **Cache**: 10 seconds client-side cache -- **Authentication**: Required - -#### 3. **POST `/api/notifications/[id]/read`** -- **File**: `app/api/notifications/[id]/read/route.ts` -- **Purpose**: Mark a specific notification as read -- **Parameters**: - - `id` (path parameter): Notification ID (format: `source-sourceId`) -- **Response**: - ```json - { - "success": boolean - } - ``` -- **Authentication**: Required - -#### 4. **POST `/api/notifications/read-all`** -- **File**: `app/api/notifications/read-all/route.ts` -- **Purpose**: Mark all notifications as read for authenticated user -- **Response**: - ```json - { - "success": boolean - } - ``` -- **Authentication**: Required - -#### 5. **GET `/api/debug/notifications`** -- **File**: `app/api/debug/notifications/route.ts` -- **Purpose**: Debug endpoint to test notification system -- **Response**: Detailed debug information including: - - Environment variables status - - User information - - Notification service test results - - Performance metrics -- **Authentication**: Required - -### Services - -#### 1. **NotificationService** (Singleton) -- **File**: `lib/services/notifications/notification-service.ts` -- **Purpose**: Core notification aggregation service -- **Features**: - - Multi-source notification aggregation (adapter pattern) - - Redis caching (30s for counts, 5min for lists) - - Background refresh scheduling - - Cache invalidation on read operations - - Lock mechanism to prevent concurrent refreshes -- **Methods**: - - `getInstance()`: Get singleton instance - - `getNotifications(userId, page, limit)`: Fetch notifications - - `getNotificationCount(userId)`: Get notification counts - - `markAsRead(userId, notificationId)`: Mark notification as read - - `markAllAsRead(userId)`: Mark all as read - - `invalidateCache(userId)`: Invalidate user caches - - `scheduleBackgroundRefresh(userId)`: Schedule background refresh - -#### 2. **NotificationAdapter Interface** -- **File**: `lib/services/notifications/notification-adapter.interface.ts` -- **Purpose**: Interface for notification source adapters -- **Methods**: - - `getNotifications(userId, page?, limit?)`: Fetch notifications - - `getNotificationCount(userId)`: Get counts - - `markAsRead(userId, notificationId)`: Mark as read - - `markAllAsRead(userId)`: Mark all as read - - `isConfigured()`: Check if adapter is configured - -#### 3. **LeantimeAdapter** (Implementation) -- **File**: `lib/services/notifications/leantime-adapter.ts` -- **Purpose**: Leantime notification source adapter -- **Features**: - - Fetches notifications from Leantime API via JSON-RPC - - Maps Leantime user IDs by email - - Transforms Leantime notifications to unified format - - Supports marking notifications as read -- **Configuration**: - - `LEANTIME_API_URL` environment variable - - `LEANTIME_TOKEN` environment variable - -### Components - -#### 1. **NotificationBadge** -- **File**: `components/notification-badge.tsx` -- **Purpose**: Notification bell icon with badge and dropdown -- **Features**: - - Displays unread count badge - - Dropdown menu with recent notifications - - Manual refresh button - - Mark as read functionality - - Mark all as read functionality - - Source badges (e.g., "Agilité" for Leantime) - - Links to source systems - - Error handling and retry -- **Used in**: `components/main-nav.tsx` - -#### 2. **MainNav** (Notification Integration) -- **File**: `components/main-nav.tsx` -- **Purpose**: Main navigation bar with notification badge -- **Notification Features**: - - Includes `` component - - Browser notification permission handling - - User status-based notification management - -### Hooks - -#### 1. **useNotifications** -- **File**: `hooks/use-notifications.ts` -- **Purpose**: React hook for notification management -- **Features**: - - Automatic polling (60 seconds interval) - - Rate limiting (5 seconds minimum between fetches) - - Debounced count fetching (300ms) - - Manual refresh support - - Mount/unmount lifecycle management - - Error handling -- **Returns**: - ```typescript - { - notifications: Notification[], - notificationCount: NotificationCount, - loading: boolean, - error: string | null, - fetchNotifications: (page?, limit?) => Promise, - fetchNotificationCount: () => Promise, - markAsRead: (notificationId: string) => Promise, - markAllAsRead: () => Promise - } - ``` - -### Types - -#### 1. **Notification Types** -- **File**: `lib/types/notification.ts` -- **Interfaces**: - - `Notification`: Main notification interface - - `id`: string (format: `source-sourceId`) - - `source`: 'leantime' | 'nextcloud' | 'gitea' | 'dolibarr' | 'moodle' - - `sourceId`: string - - `type`: string - - `title`: string - - `message`: string - - `link?`: string - - `isRead`: boolean - - `timestamp`: Date - - `priority`: 'low' | 'normal' | 'high' - - `user`: { id: string, name?: string } - - `metadata?`: Record - - `NotificationCount`: Count interface - - `total`: number - - `unread`: number - - `sources`: Record - ---- - -## 🎨 Widget Update System - -### Dashboard Widgets - -The main dashboard (`app/page.tsx`) contains the following widgets: - -1. **QuoteCard** - Daily quote widget -2. **Calendar** - Upcoming events widget -3. **News** - News articles widget -4. **Duties** - Tasks/Devoirs widget (Leantime) -5. **Email** - Email inbox widget -6. **Parole** - Chat messages widget (Rocket.Chat) - -### Widget Components & Update Mechanisms - -#### 1. **Calendar Widget** -- **Files**: - - `components/calendar.tsx` (Main dashboard widget) - - `components/calendar-widget.tsx` (Alternative implementation) - - `components/calendar/calendar-widget.tsx` (Calendar-specific widget) -- **Update Mechanism**: - - **Manual Refresh**: Refresh button in header - - **Auto Refresh**: Every 5 minutes (300000ms interval) - - **API Endpoint**: `/api/calendars?refresh=true` - - **Features**: - - Fetches calendars with events - - Filters upcoming events (today and future) - - Sorts by date (oldest first) - - Shows up to 7 events - - Displays calendar color coding -- **State Management**: - - `useState` for events, loading, error - - `useEffect` for initial fetch and interval setup - -#### 2. **News Widget** -- **File**: `components/news.tsx` -- **Update Mechanism**: - - **Manual Refresh**: Refresh button in header - - **Initial Load**: On component mount when authenticated - - **API Endpoint**: `/api/news?limit=100` or `/api/news?refresh=true&limit=100` - - **Features**: - - Fetches up to 100 news articles - - Displays article count - - Click to open in new tab - - Scrollable list (max-height: 400px) -- **State Management**: - - `useState` for news, loading, error, refreshing - - `useEffect` for initial fetch on authentication - -#### 3. **Duties Widget (Tasks)** -- **File**: `components/flow.tsx` -- **Update Mechanism**: - - **Manual Refresh**: Refresh button in header - - **Initial Load**: On component mount - - **API Endpoint**: `/api/leantime/tasks?refresh=true` - - **Features**: - - Fetches tasks from Leantime - - Filters out completed tasks (status 5) - - Sorts by due date (oldest first) - - Shows up to 7 tasks - - Displays task status badges - - Links to Leantime ticket view -- **State Management**: - - `useState` for tasks, loading, error, refreshing - - `useEffect` for initial fetch - -#### 4. **Email Widget** -- **File**: `components/email.tsx` -- **Update Mechanism**: - - **Manual Refresh**: Refresh button in header - - **Initial Load**: On component mount - - **API Endpoint**: `/api/courrier?folder=INBOX&page=1&perPage=5` (+ `&refresh=true` for refresh) - - **Features**: - - Fetches 5 most recent emails from INBOX - - Sorts by date (most recent first) - - Shows read/unread status - - Displays sender, subject, date - - Link to full email view (`/courrier`) -- **State Management**: - - `useState` for emails, loading, error, mailUrl - - `useEffect` for initial fetch - -#### 5. **Parole Widget (Chat Messages)** -- **File**: `components/parole.tsx` -- **Update Mechanism**: - - **Manual Refresh**: Refresh button in header - - **Auto Polling**: Every 30 seconds (30000ms interval) - - **Initial Load**: On authentication - - **API Endpoint**: `/api/rocket-chat/messages` (+ `?refresh=true` for refresh) - - **Features**: - - Fetches recent chat messages from Rocket.Chat - - Displays sender avatar, name, message - - Shows room/channel information - - Click to navigate to full chat (`/parole`) - - Authentication check with sign-in prompt -- **State Management**: - - `useState` for messages, loading, error, refreshing - - `useEffect` for initial fetch and polling setup - - Session status checking - -#### 6. **QuoteCard Widget** -- **File**: `components/quote-card.tsx` -- **Update Mechanism**: (To be verified - likely static or daily update) - -### Widget Update Patterns - -#### Common Update Mechanisms: - -1. **Manual Refresh**: - - All widgets have a refresh button in their header - - Triggers API call with `refresh=true` parameter - - Shows loading/spinning state during refresh - -2. **Auto Refresh/Polling**: - - **Calendar**: 5 minutes interval - - **Parole**: 30 seconds interval - - Others: On component mount only - -3. **Session-Based Loading**: - - Widgets check authentication status - - Only fetch data when `status === 'authenticated'` - - Show loading state during authentication check - -4. **Error Handling**: - - All widgets display error messages - - Retry buttons available - - Graceful degradation (empty states) - -5. **State Management**: - - All widgets use React `useState` hooks - - Loading states managed locally - - Error states managed locally - -### Related API Routes for Widgets - -#### Calendar -- **GET `/api/calendars`**: Fetch calendars with events -- **GET `/api/calendars/[id]/events`**: Fetch events for specific calendar -- **GET `/api/calendars/[id]`**: Get calendar details - -#### News -- **GET `/api/news`**: Fetch news articles - - Query params: `limit`, `refresh` - -#### Tasks (Leantime) -- **GET `/api/leantime/tasks`**: Fetch tasks - - Query params: `refresh` - -#### Email (Courrier) -- **GET `/api/courrier`**: Fetch emails - - Query params: `folder`, `page`, `perPage`, `refresh` -- **POST `/api/courrier/refresh`**: Force refresh email cache - -#### Chat (Rocket.Chat) -- **GET `/api/rocket-chat/messages`**: Fetch messages - - Query params: `refresh` - ---- - -## 📁 Complete File Structure - -### Notification Files - -``` -app/api/notifications/ -├── route.ts # GET /api/notifications -├── count/ -│ └── route.ts # GET /api/notifications/count -├── read-all/ -│ └── route.ts # POST /api/notifications/read-all -└── [id]/ - └── read/ - └── route.ts # POST /api/notifications/[id]/read - -app/api/debug/ -└── notifications/ - └── route.ts # GET /api/debug/notifications - -lib/services/notifications/ -├── notification-service.ts # Core notification service -├── notification-adapter.interface.ts # Adapter interface -└── leantime-adapter.ts # Leantime adapter implementation - -lib/types/ -└── notification.ts # Notification type definitions - -hooks/ -└── use-notifications.ts # React hook for notifications - -components/ -├── notification-badge.tsx # Notification UI component -└── main-nav.tsx # Navigation with notification badge -``` - -### Widget Files - -``` -app/ -└── page.tsx # Main dashboard with widgets - -components/ -├── calendar.tsx # Calendar widget -├── calendar-widget.tsx # Alternative calendar widget -├── calendar/ -│ └── calendar-widget.tsx # Calendar-specific widget -├── news.tsx # News widget -├── flow.tsx # Duties/Tasks widget -├── email.tsx # Email widget -├── parole.tsx # Chat messages widget -└── quote-card.tsx # Quote widget - -app/api/ -├── calendars/ -│ ├── route.ts # GET /api/calendars -│ └── [id]/ -│ └── events/ -│ └── route.ts # GET /api/calendars/[id]/events -├── news/ -│ └── route.ts # GET /api/news -├── leantime/ -│ └── tasks/ -│ └── route.ts # GET /api/leantime/tasks -├── courrier/ -│ ├── route.ts # GET /api/courrier -│ └── refresh/ -│ └── route.ts # POST /api/courrier/refresh -└── rocket-chat/ - └── messages/ - └── route.ts # GET /api/rocket-chat/messages -``` - ---- - -## 🔄 Update Flow Diagrams - -### Notification Update Flow - -``` -User Action / Polling - ↓ -useNotifications Hook - ↓ -API Route (/api/notifications or /api/notifications/count) - ↓ -NotificationService.getInstance() - ↓ -Check Redis Cache - ├─ Cache Hit → Return cached data - └─ Cache Miss → Fetch from Adapters - ↓ - LeantimeAdapter (and other adapters) - ↓ - Transform & Aggregate - ↓ - Store in Redis Cache - ↓ - Return to API - ↓ - Return to Hook - ↓ - Update Component State -``` - -### Widget Update Flow - -``` -Component Mount / User Click Refresh - ↓ -useEffect / onClick Handler - ↓ -fetch() API Call - ├─ With refresh=true (manual) - └─ Without refresh (initial) - ↓ -API Route Handler - ├─ Check Cache (if applicable) - ├─ Fetch from External Service - └─ Return Data - ↓ -Update Component State - ├─ setLoading(false) - ├─ setData(response) - └─ setError(null) - ↓ -Re-render Component -``` - ---- - -## 🎯 Key Features Summary - -### Notification System -- ✅ Multi-source aggregation (adapter pattern) -- ✅ Redis caching with TTL -- ✅ Background refresh scheduling -- ✅ Polling mechanism (60s interval) -- ✅ Rate limiting (5s minimum) -- ✅ Mark as read / Mark all as read -- ✅ Cache invalidation on updates -- ✅ Error handling and retry -- ✅ Source badges and links - -### Widget System -- ✅ Manual refresh buttons -- ✅ Auto-refresh/polling (widget-specific intervals) -- ✅ Session-based loading -- ✅ Error handling -- ✅ Loading states -- ✅ Empty states -- ✅ Responsive design - ---- - -## 📝 Notes - -1. **Notification Sources**: Currently only Leantime adapter is implemented. Other adapters (Nextcloud, Gitea, Dolibarr, Moodle) are commented out in the service. - -2. **Cache Strategy**: - - Notification counts: 30 seconds TTL - - Notification lists: 5 minutes TTL - - Widget data: Varies by widget (some use API-level caching) - -3. **Polling Intervals**: - - Notifications: 60 seconds - - Calendar widget: 5 minutes - - Parole widget: 30 seconds - - Other widgets: On mount only - -4. **Authentication**: All notification and widget APIs require authentication via NextAuth session. - -5. **Error Handling**: All components implement error states with retry mechanisms. - ---- - -## 🔍 Debugging - -- Use `/api/debug/notifications` to test notification system -- Check browser console for detailed logs (all components log extensively) -- Check Redis cache keys: `notifications:count:{userId}`, `notifications:list:{userId}:{page}:{limit}` - ---- - -*Last Updated: Generated from codebase analysis* diff --git a/NOTIFICATION_DEBUG_NEXT_STEPS.md b/NOTIFICATION_DEBUG_NEXT_STEPS.md deleted file mode 100644 index 74e119b3..00000000 --- a/NOTIFICATION_DEBUG_NEXT_STEPS.md +++ /dev/null @@ -1,200 +0,0 @@ -# Notification Mark-All-As-Read Debug - Next Steps - -**Date**: 2026-01-01 -**Status**: Enhanced logging added, awaiting test results - ---- - -## 🔍 Current Situation - -**Issue**: Mark all as read fails, but no detailed error logs are visible - -**Observation from Logs**: -``` -[NOTIFICATION_API] Mark all as read - Failed { userId: '...', duration: '209ms' } -``` - -**Missing Logs**: -- No `[LEANTIME_ADAPTER] markAllAsRead` logs -- No `[NOTIFICATION_SERVICE] markAllAsRead` detailed logs - -**Possible Causes**: -1. Server not restarted with new code -2. Adapter method not being called -3. Error happening before adapter is reached - ---- - -## ✅ Fixes Applied - -### 1. Enhanced Service Layer Logging - -**File**: `lib/services/notifications/notification-service.ts` - -**Added Logging**: -- Logs when `markAllAsRead` is called -- Logs available adapters -- Logs each adapter being processed -- Logs configuration status for each adapter -- Logs when calling adapter's `markAllAsRead` -- Logs result from each adapter -- Logs overall success/failure -- Logs cache invalidation status - -### 2. Enhanced Adapter Layer Logging - -**File**: `lib/services/notifications/leantime-adapter.ts` - -**Added Logging**: -- User email and Leantime user ID -- Request body and API URL -- Response status and body -- Parsed response with error details -- Success/failure status - ---- - -## 🚀 Next Steps - -### Step 1: Restart Server - -**CRITICAL**: The server must be restarted for the new logging to take effect. - -```bash -# Stop the server -sudo npm stop -# Or if using PM2/systemd, restart appropriately - -# Start the server -sudo npm start -``` - -### Step 2: Test Mark All As Read - -1. Open the notification dropdown -2. Click "Mark all as read" -3. Immediately check the server logs - -### Step 3: Check Logs - -**Expected Log Flow** (if working correctly): -``` -[NOTIFICATION_API] Mark all as read endpoint called -[NOTIFICATION_API] Mark all as read - Processing { userId: '...', timestamp: '...' } -[NOTIFICATION_SERVICE] markAllAsRead called for user ... -[NOTIFICATION_SERVICE] Available adapters: leantime -[NOTIFICATION_SERVICE] Processing adapter: leantime -[NOTIFICATION_SERVICE] Adapter leantime is configured: true -[NOTIFICATION_SERVICE] Calling markAllAsRead on adapter leantime -[LEANTIME_ADAPTER] markAllAsRead called for ... -[LEANTIME_ADAPTER] markAllAsRead - User email: ... -[LEANTIME_ADAPTER] markAllAsRead - Leantime user ID: ... -[LEANTIME_ADAPTER] markAllAsRead - Request body: {...} -[LEANTIME_ADAPTER] markAllAsRead - API URL: ... -[LEANTIME_ADAPTER] markAllAsRead - Response status: XXX -[LEANTIME_ADAPTER] markAllAsRead - Response body: {...} -[LEANTIME_ADAPTER] markAllAsRead - Success: true/false -[NOTIFICATION_SERVICE] Adapter leantime markAllAsRead result: true/false -[NOTIFICATION_SERVICE] markAllAsRead overall success: true/false -``` - -**If Still Failing**, the logs will show: -- Which adapter is being processed -- Whether it's configured -- Whether the adapter method is called -- What error the Leantime API returns -- Where exactly it's failing - ---- - -## 🔍 What to Look For - -### If No Adapter Logs Appear - -**Possible Issues**: -1. Server not restarted → **Solution**: Restart server -2. Adapter not configured → Check `isConfigured()` result -3. Error in service layer → Check service layer logs - -### If Adapter Logs Appear But Fail - -**Check These**: -1. **User Email**: Should show email address -2. **Leantime User ID**: Should show numeric ID (e.g., `2`) -3. **Request Body**: Should show valid JSON-RPC request -4. **Response Status**: - - `200` = Success (but check result) - - `400` = Bad request (check error body) - - `401` = Authentication issue - - `500` = Server error -5. **Response Body**: Will show the actual error from Leantime - -### Common Leantime API Errors - -1. **Method Not Found**: - - Error: `"Method not found"` - - Fix: Verify method name is correct - -2. **Invalid Parameters**: - - Error: `"Invalid params"` - - Fix: Check parameter format - -3. **Authentication Failed**: - - Error: `"Unauthorized"` or `401` - - Fix: Check API token - -4. **User Not Found**: - - Error: `"User not found"` - - Fix: Verify Leantime user ID mapping - ---- - -## 📊 Expected Log Output Examples - -### Success Case -``` -[LEANTIME_ADAPTER] markAllAsRead - Response status: 200 -[LEANTIME_ADAPTER] markAllAsRead - Response body: {"jsonrpc":"2.0","result":true,"id":1} -[LEANTIME_ADAPTER] markAllAsRead - Parsed response: { hasResult: true, result: true, hasError: false } -[LEANTIME_ADAPTER] markAllAsRead - Success: true -``` - -### Failure Case - Method Not Found -``` -[LEANTIME_ADAPTER] markAllAsRead - Response status: 200 -[LEANTIME_ADAPTER] markAllAsRead - Response body: {"jsonrpc":"2.0","error":{"code":-32601,"message":"Method not found"},"id":1} -[LEANTIME_ADAPTER] markAllAsRead - Parsed response: { hasResult: false, hasError: true, error: {...} } -[LEANTIME_ADAPTER] markAllAsRead - API Error: { code: -32601, message: "Method not found" } -[LEANTIME_ADAPTER] markAllAsRead - Success: false -``` - -### Failure Case - Invalid Params -``` -[LEANTIME_ADAPTER] markAllAsRead - Response status: 200 -[LEANTIME_ADAPTER] markAllAsRead - Response body: {"jsonrpc":"2.0","error":{"code":-32602,"message":"Invalid params"},"id":1} -[LEANTIME_ADAPTER] markAllAsRead - API Error: { code: -32602, message: "Invalid params" } -``` - ---- - -## 🎯 Action Items - -1. ✅ **Restart Server** (CRITICAL) -2. ⏳ **Test Mark All As Read** -3. ⏳ **Share Complete Logs** (from endpoint call to failure) -4. ⏳ **Analyze Error Details** (once logs are available) - ---- - -## 📝 Summary - -**Status**: Enhanced logging ready, awaiting server restart and test - -**Next**: After restart, test and share logs to identify exact failure point - -**Confidence**: 🟢 **HIGH** - Enhanced logging will reveal the root cause - ---- - -**Generated**: 2026-01-01 - diff --git a/NOTIFICATION_FIXES_IMPLEMENTED.md b/NOTIFICATION_FIXES_IMPLEMENTED.md deleted file mode 100644 index eb156551..00000000 --- a/NOTIFICATION_FIXES_IMPLEMENTED.md +++ /dev/null @@ -1,239 +0,0 @@ -# Notification System Fixes - Implementation Summary - -**Date**: 2026-01-06 -**Status**: ✅ All fixes implemented - ---- - -## ✅ **Fix #1: Redis Caching for Leantime User ID** - -### **Problem**: -- `getLeantimeUserId()` fetched ALL users from Leantime API every time -- No caching, causing slow performance and inconsistent results -- Race conditions between different calls - -### **Solution**: -- Added Redis caching with 1-hour TTL -- Cache key: `leantime:userid:${email.toLowerCase()}` -- Checks cache first before making API call -- Caches result after successful fetch - -### **Implementation**: -- **File**: `lib/services/notifications/leantime-adapter.ts` -- **Method**: `getLeantimeUserId()` -- **Cache TTL**: 3600 seconds (1 hour) -- **Static helper**: `invalidateUserIdCache()` for manual cache clearing - -### **Benefits**: -- ✅ Faster performance (no API call if cached) -- ✅ More reliable (consistent results) -- ✅ Reduced API load on Leantime -- ✅ Better error recovery (can use cached value if API fails) - ---- - -## ✅ **Fix #2: Retry Logic with Exponential Backoff** - -### **Problem**: -- `getLeantimeUserId()` failed immediately on API errors -- No retry mechanism for transient failures -- Network errors caused permanent failures - -### **Solution**: -- Added retry logic with up to 3 retries -- Exponential backoff: 1s, 2s, 4s (max 5s) -- Retries on: - - Server errors (5xx) - - Rate limiting (429) - - Network errors - - Certain JSON-RPC errors - -### **Implementation**: -- **File**: `lib/services/notifications/leantime-adapter.ts` -- **Method**: `getLeantimeUserId()` with `fetchWithRetry()` -- **Max Retries**: 3 -- **Backoff**: Exponential (1s → 2s → 4s) - -### **Benefits**: -- ✅ Handles transient failures gracefully -- ✅ Better resilience to network issues -- ✅ Improved success rate for user ID lookup - ---- - -## ✅ **Fix #3: Always Invalidate Cache After Marking** - -### **Problem**: -- Cache only invalidated if marking operation succeeded -- If `getLeantimeUserId()` failed, cache stayed stale -- Count remained at old value (65) even after marking attempts - -### **Solution**: -- Always invalidate cache after marking attempt -- Even if operation failed or returned `false` -- Ensures fresh data on next fetch - -### **Implementation**: -- **File**: `lib/services/notifications/notification-service.ts` -- **Methods**: - - `markAsRead()` - Always invalidates cache - - `markAllAsRead()` - Always invalidates cache -- **Logic**: Cache invalidation happens regardless of success/failure - -### **Benefits**: -- ✅ Count always refreshes after marking attempts -- ✅ User sees accurate data even if operation partially failed -- ✅ Better UX (no stale count stuck at 65) - ---- - -## ✅ **Fix #4: Improved Count Accuracy** - -### **Problem**: -- Count only based on first 100 notifications -- If user had >100 notifications, count was inaccurate -- Used cached notifications which might be stale - -### **Solution**: -- Fetch up to 1000 notifications directly from API for counting -- Bypasses cache to get fresh data -- More accurate count for users with many notifications - -### **Implementation**: -- **File**: `lib/services/notifications/leantime-adapter.ts` -- **Method**: `getNotificationCount()` -- **Change**: Fetches directly from API (up to 1000) instead of using cached `getNotifications()` -- **Warning**: Logs if count reaches 1000 (might have more) - -### **Benefits**: -- ✅ More accurate count (up to 1000 notifications) -- ✅ Fresh data (bypasses cache) -- ✅ Better handling of users with many notifications - ---- - -## ✅ **Fix #5: Better Error Handling and Logging** - -### **Problem**: -- Errors were logged but not handled gracefully -- No way to manually clear user ID cache -- Limited error context in logs - -### **Solution**: -- Added static method to invalidate user ID cache -- Improved error messages with more context -- Better logging throughout the flow -- Graceful degradation on errors - -### **Implementation**: -- **File**: `lib/services/notifications/leantime-adapter.ts` -- **Static Method**: `invalidateUserIdCache(email)` -- **Improved Logging**: More detailed error messages -- **Error Recovery**: Continues operation even if caching fails - -### **Benefits**: -- ✅ Better debugging with detailed logs -- ✅ Manual cache clearing for troubleshooting -- ✅ More resilient to partial failures - ---- - -## 📊 **Expected Behavior After Fixes** - -### **Before Fixes**: -1. Mark all as read → `getLeantimeUserId()` fails → Returns `false` -2. Cache NOT invalidated → Count stays 65 ❌ -3. User sees stale count - -### **After Fixes**: -1. Mark all as read → `getLeantimeUserId()` checks cache first ✅ -2. If cached: Uses cached ID immediately ✅ -3. If not cached: Fetches with retry logic ✅ -4. Marks notifications as read ✅ -5. **Always invalidates cache** ✅ -6. Count refresh gets fresh data → Shows 0 ✅ - ---- - -## 🎯 **Key Improvements** - -### **Reliability**: -- ✅ User ID lookup is now cached and retried -- ✅ Cache always invalidated after marking -- ✅ Better error recovery - -### **Performance**: -- ✅ Faster user ID lookup (cached) -- ✅ Reduced API calls to Leantime -- ✅ More efficient cache usage - -### **Accuracy**: -- ✅ Count based on up to 1000 notifications -- ✅ Fresh data from API (bypasses stale cache) -- ✅ Better handling of edge cases - -### **User Experience**: -- ✅ Count updates correctly after marking -- ✅ No more stuck count at 65 -- ✅ Faster response times - ---- - -## 🚀 **Testing Checklist** - -After rebuild (`rm -rf .next && npm run build && npm start`): - -1. ✅ **Test Mark All As Read**: - - Should work even if user ID lookup was previously failing - - Count should update to 0 after marking - - Cache should be invalidated - -2. ✅ **Test Mark Single As Read**: - - Should work reliably - - Count should decrement correctly - - Cache should be invalidated - -3. ✅ **Test Count Accuracy**: - - Should show accurate count (up to 1000) - - Should refresh after marking - - Should use fresh data from API - -4. ✅ **Test User ID Caching**: - - First call should fetch from API - - Subsequent calls should use cache - - Should be faster on subsequent calls - -5. ✅ **Test Retry Logic**: - - Should retry on transient failures - - Should eventually succeed or fail gracefully - - Should log retry attempts - ---- - -## 📝 **Files Modified** - -1. **`lib/services/notifications/leantime-adapter.ts`**: - - Added Redis caching for user ID - - Added retry logic with exponential backoff - - Improved `getNotificationCount()` to fetch directly from API - - Added `invalidateUserIdCache()` static method - - Better error handling and logging - -2. **`lib/services/notifications/notification-service.ts`**: - - Always invalidate cache in `markAsRead()` - - Always invalidate cache in `markAllAsRead()` - - Better error handling and logging - ---- - -## 🔧 **Configuration** - -- **User ID Cache TTL**: 3600 seconds (1 hour) -- **Max Retries**: 3 attempts -- **Retry Backoff**: Exponential (1s, 2s, 4s, max 5s) -- **Count Fetch Limit**: 1000 notifications - ---- - -**Status**: ✅ All fixes implemented and ready for testing - diff --git a/NOTIFICATION_FIXES_IMPLEMENTED_SUMMARY.md b/NOTIFICATION_FIXES_IMPLEMENTED_SUMMARY.md deleted file mode 100644 index 1739fbb2..00000000 --- a/NOTIFICATION_FIXES_IMPLEMENTED_SUMMARY.md +++ /dev/null @@ -1,314 +0,0 @@ -# Notification System Fixes - Implementation Summary - -**Date**: 2026-01-06 -**Status**: ✅ All High-Priority Fixes Implemented - ---- - -## ✅ **Fix #1: Integrated Unified Refresh System** - -### **Changes**: -- **File**: `hooks/use-notifications.ts` -- **Removed**: Custom polling logic (60s interval, debouncing) -- **Added**: `useUnifiedRefresh` hook integration -- **Result**: Uses centralized `RefreshManager` with 30s interval - -### **Benefits**: -- ✅ Consistent refresh intervals across all widgets -- ✅ Reduced code duplication -- ✅ Better coordination with other refresh systems -- ✅ Automatic deduplication built-in - -### **Code Changes**: -```typescript -// Before: Custom polling -pollingIntervalRef.current = setInterval(() => { - debouncedFetchCount(); -}, 60000); - -// After: Unified refresh -const { refresh: refreshCount } = useUnifiedRefresh({ - resource: 'notifications-count', - interval: REFRESH_INTERVALS.NOTIFICATIONS_COUNT, // 30s - enabled: status === 'authenticated', - onRefresh: async () => { - await fetchNotificationCount(false); - }, - priority: 'high', -}); -``` - ---- - -## ✅ **Fix #2: Batch Processing for Mark All As Read** - -### **Changes**: -- **File**: `lib/services/notifications/leantime-adapter.ts` -- **Added**: Batch processing (15 notifications per batch) -- **Added**: Delay between batches (200ms) -- **Added**: Automatic retry for failed notifications -- **Added**: Success rate threshold (80% = success) - -### **Benefits**: -- ✅ Prevents API overload -- ✅ Reduces connection resets -- ✅ Better error recovery -- ✅ More reliable marking - -### **Implementation**: -```typescript -// Process in batches of 15 -const BATCH_SIZE = 15; -const BATCH_DELAY = 200; -const MAX_RETRIES = 2; - -// Process each batch with delay -for (let i = 0; i < notificationIds.length; i += BATCH_SIZE) { - const batch = notificationIds.slice(i, i + BATCH_SIZE); - await Promise.all(batch.map(n => markSingleNotification(n))); - await delay(BATCH_DELAY); // Delay between batches -} - -// Retry failed notifications -if (failedNotifications.length > 0) { - await retryFailedNotifications(); -} -``` - ---- - -## ✅ **Fix #3: Fixed Cache TTL Consistency** - -### **Changes**: -- **File**: `lib/services/notifications/notification-service.ts` -- **Changed**: List cache TTL: 5 minutes → 30 seconds -- **Aligned**: All cache TTLs to 30 seconds -- **File**: `app/api/notifications/route.ts` & `count/route.ts` -- **Changed**: Client cache: `max-age=30/10` → `max-age=0, must-revalidate` - -### **Benefits**: -- ✅ Count and list always in sync -- ✅ Consistent behavior -- ✅ Predictable cache expiration -- ✅ No stale data inconsistencies - -### **Before/After**: -```typescript -// Before -COUNT_CACHE_TTL = 30; // 30 seconds -LIST_CACHE_TTL = 300; // 5 minutes ❌ - -// After -COUNT_CACHE_TTL = 30; // 30 seconds ✅ -LIST_CACHE_TTL = 30; // 30 seconds ✅ -``` - ---- - -## ✅ **Fix #4: Added Progress Feedback** - -### **Changes**: -- **File**: `hooks/use-notifications.ts` -- **Added**: `markingProgress` state: `{ current: number; total: number }` -- **File**: `components/notification-badge.tsx` -- **Added**: Progress bar UI during mark all as read -- **Added**: Progress text: "Marking X of Y..." - -### **Benefits**: -- ✅ User knows operation is in progress -- ✅ Better UX (no silent waiting) -- ✅ Prevents multiple clicks -- ✅ Visual feedback - -### **UI Changes**: -```tsx -{markingProgress && ( -
-
- Marking {markingProgress.current} of {markingProgress.total}... -
-)} -``` - ---- - -## ✅ **Fix #5: Improved Optimistic Updates** - -### **Changes**: -- **File**: `hooks/use-notifications.ts` -- **Added**: Polling mechanism to verify count updates -- **Changed**: Better timing for count refresh -- **Added**: Poll until count matches expected value - -### **Benefits**: -- ✅ More accurate UI updates -- ✅ Less confusing count jumps -- ✅ Better error recovery -- ✅ Verifies server state matches UI - -### **Implementation**: -```typescript -// Poll until count matches expected value -let pollCount = 0; -const maxPolls = 5; -const pollInterval = 500; - -const pollForCount = async () => { - if (pollCount >= maxPolls) return; - pollCount++; - await fetchNotificationCount(true); - if (pollCount < maxPolls) { - setTimeout(pollForCount, pollInterval); - } -}; -``` - ---- - -## ✅ **Fix #6: Added Request Deduplication** - -### **Changes**: -- **File**: `hooks/use-notifications.ts` -- **Added**: `requestDeduplicator` for all fetch calls -- **Result**: Prevents duplicate API calls within 2-second window - -### **Benefits**: -- ✅ Fewer API calls -- ✅ Better performance -- ✅ Reduced server load -- ✅ Prevents race conditions - -### **Implementation**: -```typescript -// Before: Direct fetch -const response = await fetch(url); - -// After: Deduplicated fetch -const data = await requestDeduplicator.execute( - `notifications-count-${userId}`, - async () => { - const response = await fetch(url); - return response.json(); - }, - 2000 // 2 second deduplication window -); -``` - ---- - -## ✅ **Fix #7: Cached User Email** - -### **Changes**: -- **File**: `lib/services/notifications/leantime-adapter.ts` -- **Added**: Redis cache for user email (30-minute TTL) -- **Result**: Reduces session lookups - -### **Benefits**: -- ✅ Better performance -- ✅ Fewer session calls -- ✅ More consistent -- ✅ Reduced overhead - ---- - -## 📊 **Performance Improvements** - -### **Before**: -- Polling: Every 60 seconds -- Cache TTL: Inconsistent (30s / 5min) -- Mark all: All parallel (can timeout) -- No deduplication -- No progress feedback - -### **After**: -- Refresh: Every 30 seconds (unified) -- Cache TTL: Consistent (30s / 30s) -- Mark all: Batched (15 at a time, 200ms delay) -- Request deduplication: 2-second window -- Progress feedback: Real-time UI updates - -### **Expected Results**: -- **50-70% reduction** in API calls -- **30-40% faster** response times -- **80-90% success rate** for mark all (vs 60-70% before) -- **Better UX** with progress indicators - ---- - -## 🎯 **Files Modified** - -1. ✅ `hooks/use-notifications.ts` - - Integrated unified refresh - - Added request deduplication - - Added progress tracking - - Improved optimistic updates - -2. ✅ `lib/services/notifications/leantime-adapter.ts` - - Batch processing for mark all - - Retry logic with exponential backoff - - User email caching - -3. ✅ `lib/services/notifications/notification-service.ts` - - Fixed cache TTL consistency (30s for all) - -4. ✅ `app/api/notifications/route.ts` - - Updated client cache headers - -5. ✅ `app/api/notifications/count/route.ts` - - Updated client cache headers - -6. ✅ `components/notification-badge.tsx` - - Added progress UI - - Better loading states - ---- - -## 🚀 **Testing Checklist** - -After rebuild (`rm -rf .next && npm run build && npm start`): - -1. ✅ **Unified Refresh**: - - Count should refresh every 30 seconds - - Should use centralized refresh manager - - No duplicate polling - -2. ✅ **Batch Processing**: - - Mark all as read should process in batches - - Should show progress (if implemented) - - Should be more reliable (80-90% success) - -3. ✅ **Cache Consistency**: - - Count and list should always be in sync - - Cache should expire after 30 seconds - - No stale data - -4. ✅ **Progress Feedback**: - - Should show progress bar during mark all - - Should display "Marking X of Y..." - - Should prevent multiple clicks - -5. ✅ **Request Deduplication**: - - Multiple rapid calls should be deduplicated - - Should see fewer API calls in logs - - Better performance - ---- - -## 📝 **Next Steps (Optional)** - -### **Medium Priority** (Future): -1. Real-time progress updates (WebSocket/SSE) -2. Connection pooling for API calls -3. Better error messages for users -4. Cancel operation button - -### **Low Priority** (Nice to Have): -1. WebSocket for real-time notifications -2. Push notifications -3. Notification grouping -4. Filtering and sorting - ---- - -**Status**: ✅ All high-priority fixes implemented and ready for testing - diff --git a/NOTIFICATION_FLOW_ANALYSIS.md b/NOTIFICATION_FLOW_ANALYSIS.md deleted file mode 100644 index dc9f22e2..00000000 --- a/NOTIFICATION_FLOW_ANALYSIS.md +++ /dev/null @@ -1,526 +0,0 @@ -# Complete Notification Flow Analysis - -**Date**: 2026-01-06 -**Purpose**: Trace the entire notification system flow to identify issues and improvements - ---- - -## 🔍 **FLOW 1: Initial Page Load & Count Display** - -### Step-by-Step Flow: - -1. **Component Mount** (`notification-badge.tsx`) - - `useNotifications()` hook initializes - - `useEffect` triggers when `status === 'authenticated'` - - Calls `fetchNotificationCount(true)` (force refresh) - - Calls `fetchNotifications()` - - Starts polling every 60 seconds - -2. **Count Fetch** (`use-notifications.ts` → `/api/notifications/count`) - - Hook calls `/api/notifications/count?_t=${Date.now()}` (cache-busting) - - API route authenticates user - - Calls `NotificationService.getNotificationCount(userId)` - -3. **Service Layer** (`notification-service.ts`) - - **Checks Redis cache first** (`notifications:count:${userId}`) - - If cached: Returns cached data immediately - - If not cached: Fetches from adapters - -4. **Adapter Layer** (`leantime-adapter.ts`) - - `getNotificationCount()` calls `getNotifications(userId, 1, 100)` - - **⚠️ ISSUE**: Only fetches first 100 notifications for counting - - Filters unread: `notifications.filter(n => !n.isRead).length` - - Returns count object - -5. **Cache Storage** - - Service stores count in Redis with 30-second TTL - - Returns to API route - - API returns to hook - - Hook updates React state: `setNotificationCount(data)` - -6. **UI Update** - - Badge displays `notificationCount.unread` - - Shows "65" if 65 unread notifications - ---- - -## 🔍 **FLOW 2: Mark Single Notification as Read** - -### Step-by-Step Flow: - -1. **User Action** (`notification-badge.tsx`) - - User clicks "Mark as read" button - - Calls `handleMarkAsRead(notificationId)` - - Calls `markAsRead(notificationId)` from hook - -2. **Hook Action** (`use-notifications.ts`) - - Makes POST to `/api/notifications/${notificationId}/read` - - **Optimistic UI Update**: - - Updates notification in state: `isRead: true` - - Decrements count: `unread: Math.max(0, prev.unread - 1)` - - Waits 100ms, then calls `fetchNotificationCount(true)` - -3. **API Route** (`app/api/notifications/[id]/read/route.ts`) - - Authenticates user - - Extracts notification ID: `leantime-2732` → splits to get source and ID - - Calls `NotificationService.markAsRead(userId, notificationId)` - -4. **Service Layer** (`notification-service.ts`) - - Extracts source: `leantime` from ID - - Gets adapter: `this.adapters.get('leantime')` - - Calls `adapter.markAsRead(userId, notificationId)` - -5. **Adapter Layer** (`leantime-adapter.ts`) - - **Gets user email from session**: `getUserEmail()` - - **Gets Leantime user ID**: `getLeantimeUserId(email)` - - **⚠️ CRITICAL ISSUE**: If `getLeantimeUserId()` fails → returns `false` - - If successful: Calls Leantime API `markNotificationRead` - - Returns success/failure - -6. **Cache Invalidation** (`notification-service.ts`) - - If `markAsRead()` returns `true`: - - Calls `invalidateCache(userId)` - - Deletes count cache: `notifications:count:${userId}` - - Deletes all list caches: `notifications:list:${userId}:*` - - If returns `false`: **Cache NOT invalidated** ❌ - -7. **Count Refresh** (`use-notifications.ts`) - - After 100ms delay, calls `fetchNotificationCount(true)` - - Fetches fresh count from API - - **⚠️ ISSUE**: If cache wasn't invalidated, might get stale count - ---- - -## 🔍 **FLOW 3: Mark All Notifications as Read** - -### Step-by-Step Flow: - -1. **User Action** (`notification-badge.tsx`) - - User clicks "Mark all read" button - - Calls `handleMarkAllAsRead()` - - Calls `markAllAsRead()` from hook - -2. **Hook Action** (`use-notifications.ts`) - - Makes POST to `/api/notifications/read-all` - - **Optimistic UI Update**: - - Sets all notifications: `isRead: true` - - Sets count: `unread: 0` - - Waits 200ms, then calls `fetchNotificationCount(true)` - -3. **API Route** (`app/api/notifications/read-all/route.ts`) - - Authenticates user - - Calls `NotificationService.markAllAsRead(userId)` - -4. **Service Layer** (`notification-service.ts`) - - Loops through all adapters - - For each adapter: - - Checks if configured - - Calls `adapter.markAllAsRead(userId)` - - Collects results: `[true/false, ...]` - - Determines: `success = results.every(r => r)`, `anySuccess = results.some(r => r)` - - **Cache Invalidation**: - - If `anySuccess === true`: Invalidates cache ✅ - - If `anySuccess === false`: **Cache NOT invalidated** ❌ - -5. **Adapter Layer** (`leantime-adapter.ts`) - - **Gets user email**: `getUserEmail()` - - **Gets Leantime user ID**: `getLeantimeUserId(email)` - - **⚠️ CRITICAL ISSUE**: If this fails → returns `false` immediately - - If successful: - - Fetches all notifications directly from API (up to 1000) - - Filters unread: `rawNotifications.filter(n => n.read === 0)` - - Marks each individually using `markNotificationRead` - - Returns success if any were marked - -6. **Cache Invalidation** (`notification-service.ts`) - - Only happens if `anySuccess === true` - - **⚠️ ISSUE**: If `getLeantimeUserId()` fails, `anySuccess = false` - - Cache stays stale → count remains 65 - -7. **Count Refresh** (`use-notifications.ts`) - - After 200ms, calls `fetchNotificationCount(true)` - - **⚠️ ISSUE**: If cache wasn't invalidated, gets stale count from cache - ---- - -## 🔍 **FLOW 4: Fetch Notification List** - -### Step-by-Step Flow: - -1. **User Opens Dropdown** (`notification-badge.tsx`) - - `handleOpenChange(true)` called - - Calls `manualFetch()` which calls `fetchNotifications(1, 10)` - -2. **Hook Action** (`use-notifications.ts`) - - Makes GET to `/api/notifications?page=1&limit=20` - - Updates state: `setNotifications(data.notifications)` - -3. **API Route** (`app/api/notifications/route.ts`) - - Authenticates user - - Calls `NotificationService.getNotifications(userId, page, limit)` - -4. **Service Layer** (`notification-service.ts`) - - **Checks Redis cache first**: `notifications:list:${userId}:${page}:${limit}` - - If cached: Returns cached data immediately - - If not cached: Fetches from adapters - -5. **Adapter Layer** (`leantime-adapter.ts`) - - Gets user email and Leantime user ID - - Calls Leantime API `getAllNotifications` with pagination - - Transforms notifications to our format - - Returns array - -6. **Cache Storage** - - Service stores list in Redis with 5-minute TTL - - Returns to API - - API returns to hook - - Hook updates React state - ---- - -## 🐛 **IDENTIFIED ISSUES** - -### **Issue #1: getLeantimeUserId() Fails Inconsistently** - -**Problem**: -- `getLeantimeUserId()` works in `getNotifications()` and `getNotificationCount()` -- But fails in `markAllAsRead()` and sometimes in `markAsRead()` -- Logs show: `"User not found in Leantime: a.tmiri@clm.foundation"` - -**Root Cause**: -- `getLeantimeUserId()` calls Leantime API `getAll` users endpoint -- Fetches ALL users, then searches for matching email -- **Possible causes**: - 1. **Race condition**: API call happens at different times - 2. **Session timing**: Session might be different between calls - 3. **API rate limiting**: Leantime API might throttle requests - 4. **Caching issue**: No caching of user ID lookup - -**Impact**: -- Mark all as read fails → cache not invalidated → count stays 65 -- Mark single as read might fail → cache not invalidated → count doesn't update - -**Solution**: -- Cache Leantime user ID in Redis with longer TTL -- Add retry logic with exponential backoff -- Add better error handling and logging - ---- - -### **Issue #2: Cache Invalidation Only on Success** - -**Problem**: -- Cache is only invalidated if `markAsRead()` or `markAllAsRead()` returns `true` -- If operation fails (e.g., `getLeantimeUserId()` fails), cache stays stale -- Count remains at old value (65) - -**Root Cause**: -```typescript -if (success) { - await this.invalidateCache(userId); -} -``` - -**Impact**: -- User sees stale count even after attempting to mark as read -- UI shows optimistic update, but server count doesn't match - -**Solution**: -- Always invalidate cache after marking attempt (even on failure) -- Or: Invalidate cache before marking, then refresh after -- Or: Use optimistic updates with eventual consistency - ---- - -### **Issue #3: Count Based on First 100 Notifications** - -**Problem**: -- `getNotificationCount()` only fetches first 100 notifications -- If user has 200 notifications with 66 unread, count shows 66 -- But if 66 unread are beyond first 100, count is wrong - -**Root Cause**: -```typescript -const notifications = await this.getNotifications(userId, 1, 100); -const unreadCount = notifications.filter(n => !n.isRead).length; -``` - -**Impact**: -- Count might be inaccurate if >100 notifications exist -- User might see "66 unread" but only 10 displayed (pagination) - -**Solution**: -- Use dedicated count API if Leantime provides one -- Or: Fetch all notifications for counting (up to reasonable limit) -- Or: Show "66+ unread" if count reaches 100 - ---- - -### **Issue #4: Race Condition Between Cache Invalidation and Count Fetch** - -**Problem**: -- Hook calls `fetchNotificationCount(true)` after 100-200ms delay -- But cache invalidation might not be complete -- Count fetch might still get stale cache - -**Root Cause**: -```typescript -setTimeout(() => { - fetchNotificationCount(true); -}, 200); -``` - -**Impact**: -- Count might not update immediately after marking -- User sees optimistic update, then stale count - -**Solution**: -- Increase delay to 500ms -- Or: Poll count until it matches expected value -- Or: Use WebSocket/SSE for real-time updates - ---- - -### **Issue #5: No Caching of Leantime User ID** - -**Problem**: -- `getLeantimeUserId()` fetches ALL users from Leantime API every time -- No caching, so repeated calls are slow and might fail -- Different calls might get different results (race condition) - -**Root Cause**: -- No Redis cache for user ID mapping -- Each call makes full API request - -**Impact**: -- Slow performance -- Inconsistent results -- API rate limiting issues - -**Solution**: -- Cache user ID in Redis: `leantime:userid:${email}` with 1-hour TTL -- Invalidate cache only when user changes or on explicit refresh - ---- - -### **Issue #6: getNotificationCount Uses Cached getNotifications** - -**Problem**: -- `getNotificationCount()` calls `getNotifications(userId, 1, 100)` -- `getNotifications()` uses cache if available -- Count might be based on stale cached notifications - -**Root Cause**: -```typescript -async getNotificationCount(userId: string): Promise { - const notifications = await this.getNotifications(userId, 1, 100); - // Uses cached data if available -} -``` - -**Impact**: -- Count might be stale even if notifications were marked as read -- Cache TTL mismatch: count cache (30s) vs list cache (5min) - -**Solution**: -- Fetch notifications directly from API for counting (bypass cache) -- Or: Use dedicated count endpoint -- Or: Invalidate list cache when count cache is invalidated - ---- - -### **Issue #7: Optimistic Updates Don't Match Server State** - -**Problem**: -- Hook optimistically updates count: `unread: 0` -- But server count might still be 65 (cache not invalidated) -- After refresh, count jumps back to 65 - -**Root Cause**: -- Optimistic update happens immediately -- Server cache invalidation might fail -- Count refresh gets stale data - -**Impact**: -- Confusing UX: count goes to 0, then back to 65 -- User thinks operation failed when it might have succeeded - -**Solution**: -- Only show optimistic update if we're confident operation will succeed -- Or: Show loading state until server confirms -- Or: Poll until count matches expected value - ---- - -## 🎯 **RECOMMENDED IMPROVEMENTS** - -### **Priority 1: Fix getLeantimeUserId() Reliability** - -1. **Cache User ID Mapping** - ```typescript - // Cache key: leantime:userid:${email} - // TTL: 1 hour - // Invalidate on user update or explicit refresh - ``` - -2. **Add Retry Logic** - ```typescript - // Retry 3 times with exponential backoff - // Log each attempt - // Return cached value if API fails - ``` - -3. **Better Error Handling** - ```typescript - // Log full error details - // Return null only after all retries fail - // Don't fail entire operation on user ID lookup failure - ``` - ---- - -### **Priority 2: Always Invalidate Cache After Marking** - -1. **Invalidate Before Marking** - ```typescript - // Invalidate cache first - // Then mark as read - // Then refresh count - ``` - -2. **Or: Always Invalidate After Attempt** - ```typescript - // Always invalidate cache after marking attempt - // Even if operation failed - // This ensures fresh data on next fetch - ``` - ---- - -### **Priority 3: Fix Count Accuracy** - -1. **Use Dedicated Count API** (if available) - ```typescript - // Check if Leantime has count-only endpoint - // Use that instead of fetching all notifications - ``` - -2. **Or: Fetch All for Counting** - ```typescript - // Fetch up to 1000 notifications for counting - // Or use pagination to count all - ``` - -3. **Or: Show "66+ unread" if limit reached** - ```typescript - // If count === 100, show "100+ unread" - // Indicate there might be more - ``` - ---- - -### **Priority 4: Improve Cache Strategy** - -1. **Unified Cache Invalidation** - ```typescript - // When count cache is invalidated, also invalidate list cache - // When list cache is invalidated, also invalidate count cache - // Keep them in sync - ``` - -2. **Shorter Cache TTLs** - ```typescript - // Count cache: 10 seconds (currently 30s) - // List cache: 1 minute (currently 5min) - // More frequent updates - ``` - -3. **Cache Tags/Versioning** - ```typescript - // Use cache version numbers - // Increment on invalidation - // Check version before using cache - ``` - ---- - -### **Priority 5: Better Error Recovery** - -1. **Graceful Degradation** - ```typescript - // If mark as read fails, still invalidate cache - // Show error message to user - // Allow retry - ``` - -2. **Retry Logic** - ```typescript - // Retry failed operations automatically - // Exponential backoff - // Max 3 retries - ``` - ---- - -## 📊 **FLOW DIAGRAM: Current vs Improved** - -### **Current Flow (Mark All As Read)**: -``` -User clicks → Hook → API → Service → Adapter - ↓ -getLeantimeUserId() → FAILS ❌ - ↓ -Returns false → Service: anySuccess = false - ↓ -Cache NOT invalidated ❌ - ↓ -Count refresh → Gets stale cache → Shows 65 ❌ -``` - -### **Improved Flow (Mark All As Read)**: -``` -User clicks → Hook → API → Service → Adapter - ↓ -getLeantimeUserId() → Check cache first - ↓ -If cached: Use cached ID ✅ -If not cached: Fetch from API → Cache result ✅ - ↓ -Mark all as read → Success ✅ - ↓ -Always invalidate cache (even on partial failure) ✅ - ↓ -Count refresh → Gets fresh data → Shows 0 ✅ -``` - ---- - -## 🚀 **IMPLEMENTATION PRIORITY** - -1. **Fix getLeantimeUserId() caching** (High Priority) - - Add Redis cache for user ID mapping - - Add retry logic - - Better error handling - -2. **Always invalidate cache** (High Priority) - - Invalidate cache even on failure - - Or invalidate before marking - -3. **Fix count accuracy** (Medium Priority) - - Use dedicated count API or fetch all - - Show "66+ unread" if limit reached - -4. **Improve cache strategy** (Medium Priority) - - Unified invalidation - - Shorter TTLs - - Cache versioning - -5. **Better error recovery** (Low Priority) - - Graceful degradation - - Retry logic - - Better UX - ---- - -**Status**: Analysis complete. Ready for implementation. - diff --git a/NOTIFICATION_ISSUES_FIX.md b/NOTIFICATION_ISSUES_FIX.md deleted file mode 100644 index 6713ed09..00000000 --- a/NOTIFICATION_ISSUES_FIX.md +++ /dev/null @@ -1,194 +0,0 @@ -# Notification Issues - Analysis & Fixes - -**Date**: 2026-01-01 -**Issues Reported**: -1. Count shows 66 messages, but only 10 are displayed -2. "Mark all as read" fails -3. Count doesn't update after marking as read - ---- - -## 🔍 Issue Analysis - -### Issue 1: Count vs Display Discrepancy - -**Symptom**: -- Badge shows: **66 unread notifications** -- Dropdown shows: **Only 10 notifications** - -**Root Cause**: -1. **Count Logic**: `getNotificationCount()` calls `getNotifications(userId, 1, 100)` to count - - Gets first 100 notifications from Leantime - - Counts unread: 66 - - This is correct for the first 100 notifications - -2. **Display Logic**: `getNotifications()` is called with `limit: 20` (default) - - But only 10 are shown (possibly due to pagination or filtering) - - This is a display/pagination issue - -**The Problem**: -- If Leantime has more than 100 notifications total, the count will be inaccurate -- The count only reflects the first 100 notifications -- Display shows fewer notifications than the count - -**Solution**: -- ✅ Added warning log when count reaches 100 (may have more) -- ⚠️ Consider using a dedicated count API if Leantime provides one -- ⚠️ Consider fetching all notifications for accurate count (may be slow) - ---- - -### Issue 2: Mark All As Read Fails - -**Symptom**: -``` -[NOTIFICATION_API] Mark all as read - Failed { userId: '...', duration: '197ms' } -``` - -**Root Cause**: -- Leantime API call is failing -- No detailed error logging to see why - -**Solution Applied**: -- ✅ Added comprehensive error logging to `markAllAsRead()`: - - Logs user email and Leantime user ID - - Logs request body and API URL - - Logs response status and body - - Logs parsed response with error details - - Logs exceptions with stack traces - -**Next Steps**: -1. Test mark-all-as-read again -2. Check logs for detailed error information -3. Verify Leantime API method name is correct -4. Check if Leantime API requires different parameters - ---- - -## 🔧 Fixes Applied - -### 1. Enhanced Error Logging in `markAllAsRead` - -**File**: `lib/services/notifications/leantime-adapter.ts` - -**Changes**: -- Added detailed logging at each step -- Logs request details (body, URL) -- Logs response details (status, body, parsed data) -- Logs errors with full context -- Logs success/failure status - -**Expected Log Output**: -``` -[LEANTIME_ADAPTER] markAllAsRead called for ... -[LEANTIME_ADAPTER] markAllAsRead - User email: ... -[LEANTIME_ADAPTER] markAllAsRead - Leantime user ID: ... -[LEANTIME_ADAPTER] markAllAsRead - Request body: {...} -[LEANTIME_ADAPTER] markAllAsRead - API URL: ... -[LEANTIME_ADAPTER] markAllAsRead - Response status: 200 -[LEANTIME_ADAPTER] markAllAsRead - Response body: {...} -[LEANTIME_ADAPTER] markAllAsRead - Parsed response: {...} -[LEANTIME_ADAPTER] markAllAsRead - Success: true/false -``` - ---- - -### 2. Enhanced Count Logging - -**File**: `lib/services/notifications/leantime-adapter.ts` - -**Changes**: -- Added warning when count reaches 100 (may have more notifications) -- Added read count to logging -- Added note about potential inaccuracy - ---- - -## 🎯 Next Steps - -### Immediate Testing - -1. **Test Mark All As Read** - - Click "Mark all as read" - - Check logs for detailed error information - - Look for `[LEANTIME_ADAPTER] markAllAsRead` entries - -2. **Verify Count Accuracy** - - Check if Leantime has more than 100 notifications - - Verify count matches actual unread notifications - - Check if count updates after marking as read - -### Potential Issues to Check - -1. **Leantime API Method Name** - - Current: `leantime.rpc.Notifications.Notifications.markAllNotificationsAsRead` - - Verify this is the correct method name in Leantime API - -2. **Leantime API Parameters** - - Current: `{ userId: leantimeUserId }` - - May need additional parameters - -3. **Leantime API Response Format** - - Check if response format matches expected format - - May need to handle different response structures - ---- - -## 📊 Expected Behavior After Fixes - -### Mark All As Read - -**Success Case**: -``` -[NOTIFICATION_API] Mark all as read endpoint called -[NOTIFICATION_API] Mark all as read - Processing { userId: '...', timestamp: '...' } -[LEANTIME_ADAPTER] markAllAsRead called for ... -[LEANTIME_ADAPTER] markAllAsRead - Success: true -[NOTIFICATION_API] Mark all as read - Success { userId: '...', duration: 'Xms' } -[NOTIFICATION_SERVICE] Invalidated notification caches for user ... -``` - -**Failure Case** (with detailed error): -``` -[NOTIFICATION_API] Mark all as read endpoint called -[LEANTIME_ADAPTER] markAllAsRead called for ... -[LEANTIME_ADAPTER] markAllAsRead - Response status: 400 -[LEANTIME_ADAPTER] markAllAsRead - Response body: {"error": {...}} -[LEANTIME_ADAPTER] markAllAsRead - API Error: {...} -[NOTIFICATION_API] Mark all as read - Failed { userId: '...', duration: 'Xms' } -``` - ---- - -## 🔍 Debugging Checklist - -When testing, check logs for: - -- [ ] `[LEANTIME_ADAPTER] markAllAsRead - User email:` (should show email) -- [ ] `[LEANTIME_ADAPTER] markAllAsRead - Leantime user ID:` (should show ID) -- [ ] `[LEANTIME_ADAPTER] markAllAsRead - Request body:` (should show JSON-RPC request) -- [ ] `[LEANTIME_ADAPTER] markAllAsRead - Response status:` (should be 200 for success) -- [ ] `[LEANTIME_ADAPTER] markAllAsRead - Response body:` (should show API response) -- [ ] `[LEANTIME_ADAPTER] markAllAsRead - Parsed response:` (should show result/error) -- [ ] `[LEANTIME_ADAPTER] markAllAsRead - Success:` (should be true/false) - ---- - -## 📝 Summary - -**Fixes Applied**: -1. ✅ Enhanced error logging in `markAllAsRead` -2. ✅ Enhanced count logging with warnings - -**Next Actions**: -1. Test mark-all-as-read functionality -2. Review detailed error logs -3. Fix Leantime API call based on error details -4. Verify count accuracy - -**Status**: ⏳ **AWAITING TESTING** - Enhanced logging will reveal the root cause - ---- - -**Generated**: 2026-01-01 - diff --git a/NOTIFICATION_ISSUE_ANALYSIS.md b/NOTIFICATION_ISSUE_ANALYSIS.md deleted file mode 100644 index 8e1800cb..00000000 --- a/NOTIFICATION_ISSUE_ANALYSIS.md +++ /dev/null @@ -1,202 +0,0 @@ -# Notification Issue Analysis - Mark All Read Behavior - -**Date**: 2026-01-06 -**Issue**: Mark all read works initially, then connection issues occur - ---- - -## 🔍 **What's Happening** - -### **Initial Success**: -1. ✅ Dashboard shows 60 messages (count is working) -2. ✅ User clicks "Mark all read" -3. ✅ **First step works** - Marking operation starts successfully - -### **Then Connection Issues**: -``` -failed to get redirect response [TypeError: fetch failed] { - [cause]: [Error: read ECONNRESET] { - errno: -104, - code: 'ECONNRESET', - syscall: 'read' - } -} -Redis reconnect attempt 1, retrying in 100ms -Reconnecting to Redis.. -``` - ---- - -## 📊 **Analysis** - -### **What the Logs Show**: - -1. **IMAP Pool Activity**: - ``` - [IMAP POOL] Size: 1, Active: 1, Connecting: 0, Max: 20 - [IMAP POOL] Size: 0, Active: 0, Connecting: 0, Max: 20 - ``` - - IMAP connections are being used and released - - This is normal behavior - -2. **Connection Reset Error**: - - `ECONNRESET` - Connection was reset by peer - - Happens during a fetch request (likely to Leantime API) - - This is a **network/connection issue**, not a code issue - -3. **Redis Reconnection**: - - Redis is trying to reconnect (expected behavior) - - Our retry logic is working - ---- - -## 🎯 **Root Cause** - -### **Scenario**: -1. User clicks "Mark all read" -2. System starts marking notifications (works initially) -3. During the process, a network connection to Leantime API is reset -4. This could happen because: - - **Network instability** between your server and Leantime - - **Leantime API timeout** (if marking many notifications takes too long) - - **Connection pool exhaustion** (too many concurrent requests) - - **Server-side rate limiting** (Leantime might be throttling requests) - -### **Why It Works Initially Then Fails**: -- **First few notifications**: Marked successfully ✅ -- **After some time**: Connection resets ❌ -- **Result**: Partial success (some marked, some not) - ---- - -## 🔧 **What Our Fixes Handle** - -### **✅ What's Working**: -1. **User ID Caching**: Should prevent the "user not found" error -2. **Retry Logic**: Will retry failed requests automatically -3. **Cache Invalidation**: Always happens, so count will refresh -4. **Count Accuracy**: Fetches up to 1000 notifications - -### **⚠️ What's Not Handled**: -1. **Long-running operations**: Marking 60 notifications individually can take time -2. **Connection timeouts**: If Leantime API is slow or times out -3. **Rate limiting**: If Leantime throttles too many requests -4. **Partial failures**: Some notifications marked, some not - ---- - -## 💡 **What's Likely Happening** - -### **Flow**: -``` -1. User clicks "Mark all read" - ↓ -2. System fetches 60 unread notifications ✅ - ↓ -3. Starts marking each one individually - ↓ -4. First 10-20 succeed ✅ - ↓ -5. Connection resets (ECONNRESET) ❌ - ↓ -6. Remaining notifications fail to mark - ↓ -7. Cache is invalidated (our fix) ✅ - ↓ -8. Count refresh shows remaining unread (e.g., 40 instead of 0) -``` - -### **Why Count Might Not Be 0**: -- Some notifications were marked (e.g., 20 out of 60) -- Connection reset prevented marking the rest -- Cache was invalidated (good!) -- Count refresh shows remaining unread (40 unread) - ---- - -## 🎯 **Expected Behavior** - -### **With Our Fixes**: -1. ✅ User ID lookup is cached (faster, more reliable) -2. ✅ Retry logic handles transient failures -3. ✅ Cache always invalidated (count will refresh) -4. ✅ Count shows accurate number (up to 1000) - -### **What You Should See**: -- **First attempt**: Some notifications marked, count decreases (e.g., 60 → 40) -- **Second attempt**: More notifications marked, count decreases further (e.g., 40 → 20) -- **Eventually**: All marked, count reaches 0 - -### **If Connection Issues Persist**: -- Count will show remaining unread -- User can retry "Mark all read" -- Each retry will mark more notifications -- Eventually all will be marked - ---- - -## 🔍 **Diagnostic Questions** - -1. **How many notifications are marked?** - - Check if count decreases (e.g., 60 → 40 → 20 → 0) - - If it decreases, marking is working but incomplete - -2. **Does retry help?** - - Click "Mark all read" again - - If count decreases further, retry logic is working - -3. **Is it always the same number?** - - If count always stops at same number (e.g., always 40), might be specific notifications failing - - If count varies, it's likely connection issues - -4. **Network stability?** - - Check if connection to Leantime API is stable - - Monitor for timeouts or rate limiting - ---- - -## 📝 **Recommendations** - -### **Immediate**: -1. **Retry the operation**: Click "Mark all read" again - - Should mark more notifications - - Count should decrease further - -2. **Check logs for specific errors**: - - Look for which notification IDs are failing - - Check if it's always the same ones - -3. **Monitor network**: - - Check connection stability to Leantime - - Look for timeout patterns - -### **Future Improvements** (if needed): -1. **Batch marking**: Mark notifications in smaller batches (e.g., 10 at a time) -2. **Progress indicator**: Show "Marking X of Y..." to user -3. **Resume on failure**: Track which notifications were marked, resume from where it failed -4. **Connection pooling**: Better management of concurrent requests - ---- - -## ✅ **Summary** - -### **What's Working**: -- ✅ Initial marking starts successfully -- ✅ User ID caching prevents lookup failures -- ✅ Cache invalidation ensures count refreshes -- ✅ Retry logic handles transient failures - -### **What's Failing**: -- ⚠️ Connection resets during long operations -- ⚠️ Partial marking (some succeed, some fail) -- ⚠️ Network instability between server and Leantime - -### **Solution**: -- **Retry the operation**: Click "Mark all read" multiple times -- Each retry should mark more notifications -- Eventually all will be marked - ---- - -**Status**: This is expected behavior with network issues. The fixes ensure the system recovers and continues working. - diff --git a/README-MINIO-TROUBLESHOOTING.md b/README-MINIO-TROUBLESHOOTING.md deleted file mode 100644 index a21ad38b..00000000 --- a/README-MINIO-TROUBLESHOOTING.md +++ /dev/null @@ -1,115 +0,0 @@ -# Minio Troubleshooting Guide - -This document outlines the fixes implemented for the mission file upload issues with Minio. - -## Problem Description - -Mission uploads (logo and attachments) were not working correctly: -- Files weren't appearing in Minio despite upload attempts -- Mission logos weren't displaying even though they were uploaded -- Participation field showed "Non spécifié" despite values in the database -- SDG/ODD icons weren't displaying correctly - -## Implemented Fixes - -### 1. Added URL Generation Function - -Added a `getPublicUrl` function in `lib/s3.ts` that properly constructs URLs for files stored in Minio: -```typescript -export function getPublicUrl(filePath: string): string { - if (!filePath) return ''; - if (filePath.startsWith('http')) return filePath; // Already a full URL - - // Remove leading slash if present - const cleanPath = filePath.startsWith('/') ? filePath.substring(1) : filePath; - - // Construct the full URL - const endpoint = S3_CONFIG.endpoint?.replace(/\/$/, ''); // Remove trailing slash if present - const bucket = S3_CONFIG.bucket; - - // Return original path if no endpoint is configured - if (!endpoint) return cleanPath; - - // Construct and return the full URL - return `${endpoint}/${bucket}/${cleanPath}`; -} -``` - -### 2. Updated Mission Display Page - -Modified `app/missions/page.tsx` to use the `getPublicUrl` function when displaying mission logos: -```tsx -{mission.logo ? ( - {mission.name} { - // Error handling... - }} - /> -) : null} -``` - -### 3. Enhanced Upload API - -Updated `/app/api/missions/upload/route.ts` to: -- Include additional logging -- Generate and return proper public URLs -- Improve error handling - -### 4. Enhanced Mission Detail API - -Modified `/app/api/missions/[missionId]/route.ts` to include public URLs in the response: -```typescript -const missionWithUrls = { - ...mission, - logoUrl: mission.logo ? getPublicUrl(mission.logo) : null, - attachments: mission.attachments.map((attachment) => ({ - ...attachment, - publicUrl: getPublicUrl(attachment.filePath) - })) -}; -``` - -### 5. Added Testing Tools - -1. Browser Console Utilities: - - `window.testMinioConnection()` - Test Minio connectivity - - `window.getMinioUrl(path)` - Generate a public URL for debugging - -2. Server-side Test Script: - - Created `scripts/test-minio-upload.js` to test uploads from the command line - - Tests uploading, downloading, and URL generation - -## How to Test - -1. **Using the browser console:** - ```javascript - // Test connection and list files - window.testMinioConnection() - - // Generate URL for a specific path - window.getMinioUrl('user-123/missions/456/logo.jpg') - ``` - -2. **Using the server-side script:** - ```bash - node scripts/test-minio-upload.js - ``` - -## Required Environment Variables - -Make sure these are properly set in your environment: -- `MINIO_S3_UPLOAD_BUCKET_URL` - The Minio endpoint URL -- `MINIO_AWS_REGION` - The AWS region (often 'us-east-1' for Minio) -- `MINIO_AWS_S3_UPLOAD_BUCKET_NAME` - The bucket name -- `MINIO_ACCESS_KEY` - Access key for Minio -- `MINIO_SECRET_KEY` - Secret key for Minio - -## Additional Notes - -1. The same Minio bucket is used for both Pages and Missions. -2. Pages functionality is working properly, suggesting the Minio configuration itself is correct. -3. Make sure that the bucket has proper permissions for public read access. -4. The URL paths for SDG/ODD icons were corrected to use `/F SDG Icons 2019 WEB/F-WEB-Goal-XX.png` \ No newline at end of file diff --git a/README.md b/README.md deleted file mode 100644 index 550c914d..00000000 --- a/README.md +++ /dev/null @@ -1,108 +0,0 @@ -# Neah Email Application - -A modern email client built with Next.js, featuring email composition, viewing, and management capabilities. - -## Email Processing Workflow - -The application handles email processing through a centralized workflow: - -1. **Email Fetching**: Emails are fetched through the `/api/courrier` endpoints using user credentials stored in the database. - -2. **Email Parsing**: Raw email content is parsed using: - - Server-side: `parseEmail` function from `lib/server/email-parser.ts` (which uses `simpleParser` from the `mailparser` library) - - API route: `/api/parse-email` provides a REST interface to the parser - -3. **HTML Sanitization**: Email HTML content is sanitized and processed using: - - `sanitizeHtml` function in `lib/utils/email-utils.ts` (centralized implementation) - - DOMPurify with specific configuration to handle email content safely - -4. **Email Display**: Sanitized content is rendered in the UI with proper styling and security measures - -5. **Email Composition**: The `ComposeEmail` component handles email creation, replying, and forwarding - - Email is sent through the `/api/courrier/send` endpoint - -## Key Features - -- **Email Fetching and Management**: Connect to IMAP servers and manage email fetching and caching logic -- **Email Composition**: Rich text editor with reply and forwarding capabilities -- **Email Display**: Secure rendering of HTML emails -- **Attachment Handling**: View and download attachments - -## Project Structure - -The project follows a modular structure: - -- `/app` - Next.js App Router structure with routes and API endpoints -- `/components` - React components organized by domain -- `/lib` - Core library code: - - `/server` - Server-only code like email parsing - - `/services` - Domain-specific services, including email service - - `/reducers` - State management logic - - `/utils` - Utility functions including the centralized email formatter - -## Technologies - -- Next.js 14+ with App Router -- React Server Components -- TailwindCSS for styling -- Mailparser for email parsing -- ImapFlow for email fetching -- DOMPurify for HTML sanitization -- Redis for caching - -## State Management - -Email state is managed through React context and reducers, with server data fetched through React Server Components or client-side API calls as needed. - -# Email Formatting - -## Centralized Email Formatter - -All email formatting is now handled by a centralized formatter in `lib/utils/email-utils.ts`. This ensures consistent handling of: - -- Reply and forward formatting -- HTML sanitization -- RTL/LTR text direction -- MIME encoding and decoding for email composition - -Key functions include: -- `formatForwardedEmail`: Format emails for forwarding -- `formatReplyEmail`: Format emails for replying -- `sanitizeHtml`: Safely sanitize HTML email content -- `formatEmailForReplyOrForward`: Compatibility function for both -- `decodeComposeContent`: Parse MIME content for email composition -- `encodeComposeContent`: Create MIME-formatted content for sending emails - -This centralized approach prevents formatting inconsistencies and direction problems when dealing with emails in different languages. - -## Deprecated Functions - -Several functions have been deprecated and removed in favor of centralized implementations: - -- Check the `DEPRECATED_FUNCTIONS.md` file for a complete list of deprecated functions and their replacements. - -## User Management API - -The application provides endpoints for managing users in multiple systems: - -- **Create User**: - - Endpoint: `POST /api/users` - - Creates users in Keycloak, Leantime, and Dolibarr (if they have "mediation" or "expression" roles) - -- **Update User**: - - Endpoint: `PUT /api/users/[userId]` - - Updates user details in Keycloak - -- **Delete User**: - - Endpoint: `DELETE /api/users?id=[userId]&email=[userEmail]` - - Deletes users from Keycloak, Leantime, and Dolibarr systems - - **Important**: Always include both `id` and `email` parameters for complete deletion across all systems - - The legacy endpoint `DELETE /api/users/[userId]` forwards to the above endpoint - -- **Manage Roles**: - - Endpoint: `PUT /api/users/[userId]/roles` - - Updates user roles in Keycloak - -- **Reset Password**: - - Endpoint: `PUT /api/users/[userId]/password` - - Resets user password in Keycloak \ No newline at end of file diff --git a/SEPARATED_AUTHENTICATION_FLOWS_EXPLANATION.md b/SEPARATED_AUTHENTICATION_FLOWS_EXPLANATION.md deleted file mode 100644 index 5e9111bc..00000000 --- a/SEPARATED_AUTHENTICATION_FLOWS_EXPLANATION.md +++ /dev/null @@ -1,366 +0,0 @@ -# Why Dashboard and Applications Have Separated Authentication Flows - -## Executive Summary - -The dashboard and applications use **two completely separate authentication mechanisms** that operate independently: - -1. **Dashboard**: Uses **NextAuth.js** with JWT-based sessions (30 days) -2. **Applications**: Use **Keycloak SSO** directly via browser cookies - -This separation is why logging out from the dashboard doesn't automatically log you out from applications opened directly in the browser. - ---- - -## Architecture Overview - -### Two Independent Authentication Systems - -``` -┌─────────────────────────────────────────────────────────────┐ -│ AUTHENTICATION LAYERS │ -├─────────────────────────────────────────────────────────────┤ -│ │ -│ ┌──────────────────────┐ ┌──────────────────────┐ │ -│ │ DASHBOARD AUTH │ │ APPLICATION AUTH │ │ -│ │ │ │ │ │ -│ │ NextAuth.js │ │ Keycloak SSO │ │ -│ │ (JWT Strategy) │ │ (Cookie-based) │ │ -│ │ │ │ │ │ -│ │ - Session: 30 days │ │ - Session: Variable │ │ -│ │ - Stored in: Cookie │ │ - Stored in: Cookie │ │ -│ │ - Domain: Dashboard │ │ - Domain: Keycloak │ │ -│ │ - Independent │ │ - Independent │ │ -│ └──────────────────────┘ └──────────────────────┘ │ -│ │ │ │ -│ └──────────┬───────────────────┘ │ -│ │ │ -│ ┌───────▼────────┐ │ -│ │ KEYCLOAK │ │ -│ │ (IdP Server) │ │ -│ └────────────────┘ │ -│ │ -└─────────────────────────────────────────────────────────────┘ -``` - ---- - -## Why They're Separated - -### 1. Different Authentication Purposes - -**Dashboard Authentication (NextAuth.js)**: -- Purpose: Authenticate the **Next.js dashboard application** -- Method: OAuth 2.0 flow → Get tokens → Store in JWT -- Session Management: NextAuth manages its own session lifecycle -- Storage: Encrypted JWT in HTTP-only cookie on dashboard domain -- Duration: 30 days (configurable in `app/api/auth/options.ts`) - -**Application Authentication (Keycloak SSO)**: -- Purpose: Authenticate **standalone applications** (not embedded in dashboard) -- Method: Direct Keycloak authentication via browser cookies -- Session Management: Keycloak manages SSO session lifecycle -- Storage: Keycloak session cookies on Keycloak domain -- Duration: Configured in Keycloak (typically 30 minutes to a few hours) - -### 2. Different Session Storage Locations - -**Dashboard Session**: -``` -Cookie Name: next-auth.session-token -Domain: dashboard.example.com -Path: / -HttpOnly: Yes -Secure: Yes (if HTTPS) -SameSite: Lax -Content: Encrypted JWT containing: - - accessToken (Keycloak OAuth token) - - refreshToken (Keycloak refresh token) - - idToken (Keycloak ID token) - - User info (id, email, roles, etc.) -``` - -**Application Session**: -``` -Cookie Name: KEYCLOAK_SESSION -Domain: keycloak.example.com (or configured domain) -Path: / -HttpOnly: Yes -Secure: Yes -SameSite: Lax or None (for cross-site) -Content: Keycloak session identifier -``` - -### 3. Different Authentication Flows - -**Dashboard Flow**: -``` -1. User visits dashboard → /signin -2. NextAuth redirects to Keycloak OAuth endpoint -3. Keycloak authenticates user -4. Keycloak redirects back with authorization code -5. NextAuth exchanges code for tokens -6. NextAuth creates JWT session -7. JWT stored in dashboard cookie -8. Dashboard uses JWT for authentication -``` - -**Application Flow** (when opened directly): -``` -1. User visits application directly (not via dashboard) -2. Application checks for Keycloak session cookie -3. If cookie exists → User is authenticated (SSO) -4. If cookie doesn't exist → Redirect to Keycloak login -5. Keycloak authenticates user -6. Keycloak sets session cookie -7. Application uses cookie for authentication -``` - ---- - -## Why Dashboard Logout Doesn't Log Out Applications - -### The Problem - -When you log out from the dashboard: - -1. **Dashboard logout process**: - - Clears NextAuth session cookie (`next-auth.session-token`) - - Calls Keycloak logout endpoint with `id_token_hint` - - Keycloak clears **client session** for dashboard OAuth client - - Keycloak may clear SSO session (if it's the last client session) - -2. **What happens to applications**: - - Applications don't know about dashboard logout - - Applications still have Keycloak SSO session cookie - - Applications continue to work because they use Keycloak cookies, not NextAuth - -### Technical Reasons - -#### Reason 1: Different Cookie Domains - -**Dashboard Cookie**: -- Domain: `dashboard.example.com` -- Cleared when dashboard logs out -- Applications can't access this cookie (different domain) - -**Keycloak SSO Cookie**: -- Domain: `keycloak.example.com` (or configured domain) -- Not cleared by dashboard logout (unless SSO session is cleared) -- Applications can access this cookie (same domain as Keycloak) - -#### Reason 2: Independent Session Lifecycles - -**NextAuth Session**: -- Managed by NextAuth.js -- Lifecycle: Created on login → Valid for 30 days → Cleared on logout -- Independent of Keycloak SSO session - -**Keycloak SSO Session**: -- Managed by Keycloak server -- Lifecycle: Created on login → Valid until timeout or explicit logout → Cleared on logout -- Independent of NextAuth session - -#### Reason 3: Different Authentication Mechanisms - -**Dashboard**: -- Uses OAuth 2.0 tokens (access token, refresh token) -- Tokens stored in NextAuth JWT -- Authentication: Validate JWT → Extract tokens → Use tokens for API calls - -**Applications**: -- Use Keycloak session cookies directly -- No OAuth tokens involved -- Authentication: Check for Keycloak session cookie → If exists, user is authenticated - -#### Reason 4: Keycloak SSO Session Persistence - -**Keycloak maintains two types of sessions**: - -1. **Client Session** (per OAuth client): - - Specific to each OAuth client (dashboard, app1, app2, etc.) - - Cleared when that specific client logs out - - Dashboard logout clears dashboard's client session - -2. **SSO Session** (realm-wide): - - Shared across all clients in the realm - - Persists even after individual client logouts - - Only cleared when: - - All client sessions are logged out - - Explicit SSO session logout - - Session timeout - - Admin API logout - -**When dashboard logs out**: -- Dashboard's client session is cleared ✅ -- SSO session may persist if other applications have active sessions ❌ -- Applications continue to work because SSO session is still valid ❌ - ---- - -## Current Logout Flow Analysis - -### What Happens When You Log Out from Dashboard - -``` -Step 1: User clicks logout in dashboard - ↓ -Step 2: Dashboard calls NextAuth signOut() - → Clears: next-auth.session-token cookie - → Clears: Dashboard's NextAuth session - ↓ -Step 3: Dashboard calls /api/auth/end-sso-session - → Uses Keycloak Admin API - → Calls: adminClient.users.logout({ id: userId }) - → Clears: All client sessions for user - → May clear: SSO session (if it's the last client session) - ↓ -Step 4: Dashboard redirects to Keycloak logout endpoint - → URL: ${KEYCLOAK_ISSUER}/protocol/openid-connect/logout - → Parameters: id_token_hint, post_logout_redirect_uri - → Clears: Dashboard's client session - → May clear: SSO session (if it's the last client session) - ↓ -Step 5: Keycloak redirects back to /signin?logout=true - → Dashboard shows logout message -``` - -### What Happens to Applications - -``` -Applications opened directly in browser: - ↓ -Step 1: Application checks for Keycloak session cookie - → Cookie: KEYCLOAK_SESSION - → Domain: keycloak.example.com - ↓ -Step 2: If SSO session still exists: - → Application finds valid SSO session cookie ✅ - → Application authenticates user automatically ✅ - → User remains logged in ❌ - ↓ -Step 3: If SSO session was cleared: - → Application doesn't find session cookie ✅ - → Application redirects to Keycloak login ✅ - → User must log in again ✅ -``` - -### Why Applications Stay Logged In - -**Scenario 1: SSO Session Persists** -- Dashboard logout clears client sessions -- But SSO session cookie still exists -- Applications check SSO session cookie → Still valid → User stays logged in - -**Scenario 2: Other Applications Have Active Sessions** -- If other applications are open in other tabs/windows -- They have active client sessions -- Keycloak won't clear SSO session (because other clients are still active) -- All applications stay logged in - -**Scenario 3: Cookie Domain Mismatch** -- Dashboard tries to clear Keycloak cookies client-side -- But cookies are on different domain (keycloak.example.com) -- Browser security prevents clearing cross-domain cookies -- Applications keep their cookies → Stay logged in - ---- - -## Why This Architecture Exists - -### Historical/Design Reasons - -1. **Legacy Applications**: - - Applications may have existed before the dashboard - - They were designed to use Keycloak directly - - Dashboard was added later as a wrapper/portal - -2. **Separation of Concerns**: - - Dashboard: Portal/aggregator (doesn't need to know about app internals) - - Applications: Standalone services (don't depend on dashboard) - -3. **Flexibility**: - - Applications can be accessed directly (not just via dashboard) - - Applications can be used independently - - Dashboard is optional, not required - -4. **SSO Design**: - - Keycloak SSO is designed to work across multiple applications - - Logging out from one app shouldn't log out from all apps - - This is by design for SSO functionality - -### Technical Constraints - -1. **Cookie Security**: - - Browsers prevent cross-domain cookie access - - Dashboard can't directly clear Keycloak cookies (different domain) - - Must use Keycloak logout endpoint or Admin API - -2. **Stateless vs Stateful**: - - NextAuth: Stateless (JWT, no server-side session) - - Keycloak: Stateful (server-side session, cookies) - -3. **OAuth vs Direct Authentication**: - - Dashboard: Uses OAuth 2.0 (tokens) - - Applications: Use direct Keycloak authentication (cookies) - ---- - -## What Would Be Needed for Unified Logout - -To make dashboard logout also log out applications, you would need: - -### Option 1: Keycloak Front-Channel Logout (Recommended) -- Configure all applications to participate in Front-Channel Logout -- When dashboard logs out, Keycloak notifies all registered applications -- Applications receive logout notification and clear their sessions -- **Requires**: Keycloak configuration + Application support - -### Option 2: Keycloak Single Logout (SLO) -- Configure all applications to participate in SLO -- When one application logs out, all applications are logged out -- **Requires**: Keycloak configuration + Application support - -### Option 3: Clear SSO Session Explicitly -- Use Keycloak Admin API to end SSO session -- This clears the realm-wide SSO session -- All applications lose their authentication -- **Current Implementation**: Partially implemented (`/api/auth/end-sso-session`) -- **Issue**: May not clear SSO session cookie if other clients are active - -### Option 4: Application Logout Endpoints -- Each application exposes a logout endpoint -- Dashboard calls all application logout endpoints -- Applications clear their own sessions -- **Requires**: Application modifications + Dashboard coordination - ---- - -## Summary - -### Why They're Separated - -1. **Different purposes**: Dashboard is a portal, applications are standalone services -2. **Different storage**: Dashboard uses NextAuth JWT, applications use Keycloak cookies -3. **Different domains**: Cookies are on different domains (security prevents cross-domain access) -4. **Different lifecycles**: NextAuth session (30 days) vs Keycloak SSO session (variable) -5. **SSO design**: Keycloak SSO is designed to persist across client logouts - -### Why Dashboard Logout Doesn't Log Out Applications - -1. **SSO session persists**: Keycloak SSO session may not be cleared -2. **Other active sessions**: If other applications are open, SSO session stays active -3. **Cookie domain**: Dashboard can't directly clear Keycloak cookies (different domain) -4. **Independent mechanisms**: Applications don't know about NextAuth session state - -### The Solution - -To achieve unified logout, you need to: -- Configure Keycloak Front-Channel Logout or SLO -- Ensure all applications participate in logout notifications -- Or use Admin API to explicitly end SSO session (current implementation attempts this) - -The current implementation (`/api/auth/end-sso-session`) tries to clear the SSO session, but it may not work if: -- Other applications have active sessions -- SSO session cookie is on a different domain -- Keycloak configuration prevents SSO session clearing - diff --git a/SESSION_CALLBACK_LOGGING_IMPACT_ANALYSIS.md b/SESSION_CALLBACK_LOGGING_IMPACT_ANALYSIS.md deleted file mode 100644 index a6a76ba9..00000000 --- a/SESSION_CALLBACK_LOGGING_IMPACT_ANALYSIS.md +++ /dev/null @@ -1,335 +0,0 @@ -# Session Callback Logging - Impact Analysis - -**Date**: 2026-01-01 -**Purpose**: Analyze the impact of reducing session callback logging on the multi-stack architecture - ---- - -## 🏗️ Architecture Overview - -### Stack Components -1. **Next.js Dashboard** (this application) -2. **Keycloak** (SSO/Authentication provider) -3. **MinIO** (Object storage for files) -4. **External Services** (Leantime, Rocket.Chat, News API, etc.) - -### Integration Points -- **Keycloak**: OAuth2/OIDC provider, session tokens, role extraction -- **MinIO**: File storage (mission logos, attachments), S3-compatible API -- **External APIs**: All require authenticated session - ---- - -## 📋 Current Session Callback Logging - -### What's Being Logged -```typescript -// Lines 407-472 in app/api/auth/options.ts -- === SESSION CALLBACK START === -- Token error status -- Access token presence -- Refresh token presence -- Token roles -- Token sub (user ID) -- Token email -- Token name -- Token username -- User roles for session -- Creating session user object -- Setting session tokens -- ✅ Session created successfully -- Session user details -- === SESSION CALLBACK END === -``` - -### Why It Was Added -**Historical Context** (from `DEBUG_502_CALLBACK.md`): -- Added specifically to debug **502 errors** with Keycloak callbacks -- Critical for diagnosing authentication failures -- Helps identify when session callback doesn't execute -- Essential for troubleshooting SSO flow issues - ---- - -## 🔍 Impact Analysis - -### 1. Keycloak Integration Impact - -**Dependencies**: -- ✅ **No functional impact**: Logging doesn't affect Keycloak authentication -- ⚠️ **Debugging impact**: Removing logs makes troubleshooting harder -- ✅ **Error logging preserved**: Critical errors still logged - -**Keycloak Flow**: -``` -1. User authenticates → Keycloak -2. Keycloak redirects → Next.js callback -3. JWT callback extracts tokens -4. Session callback builds session ← LOGGING HERE -5. Session used for all API calls -``` - -**Recommendation**: -- Keep error logging (always) -- Make success logging conditional (DEBUG_SESSION flag) - ---- - -### 2. MinIO Integration Impact - -**Dependencies**: -- ✅ **No direct dependency**: MinIO doesn't use session callback logs -- ✅ **Uses session for auth**: Session object used to verify user permissions -- ✅ **No impact**: Logging changes won't affect MinIO operations - -**MinIO Flow**: -``` -1. API route calls getServerSession() -2. Session callback executes (builds session) -3. Session used to verify user authentication -4. MinIO operations proceed with authenticated user -``` - -**Recommendation**: -- ✅ **Safe to reduce logging**: No impact on MinIO functionality - ---- - -### 3. External Services Impact - -**Services**: -- Leantime (project management) -- Rocket.Chat (messaging) -- News API -- Email/IMAP - -**Dependencies**: -- ✅ **No functional impact**: Services don't read logs -- ✅ **Session still created**: Logging doesn't affect session creation -- ✅ **Authentication works**: Session object still valid - -**Recommendation**: -- ✅ **Safe to reduce logging**: No impact on external services - ---- - -### 4. Monitoring & Debugging Impact - -**Current Usage**: -- Debugging 502 errors (Keycloak callbacks) -- Troubleshooting authentication issues -- Monitoring session creation frequency -- Identifying session callback failures - -**Impact of Reducing Logging**: -- ⚠️ **Harder to debug**: Less visibility into session creation -- ✅ **Still debuggable**: Error logging preserved -- ✅ **Can enable on-demand**: DEBUG_SESSION flag for troubleshooting - -**Recommendation**: -- Use conditional logging with DEBUG_SESSION flag -- Keep error logging always enabled -- Document how to enable debug logging - ---- - -## ✅ Safe Implementation Strategy - -### Phase 1: Conditional Logging (Recommended) - -**Approach**: Make success logging conditional, keep error logging always - -```typescript -async session({ session, token }) { - try { - // Always log errors - if (token.error) { - console.error("❌ Session callback error:", token.error); - } - - // Conditional verbose logging - const DEBUG_SESSION = process.env.DEBUG_SESSION === 'true' || - process.env.NODE_ENV === 'development'; - - if (DEBUG_SESSION) { - console.log('=== SESSION CALLBACK START ==='); - console.log('Token error:', token.error); - console.log('Has accessToken:', !!token.accessToken); - // ... rest of verbose logging - } - - // Always log critical errors - if (token.error === "SessionNotActive" || - token.error === "NoRefreshToken" || - !token.accessToken || - !token.refreshToken) { - console.log("❌ Session invalidated or tokens missing", { - error: token.error, - hasAccessToken: !!token.accessToken, - hasRefreshToken: !!token.refreshToken - }); - return null as any; - } - - // ... rest of callback logic - - if (DEBUG_SESSION) { - console.log('✅ Session created successfully'); - console.log('Session user id:', session.user.id); - console.log('=== SESSION CALLBACK END ==='); - } - - return session; - } catch (error) { - // Always log critical errors - console.error('❌❌❌ CRITICAL ERROR IN SESSION CALLBACK ❌❌❌'); - console.error('Error:', error); - throw error; - } -} -``` - -**Benefits**: -- ✅ Production: Minimal logging (errors only) -- ✅ Development: Full logging for debugging -- ✅ On-demand: Enable with DEBUG_SESSION=true -- ✅ No functional impact - ---- - -### Phase 2: Environment-Based Logging - -**Alternative**: Use NODE_ENV - -```typescript -const isDevelopment = process.env.NODE_ENV === 'development'; - -if (isDevelopment || token.error) { - // Verbose logging -} -``` - -**Benefits**: -- ✅ Simple implementation -- ✅ Automatic in development -- ⚠️ Less flexible than DEBUG_SESSION flag - ---- - -## 🎯 Recommended Approach - -### Option 1: DEBUG_SESSION Flag (Best) - -**Implementation**: -- Add `DEBUG_SESSION` environment variable -- Default: `false` (minimal logging) -- Set to `true` when debugging needed - -**Usage**: -```bash -# Production (minimal logging) -DEBUG_SESSION=false npm start - -# Debugging (verbose logging) -DEBUG_SESSION=true npm start -``` - -**Pros**: -- ✅ Flexible (can enable on-demand) -- ✅ Production-friendly (minimal logs) -- ✅ Debug-friendly (full logs when needed) -- ✅ No code changes needed to toggle - -**Cons**: -- ⚠️ Requires environment variable management - ---- - -### Option 2: NODE_ENV Based (Simpler) - -**Implementation**: -- Use `NODE_ENV === 'development'` for verbose logging -- Always log errors - -**Pros**: -- ✅ Simple (no new env vars) -- ✅ Automatic (works with existing setup) - -**Cons**: -- ⚠️ Less flexible (can't enable in production easily) - ---- - -## 📊 Risk Assessment - -| Risk | Impact | Mitigation | -|------|--------|------------| -| **Lost debugging capability** | Medium | Keep error logging, add DEBUG_SESSION flag | -| **Harder to troubleshoot 502 errors** | Medium | Document how to enable debug logging | -| **Performance impact** | Low | Logging overhead is minimal | -| **Functional impact** | None | Logging doesn't affect functionality | - ---- - -## ✅ Final Recommendation - -### Implementation Plan - -1. **Keep Error Logging Always** ✅ - - Critical errors always logged - - Session invalidation always logged - - Exception handling always logged - -2. **Make Success Logging Conditional** ✅ - - Use `DEBUG_SESSION` environment variable - - Default: `false` (production-friendly) - - Can enable: `DEBUG_SESSION=true` (debugging) - -3. **Document Debugging Process** ✅ - - Add to README or troubleshooting guide - - Explain when to enable DEBUG_SESSION - - Document what logs to look for - -4. **Test in Staging** ✅ - - Verify error logging still works - - Test with DEBUG_SESSION=true - - Test with DEBUG_SESSION=false - ---- - -## 🔧 Implementation Checklist - -- [ ] Update `app/api/auth/options.ts` with conditional logging -- [ ] Add `DEBUG_SESSION` to environment variable documentation -- [ ] Test error logging (should always work) -- [ ] Test success logging with DEBUG_SESSION=true -- [ ] Test success logging with DEBUG_SESSION=false -- [ ] Verify Keycloak authentication still works -- [ ] Verify MinIO operations still work -- [ ] Verify external services still work -- [ ] Update troubleshooting documentation - ---- - -## 📝 Summary - -**Impact Level**: 🟢 **LOW RISK** - -**Key Findings**: -1. ✅ No functional impact on Keycloak, MinIO, or external services -2. ✅ Logging was added for debugging, not functionality -3. ✅ Error logging preserved (critical for troubleshooting) -4. ✅ Conditional logging provides flexibility - -**Recommendation**: -- ✅ **Proceed with conditional logging** -- ✅ **Use DEBUG_SESSION flag for flexibility** -- ✅ **Keep error logging always enabled** - -**Confidence**: 🟢 **HIGH** - Safe to implement - ---- - -**Generated**: 2026-01-01 -**Next Step**: Implement conditional logging in `app/api/auth/options.ts` - diff --git a/SESSION_DURATION_SECURITY_ANALYSIS.md b/SESSION_DURATION_SECURITY_ANALYSIS.md deleted file mode 100644 index f6636bbe..00000000 --- a/SESSION_DURATION_SECURITY_ANALYSIS.md +++ /dev/null @@ -1,233 +0,0 @@ -# NextAuth Session Duration: 30 Days vs 4 Hours - Security Analysis - -## Current Configuration - -**Current Setting** (`app/api/auth/options.ts:190`): -```typescript -session: { - strategy: "jwt", - maxAge: 30 * 24 * 60 * 60, // 30 days (2,592,000 seconds) -} -``` - -**Proposed Setting**: -```typescript -session: { - strategy: "jwt", - maxAge: 4 * 60 * 60, // 4 hours (14,400 seconds) -} -``` - ---- - -## Security Analysis - -### ✅ **Why 4 Hours is Better for Security** - -1. **Reduced Attack Window**: - - **30 days**: If session is compromised, attacker has 30 days of access - - **4 hours**: If session is compromised, attacker has maximum 4 hours of access - - **Risk Reduction**: 99.4% reduction in maximum exposure time - -2. **Industry Best Practices**: - - **NIST Guidelines**: Recommend session timeouts of 2-8 hours for high-security applications - - **OWASP**: Recommends session timeouts based on risk level (typically 2-8 hours) - - **Common Practice**: Most enterprise applications use 4-8 hour sessions - -3. **Device Security**: - - **30 days**: Device left unattended = 30 days of potential unauthorized access - - **4 hours**: Device left unattended = maximum 4 hours of potential access - - **Better for**: Shared devices, public computers, unattended workstations - -4. **Compliance**: - - Many security standards (ISO 27001, SOC 2) require reasonable session timeouts - - 30 days is often considered too long for compliance - - 4 hours aligns better with security compliance requirements - -5. **Stolen Session Cookie**: - - If session cookie is stolen (XSS, MITM), shorter duration limits damage - - 4 hours gives attacker limited time to exploit - - 30 days gives attacker extensive time to exploit - -### ⚠️ **Considerations & Trade-offs** - -1. **User Experience Impact**: - - **30 days**: Users rarely need to re-authenticate (convenient) - - **4 hours**: Users need to re-authenticate every 4 hours (less convenient) - - **Impact**: Moderate - users will need to log in more frequently - -2. **Token Refresh Behavior**: - - **Good News**: Your code already handles token refresh automatically - - **How it works**: - - When NextAuth session expires (4 hours), JWT callback runs - - If `accessToken` is expired, it calls `refreshAccessToken()` - - Uses `refreshToken` to get new tokens from Keycloak - - Session is automatically renewed (if refresh token is still valid) - - **Result**: Users may not notice the 4-hour expiration if they're active - -3. **Keycloak Refresh Token Lifetime**: - - **Important**: Keycloak refresh tokens typically last 7-30 days - - **What this means**: - - NextAuth session expires after 4 hours - - But refresh token is still valid (e.g., 7 days) - - NextAuth automatically refreshes tokens - - User stays logged in seamlessly (if active) - - **Only expires if**: User is inactive for longer than refresh token lifetime - -4. **Keycloak Session Alignment**: - - **Current Issue**: Keycloak sessions typically expire in 30 minutes to a few hours - - **With 4-hour NextAuth session**: - - Better alignment with Keycloak session timeouts - - Reduces session mismatch issues - - Iframe applications will have more consistent session state - ---- - -## How It Will Work - -### Session Lifecycle with 4-Hour maxAge - -``` -User logs in - ↓ -NextAuth creates JWT session (expires in 4 hours) - ↓ -User is active for 2 hours - ↓ -User makes request → NextAuth checks session - ↓ -Session still valid (< 4 hours) → Continue - ↓ -User is active for 3 hours - ↓ -User makes request → NextAuth checks session - ↓ -Session still valid (< 4 hours) → Continue - ↓ -User is active for 4.5 hours (session expired) - ↓ -User makes request → NextAuth checks session - ↓ -Session expired → JWT callback runs - ↓ -Checks accessToken expiration - ↓ -If accessToken expired → Calls refreshAccessToken() - ↓ -Uses refreshToken to get new tokens from Keycloak - ↓ -If refreshToken still valid → New session created (another 4 hours) - ↓ -User continues seamlessly (no re-authentication needed) - ↓ -If refreshToken expired → User must re-authenticate -``` - -### When User Must Re-authenticate - -**User must re-authenticate if**: -1. **Inactive for longer than refresh token lifetime** (typically 7-30 days) -2. **Refresh token is revoked** (logout, admin action, security event) -3. **Keycloak session is invalidated** (logout from another application) - -**User does NOT need to re-authenticate if**: -1. **Active within refresh token lifetime** (automatic token refresh) -2. **Session expires but refresh token is valid** (automatic renewal) - ---- - -## Recommendations - -### ✅ **Recommendation: Implement 4-Hour Session** - -**Reasons**: -1. ✅ **Significantly better security** (99.4% reduction in exposure window) -2. ✅ **Aligns with industry best practices** (NIST, OWASP) -3. ✅ **Better compliance** (meets security standards) -4. ✅ **Better alignment with Keycloak sessions** -5. ✅ **Minimal UX impact** (automatic token refresh handles renewal) -6. ✅ **Code already supports it** (token refresh mechanism exists) - -### ⚠️ **Important Considerations** - -1. **Verify Keycloak Refresh Token Lifetime**: - - Check Keycloak configuration for refresh token lifetime - - Ensure it's longer than 4 hours (typically 7-30 days) - - If shorter, users will need to re-authenticate frequently - -2. **Monitor User Experience**: - - Track how often users need to re-authenticate - - If too frequent, consider increasing to 6-8 hours - - Balance security with usability - -3. **Consider Activity-Based Extension**: - - Current implementation: Fixed 4-hour expiration - - Alternative: Extend session on activity (sliding window) - - Requires additional implementation (activity tracking) - -4. **Keycloak Session Configuration**: - - Consider aligning Keycloak SSO session timeout with NextAuth - - Or ensure Keycloak session is longer than NextAuth session - - Prevents session mismatch issues - -### 📋 **Implementation Checklist** - -Before implementing: - -- [ ] Verify Keycloak refresh token lifetime (should be > 4 hours) -- [ ] Test token refresh flow with 4-hour session -- [ ] Monitor user re-authentication frequency -- [ ] Consider user feedback on session duration -- [ ] Document the change for users (if needed) -- [ ] Update security documentation - ---- - -## Comparison Table - -| Aspect | 30 Days | 4 Hours | Winner | -|--------|---------|---------|--------| -| **Security** | Low (long exposure window) | High (short exposure window) | ✅ 4 Hours | -| **User Convenience** | High (rare re-authentication) | Medium (automatic refresh) | ✅ 30 Days | -| **Compliance** | Poor (too long) | Good (meets standards) | ✅ 4 Hours | -| **Risk Reduction** | Low | High (99.4% reduction) | ✅ 4 Hours | -| **Keycloak Alignment** | Poor (mismatch) | Good (better alignment) | ✅ 4 Hours | -| **Token Refresh** | Works | Works (same mechanism) | ✅ Tie | - ---- - -## Conclusion - -**Recommendation: Change to 4 hours** - -**Why**: -- Significantly better security posture -- Aligns with industry best practices -- Better compliance with security standards -- Minimal UX impact (automatic token refresh) -- Better alignment with Keycloak session timeouts -- Code already supports it - -**Implementation**: -- Simple change: `maxAge: 4 * 60 * 60` -- No code changes needed (token refresh already works) -- Monitor user experience and adjust if needed - -**Alternative Consideration**: -- If 4 hours is too aggressive, consider 6-8 hours as a middle ground -- Still provides significant security improvement over 30 days -- Better user experience than 4 hours - ---- - -## Final Verdict - -**✅ Yes, change to 4 hours** - This is a good security practice that: -- Significantly reduces security risk -- Aligns with industry standards -- Has minimal UX impact (automatic refresh) -- Works with existing code -- Better aligns with Keycloak sessions - -The only trade-off is slightly more frequent re-authentication for inactive users, but this is a reasonable security trade-off. - diff --git a/SSO_FLOW_ANALYSIS.md b/SSO_FLOW_ANALYSIS.md deleted file mode 100644 index 4d296177..00000000 --- a/SSO_FLOW_ANALYSIS.md +++ /dev/null @@ -1,250 +0,0 @@ -# SSO Flow Analysis - Keycloak External Logout Issue - -## Current Flow Trace - -### Scenario: User logs out from Keycloak directly, then accesses dashboard - -**Step-by-step flow:** - -1. **Initial State (Before Keycloak Logout)** - - User is logged into Dashboard via NextAuth - - NextAuth JWT contains: - - `accessToken`: Valid Keycloak OAuth token - - `refreshToken`: Valid Keycloak refresh token - - `idToken`: Valid Keycloak ID token - - Keycloak session cookies are set in browser - - Iframe applications can authenticate via Keycloak cookies - -2. **User Logs Out from Keycloak Directly (External Application)** - - External application calls: `POST /realms/{realm}/protocol/openid-connect/logout` - - Keycloak invalidates: - - ✅ Keycloak session cookies (cleared) - - ✅ Keycloak refresh token (invalidated) - - ✅ Keycloak access token (invalidated) - - ❌ **NextAuth JWT still contains old tokens** (NextAuth doesn't know about logout) - - ❌ **NextAuth session cookie still valid** (30-day expiration) - -3. **User Accesses Dashboard** - - Browser sends NextAuth session cookie - - NextAuth decrypts JWT - - JWT contains old (now invalid) tokens - - **Token expiration check**: `Date.now() < (token.accessTokenExpires as number) * 1000` - - If token hasn't expired yet (by timestamp), NextAuth returns existing token - - **Problem**: Token is invalid in Keycloak, but NextAuth doesn't know yet - -4. **User Navigates to Iframe Application** - - `ResponsiveIframe` component mounts - - `useEffect` triggers: `refreshSession()` - - Calls: `GET /api/auth/refresh-keycloak-session` - -5. **Refresh Endpoint Execution** - ``` - GET /api/auth/refresh-keycloak-session - → getServerSession(authOptions) - → Reads NextAuth JWT from cookie - → JWT contains old refreshToken (invalid) - → Calls Keycloak: POST /token with old refreshToken - → Keycloak responds: { error: 'invalid_grant', error_description: 'Token is not active' } - → Returns 401 with SessionInvalidated error - ``` - -6. **ResponsiveIframe Handles Error** - - Detects `SessionInvalidated` error - - Redirects to `/signin` - - User signs in again - - Gets NEW tokens from Keycloak - -7. **User Returns to Iframe (After Re-login)** - - **Problem**: If NextAuth JWT callback hasn't run yet, it might still have old tokens - - OR: The new session is created, but iframe component might be using cached session - - OR: The refresh endpoint is called again before new session is fully established - -## Root Cause Analysis - -### Issue 1: Stale Token Detection - -**Problem**: NextAuth only tries to refresh tokens when they're expired (by timestamp). If a token is invalidated externally (Keycloak logout), NextAuth won't know until it tries to refresh. - -**Current Flow**: -``` -JWT Callback: - if (Date.now() < token.accessTokenExpires * 1000) { - return token; // Returns stale token without checking Keycloak - } - // Only refreshes if expired by timestamp -``` - -**What Should Happen**: -- When accessing iframe, we proactively refresh to validate token -- But if refresh fails, we need to clear the NextAuth session immediately - -### Issue 2: Session Invalidation Timing - -**Problem**: When refresh fails: -1. Refresh endpoint returns `SessionInvalidated` -2. ResponsiveIframe redirects to `/signin` -3. User signs in, gets new tokens -4. **But**: NextAuth JWT might still have old tokens cached until next JWT callback execution - -**Current Behavior**: -- Redirect to signin happens -- User re-authenticates -- New session is created -- But old session might still be in browser cache/cookies - -### Issue 3: Infinite Redirect Loop Potential - -**Problem**: If the refresh endpoint keeps failing: -- ResponsiveIframe redirects to `/signin` -- User signs in -- Returns to iframe -- Refresh endpoint called again -- If new session isn't fully established, it might still use old tokens -- Loop continues - -## Current Code Flow - -### ResponsiveIframe Component Flow - -```typescript -1. Component mounts with session -2. useEffect triggers refreshSession() -3. Calls GET /api/auth/refresh-keycloak-session -4. If 401 + SessionInvalidated: - → window.location.href = '/signin' - → User redirected -5. User signs in again -6. Returns to iframe page -7. Component mounts again -8. useEffect triggers refreshSession() again -9. If session still has old tokens → fails again -``` - -### Refresh Endpoint Flow - -```typescript -GET /api/auth/refresh-keycloak-session -1. getServerSession(authOptions) - → Reads JWT from cookie - → JWT callback runs - → If token expired: refreshAccessToken() - → If token not expired: returns existing token (might be invalid!) -2. Uses session.refreshToken -3. Calls Keycloak refresh endpoint -4. If invalid_grant: Returns SessionInvalidated -``` - -### JWT Callback Flow - -```typescript -async jwt({ token, account, profile }) { - // Initial login: account & profile present - if (account && profile) { - // Store tokens - } - - // Subsequent requests - else if (token.accessToken) { - // Check expiration - if (Date.now() < token.accessTokenExpires * 1000) { - return token; // ⚠️ Returns token without validating with Keycloak - } - - // Only refreshes if expired by timestamp - return refreshAccessToken(token); - } -} -``` - -## The Problem - -**Key Issue**: NextAuth JWT callback only checks token expiration by timestamp. It doesn't validate that the token is still valid in Keycloak. So: - -1. User logs out from Keycloak → Token invalidated -2. NextAuth JWT still has token (not expired by timestamp) -3. JWT callback returns existing token (assumes it's valid) -4. Refresh endpoint tries to use invalid refresh token -5. Fails, redirects to signin -6. User signs in, but if JWT callback hasn't run with new account, might still have old token - -## Why It Gets Stuck - -Looking at the logs: -``` -Failed to refresh Keycloak session: { error: 'invalid_grant', error_description: 'Token is not active' } -GET /api/auth/refresh-keycloak-session 401 -→ Redirects to /signin -→ User signs in -→ Returns to iframe -→ refresh-keycloak-session called again -→ Still fails (401) -``` - -**Possible reasons**: -1. **Session not fully updated**: After signin, NextAuth creates new session, but refresh endpoint might be reading old session from cookie before it's updated -2. **Token not refreshed in JWT**: The new tokens from signin might not be stored in JWT yet when refresh endpoint is called -3. **Cookie caching**: Browser might be sending old session cookie -4. **Race condition**: Refresh endpoint called before new session is established - -## Recommendations (Without Code Changes) - -### 1. Check Session State After Signin - -After user signs in and is redirected back: -- Verify that `getServerSession()` returns new session with valid tokens -- Check that JWT callback has run and stored new tokens -- Ensure session cookie is updated in browser - -### 2. Add Delay/Retry Logic - -In ResponsiveIframe: -- After redirect from signin, wait a moment before calling refresh endpoint -- Or check if session has been updated before calling refresh -- Add retry logic with exponential backoff - -### 3. Validate Token Before Using - -In refresh endpoint: -- Before using refreshToken, validate that accessToken is still valid -- Or check token age - if token is old, force refresh even if not expired - -### 4. Clear Session on Invalid Token - -When refresh fails with invalid_grant: -- Don't just redirect - also clear NextAuth session cookie -- Force complete re-authentication -- Ensure old session is completely removed - -### 5. Check Keycloak Session Status - -Before calling refresh endpoint: -- Check if Keycloak session is still active -- Use Keycloak's userinfo endpoint to validate access token -- Only refresh if token is actually invalid - -## Current Behavior Summary - -**What's Happening**: -1. ✅ User logs out from Keycloak → Keycloak invalidates tokens -2. ✅ User accesses dashboard → NextAuth still has old tokens (not expired by timestamp) -3. ✅ User goes to iframe → Refresh endpoint called -4. ✅ Refresh fails → Detects invalid token -5. ✅ Redirects to signin → User re-authenticates -6. ⚠️ **Issue 1**: Storage initialization fails during signin (`createUserFolderStructure` not exported) -7. ⚠️ **Issue 2**: After re-authentication, refresh endpoint might still be using old session -8. ⚠️ **Result**: Gets stuck in redirect loop or keeps failing - -**Root Cause**: NextAuth doesn't proactively validate tokens with Keycloak. It only checks expiration timestamps. When tokens are invalidated externally, NextAuth doesn't know until it tries to use them. - -**Additional Issue Confirmed**: -- Storage initialization fails during signin process -- Error: `createUserFolderStructure is not a function` -- This prevents complete signin initialization -- May contribute to session not being fully established - ---- - -**Analysis Date**: 2024 -**Status**: Issue Identified -**Next Steps**: Implement proactive token validation or improve session invalidation handling - diff --git a/SSO_FLOW_CONFIRMED.md b/SSO_FLOW_CONFIRMED.md deleted file mode 100644 index 207e902a..00000000 --- a/SSO_FLOW_CONFIRMED.md +++ /dev/null @@ -1,207 +0,0 @@ -# SSO Flow Analysis - Confirmed Issues - -## Browser Console Evidence - -Based on the browser console error provided, here's what's happening: - -### Confirmed Flow - -1. **User logs out from Keycloak directly** (external application) - - Keycloak invalidates all tokens and session - -2. **User accesses Dashboard** - - NextAuth session still exists (30-day expiration) - - JWT contains old, now-invalid tokens - - Token expiration check: `Date.now() < token.accessTokenExpires * 1000` - - If token hasn't expired by timestamp → JWT callback returns old token - - **Problem**: Token is invalid in Keycloak, but NextAuth doesn't validate it - -3. **User navigates to iframe application** - - `ResponsiveIframe` component mounts - - Calls `GET /api/auth/refresh-keycloak-session` - - Refresh endpoint uses old `refreshToken` from session - - Keycloak responds: `{ error: 'invalid_grant', error_description: 'Token is not active' }` - - Returns 401 with `SessionInvalidated` error - -4. **Redirect to Signin** - - `ResponsiveIframe` detects `SessionInvalidated` - - Redirects: `window.location.href = '/signin'` - - User lands on signin page - -5. **Signin Process Starts** - - `app/signin/page.tsx` detects unauthenticated status - - Triggers: `signIn("keycloak", { callbackUrl: "/" })` - - User authenticates with Keycloak - - Gets NEW tokens from Keycloak - - NextAuth callback stores new tokens in JWT - -6. **Storage Initialization Fails** ⚠️ **CONFIRMED ISSUE** - - Signin page detects session available - - Calls: `POST /api/storage/init` - - Storage endpoint tries to call: `createUserFolderStructure(session.user.id)` - - **Error**: `createUserFolderStructure is not a function` - - Storage initialization fails - - Signin page shows "Échec de l'initialisation" - - **Impact**: User might not be fully signed in, or session might not be complete - -7. **User Returns to Iframe** (After Signin) - - Navigates to iframe application again - - `ResponsiveIframe` component mounts - - Calls refresh endpoint again - - **If storage init failed**: Session might not be fully established - - **If new session not ready**: Might still use old tokens - - Refresh fails again → Redirects to signin → Loop - -## Confirmed Issues - -### Issue 1: Storage Initialization Failure ✅ CONFIRMED - -**Error**: `createUserFolderStructure is not a function` - -**Location**: `app/api/storage/init/route.ts:16` - -**Impact on SSO Flow**: -- Storage initialization is part of signin process -- If it fails, signin might not complete properly -- Session might not be fully established -- When user tries to access iframe, refresh endpoint might fail because: - - Session not complete - - Or still using old tokens if new session wasn't saved - -**Evidence from Browser**: -``` -Failed to initialize storage: "{\"error\":\"Failed to initialize storage\",\"details\":\"(0 , _lib_s3__WEBPACK_IMPORTED_MODULE_3__.createUserFolderStructure) is not a function\"}" -``` - -### Issue 2: Stale Token in NextAuth JWT ✅ CONFIRMED - -**Problem**: When Keycloak session is invalidated externally: -- NextAuth JWT still contains old tokens -- JWT callback only checks expiration timestamp -- Doesn't validate with Keycloak that token is still valid -- Returns stale token until expiration timestamp is reached - -**Evidence from Terminal**: -``` -Failed to refresh Keycloak session: { error: 'invalid_grant', error_description: 'Token is not active' } -GET /api/auth/refresh-keycloak-session 401 -``` - -### Issue 3: Race Condition After Re-authentication ✅ CONFIRMED - -**Problem**: After user signs in again: -- New session is created -- But refresh endpoint might be called before: - - JWT callback has run with new account - - New tokens are stored in JWT - - Session cookie is updated in browser -- Result: Refresh endpoint still uses old tokens - -**Evidence**: Multiple failed refresh attempts after signin - -## Complete Flow Diagram (Confirmed) - -``` -1. User logs out from Keycloak (external) - ↓ -2. Keycloak invalidates: - - Session cookies ✅ - - Refresh token ✅ - - Access token ✅ - ↓ -3. User accesses Dashboard - - NextAuth JWT has old tokens (not expired by timestamp) - - JWT callback returns old token (doesn't validate with Keycloak) - ↓ -4. User navigates to iframe - - ResponsiveIframe calls refresh endpoint - - Uses old refreshToken from session - ↓ -5. Keycloak rejects: "Token is not active" - ↓ -6. Refresh endpoint returns 401 SessionInvalidated - ↓ -7. Redirect to /signin - ↓ -8. User authenticates with Keycloak - - Gets NEW tokens - - NextAuth stores new tokens in JWT - ↓ -9. Storage initialization called - - ⚠️ FAILS: createUserFolderStructure not found - - Signin process incomplete - ↓ -10. User navigates to iframe again - - Refresh endpoint called - - ⚠️ Might still use old tokens (if new session not ready) - - OR: Session incomplete due to storage init failure - - Fails again → Redirects to signin - ↓ -11. LOOP or stuck state -``` - -## Root Causes (Confirmed) - -1. **No Proactive Token Validation** - - NextAuth only checks expiration timestamps - - Doesn't validate tokens with Keycloak - - Stale tokens remain in JWT until timestamp expiration - -2. **Storage Initialization Failure** - - Missing function: `createUserFolderStructure` - - Prevents complete signin initialization - - May cause session to be incomplete - -3. **Race Condition** - - Refresh endpoint called before new session fully established - - Browser might send old session cookie - - JWT callback might not have run yet with new account - -4. **No Session Invalidation on External Logout** - - When Keycloak session invalidated externally - - NextAuth doesn't know about it - - Continues using invalid tokens - -## Impact on User Experience - -**What User Sees**: -1. Logs out from Keycloak (external app) -2. Accesses Dashboard → Still logged in (NextAuth session valid) -3. Tries to access iframe application -4. Gets redirected to signin -5. Signs in again -6. Storage initialization fails (error message) -7. Tries to access iframe again -8. Gets redirected to signin again -9. **Stuck in loop or keeps getting disconnected** - -## Recommendations - -### Immediate Fixes Needed - -1. **Fix Storage Initialization** - - Export `createUserFolderStructure` from `lib/s3.ts` - - Or remove storage init from signin flow if not critical - - Prevents signin from failing - -2. **Proactive Token Validation** - - Before using tokens, validate with Keycloak - - Use Keycloak's userinfo endpoint to check token validity - - Clear session if token invalid - -3. **Session Invalidation on Refresh Failure** - - When refresh fails with invalid_grant - - Immediately clear NextAuth session cookie - - Force complete re-authentication - -4. **Delay Refresh After Signin** - - After redirect from signin, wait for session to be established - - Check session status before calling refresh endpoint - - Add retry logic with backoff - ---- - -**Analysis Date**: 2024 -**Status**: Issues Confirmed -**Evidence**: Browser console + Terminal logs - diff --git a/STACK_QUALITY_AND_FLOW_ANALYSIS.md b/STACK_QUALITY_AND_FLOW_ANALYSIS.md deleted file mode 100644 index c6409908..00000000 --- a/STACK_QUALITY_AND_FLOW_ANALYSIS.md +++ /dev/null @@ -1,540 +0,0 @@ -# Stack Quality & Flow Analysis Report - -## Executive Summary - -This document provides a comprehensive analysis of the codebase quality, architecture patterns, and identifies critical issues in the notification and widget update flows. - -**Overall Assessment**: ⚠️ **Moderate Quality** - Good foundation with several critical issues that need attention. - ---- - -## 🔴 Critical Issues - -### 1. **Memory Leak: Multiple Polling Intervals** - -**Location**: `hooks/use-notifications.ts`, `components/parole.tsx`, `components/calendar/calendar-widget.tsx` - -**Problem**: -- `useNotifications` hook creates polling intervals that may not be properly cleaned up -- Multiple components using the hook can create duplicate intervals -- `startPolling()` returns a cleanup function but it's not properly used in the useEffect - -**Code Issue**: -```typescript -// Line 226 in use-notifications.ts -return () => stopPolling(); // This return is inside startPolling, not useEffect! -``` - -**Impact**: Memory leaks, excessive API calls, degraded performance - -**Fix Required**: -```typescript -useEffect(() => { - isMountedRef.current = true; - - if (status === 'authenticated' && session?.user) { - fetchNotificationCount(true); - fetchNotifications(); - startPolling(); - } - - return () => { - isMountedRef.current = false; - stopPolling(); // ✅ Correct placement - }; -}, [status, session?.user, fetchNotificationCount, fetchNotifications, startPolling, stopPolling]); -``` - ---- - -### 2. **Race Condition: Notification Badge Double Fetching** - -**Location**: `components/notification-badge.tsx` - -**Problem**: -- Multiple `useEffect` hooks trigger `manualFetch()` simultaneously -- Lines 65-70, 82-87, and 92-99 all trigger fetches -- No debouncing or request deduplication - -**Code Issue**: -```typescript -// Line 65-70: Fetch on dropdown open -useEffect(() => { - if (isOpen && status === 'authenticated') { - manualFetch(); - } -}, [isOpen, status]); - -// Line 82-87: Fetch on mount -useEffect(() => { - if (status === 'authenticated') { - manualFetch(); - } -}, [status]); - -// Line 92-99: Fetch on handleOpenChange -const handleOpenChange = (open: boolean) => { - setIsOpen(open); - if (open && status === 'authenticated') { - manualFetch(); // Duplicate fetch! - } -}; -``` - -**Impact**: Unnecessary API calls, potential race conditions, poor UX - -**Fix Required**: Consolidate fetch logic, add request deduplication - ---- - -### 3. **Redis KEYS Command Performance Issue** - -**Location**: `lib/services/notifications/notification-service.ts` (line 293) - -**Problem**: -- Using `redis.keys()` which is O(N) and blocks Redis -- Can cause performance degradation in production - -**Code Issue**: -```typescript -// Line 293 - BAD -const listKeys = await redis.keys(listKeysPattern); -if (listKeys.length > 0) { - await redis.del(...listKeys); -} -``` - -**Impact**: Redis blocking, slow response times, potential timeouts - -**Fix Required**: Use `SCAN` instead of `KEYS`: -```typescript -// GOOD - Use SCAN -let cursor = '0'; -do { - const [nextCursor, keys] = await redis.scan(cursor, 'MATCH', listKeysPattern, 'COUNT', 100); - cursor = nextCursor; - if (keys.length > 0) { - await redis.del(...keys); - } -} while (cursor !== '0'); -``` - ---- - -### 4. **Infinite Loop Risk: useEffect Dependencies** - -**Location**: `hooks/use-notifications.ts` (line 255) - -**Problem**: -- `useEffect` includes functions in dependencies that are recreated on every render -- `fetchNotificationCount`, `fetchNotifications`, `startPolling`, `stopPolling` are in deps -- These functions depend on `session?.user` which changes, causing re-renders - -**Code Issue**: -```typescript -useEffect(() => { - // ... -}, [status, session?.user, fetchNotificationCount, fetchNotifications, startPolling, stopPolling]); -// ❌ Functions are recreated, causing infinite loops -``` - -**Impact**: Infinite re-renders, excessive API calls, browser freezing - -**Fix Required**: Remove function dependencies or use `useCallback` properly - ---- - -### 5. **Background Refresh Memory Leak** - -**Location**: `lib/services/notifications/notification-service.ts` (line 326) - -**Problem**: -- `setTimeout` in `scheduleBackgroundRefresh` creates closures that may not be cleaned up -- No way to cancel pending background refreshes -- Can accumulate in serverless environments - -**Code Issue**: -```typescript -setTimeout(async () => { - // This closure holds references and may not be garbage collected - await this.getNotificationCount(userId); - await this.getNotifications(userId, 1, 20); -}, 0); -``` - -**Impact**: Memory leaks, especially in serverless/edge environments - -**Fix Required**: Use proper cleanup mechanism or job queue - ---- - -## ⚠️ High Priority Issues - -### 6. **Widget Update Race Conditions** - -**Location**: Multiple widget components - -**Problem**: -- Widgets don't coordinate updates -- Multiple widgets can trigger simultaneous API calls -- No request deduplication - -**Affected Widgets**: -- `components/calendar.tsx` - Auto-refresh every 5 minutes -- `components/parole.tsx` - Auto-polling every 30 seconds -- `components/news.tsx` - Manual refresh only -- `components/flow.tsx` - Manual refresh only -- `components/email.tsx` - Manual refresh only - -**Impact**: Unnecessary load on backend, potential rate limiting - -**Fix Required**: Implement request deduplication layer or use React Query/SWR - ---- - -### 7. **Redis Connection Singleton Issues** - -**Location**: `lib/redis.ts` - -**Problem**: -- Singleton pattern but no proper connection pooling -- In serverless environments, connections may not be reused -- No connection health monitoring -- Race condition in `getRedisClient()` when `isConnecting` is true - -**Code Issue**: -```typescript -if (isConnecting) { - if (redisClient) return redisClient; - // ⚠️ What if redisClient is null but isConnecting is true? - console.warn('Redis connection in progress, creating temporary client'); -} -``` - -**Impact**: Connection leaks, connection pool exhaustion, degraded performance - -**Fix Required**: Implement proper connection pool or use Redis connection manager - ---- - -### 8. **Error Handling Gaps** - -**Location**: Multiple files - -**Problems**: -- Errors are logged but not always handled gracefully -- No retry logic for transient failures -- No circuit breaker pattern -- Widgets show errors but don't recover automatically - -**Examples**: -- `components/notification-badge.tsx` - Shows error but no auto-retry -- `lib/services/notifications/notification-service.ts` - Errors return empty arrays silently -- Widget components - Errors stop updates, no recovery - -**Impact**: Poor UX, silent failures, degraded functionality - ---- - -### 9. **Cache Invalidation Issues** - -**Location**: `lib/services/notifications/notification-service.ts` - -**Problem**: -- Cache invalidation uses `KEYS` command (blocking) -- No partial cache invalidation -- Background refresh may not invalidate properly -- Race condition: cache can be invalidated while being refreshed - -**Impact**: Stale data, inconsistent state - ---- - -### 10. **Excessive Logging** - -**Location**: Throughout codebase - -**Problem**: -- Console.log statements everywhere -- No log levels -- Production code has debug logs -- Performance impact from string concatenation - -**Impact**: Performance degradation, log storage costs, security concerns - -**Fix Required**: Use proper logging library with levels (e.g., Winston, Pino) - ---- - -## 📊 Architecture Quality Assessment - -### Strengths ✅ - -1. **Adapter Pattern**: Well-implemented notification adapter pattern -2. **Separation of Concerns**: Clear separation between services, hooks, and components -3. **Type Safety**: Good TypeScript usage -4. **Caching Strategy**: Redis caching implemented -5. **Error Boundaries**: Some error handling present - -### Weaknesses ❌ - -1. **No State Management**: Using local state instead of global state management -2. **No Request Deduplication**: Multiple components can trigger same API calls -3. **No Request Cancellation**: No way to cancel in-flight requests -4. **No Optimistic Updates**: UI doesn't update optimistically -5. **No Offline Support**: No handling for offline scenarios -6. **No Request Queue**: No queuing mechanism for API calls - ---- - -## 🔄 Flow Analysis - -### Notification Flow Issues - -#### Flow Diagram (Current - Problematic): -``` -User Action / Polling - ↓ -useNotifications Hook (multiple instances) - ↓ -Multiple API Calls (no deduplication) - ↓ -NotificationService (Redis cache check) - ↓ -Adapter Calls (parallel, but no error aggregation) - ↓ -Response (may be stale due to race conditions) -``` - -#### Issues: -1. **Multiple Hook Instances**: `NotificationBadge` and potentially other components use `useNotifications`, creating multiple polling intervals -2. **No Request Deduplication**: Same request can be made multiple times simultaneously -3. **Cache Race Conditions**: Background refresh can conflict with user requests -4. **No Request Cancellation**: Old requests aren't cancelled when new ones start - -### Widget Update Flow Issues - -#### Flow Diagram (Current - Problematic): -``` -Component Mount - ↓ -useEffect triggers fetch - ↓ -API Call (no coordination with other widgets) - ↓ -State Update (may cause unnecessary re-renders) - ↓ -Auto-refresh interval (no cleanup guarantee) -``` - -#### Issues: -1. **No Coordination**: Widgets don't know about each other's updates -2. **Duplicate Requests**: Same data fetched multiple times -3. **Cleanup Issues**: Intervals may not be cleaned up properly -4. **No Stale-While-Revalidate**: No background updates - ---- - -## 🎯 Recommendations - -### Immediate Actions (Critical) - -1. **Fix Memory Leaks** - - Fix `useNotifications` cleanup - - Ensure all intervals are cleared - - Add cleanup in all widget components - -2. **Fix Race Conditions** - - Implement request deduplication - - Fix notification badge double fetching - - Add request cancellation - -3. **Fix Redis Performance** - - Replace `KEYS` with `SCAN` - - Implement proper connection pooling - - Add connection health checks - -### Short-term Improvements (High Priority) - -1. **Implement Request Management** - - Use React Query or SWR for request deduplication - - Implement request cancellation - - Add request queuing - -2. **Improve Error Handling** - - Add retry logic with exponential backoff - - Implement circuit breaker pattern - - Add error boundaries - -3. **Optimize Caching** - - Implement stale-while-revalidate pattern - - Add cache versioning - - Improve cache invalidation strategy - -### Long-term Improvements (Medium Priority) - -1. **State Management** - - Consider Zustand or Redux for global state - - Centralize notification state - - Implement optimistic updates - -2. **Monitoring & Observability** - - Add proper logging (Winston/Pino) - - Implement metrics collection - - Add performance monitoring - -3. **Testing** - - Add unit tests for hooks - - Add integration tests for flows - - Add E2E tests for critical paths - ---- - -## 📈 Performance Metrics (Estimated) - -### Current Performance Issues: - -1. **API Calls**: - - Estimated 2-3x more calls than necessary due to race conditions - - No request deduplication - -2. **Memory Usage**: - - Potential memory leaks from uncleaned intervals - - Closures holding references - -3. **Redis Performance**: - - `KEYS` command can block for seconds with many keys - - No connection pooling - -4. **Bundle Size**: - - Excessive logging increases bundle size - - No code splitting for widgets - ---- - -## 🔍 Code Quality Metrics - -### Code Smells Found: - -1. **Long Functions**: Some functions exceed 50 lines -2. **High Cyclomatic Complexity**: `useNotifications` hook has high complexity -3. **Duplicate Code**: Similar fetch patterns across widgets -4. **Magic Numbers**: Hardcoded intervals (300000, 60000, etc.) -5. **Inconsistent Error Handling**: Different error handling patterns - -### Technical Debt: - -- **Estimated**: Medium-High -- **Areas**: - - Memory management - - Request management - - Error handling - - Caching strategy - - Logging infrastructure - ---- - -## 🛠️ Specific Code Fixes Needed - -### Fix 1: useNotifications Hook Cleanup - -```typescript -// BEFORE (Current - Problematic) -useEffect(() => { - isMountedRef.current = true; - - if (status === 'authenticated' && session?.user) { - fetchNotificationCount(true); - fetchNotifications(); - startPolling(); - } - - return () => { - isMountedRef.current = false; - stopPolling(); - }; -}, [status, session?.user, fetchNotificationCount, fetchNotifications, startPolling, stopPolling]); - -// AFTER (Fixed) -useEffect(() => { - if (status !== 'authenticated' || !session?.user) return; - - isMountedRef.current = true; - - // Initial fetch - fetchNotificationCount(true); - fetchNotifications(); - - // Start polling - const intervalId = setInterval(() => { - if (isMountedRef.current) { - debouncedFetchCount(); - } - }, POLLING_INTERVAL); - - // Cleanup - return () => { - isMountedRef.current = false; - clearInterval(intervalId); - }; -}, [status, session?.user?.id]); // Only depend on primitive values -``` - -### Fix 2: Notification Badge Deduplication - -```typescript -// Add request deduplication -const fetchInProgressRef = useRef(false); - -const manualFetch = async () => { - if (fetchInProgressRef.current) { - console.log('[NOTIFICATION_BADGE] Fetch already in progress, skipping'); - return; - } - - fetchInProgressRef.current = true; - try { - await fetchNotifications(1, 10); - } finally { - fetchInProgressRef.current = false; - } -}; -``` - -### Fix 3: Redis SCAN Instead of KEYS - -```typescript -// BEFORE -const listKeys = await redis.keys(listKeysPattern); - -// AFTER -const listKeys: string[] = []; -let cursor = '0'; -do { - const [nextCursor, keys] = await redis.scan(cursor, 'MATCH', listKeysPattern, 'COUNT', 100); - cursor = nextCursor; - listKeys.push(...keys); -} while (cursor !== '0'); -``` - ---- - -## 📝 Conclusion - -The codebase has a solid foundation with good architectural patterns (adapter pattern, separation of concerns), but suffers from several critical issues: - -1. **Memory leaks** from improper cleanup -2. **Race conditions** from lack of request coordination -3. **Performance issues** from blocking Redis operations -4. **Error handling gaps** that degrade UX - -**Priority**: Fix critical issues immediately, then implement improvements incrementally. - -**Estimated Effort**: -- Critical fixes: 2-3 days -- High priority improvements: 1-2 weeks -- Long-term improvements: 1-2 months - ---- - -*Generated: Comprehensive codebase analysis* diff --git a/UNIFIED_REFRESH_SUMMARY.md b/UNIFIED_REFRESH_SUMMARY.md deleted file mode 100644 index c36a596d..00000000 --- a/UNIFIED_REFRESH_SUMMARY.md +++ /dev/null @@ -1,302 +0,0 @@ -# Unified Refresh System - Implementation Summary - -## ✅ What Has Been Created - -### Core Infrastructure Files - -1. **`lib/constants/refresh-intervals.ts`** - - Standardized refresh intervals for all resources - - Helper functions for interval management - - All intervals harmonized and documented - -2. **`lib/utils/request-deduplication.ts`** - - Request deduplication utility - - Prevents duplicate API calls within 5 seconds - - Automatic cleanup of stale requests - -3. **`lib/services/refresh-manager.ts`** - - Centralized refresh management - - Handles all refresh intervals - - Provides pause/resume functionality - - Prevents duplicate refreshes - -4. **`hooks/use-unified-refresh.ts`** - - React hook for easy integration - - Automatic registration/cleanup - - Manual refresh support - -### Documentation Files - -1. **`IMPLEMENTATION_PLAN_UNIFIED_REFRESH.md`** - - Complete architecture overview - - Detailed implementation guide - - Code examples for all widgets - -2. **`IMPLEMENTATION_CHECKLIST.md`** - - Step-by-step checklist - - Daily progress tracking - - Success criteria - ---- - -## 🎯 Next Steps - -### Immediate Actions (Start Here) - -#### 1. Fix Critical Memory Leaks (30 minutes) - -**File**: `lib/services/notifications/notification-service.ts` - -Replace `redis.keys()` with `redis.scan()`: - -```typescript -// Line 293 - BEFORE -const listKeys = await redis.keys(listKeysPattern); - -// AFTER -const listKeys: string[] = []; -let cursor = '0'; -do { - const [nextCursor, keys] = await redis.scan( - cursor, - 'MATCH', - listKeysPattern, - 'COUNT', - 100 - ); - cursor = nextCursor; - if (keys.length > 0) { - listKeys.push(...keys); - } -} while (cursor !== '0'); -``` - ---- - -#### 2. Test Core Infrastructure (1 hour) - -Create a test file to verify everything works: - -**File**: `lib/services/__tests__/refresh-manager.test.ts` (optional) - -Or test manually: -1. Import refresh manager in a component -2. Register a test resource -3. Verify it refreshes at correct interval -4. Verify cleanup on unmount - ---- - -#### 3. Refactor Notifications (2-3 hours) - -**File**: `hooks/use-notifications.ts` - -Key changes: -- Remove manual polling logic -- Use `useUnifiedRefresh` hook -- Add `requestDeduplicator` for API calls -- Fix useEffect dependencies - -See `IMPLEMENTATION_PLAN_UNIFIED_REFRESH.md` Section 3.1 for full code. - ---- - -#### 4. Refactor Notification Badge (1 hour) - -**File**: `components/notification-badge.tsx` - -Key changes: -- Remove duplicate `useEffect` hooks -- Use hook's `refresh` function for manual refresh -- Remove manual fetch logic - ---- - -#### 5. Refactor Navigation Bar Time (30 minutes) - -**File**: `components/main-nav.tsx` + `components/main-nav-time.tsx` (new) - -Key changes: -- Extract time display to separate component -- Use `useUnifiedRefresh` hook (1 second interval) -- Fix static time issue - -See `IMPLEMENTATION_PLAN_UNIFIED_REFRESH.md` Section 3.7 for full code. - ---- - -#### 6. Refactor Widgets (1 hour each) - -Start with high-frequency widgets: -1. **Parole** (`components/parole.tsx`) - 30s interval -2. **Calendar** (`components/calendar.tsx`) - 5min interval -3. **News** (`components/news.tsx`) - 10min interval -4. **Email** (`components/email.tsx`) - 1min interval -5. **Duties** (`components/flow.tsx`) - 2min interval - -See `IMPLEMENTATION_PLAN_UNIFIED_REFRESH.md` Section 3.2 for example code. - ---- - -## 📊 Expected Results - -### Before Implementation: -- ❌ 120-150 API calls/minute -- ❌ Memory leaks from uncleaned intervals -- ❌ Duplicate requests -- ❌ No coordination between widgets - -### After Implementation: -- ✅ 40-50 API calls/minute (60-70% reduction) -- ✅ No memory leaks -- ✅ Request deduplication working -- ✅ Centralized refresh coordination - ---- - -## 🔍 Testing Checklist - -After each phase, verify: - -- [ ] No console errors -- [ ] Widgets refresh at correct intervals -- [ ] Manual refresh buttons work -- [ ] No duplicate API calls (check Network tab) -- [ ] No memory leaks (check Memory tab) -- [ ] Cleanup on component unmount -- [ ] Multiple tabs don't cause issues - ---- - -## 🚨 Important Notes - -### Backward Compatibility - -All new code is designed to be: -- ✅ Non-breaking (old code still works) -- ✅ Gradual migration (one widget at a time) -- ✅ Easy rollback (keep old implementations) - -### Migration Strategy - -1. **Phase 1**: Core infrastructure (DONE ✅) -2. **Phase 2**: Fix critical issues -3. **Phase 3**: Migrate notifications -4. **Phase 4**: Migrate widgets one by one -5. **Phase 5**: Remove old code - -### Feature Flags (Optional) - -If you want to toggle the new system: - -```typescript -// In refresh manager -const USE_UNIFIED_REFRESH = process.env.NEXT_PUBLIC_USE_UNIFIED_REFRESH !== 'false'; - -if (USE_UNIFIED_REFRESH) { - // Use new system -} else { - // Use old system -} -``` - ---- - -## 📈 Performance Monitoring - -### Metrics to Track - -1. **API Call Count** - - Before: ~120-150/min - - Target: ~40-50/min - - Monitor in Network tab - -2. **Memory Usage** - - Before: Growing over time - - Target: Stable - - Monitor in Memory tab - -3. **Refresh Accuracy** - - Verify intervals are correct - - Check last refresh times - - Monitor refresh manager status - -### Debug Tools - -```typescript -// Get refresh manager status -const status = refreshManager.getStatus(); -console.log('Refresh Manager Status:', status); - -// Get pending requests -const pendingCount = requestDeduplicator.getPendingCount(); -console.log('Pending Requests:', pendingCount); -``` - ---- - -## 🎓 Learning Resources - -### Key Concepts - -1. **Singleton Pattern**: Refresh manager uses singleton -2. **Request Deduplication**: Prevents duplicate calls -3. **React Hooks**: Proper cleanup with useEffect -4. **Memory Management**: Clearing intervals and refs - -### Code Patterns - -- **useRef for callbacks**: Prevents dependency issues -- **Map for tracking**: Efficient resource management -- **Promise tracking**: Prevents duplicate requests - ---- - -## 🐛 Troubleshooting - -### Issue: Widgets not refreshing - -**Check**: -1. Is refresh manager started? (`refreshManager.start()`) -2. Is resource registered? (`refreshManager.getStatus()`) -3. Is user authenticated? (`status === 'authenticated'`) - -### Issue: Duplicate API calls - -**Check**: -1. Is request deduplication working? (`requestDeduplicator.getPendingCount()`) -2. Are multiple components using the same resource? -3. Is TTL too short? - -### Issue: Memory leaks - -**Check**: -1. Are intervals cleaned up? (check cleanup functions) -2. Are refs cleared? (`isMountedRef.current = false`) -3. Are pending requests cleared? (check cleanup) - ---- - -## 📝 Next Session Goals - -1. ✅ Core infrastructure created -2. ⏭️ Fix Redis KEYS → SCAN -3. ⏭️ Refactor notifications hook -4. ⏭️ Refactor notification badge -5. ⏭️ Refactor first widget (Parole) - ---- - -## 🎉 Success! - -Once all widgets are migrated: - -- ✅ Unified refresh system -- ✅ 60%+ reduction in API calls -- ✅ No memory leaks -- ✅ Better user experience -- ✅ Easier maintenance - ---- - -*Last Updated: Implementation Summary v1.0* diff --git a/app/api/missions/test-n8n-config/route.ts b/app/api/missions/test-n8n-config/route.ts new file mode 100644 index 00000000..5896e2d9 --- /dev/null +++ b/app/api/missions/test-n8n-config/route.ts @@ -0,0 +1,139 @@ +import { NextResponse } from 'next/server'; +import { getServerSession } from 'next-auth'; +import { authOptions } from "@/app/api/auth/options"; +import { logger } from '@/lib/logger'; + +/** + * GET /api/missions/test-n8n-config + * + * Endpoint de test pour vérifier la configuration N8N + * Permet de diagnostiquer les problèmes de connexion entre Next.js et N8N + * + * Authentification: Requise (session utilisateur) + */ +export async function GET(request: Request) { + try { + // Vérifier l'authentification + const session = await getServerSession(authOptions); + if (!session?.user) { + return NextResponse.json( + { error: 'Unauthorized' }, + { status: 401 } + ); + } + + // Récupérer les variables d'environnement + const n8nApiKey = process.env.N8N_API_KEY; + const n8nWebhookUrl = process.env.N8N_WEBHOOK_URL || 'https://brain.slm-lab.net/webhook/mission-created'; + const n8nRollbackWebhookUrl = process.env.N8N_ROLLBACK_WEBHOOK_URL || 'https://brain.slm-lab.net/webhook/mission-rollback'; + const missionApiUrl = process.env.NEXT_PUBLIC_API_URL || 'https://api.slm-lab.net/api'; + const n8nDeleteWebhookUrl = process.env.N8N_DELETE_WEBHOOK_URL; + + // Construire la réponse + const config = { + // Variables d'environnement + environment: { + hasN8NApiKey: !!n8nApiKey, + n8nApiKeyLength: n8nApiKey?.length || 0, + n8nApiKeyPrefix: n8nApiKey ? `${n8nApiKey.substring(0, 4)}...` : 'none', + n8nWebhookUrl, + n8nRollbackWebhookUrl, + n8nDeleteWebhookUrl: n8nDeleteWebhookUrl || 'not configured', + missionApiUrl, + }, + + // URLs construites + urls: { + webhookUrl: n8nWebhookUrl, + callbackUrl: `${missionApiUrl}/api/missions/mission-created`, + rollbackUrl: n8nRollbackWebhookUrl, + deleteUrl: n8nDeleteWebhookUrl || 'not configured', + }, + + // Statut de configuration + status: { + configured: !!n8nApiKey && !!missionApiUrl, + missingApiKey: !n8nApiKey, + missingApiUrl: !missionApiUrl, + ready: !!n8nApiKey && !!missionApiUrl, + }, + + // Recommandations + recommendations: [] as string[], + }; + + // Ajouter des recommandations basées sur la configuration + if (!n8nApiKey) { + config.recommendations.push('❌ N8N_API_KEY n\'est pas défini. Ajoutez-le à vos variables d\'environnement.'); + } else { + config.recommendations.push('✅ N8N_API_KEY est configuré'); + } + + if (!missionApiUrl) { + config.recommendations.push('⚠️ NEXT_PUBLIC_API_URL n\'est pas défini. Utilisation de la valeur par défaut.'); + } else { + config.recommendations.push('✅ NEXT_PUBLIC_API_URL est configuré'); + } + + if (n8nApiKey && n8nApiKey.length < 10) { + config.recommendations.push('⚠️ N8N_API_KEY semble trop court. Vérifiez qu\'il est correct.'); + } + + // Tester la connectivité au webhook N8N (optionnel, peut être lent) + const testWebhook = request.headers.get('x-test-webhook') === 'true'; + if (testWebhook) { + try { + logger.debug('Testing N8N webhook connectivity', { url: n8nWebhookUrl }); + const testResponse = await fetch(n8nWebhookUrl, { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + 'x-api-key': n8nApiKey || '', + }, + body: JSON.stringify({ test: true }), + signal: AbortSignal.timeout(5000), // 5 secondes timeout + }); + + config.urls.webhookTest = { + status: testResponse.status, + statusText: testResponse.statusText, + reachable: testResponse.status !== 0, + note: testResponse.status === 404 + ? 'Webhook non enregistré (workflow inactif?)' + : testResponse.status === 200 || testResponse.status === 400 || testResponse.status === 500 + ? 'Webhook actif (peut échouer avec des données de test)' + : 'Réponse inattendue', + }; + } catch (error) { + config.urls.webhookTest = { + error: error instanceof Error ? error.message : 'Unknown error', + reachable: false, + note: 'Impossible de joindre le webhook N8N', + }; + } + } else { + config.urls.webhookTest = { + note: 'Ajoutez le header "x-test-webhook: true" pour tester la connectivité', + }; + } + + return NextResponse.json({ + success: true, + timestamp: new Date().toISOString(), + ...config, + }); + } catch (error) { + logger.error('Error in test-n8n-config endpoint', { + error: error instanceof Error ? error.message : String(error) + }); + return NextResponse.json( + { + success: false, + error: 'Failed to check N8N configuration', + details: error instanceof Error ? error.message : 'Unknown error' + }, + { status: 500 } + ); + } +} + diff --git a/log b/log deleted file mode 100644 index 8b137891..00000000 --- a/log +++ /dev/null @@ -1 +0,0 @@ -