| 
									
										
										
										
											2024-09-03 16:29:51 +02:00
										 |  |  | using Coravel; | 
					
						
							|  |  |  | using Coravel.Queuing.Interfaces; | 
					
						
							| 
									
										
										
										
											2024-05-28 15:29:18 +02:00
										 |  |  | using Foxnouns.Backend.Database; | 
					
						
							| 
									
										
										
										
											2024-07-08 19:03:04 +02:00
										 |  |  | using Foxnouns.Backend.Jobs; | 
					
						
							| 
									
										
										
										
											2024-05-28 15:29:18 +02:00
										 |  |  | using Foxnouns.Backend.Middleware; | 
					
						
							| 
									
										
										
										
											2024-05-28 17:09:50 +02:00
										 |  |  | using Foxnouns.Backend.Services; | 
					
						
							| 
									
										
										
										
											2024-05-28 15:29:18 +02:00
										 |  |  | using Microsoft.EntityFrameworkCore; | 
					
						
							| 
									
										
										
										
											2024-09-04 01:46:39 +02:00
										 |  |  | using Minio; | 
					
						
							| 
									
										
										
										
											2024-05-28 15:29:18 +02:00
										 |  |  | using NodaTime; | 
					
						
							| 
									
										
										
										
											2024-09-03 17:00:14 +02:00
										 |  |  | using Prometheus; | 
					
						
							| 
									
										
										
										
											2024-05-27 15:53:54 +02:00
										 |  |  | using Serilog; | 
					
						
							|  |  |  | using Serilog.Events; | 
					
						
							| 
									
										
										
										
											2024-07-13 17:23:52 +02:00
										 |  |  | using IClock = NodaTime.IClock; | 
					
						
							| 
									
										
										
										
											2024-05-27 15:53:54 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Foxnouns.Backend.Extensions; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-28 15:29:18 +02:00
										 |  |  | public static class WebApplicationExtensions | 
					
						
							| 
									
										
										
										
											2024-05-27 15:53:54 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     /// <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> | 
					
						
							| 
									
										
										
										
											2024-05-28 17:09:50 +02:00
										 |  |  |     public static WebApplicationBuilder AddSerilog(this WebApplicationBuilder builder) | 
					
						
							| 
									
										
										
										
											2024-05-27 15:53:54 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         var config = builder.Configuration.Get<Config>() ?? new(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         var logCfg = new LoggerConfiguration() | 
					
						
							|  |  |  |             .Enrich.FromLogContext() | 
					
						
							| 
									
										
										
										
											2024-07-08 19:03:04 +02:00
										 |  |  |             .MinimumLevel.Is(config.Logging.LogEventLevel) | 
					
						
							| 
									
										
										
										
											2024-05-27 15:53:54 +02:00
										 |  |  |             // 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) | 
					
						
							| 
									
										
										
										
											2024-07-13 03:09:00 +02:00
										 |  |  |             .MinimumLevel.Override("Microsoft.EntityFrameworkCore.Database.Command", | 
					
						
							| 
									
										
										
										
											2024-07-14 21:25:23 +02:00
										 |  |  |                 config.Logging.LogQueries ? LogEventLevel.Information : LogEventLevel.Fatal) | 
					
						
							| 
									
										
										
										
											2024-05-27 15:53:54 +02:00
										 |  |  |             .MinimumLevel.Override("Microsoft.AspNetCore.Hosting", LogEventLevel.Warning) | 
					
						
							|  |  |  |             .MinimumLevel.Override("Microsoft.AspNetCore.Mvc", LogEventLevel.Warning) | 
					
						
							|  |  |  |             .MinimumLevel.Override("Microsoft.AspNetCore.Routing", LogEventLevel.Warning) | 
					
						
							| 
									
										
										
										
											2024-07-13 17:23:52 +02:00
										 |  |  |             .MinimumLevel.Override("Hangfire", LogEventLevel.Information) | 
					
						
							| 
									
										
										
										
											2024-05-28 15:29:18 +02:00
										 |  |  |             .WriteTo.Console(); | 
					
						
							| 
									
										
										
										
											2024-05-27 15:53:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-08 19:03:04 +02:00
										 |  |  |         if (config.Logging.SeqLogUrl != null) | 
					
						
							| 
									
										
										
										
											2024-05-27 15:53:54 +02:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-07-08 19:03:04 +02:00
										 |  |  |             logCfg.WriteTo.Seq(config.Logging.SeqLogUrl, restrictedToMinimumLevel: LogEventLevel.Verbose); | 
					
						
							| 
									
										
										
										
											2024-05-27 15:53:54 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // AddSerilog doesn't seem to add an ILogger to the service collection, so add that manually. | 
					
						
							| 
									
										
										
										
											2024-07-13 03:09:00 +02:00
										 |  |  |         builder.Services.AddSerilog().AddSingleton(Log.Logger = logCfg.CreateLogger()); | 
					
						
							| 
									
										
										
										
											2024-05-27 15:53:54 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         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("FOXNOUNS_CONFIG_FILE") ?? "config.ini"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return builder | 
					
						
							|  |  |  |             .SetBasePath(Directory.GetCurrentDirectory()) | 
					
						
							| 
									
										
										
										
											2024-09-03 16:29:51 +02:00
										 |  |  |             .AddJsonFile("appSettings.json", true) | 
					
						
							| 
									
										
										
										
											2024-05-27 15:53:54 +02:00
										 |  |  |             .AddIniFile(file, optional: false, reloadOnChange: true) | 
					
						
							|  |  |  |             .AddEnvironmentVariables(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-28 15:29:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-04 01:46:39 +02:00
										 |  |  |     /// <summary> | 
					
						
							|  |  |  |     /// Adds required services to the IServiceCollection. | 
					
						
							|  |  |  |     /// This should only add services that are not ASP.NET-related (i.e. no middleware). | 
					
						
							|  |  |  |     /// </summary> | 
					
						
							|  |  |  |     public static IServiceCollection AddServices(this IServiceCollection services, Config config) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         services | 
					
						
							|  |  |  |             .AddQueue() | 
					
						
							|  |  |  |             .AddDbContext<DatabaseContext>() | 
					
						
							|  |  |  |             .AddMetricServer(o => o.Port = config.Logging.MetricsPort) | 
					
						
							|  |  |  |             .AddMinio(c => | 
					
						
							|  |  |  |                 c.WithEndpoint(config.Storage.Endpoint) | 
					
						
							|  |  |  |                     .WithCredentials(config.Storage.AccessKey, config.Storage.SecretKey) | 
					
						
							|  |  |  |                     .Build()) | 
					
						
							|  |  |  |             .AddSingleton<MetricsCollectionService>() | 
					
						
							|  |  |  |             .AddSingleton<IClock>(SystemClock.Instance) | 
					
						
							|  |  |  |             .AddSnowflakeGenerator() | 
					
						
							|  |  |  |             .AddScoped<UserRendererService>() | 
					
						
							|  |  |  |             .AddScoped<MemberRendererService>() | 
					
						
							|  |  |  |             .AddScoped<AuthService>() | 
					
						
							|  |  |  |             .AddScoped<KeyCacheService>() | 
					
						
							|  |  |  |             .AddScoped<RemoteAuthService>() | 
					
						
							|  |  |  |             .AddScoped<ObjectStorageService>() | 
					
						
							|  |  |  |             // Background services | 
					
						
							|  |  |  |             .AddHostedService<PeriodicTasksService>() | 
					
						
							|  |  |  |             // Transient jobs | 
					
						
							|  |  |  |             .AddTransient<MemberAvatarUpdateInvocable>() | 
					
						
							|  |  |  |             .AddTransient<UserAvatarUpdateInvocable>(); | 
					
						
							| 
									
										
										
										
											2024-09-04 14:25:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-04 01:46:39 +02:00
										 |  |  |         if (!config.Logging.EnableMetrics) | 
					
						
							|  |  |  |             services.AddHostedService<BackgroundMetricsCollectionService>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return services; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-28 15:29:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     public static IServiceCollection AddCustomMiddleware(this IServiceCollection services) => services | 
					
						
							|  |  |  |         .AddScoped<ErrorHandlerMiddleware>() | 
					
						
							|  |  |  |         .AddScoped<AuthenticationMiddleware>() | 
					
						
							|  |  |  |         .AddScoped<AuthorizationMiddleware>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public static IApplicationBuilder UseCustomMiddleware(this IApplicationBuilder app) => app | 
					
						
							|  |  |  |         .UseMiddleware<ErrorHandlerMiddleware>() | 
					
						
							|  |  |  |         .UseMiddleware<AuthenticationMiddleware>() | 
					
						
							|  |  |  |         .UseMiddleware<AuthorizationMiddleware>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public static async Task Initialize(this WebApplication app, string[] args) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         await BuildInfo.ReadBuildInfo(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-03 16:29:51 +02:00
										 |  |  |         app.Services.ConfigureQueue().LogQueuedTaskProgress(app.Services.GetRequiredService<ILogger<IQueue>>()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-28 15:29:18 +02:00
										 |  |  |         await using var scope = app.Services.CreateAsyncScope(); | 
					
						
							|  |  |  |         var logger = scope.ServiceProvider.GetRequiredService<ILogger>().ForContext<WebApplication>(); | 
					
						
							|  |  |  |         var db = scope.ServiceProvider.GetRequiredService<DatabaseContext>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         logger.Information("Starting Foxnouns.NET {Version} ({Hash})", BuildInfo.Version, BuildInfo.Hash); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         var pendingMigrations = (await db.Database.GetPendingMigrationsAsync()).ToList(); | 
					
						
							|  |  |  |         if (args.Contains("--migrate") || args.Contains("--migrate-and-start")) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (pendingMigrations.Count == 0) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 logger.Information("Migrations requested but no migrations are required"); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 logger.Information("Migrating database to the latest version"); | 
					
						
							|  |  |  |                 await db.Database.MigrateAsync(); | 
					
						
							|  |  |  |                 logger.Information("Successfully migrated database"); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!args.Contains("--migrate-and-start")) Environment.Exit(0); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (pendingMigrations.Count > 0) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             logger.Fatal( | 
					
						
							|  |  |  |                 "There are {Count} pending migrations, run server with --migrate or --migrate-and-start to run migrations.", | 
					
						
							|  |  |  |                 pendingMigrations.Count); | 
					
						
							|  |  |  |             Environment.Exit(1); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-05-30 16:59:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-28 15:29:18 +02:00
										 |  |  |         logger.Information("Initializing frontend OAuth application"); | 
					
						
							|  |  |  |         _ = await db.GetFrontendApplicationAsync(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |