285 lines
9.0 KiB
C#
285 lines
9.0 KiB
C#
using DysonNetwork.Shared.Data;
|
|
using DysonNetwork.Shared.Proto;
|
|
using Microsoft.AspNetCore.Authorization;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.EntityFrameworkCore;
|
|
|
|
namespace DysonNetwork.Sphere.Post;
|
|
|
|
[ApiController]
|
|
[Route("/api/posts")]
|
|
public class PostCategoryController(AppDatabase db) : ControllerBase
|
|
{
|
|
[HttpGet("categories")]
|
|
public async Task<ActionResult<List<PostCategory>>> ListCategories(
|
|
[FromQuery] string? query = null,
|
|
[FromQuery] int offset = 0,
|
|
[FromQuery] int take = 20,
|
|
[FromQuery] string? order = null
|
|
)
|
|
{
|
|
var categoriesQuery = db.PostCategories
|
|
.OrderBy(e => e.Name)
|
|
.AsQueryable();
|
|
|
|
if (!string.IsNullOrEmpty(query))
|
|
categoriesQuery = categoriesQuery
|
|
.Where(e => EF.Functions.ILike(e.Slug, $"%{query}%"));
|
|
if (!string.IsNullOrEmpty(order))
|
|
{
|
|
categoriesQuery = order switch
|
|
{
|
|
"usage" => categoriesQuery.OrderByDescending(e => e.Posts.Count),
|
|
_ => categoriesQuery.OrderByDescending(e => e.CreatedAt)
|
|
};
|
|
}
|
|
|
|
var totalCount = await categoriesQuery.CountAsync();
|
|
Response.Headers.Append("X-Total", totalCount.ToString());
|
|
|
|
// Get categories with their post counts in a single query
|
|
var categories = await categoriesQuery
|
|
.Skip(offset)
|
|
.Take(take)
|
|
.Select(c => new
|
|
{
|
|
Category = c,
|
|
PostCount = c.Posts.Count
|
|
})
|
|
.ToListAsync();
|
|
|
|
// Project results back to the original type and set the Usage property
|
|
var result = categories.Select(x =>
|
|
{
|
|
x.Category.Usage = x.PostCount;
|
|
return x.Category;
|
|
}).ToList();
|
|
|
|
return Ok(result);
|
|
}
|
|
|
|
[HttpGet("tags")]
|
|
public async Task<ActionResult<List<PostTag>>> ListTags(
|
|
[FromQuery] string? query = null,
|
|
[FromQuery] int offset = 0,
|
|
[FromQuery] int take = 20,
|
|
[FromQuery] string? order = null
|
|
)
|
|
{
|
|
var tagsQuery = db.PostTags
|
|
.OrderBy(e => e.Name)
|
|
.AsQueryable();
|
|
|
|
if (!string.IsNullOrEmpty(query))
|
|
tagsQuery = tagsQuery
|
|
.Where(e => EF.Functions.ILike(e.Slug, $"%{query}%"));
|
|
if (!string.IsNullOrEmpty(order))
|
|
{
|
|
tagsQuery = order switch
|
|
{
|
|
"usage" => tagsQuery.OrderByDescending(e => e.Posts.Count),
|
|
_ => tagsQuery.OrderByDescending(e => e.CreatedAt)
|
|
};
|
|
}
|
|
|
|
var totalCount = await tagsQuery.CountAsync();
|
|
Response.Headers.Append("X-Total", totalCount.ToString());
|
|
|
|
// Get tags with their post counts in a single query
|
|
var tags = await tagsQuery
|
|
.Skip(offset)
|
|
.Take(take)
|
|
.Select(t => new
|
|
{
|
|
Tag = t,
|
|
PostCount = t.Posts.Count
|
|
})
|
|
.ToListAsync();
|
|
|
|
// Project results back to the original type and set the Usage property
|
|
var result = tags.Select(x =>
|
|
{
|
|
x.Tag.Usage = x.PostCount;
|
|
return x.Tag;
|
|
}).ToList();
|
|
|
|
return Ok(result);
|
|
}
|
|
|
|
[HttpGet("categories/{slug}")]
|
|
public async Task<ActionResult<PostCategory>> GetCategory(string slug)
|
|
{
|
|
var category = await db.PostCategories.FirstOrDefaultAsync(e => e.Slug == slug);
|
|
if (category is null)
|
|
return NotFound();
|
|
return Ok(category);
|
|
}
|
|
|
|
[HttpGet("tags/{slug}")]
|
|
public async Task<ActionResult<PostTag>> GetTag(string slug)
|
|
{
|
|
var tag = await db.PostTags.FirstOrDefaultAsync(e => e.Slug == slug);
|
|
if (tag is null)
|
|
return NotFound();
|
|
return Ok(tag);
|
|
}
|
|
|
|
[HttpPost("categories/{slug}/subscribe")]
|
|
[Authorize]
|
|
public async Task<ActionResult<PostCategorySubscription>> SubscribeCategory(string slug)
|
|
{
|
|
if (HttpContext.Items["CurrentUser"] is not Account currentUser) return Unauthorized();
|
|
var accountId = Guid.Parse(currentUser.Id);
|
|
|
|
var category = await db.PostCategories.FirstOrDefaultAsync(c => c.Slug == slug);
|
|
if (category == null)
|
|
{
|
|
return NotFound("Category not found.");
|
|
}
|
|
|
|
var existingSubscription = await db.PostCategorySubscriptions
|
|
.FirstOrDefaultAsync(s => s.CategoryId == category.Id && s.AccountId == accountId);
|
|
|
|
if (existingSubscription != null)
|
|
return Ok(existingSubscription);
|
|
|
|
var subscription = new PostCategorySubscription
|
|
{
|
|
AccountId = accountId,
|
|
CategoryId = category.Id
|
|
};
|
|
|
|
db.PostCategorySubscriptions.Add(subscription);
|
|
await db.SaveChangesAsync();
|
|
|
|
return CreatedAtAction(nameof(GetCategorySubscription), new { slug }, subscription);
|
|
}
|
|
|
|
[HttpPost("categories/{slug}/unsubscribe")]
|
|
[Authorize]
|
|
public async Task<IActionResult> UnsubscribeCategory(string slug)
|
|
{
|
|
if (HttpContext.Items["CurrentUser"] is not Account currentUser) return Unauthorized();
|
|
var accountId = Guid.Parse(currentUser.Id);
|
|
|
|
var category = await db.PostCategories.FirstOrDefaultAsync(c => c.Slug == slug);
|
|
if (category == null)
|
|
return NotFound("Category not found.");
|
|
|
|
var subscription = await db.PostCategorySubscriptions
|
|
.FirstOrDefaultAsync(s => s.CategoryId == category.Id && s.AccountId == accountId);
|
|
|
|
if (subscription == null)
|
|
return NoContent();
|
|
|
|
db.PostCategorySubscriptions.Remove(subscription);
|
|
await db.SaveChangesAsync();
|
|
|
|
return NoContent();
|
|
}
|
|
|
|
[HttpGet("categories/{slug}/subscription")]
|
|
[Authorize]
|
|
public async Task<ActionResult<PostCategorySubscription>> GetCategorySubscription(string slug)
|
|
{
|
|
if (HttpContext.Items["CurrentUser"] is not Account currentUser) return Unauthorized();
|
|
var accountId = Guid.Parse(currentUser.Id);
|
|
|
|
var category = await db.PostCategories.FirstOrDefaultAsync(c => c.Slug == slug);
|
|
if (category == null)
|
|
return NotFound("Category not found.");
|
|
|
|
var subscription = await db.PostCategorySubscriptions
|
|
.FirstOrDefaultAsync(s => s.CategoryId == category.Id && s.AccountId == accountId);
|
|
|
|
if (subscription == null)
|
|
return NotFound("Subscription not found.");
|
|
|
|
return Ok(subscription);
|
|
}
|
|
|
|
[HttpPost("tags/{slug}/subscribe")]
|
|
[Authorize]
|
|
public async Task<ActionResult<PostCategorySubscription>> SubscribeTag(string slug)
|
|
{
|
|
if (HttpContext.Items["CurrentUser"] is not Account currentUser) return Unauthorized();
|
|
var accountId = Guid.Parse(currentUser.Id);
|
|
|
|
var tag = await db.PostTags.FirstOrDefaultAsync(t => t.Slug == slug);
|
|
if (tag == null)
|
|
{
|
|
return NotFound("Tag not found.");
|
|
}
|
|
|
|
var existingSubscription = await db.PostCategorySubscriptions
|
|
.FirstOrDefaultAsync(s => s.TagId == tag.Id && s.AccountId == accountId);
|
|
|
|
if (existingSubscription != null)
|
|
{
|
|
return Ok(existingSubscription);
|
|
}
|
|
|
|
var subscription = new PostCategorySubscription
|
|
{
|
|
AccountId = accountId,
|
|
TagId = tag.Id
|
|
};
|
|
|
|
db.PostCategorySubscriptions.Add(subscription);
|
|
await db.SaveChangesAsync();
|
|
|
|
return CreatedAtAction(nameof(GetTagSubscription), new { slug }, subscription);
|
|
}
|
|
|
|
[HttpPost("tags/{slug}/unsubscribe")]
|
|
[Authorize]
|
|
public async Task<IActionResult> UnsubscribeTag(string slug)
|
|
{
|
|
if (HttpContext.Items["CurrentUser"] is not Account currentUser) return Unauthorized();
|
|
var accountId = Guid.Parse(currentUser.Id);
|
|
|
|
var tag = await db.PostTags.FirstOrDefaultAsync(t => t.Slug == slug);
|
|
if (tag == null)
|
|
{
|
|
return NotFound("Tag not found.");
|
|
}
|
|
|
|
var subscription = await db.PostCategorySubscriptions
|
|
.FirstOrDefaultAsync(s => s.TagId == tag.Id && s.AccountId == accountId);
|
|
|
|
if (subscription == null)
|
|
{
|
|
return NoContent();
|
|
}
|
|
|
|
db.PostCategorySubscriptions.Remove(subscription);
|
|
await db.SaveChangesAsync();
|
|
|
|
return NoContent();
|
|
}
|
|
|
|
[HttpGet("tags/{slug}/subscription")]
|
|
[Authorize]
|
|
public async Task<ActionResult<PostCategorySubscription>> GetTagSubscription(string slug)
|
|
{
|
|
if (HttpContext.Items["CurrentUser"] is not Account currentUser) return Unauthorized();
|
|
var accountId = Guid.Parse(currentUser.Id);
|
|
|
|
var tag = await db.PostTags.FirstOrDefaultAsync(t => t.Slug == slug);
|
|
if (tag == null)
|
|
{
|
|
return NotFound("Tag not found.");
|
|
}
|
|
|
|
var subscription = await db.PostCategorySubscriptions
|
|
.FirstOrDefaultAsync(s => s.TagId == tag.Id && s.AccountId == accountId);
|
|
|
|
if (subscription == null)
|
|
{
|
|
return NotFound("Subscription not found.");
|
|
}
|
|
|
|
return Ok(subscription);
|
|
}
|
|
}
|