getHeader('Authorization'); $token = null; if ($authHeader && str_starts_with($authHeader, 'Bearer ')) { $token = substr($authHeader, 7); } elseif (isset($_COOKIE['access_token'])) { $token = $_COOKIE['access_token']; // CSRF Check for browser sessions using cookies if (in_array($request->getMethod(), ['POST', 'PUT', 'DELETE', 'PATCH'])) { $csrfHeader = $request->getHeader('X-CSRF-TOKEN'); $csrfCookie = $_COOKIE['csrf_token'] ?? null; if (!$csrfHeader || !$csrfCookie || !hash_equals($csrfCookie, $csrfHeader)) { Response::error('انتهت صلاحية الجلسة أو فشل التحقق الأمني (CSRF)', 'CSRF_FAILED', 403); return null; } } } if (!$token) { Response::error('يجب تسجيل الدخول للوصول إلى هذا المورد', 'UNAUTHORIZED', 401); return null; } try { $decoded = $this->jwtService->verifyToken($token); // Check if JTI is blacklisted $jti = $decoded['jti'] ?? null; if ($jti) { try { $redis = \App\Core\Redis::getInstance(); if ($redis->exists('jwt_blacklist:' . $jti)) { Response::error('الجلسة منتهية، يرجى تسجيل الدخول من جديد', 'TOKEN_REVOKED', 401); return null; } } catch (\Throwable $e) { // Redis down — allow (fail open, log security event) error_log('[AUTH] JWT blacklist check failed: ' . $e->getMessage()); } } $request->user = (object) $decoded; $request->tenantId = $decoded['tenant_id'] ?? null; } catch (Exception $e) { Response::error('جلسة العمل منتهية أو غير صالحة', 'UNAUTHORIZED', 401); return null; } return $next($request); } }