123 lines
4.0 KiB
C#
123 lines
4.0 KiB
C#
|
|
using Microsoft.AspNetCore.Mvc;
|
||
|
|
using Microsoft.AspNetCore.Authorization;
|
||
|
|
using FutureMailAPI.Services;
|
||
|
|
using FutureMailAPI.DTOs;
|
||
|
|
using FutureMailAPI.Extensions;
|
||
|
|
|
||
|
|
namespace FutureMailAPI.Controllers
|
||
|
|
{
|
||
|
|
[ApiController]
|
||
|
|
[Route("api/v1/auth")]
|
||
|
|
public class AuthController : ControllerBase
|
||
|
|
{
|
||
|
|
private readonly IAuthService _authService;
|
||
|
|
private readonly ILogger<AuthController> _logger;
|
||
|
|
|
||
|
|
public AuthController(IAuthService authService, ILogger<AuthController> logger)
|
||
|
|
{
|
||
|
|
_authService = authService;
|
||
|
|
_logger = logger;
|
||
|
|
}
|
||
|
|
|
||
|
|
[HttpPost("register")]
|
||
|
|
[AllowAnonymous]
|
||
|
|
public async Task<ActionResult<ApiResponse<AuthResponseDto>>> Register([FromBody] UserRegisterDto registerDto)
|
||
|
|
{
|
||
|
|
if (!ModelState.IsValid)
|
||
|
|
{
|
||
|
|
return BadRequest(ApiResponse<AuthResponseDto>.ErrorResult("输入数据无效"));
|
||
|
|
}
|
||
|
|
|
||
|
|
var result = await _authService.RegisterAsync(registerDto);
|
||
|
|
|
||
|
|
if (!result.Success)
|
||
|
|
{
|
||
|
|
return BadRequest(result);
|
||
|
|
}
|
||
|
|
|
||
|
|
return Ok(result);
|
||
|
|
}
|
||
|
|
|
||
|
|
[HttpPost("login")]
|
||
|
|
[AllowAnonymous]
|
||
|
|
public async Task<ActionResult<ApiResponse<AuthResponseDto>>> Login([FromBody] UserLoginDto loginDto)
|
||
|
|
{
|
||
|
|
if (!ModelState.IsValid)
|
||
|
|
{
|
||
|
|
return BadRequest(ApiResponse<AuthResponseDto>.ErrorResult("输入数据无效"));
|
||
|
|
}
|
||
|
|
|
||
|
|
var result = await _authService.LoginAsync(loginDto);
|
||
|
|
|
||
|
|
if (!result.Success)
|
||
|
|
{
|
||
|
|
return BadRequest(result);
|
||
|
|
}
|
||
|
|
|
||
|
|
return Ok(result);
|
||
|
|
}
|
||
|
|
|
||
|
|
[HttpPost("refresh")]
|
||
|
|
[AllowAnonymous]
|
||
|
|
public async Task<ActionResult<ApiResponse<AuthResponseDto>>> RefreshToken([FromBody] RefreshTokenRequestDto request)
|
||
|
|
{
|
||
|
|
if (request == null || string.IsNullOrEmpty(request.Token))
|
||
|
|
{
|
||
|
|
return BadRequest(ApiResponse<AuthResponseDto>.ErrorResult("令牌不能为空"));
|
||
|
|
}
|
||
|
|
|
||
|
|
// 使用OAuth刷新令牌
|
||
|
|
var tokenResult = await _authService.RefreshTokenAsync(request.Token);
|
||
|
|
|
||
|
|
if (!tokenResult.Success)
|
||
|
|
{
|
||
|
|
return BadRequest(ApiResponse<AuthResponseDto>.ErrorResult(tokenResult.Message));
|
||
|
|
}
|
||
|
|
|
||
|
|
// 创建认证响应DTO
|
||
|
|
var authResponse = new AuthResponseDto
|
||
|
|
{
|
||
|
|
Token = tokenResult.Data,
|
||
|
|
Expires = DateTime.UtcNow.AddHours(1) // OAuth访问令牌默认1小时过期
|
||
|
|
};
|
||
|
|
|
||
|
|
return Ok(ApiResponse<AuthResponseDto>.SuccessResult(authResponse, "令牌刷新成功"));
|
||
|
|
}
|
||
|
|
|
||
|
|
[HttpPost("logout")]
|
||
|
|
public async Task<ActionResult<ApiResponse<bool>>> Logout()
|
||
|
|
{
|
||
|
|
// 从JWT令牌中获取当前用户ID
|
||
|
|
var currentUserId = GetCurrentUserId();
|
||
|
|
|
||
|
|
if (currentUserId == null)
|
||
|
|
{
|
||
|
|
return Unauthorized(ApiResponse<bool>.ErrorResult("未授权访问"));
|
||
|
|
}
|
||
|
|
|
||
|
|
// 这里可以实现令牌黑名单或其他注销逻辑
|
||
|
|
// 目前只返回成功响应
|
||
|
|
return Ok(ApiResponse<bool>.SuccessResult(true));
|
||
|
|
}
|
||
|
|
|
||
|
|
private int? GetCurrentUserId()
|
||
|
|
{
|
||
|
|
// 从OAuth中间件获取用户ID
|
||
|
|
var userId = HttpContext.GetCurrentUserId();
|
||
|
|
if (userId.HasValue)
|
||
|
|
{
|
||
|
|
return userId.Value;
|
||
|
|
}
|
||
|
|
|
||
|
|
// 兼容旧的JWT方式
|
||
|
|
var userIdClaim = User.FindFirst(System.Security.Claims.ClaimTypes.NameIdentifier);
|
||
|
|
|
||
|
|
if (userIdClaim == null || !int.TryParse(userIdClaim.Value, out var jwtUserId))
|
||
|
|
{
|
||
|
|
return null;
|
||
|
|
}
|
||
|
|
|
||
|
|
return jwtUserId;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|