stuff
This commit is contained in:
parent
ab2e20f7e1
commit
befaa207d7
23 changed files with 244 additions and 95 deletions
|
@ -1,6 +0,0 @@
|
|||
using Femto.Modules.Blog.Contracts.Dto;
|
||||
using MediatR;
|
||||
|
||||
namespace Femto.Modules.Blog.Domain.Posts.Commands.GetAuthorPosts;
|
||||
|
||||
public record GetAuthorPostsQuery(Guid AuthorId, Guid? Cursor, int? Count) : IRequest<IList<GetAuthorPostsDto>>;
|
|
@ -1,61 +0,0 @@
|
|||
using Dapper;
|
||||
using Femto.Modules.Blog.Contracts.Dto;
|
||||
using Femto.Modules.Blog.Infrastructure.DbConnection;
|
||||
using MediatR;
|
||||
using Microsoft.Data.SqlClient;
|
||||
|
||||
namespace Femto.Modules.Blog.Domain.Posts.Commands.GetAuthorPosts;
|
||||
|
||||
public class GetAuthorPostsQueryHandler(IDbConnectionFactory connectionFactory)
|
||||
: IRequestHandler<GetAuthorPostsQuery, IList<GetAuthorPostsDto>>
|
||||
{
|
||||
public async Task<IList<GetAuthorPostsDto>> Handle(
|
||||
GetAuthorPostsQuery query,
|
||||
CancellationToken cancellationToken
|
||||
)
|
||||
{
|
||||
using var conn = connectionFactory.GetConnection();
|
||||
|
||||
var sql = $$"""
|
||||
with post_page as (
|
||||
select * from blog.post
|
||||
where blog.post.author_id = @authorId
|
||||
and (@cursor is null or blog.post.id < @cursor)
|
||||
order by blog.post.id desc
|
||||
limit @count
|
||||
)
|
||||
select
|
||||
p.id as PostId,
|
||||
p.content as Content,
|
||||
pm.url as MediaUrl
|
||||
from post_page p
|
||||
left join blog.post_media pm on pm.post_id = p.id
|
||||
order by p.id desc
|
||||
""";
|
||||
|
||||
var result = await conn.QueryAsync<QueryResult>(
|
||||
sql,
|
||||
new { authorId = query.AuthorId, cursor = query.Cursor, count = query.Count }
|
||||
);
|
||||
|
||||
return result
|
||||
.GroupBy(row => row.PostId)
|
||||
.Select(group => new GetAuthorPostsDto(
|
||||
group.Key,
|
||||
group.First().Content,
|
||||
group
|
||||
.Select(row => row.MediaUrl)
|
||||
.OfType<string>()
|
||||
.Select(url => new GetAuthorPostsMediaDto(new Uri(url)))
|
||||
.ToList()
|
||||
))
|
||||
.ToList();
|
||||
}
|
||||
|
||||
internal class QueryResult
|
||||
{
|
||||
public Guid PostId { get; set; }
|
||||
public string Content { get; set; }
|
||||
public string? MediaUrl { get; set; }
|
||||
}
|
||||
}
|
|
@ -0,0 +1,3 @@
|
|||
namespace Femto.Modules.Blog.Domain.Posts.Commands.GetPosts.Dto;
|
||||
|
||||
public record GetPostsQueryResult(IList<PostDto> Posts, Guid? Next);
|
|
@ -0,0 +1,3 @@
|
|||
namespace Femto.Modules.Blog.Domain.Posts.Commands.GetPosts.Dto;
|
||||
|
||||
public record PostAuthorDto(Guid AuthorId, string Username);
|
|
@ -0,0 +1,3 @@
|
|||
namespace Femto.Modules.Blog.Domain.Posts.Commands.GetPosts.Dto;
|
||||
|
||||
public record PostDto(Guid PostId, string Text, IList<PostMediaDto> Media, DateTime CreatedAt, PostAuthorDto Author);
|
|
@ -0,0 +1,3 @@
|
|||
namespace Femto.Modules.Blog.Domain.Posts.Commands.GetPosts.Dto;
|
||||
|
||||
public record PostMediaDto(Uri Url);
|
|
@ -0,0 +1,25 @@
|
|||
using Femto.Modules.Blog.Domain.Posts.Commands.GetPosts.Dto;
|
||||
using MediatR;
|
||||
|
||||
namespace Femto.Modules.Blog.Domain.Posts.Commands.GetPosts;
|
||||
|
||||
public class GetPostsQuery : IRequest<GetPostsQueryResult>
|
||||
{
|
||||
|
||||
public string? Username { get; init; }
|
||||
public Guid? From { get; init; }
|
||||
public int Amount { get; init; } = 20;
|
||||
public Guid? AuthorGuid { get; init; }
|
||||
|
||||
/// <summary>
|
||||
/// Default is to load in reverse chronological order
|
||||
/// TODO this is not exposed on the client as it probably wouldn't work that well
|
||||
/// </summary>
|
||||
public GetPostsDirection Direction { get; init; } = GetPostsDirection.Backward;
|
||||
}
|
||||
|
||||
public enum GetPostsDirection
|
||||
{
|
||||
Forward,
|
||||
Backward
|
||||
}
|
|
@ -0,0 +1,99 @@
|
|||
using Dapper;
|
||||
using Femto.Modules.Blog.Domain.Posts.Commands.GetPosts.Dto;
|
||||
using Femto.Modules.Blog.Infrastructure.DbConnection;
|
||||
using MediatR;
|
||||
|
||||
namespace Femto.Modules.Blog.Domain.Posts.Commands.GetPosts;
|
||||
|
||||
public class GetPostsQueryHandler(IDbConnectionFactory connectionFactory)
|
||||
: IRequestHandler<GetPostsQuery, GetPostsQueryResult>
|
||||
{
|
||||
public async Task<GetPostsQueryResult> Handle(
|
||||
GetPostsQuery query,
|
||||
CancellationToken cancellationToken
|
||||
)
|
||||
{
|
||||
using var conn = connectionFactory.GetConnection();
|
||||
|
||||
if (query.Username is not null && query.AuthorGuid is not null)
|
||||
throw new ArgumentException(
|
||||
"Cannot specify both username and authorGuid",
|
||||
nameof(query)
|
||||
);
|
||||
|
||||
var orderBy = query.Direction is GetPostsDirection.Backward ? "desc" : "asc";
|
||||
var pageFilter = query.Direction is GetPostsDirection.Backward ? "<=" : ">=";
|
||||
|
||||
// lang=sql
|
||||
var sql = $$"""
|
||||
with page as (
|
||||
select blog.post.*, blog.author.username as Username, blog.author.id as AuthorId
|
||||
from blog.post
|
||||
inner join blog.author on blog.author.id = blog.post.author_id
|
||||
where (@username is null or blog.author.username = @username)
|
||||
and (@authorGuid is null or blog.author.id = @authorGuid)
|
||||
and (@cursor is null or blog.post.id {{pageFilter}} @cursor)
|
||||
order by blog.post.id {{orderBy}}
|
||||
limit @amount
|
||||
)
|
||||
select
|
||||
page.id as PostId,
|
||||
page.content as Content,
|
||||
blog.post_media.url as MediaUrl,
|
||||
page.created_on as CreatedAt,
|
||||
page.Username,
|
||||
page.AuthorId
|
||||
from page
|
||||
left join blog.post_media on blog.post_media.post_id = page.id
|
||||
order by page.id {{orderBy}}
|
||||
""";
|
||||
|
||||
var result = await conn.QueryAsync<QueryResult>(
|
||||
sql,
|
||||
new
|
||||
{
|
||||
username = query.Username,
|
||||
authorGuid = query.AuthorGuid,
|
||||
cursor = query.From,
|
||||
// load an extra one to take for the curst
|
||||
amount = query.Amount + 1,
|
||||
}
|
||||
);
|
||||
|
||||
var rows = result.ToList();
|
||||
|
||||
var posts = rows.GroupBy(row => row.PostId)
|
||||
.Select(group =>
|
||||
{
|
||||
var postId = group.Key;
|
||||
var post = group.First();
|
||||
var media = group
|
||||
.Select(row => row.MediaUrl)
|
||||
.OfType<string>()
|
||||
.Select(url => new PostMediaDto(new Uri(url)))
|
||||
.ToList();
|
||||
return new PostDto(
|
||||
postId,
|
||||
post.Content,
|
||||
media,
|
||||
post.CreatedAt,
|
||||
new PostAuthorDto(post.AuthorId, post.Username)
|
||||
);
|
||||
})
|
||||
.ToList();
|
||||
|
||||
var next = rows.Count >= query.Amount ? rows.LastOrDefault()?.PostId : null;
|
||||
|
||||
return new GetPostsQueryResult(posts, next);
|
||||
}
|
||||
|
||||
internal class QueryResult
|
||||
{
|
||||
public Guid PostId { get; set; }
|
||||
public string Content { get; set; }
|
||||
public string? MediaUrl { get; set; }
|
||||
public DateTime CreatedAt { get; set; }
|
||||
public Guid AuthorId { get; set; }
|
||||
public string Username { get; set; }
|
||||
}
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue