Program.cs 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. using Application;
  2. using Application.Abstractions.Chat;
  3. using Infrastructure;
  4. using SharedKernel;
  5. using System.Reflection;
  6. using Microsoft.Extensions.Hosting.WindowsServices;
  7. using Serilog;
  8. using Web.Api;
  9. using Web.Api.Extensions;
  10. using Web.Api.Hubs;
  11. using Web.Api.Services;
  12. var builder = WebApplication.CreateBuilder(args);
  13. var settings = builder.Configuration.Get<AppSettings>()!;
  14. if (!WindowsServiceHelpers.IsWindowsService())
  15. {
  16. Console.Title = settings.App.Name;
  17. }
  18. Console.WriteLine($"ENV={builder.Environment.EnvironmentName}");
  19. Console.WriteLine($"현재 시간: {DateTime.Now} / {TimeZoneInfo.Local.Id}");
  20. builder.Host.UseWindowsService();
  21. // Serilog — Production에서만 파일 로그
  22. if (builder.Environment.IsProduction())
  23. {
  24. Log.Logger = new LoggerConfiguration()
  25. .ReadFrom.Configuration(builder.Configuration)
  26. .WriteTo.File("logs/webapi-.log",
  27. rollingInterval: RollingInterval.Day,
  28. retainedFileCountLimit: 30,
  29. outputTemplate: "{Timestamp:yyyy-MM-dd HH:mm:ss.fff} [{Level:u3}] {Message:lj}{NewLine}{Exception}")
  30. .CreateLogger();
  31. builder.Host.UseSerilog();
  32. }
  33. builder.Services.Configure<AppSettings>(builder.Configuration);
  34. builder.Services
  35. .AddApplication()
  36. .AddPresentation()
  37. .AddApiInfrastructure(builder.Configuration);
  38. // CORS
  39. builder.Services.AddCors(options =>
  40. {
  41. options.AddPolicy(settings.CorsPolicy.Name, policy =>
  42. {
  43. policy
  44. .WithOrigins(settings.CorsPolicy.AllowedOrigins.ToArray())
  45. .AllowAnyHeader()
  46. .AllowAnyMethod()
  47. .AllowCredentials()
  48. .SetPreflightMaxAge(TimeSpan.FromSeconds(settings.CorsPolicy.PreflightMaxAgeSeconds));
  49. });
  50. });
  51. // SignalR
  52. builder.Services.AddSignalR();
  53. // Chat
  54. builder.Services.AddSingleton<IChatHubService, ChatHubService>();
  55. builder.Services.AddHostedService<KickSubscriberService>();
  56. // Endpoints
  57. builder.Services.AddEndpoints(Assembly.GetExecutingAssembly());
  58. builder.Logging.AddConsole();
  59. /**
  60. * =======================================================================================================================================================
  61. */
  62. var app = builder.Build();
  63. /**
  64. * =======================================================================================================================================================
  65. */
  66. if (app.Environment.IsDevelopment())
  67. {
  68. app.UseSwagger();
  69. app.UseSwaggerUI();
  70. }
  71. // 상태 확인\
  72. app.MapHealthChecks("/health");
  73. app.UseExceptionHandler();
  74. app.UseCors(settings.CorsPolicy.Name);
  75. app.UseStaticFiles();
  76. app.UseAuthentication();
  77. app.UseAuthorization();
  78. app.MapEndpoints();
  79. app.MapHub<ChatHub>("/hubs/chat");
  80. // 서버 시작 시 이전 채팅 접속자 정보 초기화 (stale 연결 제거)
  81. using (var scope = app.Services.CreateScope())
  82. {
  83. await scope.ServiceProvider.GetRequiredService<IChatConnectionTracker>().ClearAllAsync();
  84. }
  85. app.Run();