init
This commit is contained in:
commit
ded4f4db26
43 changed files with 2052 additions and 0 deletions
44
Catalogger.Backend/Extensions/DiscordExtensions.cs
Normal file
44
Catalogger.Backend/Extensions/DiscordExtensions.cs
Normal file
|
|
@ -0,0 +1,44 @@
|
|||
using Remora.Discord.API.Abstractions.Objects;
|
||||
using Remora.Rest.Core;
|
||||
using Remora.Results;
|
||||
|
||||
namespace Catalogger.Backend.Extensions;
|
||||
|
||||
public static class DiscordExtensions
|
||||
{
|
||||
public static string Tag(this IPartialUser user)
|
||||
{
|
||||
var discriminator = user.Discriminator.OrDefault();
|
||||
return discriminator == 0 ? user.Username.Value : $"{user.Username.Value}#{discriminator:0000}";
|
||||
}
|
||||
|
||||
public static string AvatarUrl(this IUser user, int size = 256)
|
||||
{
|
||||
if (user.Avatar != null)
|
||||
{
|
||||
var ext = user.Avatar.HasGif ? ".gif" : ".webp";
|
||||
return $"https://cdn.discordapp.com/avatars/{user.ID}/{user.Avatar.Value}{ext}?size={size}";
|
||||
}
|
||||
|
||||
var avatarIndex = user.Discriminator == 0 ? (int)((user.ID.Value >> 22) % 6) : user.Discriminator % 5;
|
||||
return $"https://cdn.discordapp.com/embed/avatars/{avatarIndex}.png?size={size}";
|
||||
}
|
||||
|
||||
public static ulong ToUlong(this Snowflake snowflake) => snowflake.Value;
|
||||
|
||||
public static ulong ToUlong(this Optional<Snowflake> snowflake)
|
||||
{
|
||||
if (!snowflake.IsDefined()) throw new Exception("ToUlong called on an undefined Snowflake");
|
||||
return snowflake.Value.Value;
|
||||
}
|
||||
|
||||
public static T GetOrThrow<T>(this Result<T> result)
|
||||
{
|
||||
if (result.Error != null) throw new DiscordRestException(result.Error.Message);
|
||||
return result.Entity;
|
||||
}
|
||||
|
||||
public static async Task<T> GetOrThrow<T>(this Task<Result<T>> result) => (await result).GetOrThrow();
|
||||
|
||||
public class DiscordRestException(string message) : Exception(message);
|
||||
}
|
||||
121
Catalogger.Backend/Extensions/StartupExtensions.cs
Normal file
121
Catalogger.Backend/Extensions/StartupExtensions.cs
Normal file
|
|
@ -0,0 +1,121 @@
|
|||
using Catalogger.Backend.Bot.Responders;
|
||||
using Catalogger.Backend.Cache;
|
||||
using Catalogger.Backend.Database;
|
||||
using Catalogger.Backend.Database.Queries;
|
||||
using Catalogger.Backend.Services;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using NodaTime;
|
||||
using Remora.Discord.API;
|
||||
using Remora.Discord.API.Abstractions.Rest;
|
||||
using Remora.Discord.Commands.Services;
|
||||
using Serilog;
|
||||
using Serilog.Events;
|
||||
|
||||
namespace Catalogger.Backend.Extensions;
|
||||
|
||||
public static class StartupExtensions
|
||||
{
|
||||
/// <summary>
|
||||
/// Adds Serilog to this service collection. This method also initializes Serilog, so it should be called as early as possible, before any log calls.
|
||||
/// </summary>
|
||||
public static WebApplicationBuilder AddSerilog(this WebApplicationBuilder builder)
|
||||
{
|
||||
var config = builder.Configuration.Get<Config>() ?? new();
|
||||
|
||||
var logCfg = new LoggerConfiguration()
|
||||
.Enrich.FromLogContext()
|
||||
.MinimumLevel.Is(config.Logging.LogEventLevel)
|
||||
// ASP.NET's built in request logs are extremely verbose, so we use Serilog's instead.
|
||||
// Serilog doesn't disable the built-in logs, so we do it here.
|
||||
.MinimumLevel.Override("Microsoft", LogEventLevel.Information)
|
||||
.MinimumLevel.Override("Microsoft.EntityFrameworkCore.Database.Command",
|
||||
config.Logging.LogQueries ? LogEventLevel.Information : LogEventLevel.Fatal)
|
||||
.MinimumLevel.Override("Microsoft.AspNetCore.Hosting", LogEventLevel.Warning)
|
||||
.MinimumLevel.Override("Microsoft.AspNetCore.Mvc", LogEventLevel.Warning)
|
||||
.MinimumLevel.Override("Microsoft.AspNetCore.Routing", LogEventLevel.Warning)
|
||||
.WriteTo.Console();
|
||||
|
||||
// AddSerilog doesn't seem to add an ILogger to the service collection, so add that manually.
|
||||
builder.Services.AddSerilog().AddSingleton(Log.Logger = logCfg.CreateLogger());
|
||||
|
||||
return builder;
|
||||
}
|
||||
|
||||
public static Config AddConfiguration(this WebApplicationBuilder builder)
|
||||
{
|
||||
builder.Configuration.Sources.Clear();
|
||||
builder.Configuration.AddConfiguration();
|
||||
|
||||
var config = builder.Configuration.Get<Config>() ?? new();
|
||||
builder.Services.AddSingleton(config);
|
||||
return config;
|
||||
}
|
||||
|
||||
public static IConfigurationBuilder AddConfiguration(this IConfigurationBuilder builder)
|
||||
{
|
||||
var file = Environment.GetEnvironmentVariable("CATALOGGER_CONFIG_FILE") ?? "config.ini";
|
||||
|
||||
return builder
|
||||
.SetBasePath(Directory.GetCurrentDirectory())
|
||||
.AddIniFile(file, optional: false, reloadOnChange: false)
|
||||
.AddEnvironmentVariables();
|
||||
}
|
||||
|
||||
public static IServiceCollection AddCustomServices(this IServiceCollection services) => services
|
||||
.AddSingleton<IClock>(SystemClock.Instance)
|
||||
.AddSingleton<ChannelCacheService>()
|
||||
.AddSingleton<UserCacheService>()
|
||||
.AddSingleton<PluralkitApiService>()
|
||||
.AddSingleton<IWebhookCache, InMemoryWebhookCache>()
|
||||
.AddScoped<IEncryptionService, EncryptionService>()
|
||||
.AddScoped<MessageRepository>()
|
||||
.AddSingleton<WebhookExecutorService>()
|
||||
.AddSingleton<PkMessageHandler>();
|
||||
|
||||
public static async Task Initialize(this WebApplication app)
|
||||
{
|
||||
await using var scope = app.Services.CreateAsyncScope();
|
||||
var logger = scope.ServiceProvider.GetRequiredService<ILogger>().ForContext<Program>();
|
||||
logger.Information("Starting Catalogger.NET");
|
||||
|
||||
await using (var db = scope.ServiceProvider.GetRequiredService<DatabaseContext>())
|
||||
{
|
||||
var migrationCount = (await db.Database.GetPendingMigrationsAsync()).Count();
|
||||
if (migrationCount != 0)
|
||||
{
|
||||
logger.Information("Applying {Count} database migrations", migrationCount);
|
||||
await db.Database.MigrateAsync();
|
||||
}
|
||||
else logger.Information("There are no pending migrations");
|
||||
}
|
||||
|
||||
var config = scope.ServiceProvider.GetRequiredService<Config>();
|
||||
var slashService = scope.ServiceProvider.GetRequiredService<SlashService>();
|
||||
|
||||
if (config.Discord.ApplicationId == 0)
|
||||
{
|
||||
logger.Warning(
|
||||
"Application ID not set in config. Fetching and setting it now, but for future restarts, please add it to config.ini as Discord.ApplicationId.");
|
||||
var restApi = scope.ServiceProvider.GetRequiredService<IDiscordRestApplicationAPI>();
|
||||
var application = await restApi.GetCurrentApplicationAsync().GetOrThrow();
|
||||
config.Discord.ApplicationId = application.ID.ToUlong();
|
||||
logger.Information("Current application ID is {ApplicationId}", config.Discord.ApplicationId);
|
||||
}
|
||||
|
||||
if (config.Discord.SyncCommands)
|
||||
{
|
||||
if (config.Discord.CommandsGuildId != null)
|
||||
{
|
||||
logger.Information("Syncing application commands with guild {GuildId}", config.Discord.CommandsGuildId);
|
||||
await slashService.UpdateSlashCommandsAsync(
|
||||
guildID: DiscordSnowflake.New(config.Discord.CommandsGuildId.Value));
|
||||
}
|
||||
else
|
||||
{
|
||||
logger.Information("Syncing application commands globally");
|
||||
await slashService.UpdateSlashCommandsAsync();
|
||||
}
|
||||
}
|
||||
else logger.Information("Not syncing slash commands, Discord.SyncCommands is false or unset");
|
||||
}
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue