using System.Net.Mime; using Iceshrimp.Backend.Controllers.Attributes; using Iceshrimp.Backend.Controllers.Mastodon.Attributes; using Iceshrimp.Backend.Controllers.Mastodon.Renderers; using Iceshrimp.Backend.Controllers.Mastodon.Schemas; using Iceshrimp.Backend.Controllers.Mastodon.Schemas.Entities; using Iceshrimp.Backend.Core.Database; using Iceshrimp.Backend.Core.Database.Tables; using Iceshrimp.Backend.Core.Extensions; using Iceshrimp.Backend.Core.Middleware; using Microsoft.AspNetCore.Cors; using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.RateLimiting; using Microsoft.EntityFrameworkCore; using Microsoft.Extensions.Caching.Distributed; namespace Iceshrimp.Backend.Controllers.Mastodon; [MastodonApiController] [Route("/api/v1/timelines")] [Authenticate] [LinkPagination(20, 40)] [EnableRateLimiting("sliding")] [EnableCors("mastodon")] [Produces(MediaTypeNames.Application.Json)] public class TimelineController(DatabaseContext db, NoteRenderer noteRenderer, IDistributedCache cache) : ControllerBase { [Authorize("read:statuses")] [HttpGet("home")] [ProducesResponseType(StatusCodes.Status200OK, Type = typeof(IEnumerable))] public async Task GetHomeTimeline(MastodonPaginationQuery query) { var user = HttpContext.GetUserOrFail(); var heuristic = await QueryableExtensions.GetHeuristic(user, db, cache); var res = await db.Notes .IncludeCommonProperties() .FilterByFollowingAndOwn(user, db, heuristic) .EnsureVisibleFor(user) .FilterHiddenListMembers(user) .FilterBlocked(user) .FilterMuted(user) .Paginate(query, ControllerContext) .PrecomputeVisibilities(user) .RenderAllForMastodonAsync(noteRenderer, user); return Ok(res); } [Authorize("read:statuses")] [HttpGet("public")] [ProducesResponseType(StatusCodes.Status200OK, Type = typeof(IEnumerable))] public async Task GetPublicTimeline( MastodonPaginationQuery query, TimelineSchemas.PublicTimelineRequest request ) { var user = HttpContext.GetUserOrFail(); var res = await db.Notes .IncludeCommonProperties() .HasVisibility(Note.NoteVisibility.Public) .FilterByPublicTimelineRequest(request) .FilterBlocked(user) .FilterMuted(user) .Paginate(query, ControllerContext) .PrecomputeVisibilities(user) .RenderAllForMastodonAsync(noteRenderer, user); return Ok(res); } [Authorize("read:statuses")] [HttpGet("tag/{hashtag}")] [ProducesResponseType(StatusCodes.Status200OK, Type = typeof(IEnumerable))] public async Task GetHashtagTimeline( string hashtag, MastodonPaginationQuery query, TimelineSchemas.HashtagTimelineRequest request ) { var user = HttpContext.GetUserOrFail(); var res = await db.Notes .IncludeCommonProperties() .Where(p => p.Tags.Contains(hashtag.ToLowerInvariant())) .FilterByHashtagTimelineRequest(request) .FilterBlocked(user) .FilterMuted(user) .Paginate(query, ControllerContext) .PrecomputeVisibilities(user) .RenderAllForMastodonAsync(noteRenderer, user); return Ok(res); } [Authorize("read:lists")] [HttpGet("list/{id}")] [ProducesResponseType(StatusCodes.Status200OK, Type = typeof(IEnumerable))] public async Task GetListTimeline(string id, MastodonPaginationQuery query) { var user = HttpContext.GetUserOrFail(); if (!await db.UserLists.AnyAsync(p => p.Id == id && p.User == user)) throw GracefulException.RecordNotFound(); var res = await db.Notes .IncludeCommonProperties() .Where(p => db.UserListMembers.Any(l => l.UserListId == id && l.UserId == p.UserId)) .EnsureVisibleFor(user) .FilterBlocked(user) .FilterMuted(user) .Paginate(query, ControllerContext) .PrecomputeVisibilities(user) .RenderAllForMastodonAsync(noteRenderer, user); return Ok(res); } }