using AsbCloudApp.Data;
using AsbCloudApp.Services;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;

// For more information on enabling Web API for empty projects, visit https://go.microsoft.com/fwlink/?LinkID=397860
namespace AsbCloudWebApi.Controllers;


/// <summary>
/// CRUD контроллер для админки.
/// </summary>
/// <typeparam name="T"></typeparam>
/// <typeparam name="TService"></typeparam>
[ApiController]
[Route("api/[controller]")]
[Authorize]
public abstract class CrudController<T, TService> : ControllerBase
    where T : IId
    where TService : ICrudRepository<T>
{
    protected readonly TService service;

    public Func<T, CancellationToken, Task<bool>> InsertForbidAsync { get; protected set; } = null!;
    public Func<T, CancellationToken, Task<bool>> UpdateForbidAsync { get; protected set; } = null!;
    public Func<int, CancellationToken, Task<bool>> DeleteForbidAsync { get; protected set; } = null!;

    public CrudController(TService service)
    {
        this.service = service;
    }

    /// <summary>
    /// Получить все записи
    /// </summary>
    /// <param name="token">CancellationToken</param>
    /// <returns>все записи</returns>
    [HttpGet]
    [Permission]
    public virtual async Task<ActionResult<IEnumerable<T>>> GetAllAsync(CancellationToken token)
    {
        var result = await service.GetAllAsync(token).ConfigureAwait(false);
        return Ok(result);
    }

    /// <summary>
    /// Получить одну запись по Id
    /// </summary>
    /// <param name="id">id записи</param>
    /// <param name="token"></param>
    /// <returns>запись</returns>
    [HttpGet("{id}")]
    [Permission]
    public virtual async Task<ActionResult<T?>> GetOrDefaultAsync(int id, CancellationToken token)
    {
        var result = await service.GetOrDefaultAsync(id, token).ConfigureAwait(false);
        return Ok(result);
    }

    /// <summary>
    /// Добавить запись
    /// </summary>
    /// <param name="value">запись</param>
    /// <param name="token"></param>
    /// <returns>id</returns>
    [HttpPost]
    [Permission]
    public virtual async Task<ActionResult<int>> InsertAsync([FromBody] T value, CancellationToken token)
    {
        if (InsertForbidAsync is not null && await InsertForbidAsync(value, token))
            return Forbid();

        var result = await service.InsertAsync(value, token).ConfigureAwait(false);
        return Ok(result);
    }

    /// <summary>
    /// Добавить несколько записей<br/>
    /// При невозможности добавить любую из записей, все не будут добавлены.
    /// </summary>
    /// <param name="values">записи</param>
    /// <param name="token"></param>
    /// <returns>id</returns>
    [HttpPost("range")]
    [Permission]
    [ProducesResponseType(typeof(ValidationProblemDetails), (int)System.Net.HttpStatusCode.BadRequest)]
    public virtual async Task<ActionResult<int>> InsertRangeAsync([FromBody] IEnumerable<T> values, CancellationToken token)
    {
        if (!values.Any())
            return this.ValidationBadRequest(nameof(values), "there is no values to add");

        if (InsertForbidAsync is not null)
            foreach (var value in values)
                if (await InsertForbidAsync(value, token))
                    return Forbid();

        var result = await service.InsertRangeAsync(values, token).ConfigureAwait(false);
        return Ok(result);
    }

    /// <summary>
    /// Редактировать запись по id
    /// </summary>
    /// <param name="value">запись</param>
    /// <param name="token"></param>
    /// <returns>1 - успешно отредактировано, 0 - нет</returns>
    [HttpPut]
    [Permission]
    [ProducesResponseType(typeof(ValidationProblemDetails), (int)System.Net.HttpStatusCode.BadRequest)]
    public virtual async Task<ActionResult<int>> UpdateAsync([FromBody] T value, CancellationToken token)
    {
        if (UpdateForbidAsync is not null && await UpdateForbidAsync(value, token))
            return Forbid();

        var result = await service.UpdateAsync(value, token).ConfigureAwait(false);
        if (result == ICrudRepository<T>.ErrorIdNotFound)
            return this.ValidationBadRequest(nameof(value.Id), $"id:{value.Id} does not exist");
        return Ok(result);
    }

    /// <summary>
    /// Удалить запись по id
    /// </summary>
    /// <param name="id">id записи</param>
    /// <param name="token"></param>
    /// <returns>1 - успешно удалено, 0 - нет</returns>
    [HttpDelete("{id}")]
    [Permission]
    public virtual async Task<ActionResult<int>> DeleteAsync(int id, CancellationToken token)
    {
        if (DeleteForbidAsync is not null && await DeleteForbidAsync(id, token))
            return Forbid();

        var result = await service.DeleteAsync(id, token).ConfigureAwait(false);
        if (result == ICrudRepository<T>.ErrorIdNotFound)
            return NoContent();
        return Ok(result);
    }
}