PROBLÈME RÉSOLU: - Erreur "Code already used" répétée dans les logs Keycloak - Boucle infinie de tentatives d'échange du code d'autorisation OAuth - Utilisateurs bloqués à la connexion CORRECTIONS APPLIQUÉES: 1. Ajout de useRef pour protéger contre les exécutions multiples - hasExchanged.current: Flag pour prévenir les réexécutions - isProcessing.current: Protection pendant le traitement 2. Modification des dépendances useEffect - AVANT: [searchParams, router] → exécution à chaque changement - APRÈS: [] → exécution unique au montage du composant 3. Amélioration du logging - Console logs pour debug OAuth flow - Messages emoji pour faciliter le suivi 4. Nettoyage de l'URL - window.history.replaceState() pour retirer les paramètres OAuth - Évite les re-renders causés par les paramètres dans l'URL 5. Gestion d'erreurs améliorée - Capture des erreurs JSON du serveur - Messages d'erreur plus explicites FICHIERS AJOUTÉS: - app/(main)/aide/* - 4 pages du module Aide (documentation, tutoriels, support) - app/(main)/messages/* - 4 pages du module Messages (inbox, envoyés, archives) - app/auth/callback/page.tsx.backup - Sauvegarde avant modification IMPACT: ✅ Un seul échange de code par authentification ✅ Plus d'erreur "Code already used" ✅ Connexion fluide et sans boucle ✅ Logs propres et lisibles 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
125 lines
3.9 KiB
JavaScript
125 lines
3.9 KiB
JavaScript
/** @type {import('next').NextConfig} */
|
|
const nextConfig = {
|
|
reactStrictMode: false, // Disabled to prevent double OAuth code usage in dev
|
|
output: 'standalone',
|
|
|
|
// Optimisations pour la production
|
|
compress: true,
|
|
poweredByHeader: false,
|
|
generateEtags: false,
|
|
|
|
// Désactiver la génération statique pour éviter les erreurs useSearchParams
|
|
skipTrailingSlashRedirect: true,
|
|
|
|
// Configuration des images optimisées
|
|
images: {
|
|
domains: ['btpxpress.lions.dev', 'api.lions.dev', 'security.lions.dev'],
|
|
formats: ['image/webp', 'image/avif'],
|
|
minimumCacheTTL: 60,
|
|
dangerouslyAllowSVG: true,
|
|
contentSecurityPolicy: "default-src 'self'; script-src 'none'; sandbox;",
|
|
},
|
|
|
|
// Optimisations de performance
|
|
experimental: {
|
|
optimizeCss: true,
|
|
optimizePackageImports: ['primereact', 'primeicons', 'chart.js', 'axios'],
|
|
},
|
|
|
|
// Packages externes pour les composants serveur
|
|
serverExternalPackages: ['@prisma/client'],
|
|
|
|
// Configuration Turbopack (stable)
|
|
turbopack: {
|
|
rules: {
|
|
'*.svg': {
|
|
loaders: ['@svgr/webpack'],
|
|
as: '*.js',
|
|
},
|
|
},
|
|
},
|
|
|
|
// Configuration du bundler optimisée
|
|
webpack: (config, { dev, isServer }) => {
|
|
// Optimisations pour la production
|
|
if (!dev && !isServer) {
|
|
config.optimization.splitChunks = {
|
|
chunks: 'all',
|
|
cacheGroups: {
|
|
vendor: {
|
|
test: /[\\/]node_modules[\\/]/,
|
|
name: 'vendors',
|
|
chunks: 'all',
|
|
},
|
|
primereact: {
|
|
test: /[\\/]node_modules[\\/]primereact[\\/]/,
|
|
name: 'primereact',
|
|
chunks: 'all',
|
|
},
|
|
charts: {
|
|
test: /[\\/]node_modules[\\/](chart\.js|react-chartjs-2)[\\/]/,
|
|
name: 'charts',
|
|
chunks: 'all',
|
|
},
|
|
},
|
|
};
|
|
}
|
|
|
|
// Alias pour optimiser les imports
|
|
config.resolve.alias = {
|
|
...config.resolve.alias,
|
|
'@': require('path').resolve(__dirname),
|
|
'@components': require('path').resolve(__dirname, 'components'),
|
|
'@services': require('path').resolve(__dirname, 'services'),
|
|
'@types': require('path').resolve(__dirname, 'types'),
|
|
'@utils': require('path').resolve(__dirname, 'utils'),
|
|
};
|
|
|
|
return config;
|
|
},
|
|
|
|
// Headers de sécurité
|
|
async headers() {
|
|
return [
|
|
{
|
|
source: '/(.*)',
|
|
headers: [
|
|
{
|
|
key: 'X-Frame-Options',
|
|
value: 'DENY'
|
|
},
|
|
{
|
|
key: 'X-Content-Type-Options',
|
|
value: 'nosniff'
|
|
},
|
|
{
|
|
key: 'Referrer-Policy',
|
|
value: 'strict-origin-when-cross-origin'
|
|
}
|
|
]
|
|
}
|
|
];
|
|
},
|
|
|
|
async redirects() {
|
|
return [
|
|
{
|
|
source: '/apps/mail',
|
|
destination: '/apps/mail/inbox',
|
|
permanent: true
|
|
}
|
|
];
|
|
},
|
|
|
|
// Configuration des variables d'environnement
|
|
env: {
|
|
NEXT_PUBLIC_API_URL: process.env.NEXT_PUBLIC_API_URL,
|
|
NEXT_PUBLIC_KEYCLOAK_URL: process.env.NEXT_PUBLIC_KEYCLOAK_URL,
|
|
NEXT_PUBLIC_KEYCLOAK_REALM: process.env.NEXT_PUBLIC_KEYCLOAK_REALM,
|
|
NEXT_PUBLIC_KEYCLOAK_CLIENT_ID: process.env.NEXT_PUBLIC_KEYCLOAK_CLIENT_ID,
|
|
NEXT_PUBLIC_APP_ENV: process.env.NEXT_PUBLIC_APP_ENV,
|
|
}
|
|
};
|
|
|
|
module.exports = nextConfig;
|