using ApplicationLayer.Services.Applicants.Models;
using ApplicationLayer.Services.AuthServices.Common;
using ApplicationLayer.Services.AuthServices.LoginService;
using ApplicationLayer.Services.AuthServices.RegisterService;
using ApplicationLayer.Services.AuthServices.Requests;
using ApplicationLayer.Services.Users;
using ApplicationLayer.Services.Users.Models;
using ApplicationLayer.Services.Users.Requests;
using FluentValidation;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
using SchengenVisaApi.Common;
namespace SchengenVisaApi.Controllers;
/// Controller for user-auth and registration
[ApiController]
[Route("users")]
public class UsersController(
IRegisterService registerService,
ILoginService loginService,
IUsersService usersService,
IValidator registerApplicantRequestValidator,
IValidator changeUserAuthDataRequestValidator,
IValidator registerRequestValidator) : ControllerBase
{
/// Adds applicant with user account
[HttpPost("register")]
[ProducesResponseType(StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status400BadRequest)]
public async Task Register(RegisterApplicantRequest request, CancellationToken cancellationToken)
{
await registerApplicantRequestValidator.ValidateAndThrowAsync(request, cancellationToken);
await registerService.RegisterApplicant(request, cancellationToken);
return Ok();
}
/// Adds approving authority with user account
/// Accessible only for admins
[HttpPost("authorities")]
[ProducesResponseType(StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status403Forbidden)]
[ProducesResponseType(StatusCodes.Status401Unauthorized)]
[ProducesResponseType(StatusCodes.Status400BadRequest)]
[Authorize(policy: PolicyConstants.AdminPolicy)]
public async Task RegisterAuthority(RegisterRequest request, CancellationToken cancellationToken)
{
await registerRequestValidator.ValidateAndThrowAsync(request, cancellationToken);
await registerService.RegisterAuthority(request, cancellationToken);
return Ok();
}
/// Returns JWT-token for authentication
[HttpGet("login")]
[ProducesResponseType(StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status403Forbidden)]
public async Task Login(string email, string password, CancellationToken cancellationToken)
{
var loginRequest = new LoginRequest
{
AuthData = new() { Email = email, Password = password }
};
var result = await loginService.LoginAsync(loginRequest, cancellationToken);
return Ok(result);
}
/// Returns list of authority accounts
/// Accessible only for admins
[HttpGet("authorities")]
[ProducesResponseType>(StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status403Forbidden)]
[ProducesResponseType(StatusCodes.Status401Unauthorized)]
[Authorize(policy: PolicyConstants.AdminPolicy)]
public async Task GetAuthorityAccounts(CancellationToken cancellationToken)
{
var result = await usersService.GetAuthoritiesAccountsAsync(cancellationToken);
return Ok(result);
}
/// Changes authority's account authentication data
/// Accessible only for admins
[HttpPut("authorities")]
[ProducesResponseType(StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
[ProducesResponseType(StatusCodes.Status403Forbidden)]
[ProducesResponseType(StatusCodes.Status401Unauthorized)]
[ProducesResponseType(StatusCodes.Status400BadRequest)]
[Authorize(policy: PolicyConstants.AdminPolicy)]
public async Task ChangeAuthorityAuthData(ChangeUserAuthDataRequest request, CancellationToken cancellationToken)
{
await changeUserAuthDataRequestValidator.ValidateAndThrowAsync(request, cancellationToken);
await usersService.ChangeAuthorityAuthDataAsync(request, cancellationToken);
return Ok();
}
/// Removes authority's account
/// Accessible only for admins
[HttpDelete("authorities/{authorityAccountId:guid}")]
[ProducesResponseType(StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status404NotFound)]
[ProducesResponseType(StatusCodes.Status403Forbidden)]
[ProducesResponseType(StatusCodes.Status401Unauthorized)]
[Authorize(policy: PolicyConstants.AdminPolicy)]
public async Task RemoveAuthorityAccount(Guid authorityAccountId, CancellationToken cancellationToken)
{
await usersService.RemoveAuthorityAccount(authorityAccountId, cancellationToken);
return Ok();
}
/// Returns applicant info
[HttpGet("applicant")]
[ProducesResponseType(StatusCodes.Status200OK)]
[ProducesResponseType(StatusCodes.Status401Unauthorized)]
[ProducesResponseType(StatusCodes.Status403Forbidden)]
[Authorize(policy: PolicyConstants.ApplicantPolicy)]
public async Task GetApplicant(CancellationToken cancellationToken)
{
var result = await usersService.GetAuthenticatedApplicant(cancellationToken);
return Ok(result);
}
}