Iceshrimp.NET/Iceshrimp.Backend/Core/Services/InstanceService.cs

148 lines
No EOL
4.4 KiB
C#

using AsyncKeyedLock;
using Iceshrimp.Backend.Core.Database;
using Iceshrimp.Backend.Core.Database.Tables;
using Iceshrimp.Backend.Core.Extensions;
using Iceshrimp.Backend.Core.Federation.WebFinger;
using Iceshrimp.Backend.Core.Helpers;
using Microsoft.EntityFrameworkCore;
namespace Iceshrimp.Backend.Core.Services;
public class InstanceService(DatabaseContext db, HttpClient httpClient, ILogger<InstanceService> logger)
{
private static readonly AsyncKeyedLocker<string> KeyedLocker = new(o =>
{
o.PoolSize = 100;
o.PoolInitialFill = 5;
});
private async Task<Instance> GetUpdatedInstanceMetadataAsync(string host, string webDomain)
{
if (host == webDomain)
logger.LogDebug("Updating instance metadata for {host}", host);
else
logger.LogDebug("Updating instance metadata for {host} ({domain})", host, webDomain);
host = host.ToPunycodeLower();
var instance = await db.Instances.FirstOrDefaultAsync(p => p.Host == host);
if (instance == null)
{
if (!KeyedLocker.IsInUse(host))
{
using (await KeyedLocker.LockAsync(host))
{
instance = new Instance
{
Id = IdHelpers.GenerateSlowflakeId(),
Host = host,
CaughtAt = DateTime.UtcNow,
LastCommunicatedAt = DateTime.UtcNow
};
await db.AddAsync(instance);
await db.SaveChangesAsync();
}
}
else
{
using (await KeyedLocker.LockAsync(host))
{
instance = await db.Instances.FirstOrDefaultAsync(p => p.Host == host);
}
if (instance == null) throw new Exception("Failed to get instance metadata for {host}");
}
}
if (instance.NeedsUpdate && !KeyedLocker.IsInUse(host))
{
using (await KeyedLocker.LockAsync(host))
{
instance.InfoUpdatedAt = DateTime.UtcNow;
var nodeinfo = await GetNodeInfoAsync(webDomain);
if (nodeinfo != null)
{
instance.Name = nodeinfo.Metadata?.NodeName;
instance.Description = nodeinfo.Metadata?.NodeDescription;
//instance.FaviconUrl = TODO,
//instance.FollowersCount = TODO,
//instance.FollowingCount = TODO,
//instance.IconUrl = TODO,
instance.MaintainerName = nodeinfo.Metadata?.Maintainer?.Name;
instance.MaintainerEmail = nodeinfo.Metadata?.Maintainer?.Email;
instance.OpenRegistrations = nodeinfo.OpenRegistrations;
instance.SoftwareName = nodeinfo.Software?.Name;
instance.SoftwareVersion = nodeinfo.Software?.Version;
instance.ThemeColor = nodeinfo.Metadata?.ThemeColor;
}
await db.SaveChangesAsync();
}
}
return instance;
}
public async Task<Instance> GetUpdatedInstanceMetadataAsync(User user)
{
if (user.Host == null || user.Uri == null) throw new Exception("Can't fetch instance metadata for local user");
return await GetUpdatedInstanceMetadataAsync(user.Host, new Uri(user.Uri).Host);
}
private async Task<NodeInfoResponse?> GetNodeInfoAsync(string webDomain)
{
try
{
var res =
await httpClient.GetFromJsonAsync<NodeInfoIndexResponse>($"https://{webDomain}/.well-known/nodeinfo");
var url = res?.Links.FirstOrDefault(p => p.Rel == "http://nodeinfo.diaspora.software/ns/schema/2.1") ??
res?.Links.FirstOrDefault(p => p.Rel == "http://nodeinfo.diaspora.software/ns/schema/2.0");
if (url == null) return null;
return await httpClient.GetFromJsonAsync<NodeInfoResponse>(url.Href);
}
catch
{
return null;
}
}
public async Task UpdateInstanceStatusAsync(string host, string webDomain)
{
var instance = await GetUpdatedInstanceMetadataAsync(host, webDomain);
instance.LastCommunicatedAt = DateTime.UtcNow;
instance.LatestRequestReceivedAt = DateTime.UtcNow;
await db.SaveChangesAsync();
}
public async Task UpdateInstanceStatusAsync(string host, string webDomain, int statusCode, bool notResponding)
{
var instance = await GetUpdatedInstanceMetadataAsync(host, webDomain);
instance.LatestStatus = statusCode;
instance.LatestRequestSentAt = DateTime.UtcNow;
if (notResponding)
{
instance.IsNotResponding = true;
}
else
{
instance.IsNotResponding = false;
instance.LastCommunicatedAt = DateTime.UtcNow;
}
await db.SaveChangesAsync();
}
public async Task MarkInstanceAsUnresponsive(string host, string webDomain)
{
var instance = await GetUpdatedInstanceMetadataAsync(host, webDomain);
instance.LatestRequestSentAt = DateTime.UtcNow;
instance.IsNotResponding = true;
await db.SaveChangesAsync();
}
}