Post activitypub, and publisher update events

This commit is contained in:
2025-12-31 00:24:08 +08:00
parent 6aa6833163
commit 0519f2a2e6
5 changed files with 510 additions and 6 deletions

View File

@@ -3,6 +3,7 @@ using DysonNetwork.Shared.Auth;
using DysonNetwork.Shared.Models;
using DysonNetwork.Shared.Proto;
using DysonNetwork.Shared.Registry;
using DysonNetwork.Sphere.ActivityPub;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
using Microsoft.EntityFrameworkCore;
@@ -20,7 +21,8 @@ public class PublisherController(
FileService.FileServiceClient files,
FileReferenceService.FileReferenceServiceClient fileRefs,
ActionLogService.ActionLogServiceClient als,
RemoteRealmService remoteRealmService
RemoteRealmService remoteRealmService,
IServiceScopeFactory factory
) : ControllerBase
{
[HttpGet("{name}")]
@@ -668,6 +670,27 @@ public class PublisherController(
}
);
// Send ActivityPub Update activity if actor exists
var actor = await db.FediverseActors
.FirstOrDefaultAsync(a => a.PublisherId == publisher.Id);
if (actor != null)
{
_ = Task.Run(async () =>
{
try
{
using var scope = factory.CreateScope();
var deliveryService = scope.ServiceProvider.GetRequiredService<ActivityPubDeliveryService>();
await deliveryService.SendUpdateActorActivityAsync(actor);
}
catch (Exception ex)
{
Console.Error.WriteLine($"Error sending ActivityPub Update actor activity for publisher {publisher.Id}: {ex.Message}");
}
});
}
return Ok(publisher);
}
@@ -886,5 +909,87 @@ public class PublisherController(
await ps.SettlePublisherRewards();
return Ok();
}
[HttpGet("{name}/fediverse")]
[Authorize]
public async Task<ActionResult<FediverseStatus>> GetFediverseStatus(string name)
{
if (HttpContext.Items["CurrentUser"] is not Account currentUser)
return Unauthorized();
var publisher = await db.Publishers.Where(p => p.Name == name).FirstOrDefaultAsync();
if (publisher is null)
return NotFound();
var status = await ps.GetFediverseStatusAsync(publisher.Id, Guid.Parse(currentUser.Id));
return Ok(status);
}
[HttpPost("{name}/fediverse")]
[Authorize]
public async Task<ActionResult<FediverseStatus>> EnableFediverse(string name)
{
if (HttpContext.Items["CurrentUser"] is not Account currentUser)
return Unauthorized();
var accountId = Guid.Parse(currentUser.Id);
var publisher = await db.Publishers.Where(p => p.Name == name).FirstOrDefaultAsync();
if (publisher is null)
return NotFound();
var member = await db
.PublisherMembers.Where(m => m.AccountId == accountId)
.Where(m => m.PublisherId == publisher.Id)
.FirstOrDefaultAsync();
if (member is null)
return StatusCode(403, "You are not even a member of targeted publisher.");
if (member.Role < Shared.Models.PublisherMemberRole.Manager)
return StatusCode(403, "You need at least be manager to enable fediverse for this publisher.");
try
{
var actor = await ps.EnableFediverseAsync(publisher.Id, accountId);
var status = await ps.GetFediverseStatusAsync(publisher.Id);
return Ok(status);
}
catch (InvalidOperationException ex)
{
return BadRequest(ex.Message);
}
}
[HttpDelete("{name}/fediverse")]
[Authorize]
public async Task<ActionResult> DisableFediverse(string name)
{
if (HttpContext.Items["CurrentUser"] is not Account currentUser)
return Unauthorized();
var accountId = Guid.Parse(currentUser.Id);
var publisher = await db.Publishers.Where(p => p.Name == name).FirstOrDefaultAsync();
if (publisher is null)
return NotFound();
var member = await db
.PublisherMembers.Where(m => m.AccountId == accountId)
.Where(m => m.PublisherId == publisher.Id)
.FirstOrDefaultAsync();
if (member is null)
return StatusCode(403, "You are not even a member of targeted publisher.");
if (member.Role < Shared.Models.PublisherMemberRole.Manager)
return StatusCode(403, "You need at least be manager to disable fediverse for this publisher.");
try
{
await ps.DisableFediverseAsync(publisher.Id, accountId);
return NoContent();
}
catch (UnauthorizedAccessException ex)
{
return StatusCode(403, ex.Message);
}
}
}

View File

@@ -3,6 +3,7 @@ using DysonNetwork.Shared.Cache;
using DysonNetwork.Shared.Models;
using DysonNetwork.Shared.Proto;
using DysonNetwork.Shared.Registry;
using DysonNetwork.Sphere.ActivityPub;
using Microsoft.EntityFrameworkCore;
using NodaTime;
using NodaTime.Serialization.Protobuf;
@@ -11,13 +12,23 @@ using PublisherType = DysonNetwork.Shared.Models.PublisherType;
namespace DysonNetwork.Sphere.Publisher;
public class FediverseStatus
{
public bool Enabled { get; set; }
public SnFediverseActor? Actor { get; set; }
public int FollowerCount { get; set; }
public string? ActorUri { get; set; }
}
public class PublisherService(
AppDatabase db,
FileReferenceService.FileReferenceServiceClient fileRefs,
SocialCreditService.SocialCreditServiceClient socialCredits,
ExperienceService.ExperienceServiceClient experiences,
ICacheService cache,
RemoteAccountService remoteAccounts
RemoteAccountService remoteAccounts,
ActivityPubKeyService keyService,
IConfiguration configuration
)
{
public async Task<SnPublisher?> GetPublisherLoaded(Guid id)
@@ -668,4 +679,142 @@ public class PublisherService(
}
}
}
private string Domain => configuration["ActivityPub:Domain"] ?? "localhost";
public async Task<SnFediverseActor?> EnableFediverseAsync(Guid publisherId, Guid requesterAccountId)
{
var member = await db.PublisherMembers
.Where(m => m.PublisherId == publisherId && m.AccountId == requesterAccountId)
.FirstOrDefaultAsync();
if (member == null || member.Role < PublisherMemberRole.Manager)
throw new UnauthorizedAccessException("You need at least Manager role to enable fediverse for this publisher");
var existingActor = await db.FediverseActors
.FirstOrDefaultAsync(a => a.PublisherId == publisherId);
if (existingActor != null)
throw new InvalidOperationException("Fediverse actor already exists for this publisher");
var publisher = await db.Publishers
.Include(p => p.Picture)
.Include(p => p.Background)
.FirstOrDefaultAsync(p => p.Id == publisherId);
if (publisher == null)
throw new InvalidOperationException("Publisher not found");
var instance = await db.FediverseInstances
.FirstOrDefaultAsync(i => i.Domain == Domain);
if (instance == null)
{
instance = new SnFediverseInstance
{
Domain = Domain,
Name = Domain
};
db.FediverseInstances.Add(instance);
await db.SaveChangesAsync();
}
var (privateKey, publicKey) = keyService.GenerateKeyPair();
publisher.PrivateKeyPem = privateKey;
publisher.PublicKeyPem = publicKey;
var assetsBaseUrl = configuration["ActivityPub:FileBaseUrl"] ?? $"https://{Domain}/files";
var actorUrl = $"https://{Domain}/activitypub/actors/{publisher.Name}";
var actor = new SnFediverseActor
{
Uri = actorUrl,
Username = publisher.Name,
DisplayName = publisher.Nick,
Bio = publisher.Bio,
Type = "Person",
InboxUri = $"{actorUrl}/inbox",
OutboxUri = $"{actorUrl}/outbox",
FollowersUri = $"{actorUrl}/followers",
FollowingUri = $"{actorUrl}/following",
PublicKeyId = $"{actorUrl}#main-key",
PublicKey = publicKey,
AvatarUrl = publisher.Picture != null ? $"{assetsBaseUrl}/{publisher.Picture.Id}" : null,
HeaderUrl = publisher.Background != null ? $"{assetsBaseUrl}/{publisher.Background.Id}" : null,
InstanceId = instance.Id,
PublisherId = publisher.Id
};
db.FediverseActors.Add(actor);
db.Update(publisher);
await db.SaveChangesAsync();
return actor;
}
public async Task<bool> DisableFediverseAsync(Guid publisherId, Guid requesterAccountId)
{
var member = await db.PublisherMembers
.Where(m => m.PublisherId == publisherId && m.AccountId == requesterAccountId)
.FirstOrDefaultAsync();
if (member == null || member.Role < PublisherMemberRole.Manager)
throw new UnauthorizedAccessException("You need at least Manager role to disable fediverse for this publisher");
var actor = await db.FediverseActors
.FirstOrDefaultAsync(a => a.PublisherId == publisherId);
if (actor == null)
return true;
var publisher = await db.Publishers
.FirstOrDefaultAsync(p => p.Id == publisherId);
if (publisher != null)
{
publisher.PrivateKeyPem = null;
publisher.PublicKeyPem = null;
db.Update(publisher);
}
db.FediverseActors.Remove(actor);
await db.SaveChangesAsync();
return true;
}
public async Task<FediverseStatus?> GetFediverseStatusAsync(Guid publisherId, Guid? requesterAccountId = null)
{
var actor = await db.FediverseActors
.Include(a => a.Instance)
.FirstOrDefaultAsync(a => a.PublisherId == publisherId);
var followerCount = await db.FediverseRelationships
.Where(r => r.Actor.PublisherId == publisherId && r.IsFollowedBy)
.CountAsync();
var publisher = await db.Publishers
.FirstOrDefaultAsync(p => p.Id == publisherId);
if (publisher == null)
return null;
var assetsBaseUrl = configuration["ActivityPub:FileBaseUrl"] ?? $"https://{Domain}/files";
return new FediverseStatus
{
Enabled = actor != null,
Actor = actor,
FollowerCount = followerCount,
ActorUri = actor != null ? actor.Uri : null
};
}
public async Task<SnFediverseActor?> GetLocalActorAsync(Guid publisherId)
{
return await db.FediverseActors
.Include(a => a.Instance)
.FirstOrDefaultAsync(a => a.PublisherId == publisherId);
}
}