mirror of
https://github.com/zoriya/Kyoo.git
synced 2025-10-17 12:00:50 -04:00
254 lines
6.6 KiB
C#
254 lines
6.6 KiB
C#
// Kyoo - A portable and vast media library solution.
|
|
// Copyright (c) Kyoo.
|
|
//
|
|
// See AUTHORS.md and LICENSE file in the project root for full license information.
|
|
//
|
|
// Kyoo is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// any later version.
|
|
//
|
|
// Kyoo is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with Kyoo. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Threading.Tasks;
|
|
using Kyoo.Abstractions.Controllers;
|
|
using Kyoo.Abstractions.Models;
|
|
using Kyoo.Abstractions.Models.Exceptions;
|
|
using Kyoo.Abstractions.Models.Permissions;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
|
|
namespace Kyoo.Core.Api
|
|
{
|
|
/// <summary>
|
|
/// A base class to handle CRUD operations on a specific resource type <typeparamref name="T"/>.
|
|
/// </summary>
|
|
/// <typeparam name="T">The type of resource to make CRUD apis for.</typeparam>
|
|
[ApiController]
|
|
[ResourceView]
|
|
public class CrudApi<T> : ControllerBase
|
|
where T : class, IResource
|
|
{
|
|
/// <summary>
|
|
/// The repository of the resource, used to retrieve, save and do operations on the baking store.
|
|
/// </summary>
|
|
private readonly IRepository<T> _repository;
|
|
|
|
/// <summary>
|
|
/// The base URL of Kyoo. This will be used to create links for images and <see cref="Abstractions.Models.Page{T}"/>.
|
|
/// </summary>
|
|
protected Uri BaseURL { get; }
|
|
|
|
/// <summary>
|
|
/// Create a new <see cref="CrudApi{T}"/> using the given repository and base url.
|
|
/// </summary>
|
|
/// <param name="repository">
|
|
/// The repository to use as a baking store for the type <typeparamref name="T"/>.
|
|
/// </param>
|
|
/// <param name="baseURL">
|
|
/// The base URL of Kyoo to use to create links.
|
|
/// </param>
|
|
public CrudApi(IRepository<T> repository, Uri baseURL)
|
|
{
|
|
_repository = repository;
|
|
BaseURL = baseURL;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get a <typeparamref name="T"/> by ID.
|
|
/// </summary>
|
|
/// <param name="id">The ID of the resource to retrieve.</param>
|
|
/// <returns>The retrieved <typeparamref name="T"/>.</returns>
|
|
/// <response code="200">The <typeparamref name="T"/> exist and is returned.</response>
|
|
/// <response code="404">A resource with the ID <paramref name="id"/> does not exist.</response>
|
|
[HttpGet("{id:int}")]
|
|
[PartialPermission(Kind.Read)]
|
|
public virtual async Task<ActionResult<T>> Get(int id)
|
|
{
|
|
T ret = await _repository.GetOrDefault(id);
|
|
if (ret == null)
|
|
return NotFound();
|
|
return ret;
|
|
}
|
|
|
|
[HttpGet("{slug}")]
|
|
[PartialPermission(Kind.Read)]
|
|
public virtual async Task<ActionResult<T>> Get(string slug)
|
|
{
|
|
T ret = await _repository.GetOrDefault(slug);
|
|
if (ret == null)
|
|
return NotFound();
|
|
return ret;
|
|
}
|
|
|
|
[HttpGet("count")]
|
|
[PartialPermission(Kind.Read)]
|
|
public virtual async Task<ActionResult<int>> GetCount([FromQuery] Dictionary<string, string> where)
|
|
{
|
|
try
|
|
{
|
|
return await _repository.GetCount(ApiHelper.ParseWhere<T>(where));
|
|
}
|
|
catch (ArgumentException ex)
|
|
{
|
|
return BadRequest(new { Error = ex.Message });
|
|
}
|
|
}
|
|
|
|
[HttpGet]
|
|
[PartialPermission(Kind.Read)]
|
|
public virtual async Task<ActionResult<Page<T>>> GetAll([FromQuery] string sortBy,
|
|
[FromQuery] int afterID,
|
|
[FromQuery] Dictionary<string, string> where,
|
|
[FromQuery] int limit = 20)
|
|
{
|
|
try
|
|
{
|
|
ICollection<T> resources = await _repository.GetAll(ApiHelper.ParseWhere<T>(where),
|
|
new Sort<T>(sortBy),
|
|
new Pagination(limit, afterID));
|
|
|
|
return Page(resources, limit);
|
|
}
|
|
catch (ArgumentException ex)
|
|
{
|
|
return BadRequest(new { Error = ex.Message });
|
|
}
|
|
}
|
|
|
|
protected Page<TResult> Page<TResult>(ICollection<TResult> resources, int limit)
|
|
where TResult : IResource
|
|
{
|
|
return new Page<TResult>(resources,
|
|
new Uri(BaseURL, Request.Path),
|
|
Request.Query.ToDictionary(x => x.Key, x => x.Value.ToString(), StringComparer.InvariantCultureIgnoreCase),
|
|
limit);
|
|
}
|
|
|
|
[HttpPost]
|
|
[PartialPermission(Kind.Create)]
|
|
public virtual async Task<ActionResult<T>> Create([FromBody] T resource)
|
|
{
|
|
try
|
|
{
|
|
return await _repository.Create(resource);
|
|
}
|
|
catch (ArgumentException ex)
|
|
{
|
|
return BadRequest(new { Error = ex.Message });
|
|
}
|
|
catch (DuplicatedItemException)
|
|
{
|
|
T existing = await _repository.GetOrDefault(resource.Slug);
|
|
return Conflict(existing);
|
|
}
|
|
}
|
|
|
|
[HttpPut]
|
|
[PartialPermission(Kind.Write)]
|
|
public virtual async Task<ActionResult<T>> Edit([FromQuery] bool resetOld, [FromBody] T resource)
|
|
{
|
|
try
|
|
{
|
|
if (resource.ID > 0)
|
|
return await _repository.Edit(resource, resetOld);
|
|
|
|
T old = await _repository.Get(resource.Slug);
|
|
resource.ID = old.ID;
|
|
return await _repository.Edit(resource, resetOld);
|
|
}
|
|
catch (ItemNotFoundException)
|
|
{
|
|
return NotFound();
|
|
}
|
|
}
|
|
|
|
[HttpPut("{id:int}")]
|
|
[PartialPermission(Kind.Write)]
|
|
public virtual async Task<ActionResult<T>> Edit(int id, [FromQuery] bool resetOld, [FromBody] T resource)
|
|
{
|
|
resource.ID = id;
|
|
try
|
|
{
|
|
return await _repository.Edit(resource, resetOld);
|
|
}
|
|
catch (ItemNotFoundException)
|
|
{
|
|
return NotFound();
|
|
}
|
|
}
|
|
|
|
[HttpPut("{slug}")]
|
|
[PartialPermission(Kind.Write)]
|
|
public virtual async Task<ActionResult<T>> Edit(string slug, [FromQuery] bool resetOld, [FromBody] T resource)
|
|
{
|
|
try
|
|
{
|
|
T old = await _repository.Get(slug);
|
|
resource.ID = old.ID;
|
|
return await _repository.Edit(resource, resetOld);
|
|
}
|
|
catch (ItemNotFoundException)
|
|
{
|
|
return NotFound();
|
|
}
|
|
}
|
|
|
|
[HttpDelete("{id:int}")]
|
|
[PartialPermission(Kind.Delete)]
|
|
public virtual async Task<IActionResult> Delete(int id)
|
|
{
|
|
try
|
|
{
|
|
await _repository.Delete(id);
|
|
}
|
|
catch (ItemNotFoundException)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
return Ok();
|
|
}
|
|
|
|
[HttpDelete("{slug}")]
|
|
[PartialPermission(Kind.Delete)]
|
|
public virtual async Task<IActionResult> Delete(string slug)
|
|
{
|
|
try
|
|
{
|
|
await _repository.Delete(slug);
|
|
}
|
|
catch (ItemNotFoundException)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
return Ok();
|
|
}
|
|
|
|
[HttpDelete]
|
|
[PartialPermission(Kind.Delete)]
|
|
public virtual async Task<IActionResult> Delete(Dictionary<string, string> where)
|
|
{
|
|
try
|
|
{
|
|
await _repository.DeleteAll(ApiHelper.ParseWhere<T>(where));
|
|
}
|
|
catch (ItemNotFoundException)
|
|
{
|
|
return NotFound();
|
|
}
|
|
|
|
return Ok();
|
|
}
|
|
}
|
|
}
|