Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.

63 linhas
1.8KB

  1. using Campaign_Tracker.Server.Authentication;
  2. using System.Security.Claims;
  3. using Microsoft.AspNetCore.Authorization;
  4. using Microsoft.AspNetCore.Mvc;
  5. namespace Campaign_Tracker.Server.Controllers;
  6. [ApiController]
  7. [Authorize]
  8. [Route("api/auth/logout")]
  9. public sealed class AuthLogoutController : ControllerBase
  10. {
  11. private readonly IKeycloakTokenClient _tokenClient;
  12. private readonly IAuthenticationAuditStore _auditStore;
  13. private readonly ILogger<AuthLogoutController> _logger;
  14. public AuthLogoutController(
  15. IKeycloakTokenClient tokenClient,
  16. IAuthenticationAuditStore auditStore,
  17. ILogger<AuthLogoutController> logger)
  18. {
  19. _tokenClient = tokenClient;
  20. _auditStore = auditStore;
  21. _logger = logger;
  22. }
  23. [HttpPost]
  24. public async Task<IActionResult> Logout(
  25. [FromBody] LogoutRequest? request,
  26. CancellationToken cancellationToken)
  27. {
  28. if (request is null || string.IsNullOrWhiteSpace(request.IdTokenHint))
  29. {
  30. return BadRequest();
  31. }
  32. var subject = User.Identity?.Name
  33. ?? User.FindFirstValue(ClaimTypes.NameIdentifier)
  34. ?? "unknown";
  35. var succeeded = false;
  36. try
  37. {
  38. await _tokenClient.EndSessionAsync(request.IdTokenHint, cancellationToken);
  39. succeeded = true;
  40. }
  41. catch (Exception ex)
  42. {
  43. _logger.LogWarning(
  44. ex,
  45. "Keycloak end-session call failed for subject {Subject}; client tokens will still be cleared.",
  46. subject);
  47. }
  48. _auditStore.RecordLogout(subject, succeeded, HttpContext.TraceIdentifier);
  49. // Always return 200 — per AC #8, client tokens must be cleared regardless of Keycloak availability.
  50. return Ok();
  51. }
  52. }
  53. public sealed record LogoutRequest(string IdTokenHint);

Powered by TurnKey Linux.