Kavita/Kavita.Server/Controllers/FilterController.cs
2026-04-18 13:35:34 -07:00

295 lines
10 KiB
C#

using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.Linq;
using System.Threading.Tasks;
using Kavita.API.Database;
using Kavita.API.Repositories;
using Kavita.API.Services;
using Kavita.Common;
using Kavita.Models;
using Kavita.Models.Constants;
using Kavita.Models.DTOs.Dashboard;
using Kavita.Models.DTOs.Filtering.v2;
using Kavita.Models.DTOs.Filtering.v2.Requests;
using Kavita.Models.Entities.User;
using Kavita.Server.Attributes;
using Kavita.Services.Helpers.SmartFilter;
using Microsoft.AspNetCore.Mvc;
using Microsoft.Extensions.Logging;
namespace Kavita.Server.Controllers;
public class FilterController(
IUnitOfWork unitOfWork,
ILocalizationService localizationService,
IStreamService streamService,
ILogger<FilterController> logger)
: BaseApiController
{
/// <summary>
/// Creates or Updates the Series filter
/// </summary>
/// <param name="dto"></param>
/// <returns></returns>
[HttpPost("update/series")]
[DisallowRole(PolicyConstants.ReadOnlyRole)]
public async Task<ActionResult> CreateOrUpdateSeriesSmartFilter(SeriesFilterV2Dto dto)
{
try
{
if (string.IsNullOrEmpty(dto.Name)) return BadRequest("Name is required");
var encodedString = SmartFilterHelper.Encode(dto);
await ValidateAndSaveFilterUpsert(dto.Name!, encodedString, dto.EntityType);
return Ok();
}
catch (KavitaException ex)
{
return BadRequest(ex.Message);
}
}
/// <summary>
/// Creates or Updates the Reading List filter
/// </summary>
/// <param name="dto"></param>
/// <returns></returns>
[HttpPost("update/reading-list")]
[DisallowRole(PolicyConstants.ReadOnlyRole)]
public async Task<ActionResult> CreateOrUpdateReadingListSmartFilter(ReadingListFilterDto dto)
{
try
{
if (string.IsNullOrEmpty(dto.Name)) return BadRequest("Name is required");
var encodedString = SmartFilterHelper.Encode(dto);
await ValidateAndSaveFilterUpsert(dto.Name!, encodedString, dto.EntityType);
return Ok();
}
catch (KavitaException ex)
{
return BadRequest(ex.Message);
}
}
/// <summary>
/// Creates or Updates the Person filter
/// </summary>
/// <param name="dto"></param>
/// <returns></returns>
[HttpPost("update/person")]
[DisallowRole(PolicyConstants.ReadOnlyRole)]
public async Task<ActionResult> CreateOrUpdatePersonSmartFilter(PersonFilterDto dto)
{
try
{
if (string.IsNullOrEmpty(dto.Name)) return BadRequest("Name is required");
var encodedString = SmartFilterHelper.Encode(dto);
await ValidateAndSaveFilterUpsert(dto.Name!, encodedString, dto.EntityType);
return Ok();
}
catch (KavitaException ex)
{
return BadRequest(ex.Message);
}
}
/// <summary>
/// Creates or Updates the Reading List filter
/// </summary>
/// <param name="dto"></param>
/// <returns></returns>
[HttpPost("update/annotation")]
[DisallowRole(PolicyConstants.ReadOnlyRole)]
public async Task<ActionResult> CreateOrUpdateAnnotationSmartFilter(AnnotationFilterDto dto)
{
try
{
if (string.IsNullOrEmpty(dto.Name)) return BadRequest("Name is required");
var encodedString = SmartFilterHelper.Encode(dto);
await ValidateAndSaveFilterUpsert(dto.Name!, encodedString, dto.EntityType);
return Ok();
}
catch (KavitaException ex)
{
return BadRequest(ex.Message);
}
}
private async Task ValidateAndSaveFilterUpsert(string filterName, string encodedFilter, FilterEntityType entityType)
{
var user = (await unitOfWork.UserRepository.GetUserByIdAsync(UserId, AppUserIncludes.SmartFilters))!;
if (string.IsNullOrWhiteSpace(filterName)) throw new KavitaException("Name must be set");
if (Defaults.DefaultStreams.Any(s => s.Name.Equals(filterName, StringComparison.InvariantCultureIgnoreCase)))
{
// NOTE: This checks against localization keys (on-deck), so this case will almost never happen
throw new KavitaException("You cannot use the name of a system provided stream");
}
var existingFilter = user.SmartFilters.FirstOrDefault(s => s.Name.Equals(filterName, StringComparison.InvariantCultureIgnoreCase));
if (existingFilter != null)
{
// Update the filter
existingFilter.Filter = encodedFilter;
unitOfWork.AppUserSmartFilterRepository.Update(existingFilter);
}
else
{
existingFilter = new AppUserSmartFilter()
{
Name = filterName,
Filter = encodedFilter,
EntityType = entityType
};
user.SmartFilters.Add(existingFilter);
unitOfWork.UserRepository.Update(user);
}
if (!unitOfWork.HasChanges()) return;
await unitOfWork.CommitAsync();
}
/// <summary>
/// All Smart Filters for the authenticated user
/// </summary>
/// <returns></returns>
[HttpGet]
public async Task<ActionResult<IEnumerable<SmartFilterDto>>> GetFilters()
{
return Ok(await unitOfWork.AppUserSmartFilterRepository.GetAllDtosByUserId(UserId));
}
/// <summary>
/// Delete the smart filter for the authenticated user
/// </summary>
/// <remarks>User must not be in <see cref="PolicyConstants.ReadOnlyRole"/></remarks>
/// <param name="filterId"></param>
/// <returns></returns>
[HttpDelete]
[DisallowRole(PolicyConstants.ReadOnlyRole)]
public async Task<ActionResult> DeleteFilter(int filterId)
{
var filter = await unitOfWork.AppUserSmartFilterRepository.GetById(filterId);
if (filter == null) return Ok();
// This needs to delete any dashboard filters that have it too
var streams = await unitOfWork.UserRepository.GetDashboardStreamWithFilter(filter.Id);
unitOfWork.UserRepository.Delete(streams);
var streams2 = await unitOfWork.UserRepository.GetSideNavStreamWithFilter(filter.Id);
unitOfWork.UserRepository.Delete(streams2);
unitOfWork.AppUserSmartFilterRepository.Delete(filter);
await unitOfWork.CommitAsync();
return Ok();
}
/// <summary>
/// Encode a Series filter
/// </summary>
/// <param name="dto">This must be entityType Series</param>
/// <returns></returns>
[HttpPost("encode/series")]
public ActionResult<string> EncodeSeriesFilter(SeriesFilterV2Dto dto)
{
return Ok(SmartFilterHelper.Encode(dto));
}
/// <summary>
/// Encode a Reading List filter
/// </summary>
/// <param name="dto">This must be entityType ReadingList</param>
/// <returns></returns>
[HttpPost("encode/reading-list")]
public ActionResult<string> EncodeRlFilter(ReadingListFilterDto dto)
{
return Ok(SmartFilterHelper.Encode(dto));
}
/// <summary>
/// Encode a Person Filter
/// </summary>
/// <param name="dto">This must be entityType Person</param>
/// <returns></returns>
[HttpPost("encode/person")]
public ActionResult<string> EncodePersonFilter(PersonFilterDto dto)
{
return Ok(SmartFilterHelper.Encode(dto));
}
/// <summary>
/// Encode an Annotation Filter
/// </summary>
/// <param name="dto">This must be entityType Annotation</param>
/// <returns></returns>
[HttpPost("encode/annotation")]
public ActionResult<string> EncodeAnnotationFilter(AnnotationFilterDto dto)
{
return Ok(SmartFilterHelper.Encode(dto));
}
/// <summary>
/// Decodes the Filter
/// </summary>
/// <remarks>Decoded filter will always have the same shape of <see cref="IFilterDto{TStatement,TSortOption}"/>.
/// The concrete class is driven by <c>EntityType</c>.
/// Classes: <see cref="SeriesFilterV2Dto"/>, <see cref="PersonFilterDto"/>, <see cref="AnnotationFilterDto"/>, <see cref="ReadingListFilterDto"/>
/// </remarks>
/// <param name="dto"></param>
/// <returns></returns>
[HttpPost("decode")]
public ActionResult<IFilterDto> DecodeFilter(DecodeFilterDto dto)
{
return Ok(SmartFilterHelper.Decode(dto.EncodedFilter));
}
/// <summary>
/// Rename a Smart Filter given the filterId and new name
/// </summary>
/// <param name="filterId"></param>
/// <param name="name"></param>
/// <returns></returns>
[HttpPost("rename")]
[DisallowRole(PolicyConstants.ReadOnlyRole)]
public async Task<ActionResult> RenameFilter([FromQuery] int filterId, [FromQuery] [Required] string name)
{
try
{
var user = await unitOfWork.UserRepository.GetUserByIdAsync(UserId, AppUserIncludes.SmartFilters);
if (user == null) return Unauthorized();
name = name.Trim();
if (string.IsNullOrWhiteSpace(name))
{
return BadRequest(await localizationService.TranslateAsync(user.Id, "smart-filter-name-required"));
}
if (Defaults.DefaultStreams.Any(s => s.Name.Equals(name, StringComparison.InvariantCultureIgnoreCase)))
{
return BadRequest(await localizationService.TranslateAsync(user.Id, "smart-filter-system-name"));
}
var filter = user.SmartFilters.FirstOrDefault(f => f.Id == filterId);
if (filter == null)
{
return BadRequest(await localizationService.TranslateAsync(user.Id, "filter-not-found"));
}
filter.Name = name;
unitOfWork.AppUserSmartFilterRepository.Update(filter);
await unitOfWork.CommitAsync();
await streamService.RenameSmartFilterStreams(filter);
return Ok();
}
catch (Exception ex)
{
logger.LogError(ex, "There was an exception when renaming smart filter: {FilterId}", filterId);
return BadRequest(await localizationService.TranslateAsync(UserId, "generic-error"));
}
}
}