return postdto from post create

This commit is contained in:
john 2025-05-28 20:05:00 +02:00
parent 8ad4302ec8
commit d1687f276b
15 changed files with 226 additions and 111 deletions

View file

@ -1,3 +1,3 @@
namespace Femto.Api.Controllers.Posts.Dto; namespace Femto.Api.Controllers.Posts.Dto;
public record CreatePostResponse(Guid PostId); public record CreatePostResponse(PostDto Post);

View file

@ -9,5 +9,6 @@ public record PostDto(
string Content, string Content,
IEnumerable<PostMediaDto> Media, IEnumerable<PostMediaDto> Media,
IEnumerable<PostReactionDto> Reactions, IEnumerable<PostReactionDto> Reactions,
DateTimeOffset CreatedAt DateTimeOffset CreatedAt,
IEnumerable<string> PossibleReactions
); );

View file

@ -1,3 +1,3 @@
namespace Femto.Api.Controllers.Posts.Dto; namespace Femto.Api.Controllers.Posts.Dto;
public record PostReactionDto(Guid ReactionId, string Emoji, int Count, bool DidReact); public record PostReactionDto(string Emoji, int Count, bool DidReact);

View file

@ -11,7 +11,8 @@ namespace Femto.Api.Controllers.Posts;
[ApiController] [ApiController]
[Route("posts")] [Route("posts")]
public class PostsController(IBlogModule blogModule, ICurrentUserContext currentUserContext) : ControllerBase public class PostsController(IBlogModule blogModule, ICurrentUserContext currentUserContext)
: ControllerBase
{ {
[HttpGet] [HttpGet]
public async Task<ActionResult<LoadPostsResponse>> LoadPosts( public async Task<ActionResult<LoadPostsResponse>> LoadPosts(
@ -36,8 +37,9 @@ public class PostsController(IBlogModule blogModule, ICurrentUserContext current
p.PostId, p.PostId,
p.Text, p.Text,
p.Media.Select(m => new PostMediaDto(m.Url, m.Width, m.Height)), p.Media.Select(m => new PostMediaDto(m.Url, m.Width, m.Height)),
p.Reactions?.Select(r => new PostReactionDto(r.ReactionId, r.Emoji, r.Count, r.DidReact)), p.Reactions.Select(r => new PostReactionDto(r.Emoji, r.Count, r.DidReact)),
p.CreatedAt p.CreatedAt,
p.PossibleReactions
)), )),
res.Next res.Next
); );
@ -50,7 +52,7 @@ public class PostsController(IBlogModule blogModule, ICurrentUserContext current
CancellationToken cancellationToken CancellationToken cancellationToken
) )
{ {
var guid = await blogModule.Command( var post = await blogModule.Command(
new CreatePostCommand( new CreatePostCommand(
req.AuthorId, req.AuthorId,
req.Content, req.Content,
@ -65,18 +67,32 @@ public class PostsController(IBlogModule blogModule, ICurrentUserContext current
media.Height media.Height
) )
), ),
req.IsPublic req.IsPublic,
currentUserContext.CurrentUser!
), ),
cancellationToken cancellationToken
); );
return new CreatePostResponse(guid); return new CreatePostResponse(
new PostDto(
new PostAuthorDto(post.Author.AuthorId, post.Author.Username),
post.PostId,
post.Text,
post.Media.Select(m => new PostMediaDto(m.Url, m.Width, m.Height)),
post.Reactions.Select(r => new PostReactionDto(r.Emoji, r.Count, r.DidReact)).ToList(),
post.CreatedAt,
post.PossibleReactions
)
);
} }
[HttpDelete("{postId}")] [HttpDelete("{postId}")]
[Authorize] [Authorize]
public async Task DeletePost(Guid postId, CancellationToken cancellationToken) public async Task DeletePost(Guid postId, CancellationToken cancellationToken)
{ {
await blogModule.Command(new DeletePostCommand(postId, currentUserContext.CurrentUser!.Id), cancellationToken); await blogModule.Command(
new DeletePostCommand(postId, currentUserContext.CurrentUser!.Id),
cancellationToken
);
} }
} }

View file

@ -6,4 +6,4 @@ CREATE TABLE authn.user_role
user_id uuid REFERENCES authn.user_identity(id), user_id uuid REFERENCES authn.user_identity(id),
role int, role int,
primary key (user_id, role) primary key (user_id, role)
); )

View file

@ -8,5 +8,6 @@ CREATE TABLE blog.post_reaction
( (
post_id uuid REFERENCES blog.post(id), post_id uuid REFERENCES blog.post(id),
author_id uuid REFERENCES blog.author(id), author_id uuid REFERENCES blog.author(id),
emoji text not null emoji text not null,
primary key (post_id, author_id, emoji)
); );

View file

@ -35,12 +35,12 @@ public static class TestDataSeeder
; ;
INSERT INTO blog.post INSERT INTO blog.post
(id, author_id, content) (id, author_id, possible_reactions, content)
VALUES VALUES
('019691a0-48ed-7eba-b8d3-608e25e07d4b', @id, 'However, authors often misinterpret the zoology as a smothered advantage, when in actuality it feels more like a blindfold accordion. They were lost without the chastest puppy that composed their Santa.'), ('019691a0-48ed-7eba-b8d3-608e25e07d4b', @id, '["🍆", "🧢", "🧑🏾‍🎓", "🥕", "🕗"]', 'However, authors often misinterpret the zoology as a smothered advantage, when in actuality it feels more like a blindfold accordion. They were lost without the chastest puppy that composed their Santa.'),
('019691a0-4ace-7bb5-a8f3-e3362920eba0', @id, 'Extending this logic, a swim can hardly be considered a seasick duckling without also being a tornado. Some posit the whity voyage to be less than dippy.'), ('019691a0-4ace-7bb5-a8f3-e3362920eba0', @id, '["🍆", "🧢", "🧑🏾‍🎓", "🥕", "🕗"]', 'Extending this logic, a swim can hardly be considered a seasick duckling without also being a tornado. Some posit the whity voyage to be less than dippy.'),
('019691a0-4c3e-726f-b8f6-bcbaabe789ae', @id,'Few can name a springless sun that isn''t a thudding Vietnam. The burn of a competitor becomes a frosted target.'), ('019691a0-4c3e-726f-b8f6-bcbaabe789ae', @id, '["🍆", "🧢", "🧑🏾‍🎓", "🥕", "🕗"]', 'Few can name a springless sun that isn''t a thudding Vietnam. The burn of a competitor becomes a frosted target.'),
('019691a0-4dd3-7e89-909e-94a6fd19a05e', @id,'Some unwitched marbles are thought of simply as currencies. A boundary sees a nepal as a chordal railway.') ('019691a0-4dd3-7e89-909e-94a6fd19a05e', @id, '["🍆", "🧢", "🧑🏾‍🎓", "🥕", "🕗"]', 'Some unwitched marbles are thought of simply as currencies. A boundary sees a nepal as a chordal railway.')
; ;
INSERT INTO blog.post_media INSERT INTO blog.post_media
@ -54,6 +54,15 @@ public static class TestDataSeeder
('019691b6-2608-7088-8110-f0f6e35fa633', '019691a0-4dd3-7e89-909e-94a6fd19a05e', 'https://www.pinclipart.com/picdir/big/535-5356059_big-transparent-chungus-png-background-big-chungus-clipart.png', 0) ('019691b6-2608-7088-8110-f0f6e35fa633', '019691a0-4dd3-7e89-909e-94a6fd19a05e', 'https://www.pinclipart.com/picdir/big/535-5356059_big-transparent-chungus-png-background-big-chungus-clipart.png', 0)
; ;
INSERT INTO blog.post_reaction
(post_id, author_id, emoji)
VALUES
('019691a0-48ed-7eba-b8d3-608e25e07d4b', @id, '🍆'),
('019691a0-4ace-7bb5-a8f3-e3362920eba0', @id, '🍆'),
('019691a0-4c3e-726f-b8f6-bcbaabe789ae', @id, '🧑🏾'),
('019691a0-4c3e-726f-b8f6-bcbaabe789ae', @id, '🕗')
;
INSERT INTO authn.user_identity INSERT INTO authn.user_identity
(id, username, password_hash, password_salt) (id, username, password_hash, password_salt)
VALUES VALUES

View file

@ -1,8 +1,22 @@
using Femto.Common;
using Femto.Common.Domain; using Femto.Common.Domain;
using Femto.Modules.Blog.Application.Queries.GetPosts.Dto;
namespace Femto.Modules.Blog.Application.Commands.CreatePost; namespace Femto.Modules.Blog.Application.Commands.CreatePost;
public record CreatePostCommand(Guid AuthorId, string Content, IEnumerable<CreatePostMedia> Media, bool? IsPublic) public record CreatePostCommand(
: ICommand<Guid>; Guid AuthorId,
string Content,
IEnumerable<CreatePostMedia> Media,
bool? IsPublic,
CurrentUser CurrentUser
) : ICommand<PostDto>;
public record CreatePostMedia(Guid MediaId, Uri Url, string? Type, int Order, int? Width, int? Height); public record CreatePostMedia(
Guid MediaId,
Uri Url,
string? Type,
int Order,
int? Width,
int? Height
);

View file

@ -1,12 +1,16 @@
using Femto.Modules.Blog.Application.Queries.GetPosts.Dto;
using Femto.Modules.Blog.Domain.Posts; using Femto.Modules.Blog.Domain.Posts;
using MediatR; using MediatR;
namespace Femto.Modules.Blog.Application.Commands.CreatePost; namespace Femto.Modules.Blog.Application.Commands.CreatePost;
internal class CreatePostCommandHandler(BlogContext context) internal class CreatePostCommandHandler(BlogContext context)
: IRequestHandler<CreatePostCommand, Guid> : IRequestHandler<CreatePostCommand, PostDto>
{ {
public async Task<Guid> Handle(CreatePostCommand request, CancellationToken cancellationToken) public async Task<PostDto> Handle(
CreatePostCommand request,
CancellationToken cancellationToken
)
{ {
var post = new Post( var post = new Post(
request.AuthorId, request.AuthorId,
@ -22,11 +26,19 @@ internal class CreatePostCommandHandler(BlogContext context)
)) ))
.ToList() .ToList()
); );
post.IsPublic = request.IsPublic is true; post.IsPublic = request.IsPublic is true;
await context.AddAsync(post, cancellationToken); await context.AddAsync(post, cancellationToken);
return post.Id; return new PostDto(
post.Id,
post.Content,
post.Media.Select(m => new PostMediaDto(m.Url, m.Width, m.Height)).ToList(),
post.PostedOn,
new PostAuthorDto(post.AuthorId, request.CurrentUser.Username),
[],
post.PossibleReactions
);
} }
} }

View file

@ -11,5 +11,10 @@ internal class PostConfiguration : IEntityTypeConfiguration<Post>
table.ToTable("post"); table.ToTable("post");
table.OwnsMany(post => post.Media).WithOwner(); table.OwnsMany(post => post.Media).WithOwner();
table.OwnsMany(post => post.Reactions).WithOwner(); table.OwnsMany(post => post.Reactions).WithOwner();
table.Property<string>("PossibleReactionsJson")
.HasColumnName("possible_reactions");
table.Ignore(e => e.PossibleReactions);
} }
} }

View file

@ -6,4 +6,6 @@ public record PostDto(
IList<PostMediaDto> Media, IList<PostMediaDto> Media,
DateTimeOffset CreatedAt, DateTimeOffset CreatedAt,
PostAuthorDto Author, PostAuthorDto Author,
IList<PostReactionDto> Reactions); IList<PostReactionDto> Reactions,
IEnumerable<string> PossibleReactions
);

View file

@ -1,3 +1,4 @@
using System.Text.Json;
using Dapper; using Dapper;
using Femto.Common.Infrastructure.DbConnection; using Femto.Common.Infrastructure.DbConnection;
using Femto.Modules.Blog.Application.Queries.GetPosts.Dto; using Femto.Modules.Blog.Application.Queries.GetPosts.Dto;
@ -15,42 +16,29 @@ public class GetPostsQueryHandler(IDbConnectionFactory connectionFactory)
{ {
using var conn = connectionFactory.GetConnection(); using var conn = connectionFactory.GetConnection();
var orderBy = query.Direction is GetPostsDirection.Backward ? "desc" : "asc";
var pageFilter = query.Direction is GetPostsDirection.Backward ? "<=" : ">=";
var username = query.Author; var username = query.Author;
var authorGuid = query.AuthorId; var authorGuid = query.AuthorId;
var cursor = query.From; var cursor = query.From;
var showPrivate = query.CurrentUserId is not null; var showPrivate = query.CurrentUserId is not null;
// lang=sql var loadPostsResult = await conn.QueryAsync<LoadPostRow>(
var sql = $$""" """
with page as ( select
select blog.post.*, blog.author.username as Username, blog.author.id as AuthorId blog.post.id as PostId,
from blog.post blog.post.content as Content,
inner join blog.author on blog.author.id = blog.post.author_id blog.post.posted_on as PostedOn,
where (@username is null or blog.author.username = @username) blog.author.username as Username,
and (@showPrivate or blog.post.is_public = true) blog.author.id as AuthorId,
and (@authorGuid is null or blog.author.id = @authorGuid) blog.post.possible_reactions as PossibleReactions
and (@cursor is null or blog.post.id {{pageFilter}} @cursor) from blog.post
order by blog.post.id {{orderBy}} inner join blog.author on blog.author.id = blog.post.author_id
limit @amount where (@username is null or blog.author.username = @username)
) and (@showPrivate or blog.post.is_public = true)
select and (@authorGuid is null or blog.author.id = @authorGuid)
page.id as PostId, and (@cursor is null or blog.post.id <= @cursor)
page.content as Content, order by blog.post.id desc
blog.post_media.url as MediaUrl, limit @amount
blog.post_media.width as MediaWidth, """,
blog.post_media.height as MediaHeight,
page.posted_on as PostedOn,
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 new
{ {
username, username,
@ -62,54 +50,115 @@ public class GetPostsQueryHandler(IDbConnectionFactory connectionFactory)
} }
); );
var rows = result.ToList(); var loadedPosts = loadPostsResult.ToList();
var posts = loadedPosts.Take(query.Amount).ToList();
var next = loadedPosts.LastOrDefault()?.PostId;
var posts = rows.GroupBy(row => row.PostId) var postIds = posts.Select(p => p.PostId).ToList();
.Select(group =>
{
var postId = group.Key;
var post = group.First();
var media = group
.Select(row =>
{
if (row.MediaUrl is not null)
{
return new PostMediaDto(
new Uri(row.MediaUrl),
row.MediaHeight,
row.MediaHeight
);
}
else
return null;
})
.OfType<PostMediaDto>()
.ToList();
return new PostDto(
postId,
post.Content,
media,
post.PostedOn,
new PostAuthorDto(post.AuthorId, post.Username)
);
})
.ToList();
var next = rows.Count >= query.Amount ? rows.LastOrDefault()?.PostId : null; var loadMediaResult = await conn.QueryAsync<LoadMediaRow>(
"""
select
pm.url as MediaUrl,
pm.type as MediaType,
pm.width as MediaWidth,
pm.height as MediaHeight,
pm.post_id as PostId
from blog.post_media pm where pm.post_id = ANY (@postIds)
order by pm.ordering
""",
new { postIds }
);
return new GetPostsQueryResult(posts, next); var loadReactionsResult = await conn.QueryAsync<LoadReactionRow>(
"""
select
pr.post_id as PostId,
pr.author_id as AuthorId,
pr.emoji as Emoji
from blog.post_reaction pr
where pr.post_id = ANY (@postIds)
""",
new { postIds }
);
var reactionsByPostId = loadReactionsResult
.GroupBy(r => r.PostId)
.ToDictionary(
group => group.Key,
group =>
group
.GroupBy(
r => r.Emoji,
(key, g) =>
{
var reactions = g.ToList();
return new PostReactionDto(
key,
reactions.Count,
reactions.Any(r => r.AuthorId == query.CurrentUserId)
);
}
)
.ToList()
);
var mediaByPostId = loadMediaResult
.GroupBy(m => m.PostId)
.ToDictionary(
g => g.Key,
g =>
g.Select(m => new PostMediaDto(
new Uri(m.MediaUrl),
m.MediaWidth,
m.MediaHeight
))
.ToList()
);
return new GetPostsQueryResult(
posts
.Select(p => new PostDto(
p.PostId,
p.Content,
mediaByPostId.TryGetValue(p.PostId, out var mediaDtos) ? mediaDtos : [],
p.PostedOn,
new PostAuthorDto(p.AuthorId, p.Username),
reactionsByPostId.TryGetValue(p.PostId, out var reactionDtos)
? reactionDtos.ToList()
: [],
!string.IsNullOrEmpty(p.PossibleReactions)
? JsonSerializer.Deserialize<IEnumerable<string>>(p.PossibleReactions)!
: []
))
.ToList(),
next
);
} }
internal class QueryResult internal record LoadPostRow
{ {
public Guid PostId { get; set; } public Guid PostId { get; init; }
public string Content { get; set; } public string Content { get; init; }
public string? MediaUrl { get; set; } public DateTimeOffset PostedOn { get; init; }
public string? MediaType { get; set; } public string Username { get; init; }
public int? MediaWidth { get; set; } public Guid AuthorId { get; init; }
public int? MediaHeight { get; set; } public string? PossibleReactions { get; init; }
public DateTimeOffset PostedOn { get; set; } }
public Guid AuthorId { get; set; }
public string Username { get; set; } internal record LoadMediaRow
{
public string MediaUrl { get; init; }
public string? MediaType { get; init; }
public int? MediaWidth { get; init; }
public int? MediaHeight { get; init; }
public Guid PostId { get; init; }
}
internal record LoadReactionRow
{
public Guid PostId { get; init; }
public Guid AuthorId { get; init; }
public string Emoji { get; init; }
} }
} }

View file

@ -1,3 +1,4 @@
using System.Text.Json;
using Femto.Common.Domain; using Femto.Common.Domain;
using Femto.Modules.Blog.Domain.Posts.Events; using Femto.Modules.Blog.Domain.Posts.Events;
using Femto.Modules.Blog.Emoji; using Femto.Modules.Blog.Emoji;
@ -13,6 +14,16 @@ internal class Post : Entity
public IList<PostReaction> Reactions { get; private set; } = []; public IList<PostReaction> Reactions { get; private set; } = [];
public bool IsPublic { get; set; } public bool IsPublic { get; set; }
public DateTimeOffset PostedOn { get; private set; }
private string PossibleReactionsJson { get; set; } = null!;
public IEnumerable<string> PossibleReactions
{
get => JsonSerializer.Deserialize<IEnumerable<string>>(this.PossibleReactionsJson)!;
init => PossibleReactionsJson = JsonSerializer.Serialize(value);
}
private Post() { } private Post() { }
@ -22,13 +33,9 @@ internal class Post : Entity
this.AuthorId = authorId; this.AuthorId = authorId;
this.Content = content; this.Content = content;
this.Media = media; this.Media = media;
this.PossibleReactions = AllEmoji.GetRandomEmoji(5);
this.Reactions = AllEmoji this.PostedOn = DateTimeOffset.UtcNow;
.GetRandomEmoji(5)
.Select(emoji => new PostReaction(emoji, 0))
.ToList();
this.AddDomainEvent(new PostCreated(this)); this.AddDomainEvent(new PostCreated(this));
} }
} }

View file

@ -2,15 +2,14 @@ namespace Femto.Modules.Blog.Domain.Posts;
public class PostReaction public class PostReaction
{ {
public Guid Id { get; private set; } public Guid AuthorId { get; private set; }
public Guid PostId { get; private set; }
public string Emoji { get; private set; } = null!; public string Emoji { get; private set; } = null!;
public int Count { get; private set; } public PostReaction(Guid authorId, Guid postId, string emoji)
public PostReaction(string emoji, int count)
{ {
this.Id = Guid.CreateVersion7(); this.AuthorId = authorId;
this.PostId = postId;
this.Emoji = emoji; this.Emoji = emoji;
this.Count = count;
} }
private PostReaction() { } private PostReaction() { }