✨ Award post
This commit is contained in:
		| @@ -27,6 +27,7 @@ public class PaymentService( | ||||
|         Duration? expiration = null, | ||||
|         string? appIdentifier = null, | ||||
|         string? productIdentifier = null, | ||||
|         string? remarks = null, | ||||
|         Dictionary<string, object>? meta = null, | ||||
|         bool reuseable = true | ||||
|     ) | ||||
| @@ -65,6 +66,7 @@ public class PaymentService( | ||||
|             ExpiredAt = SystemClock.Instance.GetCurrentInstant().Plus(expiration ?? Duration.FromHours(24)), | ||||
|             AppIdentifier = appIdentifier, | ||||
|             ProductIdentifier = productIdentifier, | ||||
|             Remarks = remarks, | ||||
|             Meta = meta | ||||
|         }; | ||||
|  | ||||
|   | ||||
| @@ -15,6 +15,7 @@ public class PaymentServiceGrpc(PaymentService paymentService) : Shared.Proto.Pa | ||||
|             request.Expiration is not null ? Duration.FromSeconds(request.Expiration.Seconds) : null, | ||||
|             request.HasAppIdentifier ? request.AppIdentifier : Order.InternalAppIdentifier, | ||||
|             request.HasProductIdentifier ? request.ProductIdentifier : null, | ||||
|             request.HasRemarks ? request.Remarks : null, | ||||
|             request.HasMeta | ||||
|                 ? GrpcTypeHelper.ConvertByteStringToObject<Dictionary<string, object>>(request.Meta) | ||||
|                 : null, | ||||
|   | ||||
| @@ -98,6 +98,18 @@ public static class GrpcClientHelper | ||||
|             clientCertPassword)); | ||||
|     } | ||||
|      | ||||
|     public static async Task<PaymentService.PaymentServiceClient> CreatePaymentServiceClient( | ||||
|         IEtcdClient etcdClient, | ||||
|         string clientCertPath, | ||||
|         string clientKeyPath, | ||||
|         string? clientCertPassword = null | ||||
|     ) | ||||
|     { | ||||
|         var url = await GetServiceUrlFromEtcd(etcdClient, "DysonNetwork.Pass"); | ||||
|         return new PaymentService.PaymentServiceClient(CreateCallInvoker(url, clientCertPath, clientKeyPath, | ||||
|             clientCertPassword)); | ||||
|     } | ||||
|      | ||||
|     public static async Task<PusherService.PusherServiceClient> CreatePusherServiceClient( | ||||
|         IEtcdClient etcdClient, | ||||
|         string clientCertPath, | ||||
|   | ||||
| @@ -126,6 +126,7 @@ message CreateOrderRequest { | ||||
|     optional string product_identifier = 8; | ||||
|     // Using bytes for meta to represent JSON. | ||||
|     optional bytes meta = 6; | ||||
|     optional string remarks = 9; | ||||
|     bool reuseable = 7; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -71,6 +71,20 @@ public static class ServiceInjectionHelper | ||||
|                 .GetResult(); | ||||
|         });  | ||||
|          | ||||
|         services.AddSingleton<PaymentService.PaymentServiceClient>(sp => | ||||
|         { | ||||
|             var etcdClient = sp.GetRequiredService<IEtcdClient>(); | ||||
|             var config = sp.GetRequiredService<IConfiguration>(); | ||||
|             var clientCertPath = config["Service:ClientCert"]!; | ||||
|             var clientKeyPath = config["Service:ClientKey"]!; | ||||
|             var clientCertPassword = config["Service:CertPassword"]; | ||||
|  | ||||
|             return GrpcClientHelper | ||||
|                 .CreatePaymentServiceClient(etcdClient, clientCertPath, clientKeyPath, clientCertPassword) | ||||
|                 .GetAwaiter() | ||||
|                 .GetResult(); | ||||
|         }); | ||||
|          | ||||
|         return services; | ||||
|     } | ||||
|      | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| using System.ComponentModel.DataAnnotations; | ||||
| using System.Globalization; | ||||
| using DysonNetwork.Shared.Auth; | ||||
| using DysonNetwork.Shared.Content; | ||||
| using DysonNetwork.Shared.Data; | ||||
| @@ -24,6 +25,7 @@ public class PostController( | ||||
|     PublisherService pub, | ||||
|     AccountService.AccountServiceClient accounts, | ||||
|     ActionLogService.ActionLogServiceClient als, | ||||
|     PaymentService.PaymentServiceClient payments, | ||||
|     PollService polls, | ||||
|     RealmService rs | ||||
| ) | ||||
| @@ -324,7 +326,6 @@ public class PostController( | ||||
|             .OrderByDescending(p => p.CreatedAt) | ||||
|             .FilterWithVisibility(currentUser, userFriends, userPublishers) | ||||
|             .ToListAsync(); | ||||
|         if (posts is null) return NotFound(); | ||||
|         posts = await ps.LoadPostInfo(posts, currentUser); | ||||
|  | ||||
|         return Ok(posts); | ||||
| @@ -582,14 +583,22 @@ public class PostController( | ||||
|     public class PostAwardRequest | ||||
|     { | ||||
|         public decimal Amount { get; set; } | ||||
|         public PostReactionAttitude Attitude { get; set; } | ||||
|         [MaxLength(4096)] public string? Message { get; set; } | ||||
|     } | ||||
|  | ||||
|     public class PostAwardResponse | ||||
|     { | ||||
|         public Guid OrderId { get; set; } | ||||
|     } | ||||
|  | ||||
|     [HttpPost("{id:guid}/awards")] | ||||
|     [Authorize] | ||||
|     public async Task<ActionResult<PostAward>> AwardPost(Guid id, [FromBody] PostAwardRequest request) | ||||
|     public async Task<ActionResult<PostAwardResponse>> AwardPost(Guid id, [FromBody] PostAwardRequest request) | ||||
|     { | ||||
|         if (HttpContext.Items["CurrentUser"] is not Account currentUser) return Unauthorized(); | ||||
|         if (request.Attitude == PostReactionAttitude.Neutral) | ||||
|             return BadRequest("You cannot create a neutral post award"); | ||||
|  | ||||
|         var friendsResponse = | ||||
|             await accounts.ListFriendsAsync(new ListRelationshipSimpleRequest | ||||
| @@ -606,9 +615,26 @@ public class PostController( | ||||
|  | ||||
|         var accountId = Guid.Parse(currentUser.Id); | ||||
|  | ||||
|         // TODO Make payment, add record | ||||
|         var orderRemark = string.IsNullOrWhiteSpace(post.Title) ? "from @" + post.Publisher.Name : post.Title; | ||||
|         var order = await payments.CreateOrderAsync(new CreateOrderRequest | ||||
|         { | ||||
|             ProductIdentifier = "posts.award", | ||||
|             Currency = "points", // NSP - Source Points | ||||
|             Remarks = $"Award post {orderRemark}", | ||||
|             Meta = GrpcTypeHelper.ConvertObjectToByteString(new Dictionary<string, object?>() | ||||
|             { | ||||
|                 ["account_id"] = accountId, | ||||
|                 ["post_id"] = post.Id, | ||||
|                 ["amount"] = request.Amount.ToString(CultureInfo.InvariantCulture), | ||||
|                 ["message"] = request.Message, | ||||
|                 ["attitude"] = request.Attitude, | ||||
|             }) | ||||
|         }); | ||||
|  | ||||
|         return Ok(); | ||||
|         return Ok(new PostAwardResponse() | ||||
|         { | ||||
|             OrderId = Guid.Parse(order.Id), | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     public class PostPinRequest | ||||
|   | ||||
| @@ -424,10 +424,12 @@ public partial class PostService( | ||||
|         var accountId = Guid.Parse(currentUser.Id); | ||||
|         if (post.RepliedPostId != null) | ||||
|         { | ||||
|             if (pinMode != PostPinMode.ReplyPage) throw new InvalidOperationException("Replies can only be pinned in the reply page."); | ||||
|             if (pinMode != PostPinMode.ReplyPage) | ||||
|                 throw new InvalidOperationException("Replies can only be pinned in the reply page."); | ||||
|             if (post.RepliedPost == null) throw new ArgumentNullException(nameof(post.RepliedPost)); | ||||
|  | ||||
|             if (!await ps.IsMemberWithRole(post.RepliedPost.PublisherId, accountId, Publisher.PublisherMemberRole.Editor)) | ||||
|             if (!await ps.IsMemberWithRole(post.RepliedPost.PublisherId, accountId, | ||||
|                     Publisher.PublisherMemberRole.Editor)) | ||||
|                 throw new InvalidOperationException("Only editors of original post can pin replies."); | ||||
|  | ||||
|             post.PinMode = pinMode; | ||||
| @@ -453,7 +455,8 @@ public partial class PostService( | ||||
|         { | ||||
|             if (post.RepliedPost == null) throw new ArgumentNullException(nameof(post.RepliedPost)); | ||||
|  | ||||
|             if (!await ps.IsMemberWithRole(post.RepliedPost.PublisherId, accountId, Publisher.PublisherMemberRole.Editor)) | ||||
|             if (!await ps.IsMemberWithRole(post.RepliedPost.PublisherId, accountId, | ||||
|                     Publisher.PublisherMemberRole.Editor)) | ||||
|                 throw new InvalidOperationException("Only editors of original post can unpin replies."); | ||||
|         } | ||||
|         else | ||||
| @@ -815,7 +818,8 @@ public partial class PostService( | ||||
|         { | ||||
|             // The previous day highest rated posts | ||||
|             var today = SystemClock.Instance.GetCurrentInstant(); | ||||
|             var periodStart = today.InUtc().Date.AtStartOfDayInZone(DateTimeZone.Utc).ToInstant().Minus(Duration.FromDays(1)); | ||||
|             var periodStart = today.InUtc().Date.AtStartOfDayInZone(DateTimeZone.Utc).ToInstant() | ||||
|                 .Minus(Duration.FromDays(1)); | ||||
|             var periodEnd = today.InUtc().Date.AtStartOfDayInZone(DateTimeZone.Utc).ToInstant(); | ||||
|  | ||||
|             var postsInPeriod = await db.Posts | ||||
| @@ -890,6 +894,34 @@ public partial class PostService( | ||||
|  | ||||
|         return posts; | ||||
|     } | ||||
|  | ||||
|     public async Task<PostAward> AwardPost( | ||||
|         Guid postId, | ||||
|         Guid accountId, | ||||
|         decimal amount, | ||||
|         PostReactionAttitude attitude, | ||||
|         string? message | ||||
|     ) | ||||
|     { | ||||
|         var award = new PostAward | ||||
|         { | ||||
|             Amount = amount, | ||||
|             Attitude = attitude, | ||||
|             Message = message, | ||||
|             PostId = postId, | ||||
|             AccountId = accountId | ||||
|         }; | ||||
|  | ||||
|         db.PostAwards.Add(award); | ||||
|         await db.SaveChangesAsync(); | ||||
|  | ||||
|         var delta = award.Attitude == PostReactionAttitude.Positive ? amount : -amount; | ||||
|  | ||||
|         await db.Posts.Where(p => p.Id == postId) | ||||
|             .ExecuteUpdateAsync(s => s.SetProperty(p => p.AwardedScore, p => p.AwardedScore + delta)); | ||||
|  | ||||
|         return award; | ||||
|     } | ||||
| } | ||||
|  | ||||
| public static class PostQueryExtensions | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| using System.Text.Json; | ||||
| using DysonNetwork.Shared.Stream; | ||||
| using DysonNetwork.Sphere.Post; | ||||
| using Microsoft.EntityFrameworkCore; | ||||
| using NATS.Client.Core; | ||||
|  | ||||
| @@ -13,7 +14,84 @@ public class BroadcastEventHandler( | ||||
| { | ||||
|     protected override async Task ExecuteAsync(CancellationToken stoppingToken) | ||||
|     { | ||||
|         await foreach (var msg in nats.SubscribeAsync<byte[]>(AccountDeletedEvent.Type, cancellationToken: stoppingToken)) | ||||
|         await foreach (var msg in nats.SubscribeAsync<byte[]>(PaymentOrderEvent.Type, cancellationToken: stoppingToken)) | ||||
|         { | ||||
|             PaymentOrderEvent? evt = null; | ||||
|             try | ||||
|             { | ||||
|                 evt = JsonSerializer.Deserialize<PaymentOrderEvent>(msg.Data); | ||||
|  | ||||
|                 // Every order goes into the MQ is already paid, so we skipped the status validation | ||||
|  | ||||
|                 if (evt?.ProductIdentifier is null) | ||||
|                     continue; | ||||
|  | ||||
|                 switch (evt.ProductIdentifier) | ||||
|                 { | ||||
|                     case "posts.award": | ||||
|                     { | ||||
|                         logger.LogInformation("Handling post award order: {OrderId}", evt.OrderId); | ||||
|  | ||||
|                         if (!evt.Meta.TryGetValue("account_id", out var accountIdObj) || | ||||
|                             accountIdObj is not string accountIdStr || | ||||
|                             !Guid.TryParse(accountIdStr, out var accountId)) | ||||
|                         { | ||||
|                             logger.LogWarning("Post award order {OrderId} missing or invalid account_id", evt.OrderId); | ||||
|                             break; | ||||
|                         } | ||||
|  | ||||
|                         if (!evt.Meta.TryGetValue("post_id", out var postIdObj) || | ||||
|                             postIdObj is not string postIdStr || | ||||
|                             !Guid.TryParse(postIdStr, out var postId)) | ||||
|                         { | ||||
|                             logger.LogWarning("Post award order {OrderId} missing or invalid post_id", evt.OrderId); | ||||
|                             break; | ||||
|                         } | ||||
|  | ||||
|                         if (!evt.Meta.TryGetValue("amount", out var amountObj) || | ||||
|                             amountObj is not string amountStr || | ||||
|                             !decimal.TryParse(amountStr, out var amount)) | ||||
|                         { | ||||
|                             logger.LogWarning("Post award order {OrderId} missing or invalid amount", evt.OrderId); | ||||
|                             break; | ||||
|                         } | ||||
|  | ||||
|                         if (!evt.Meta.TryGetValue("attitude", out var attitudeObj) || | ||||
|                             attitudeObj is not string attitudeStr || | ||||
|                             !int.TryParse(attitudeStr, out var attitudeInt) || | ||||
|                             !Enum.IsDefined(typeof(PostReactionAttitude), attitudeInt)) | ||||
|                         { | ||||
|                             logger.LogWarning("Post award order {OrderId} missing or invalid attitude", evt.OrderId); | ||||
|                             break; | ||||
|                         } | ||||
|                         var attitude = (PostReactionAttitude)attitudeInt; | ||||
|  | ||||
|                         string? message = null; | ||||
|                         if (evt.Meta.TryGetValue("message", out var messageObj) && | ||||
|                             messageObj is string messageStr) | ||||
|                         { | ||||
|                             message = messageStr; | ||||
|                         } | ||||
|  | ||||
|                         await using var scope = serviceProvider.CreateAsyncScope(); | ||||
|                         var ps = scope.ServiceProvider.GetRequiredService<PostService>(); | ||||
|  | ||||
|                         await ps.AwardPost(postId, accountId, amount, attitude, message); | ||||
|  | ||||
|                         logger.LogInformation("Post award for order {OrderId} handled successfully.", evt.OrderId); | ||||
|  | ||||
|                         break; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             catch (Exception ex) | ||||
|             { | ||||
|                 logger.LogError(ex, "Error processing payment order event for order {OrderId}", evt?.OrderId); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         await foreach (var msg in nats.SubscribeAsync<byte[]>(AccountDeletedEvent.Type, | ||||
|                            cancellationToken: stoppingToken)) | ||||
|         { | ||||
|             try | ||||
|             { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user