using System; using System.Threading.Tasks; using Microsoft.AspNetCore.Http; using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Authorization; using Microsoft.Extensions.Logging; using AyaNova.Api.ControllerHelpers; using AyaNova.Biz; using AyaNova.Models; using System.ComponentModel.DataAnnotations; namespace AyaNova.Api.Controllers { /// /// Server state controller /// [ApiController] [ApiVersion("8.0")] [Route("api/v{version:apiVersion}/[controller]")] [Produces("application/json")] public class ServerStateController : ControllerBase { private readonly AyContext ct; private readonly ILogger log; private readonly ApiServerState serverState; /// /// ctor /// /// /// /// public ServerStateController(ILogger logger, ApiServerState apiServerState, AyContext dbcontext) { ct = dbcontext; log = logger; serverState = apiServerState; } /// /// Get server state /// /// Current server state (Closed, OpsOnly, Open) [HttpGet] public ActionResult Get() { return Ok(ApiOkResponse.Response(new ServerStateModel() { ServerState = serverState.GetState().ToString(), Reason = serverState.Reason }, !Authorized.HasModifyRole(HttpContext.Items, AyaType.ServerState))); } /// /// Set server state /// Valid parameters: /// One of "OpsOnly" or "Open" /// /// {"serverState":"Open"} /// NoContent 204 [HttpPost] [Authorize] public async Task PostServerState([FromBody] ServerStateModel state) { if (serverState.IsClosed)//no state change allowed when closed return StatusCode(503, new ApiErrorResponse(serverState.ApiErrorCode, null, serverState.Reason)); if (!Authorized.HasModifyRole(HttpContext.Items, AyaType.ServerState)) return StatusCode(403, new ApiNotAuthorizedResponse()); if (!ModelState.IsValid) return BadRequest(new ApiErrorResponse(ModelState)); ApiServerState.ServerState desiredState; if (!Enum.TryParse(state.ServerState, true, out desiredState)) return BadRequest(new ApiErrorResponse(ApiErrorCode.VALIDATION_INVALID_VALUE, null, "Invalid state - must be one of \"OpsOnly\" or \"Open\"")); //don't allow a server to be set to closed, that's for internal ops only if (desiredState == ApiServerState.ServerState.Closed) return BadRequest(new ApiErrorResponse(ApiErrorCode.VALIDATION_INVALID_VALUE, null, "Invalid state - must be one of \"OpsOnly\" or \"Open\"")); log.LogInformation($"ServerState change request by user {UserNameFromContext.Name(HttpContext.Items)} from current state of \"{serverState.GetState().ToString()}\" to \"{desiredState.ToString()}-{state.Reason}\""); serverState.SetState(desiredState, state.Reason); //Log await EventLogProcessor.LogEventToDatabaseAsync(new Event(UserIdFromContext.Id(HttpContext.Items), 0, AyaType.ServerState, AyaEvent.ServerStateChange, $"{state.ServerState}-{state.Reason}"), ct); return Ok(ApiOkResponse.Response(new ServerStateModel() { ServerState = serverState.GetState().ToString(), Reason = serverState.Reason }, false)); } /// /// Parameter object /// public class ServerStateModel { /// /// "OpsOnly" or "Open" /// /// [Required] public string ServerState { get; set; } /// /// Reason for server state /// /// public string Reason { get; set; } } //------------ } }