Startup.cs 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. using System;
  2. using System.ComponentModel;
  3. using System.Net.Http.Headers;
  4. using Jellyfin.Api.TypeConverters;
  5. using Jellyfin.Server.Extensions;
  6. using Jellyfin.Server.Implementations;
  7. using Jellyfin.Server.Middleware;
  8. using MediaBrowser.Common.Net;
  9. using MediaBrowser.Controller;
  10. using MediaBrowser.Controller.Configuration;
  11. using MediaBrowser.Controller.Extensions;
  12. using Microsoft.AspNetCore.Builder;
  13. using Microsoft.AspNetCore.Hosting;
  14. using Microsoft.Extensions.Configuration;
  15. using Microsoft.Extensions.DependencyInjection;
  16. using Microsoft.Extensions.FileProviders;
  17. using Microsoft.Extensions.Hosting;
  18. using Prometheus;
  19. namespace Jellyfin.Server
  20. {
  21. /// <summary>
  22. /// Startup configuration for the Kestrel webhost.
  23. /// </summary>
  24. public class Startup
  25. {
  26. private readonly IServerConfigurationManager _serverConfigurationManager;
  27. private readonly IServerApplicationHost _serverApplicationHost;
  28. /// <summary>
  29. /// Initializes a new instance of the <see cref="Startup" /> class.
  30. /// </summary>
  31. /// <param name="serverConfigurationManager">The server configuration manager.</param>
  32. /// <param name="serverApplicationHost">The server application host.</param>
  33. public Startup(
  34. IServerConfigurationManager serverConfigurationManager,
  35. IServerApplicationHost serverApplicationHost)
  36. {
  37. _serverConfigurationManager = serverConfigurationManager;
  38. _serverApplicationHost = serverApplicationHost;
  39. }
  40. /// <summary>
  41. /// Configures the service collection for the webhost.
  42. /// </summary>
  43. /// <param name="services">The service collection.</param>
  44. public void ConfigureServices(IServiceCollection services)
  45. {
  46. services.AddResponseCompression();
  47. services.AddHttpContextAccessor();
  48. services.AddHttpsRedirection(options =>
  49. {
  50. options.HttpsPort = _serverApplicationHost.HttpsPort;
  51. });
  52. services.AddJellyfinApi(_serverApplicationHost.GetApiPluginAssemblies());
  53. services.AddJellyfinApiSwagger();
  54. // configure custom legacy authentication
  55. services.AddCustomAuthentication();
  56. services.AddJellyfinApiAuthorization();
  57. var productHeader = new ProductInfoHeaderValue(
  58. _serverApplicationHost.Name.Replace(' ', '-'),
  59. _serverApplicationHost.ApplicationVersionString);
  60. services
  61. .AddHttpClient(NamedClient.Default, c =>
  62. {
  63. c.DefaultRequestHeaders.UserAgent.Add(productHeader);
  64. })
  65. .ConfigurePrimaryHttpMessageHandler(x => new DefaultHttpClientHandler());
  66. services.AddHttpClient(NamedClient.MusicBrainz, c =>
  67. {
  68. c.DefaultRequestHeaders.UserAgent.Add(productHeader);
  69. c.DefaultRequestHeaders.UserAgent.Add(new ProductInfoHeaderValue($"({_serverApplicationHost.ApplicationUserAgentAddress})"));
  70. })
  71. .ConfigurePrimaryHttpMessageHandler(x => new DefaultHttpClientHandler());
  72. services.AddHealthChecks()
  73. .AddDbContextCheck<JellyfinDb>();
  74. }
  75. /// <summary>
  76. /// Configures the app builder for the webhost.
  77. /// </summary>
  78. /// <param name="app">The application builder.</param>
  79. /// <param name="env">The webhost environment.</param>
  80. /// <param name="appConfig">The application config.</param>
  81. public void Configure(
  82. IApplicationBuilder app,
  83. IWebHostEnvironment env,
  84. IConfiguration appConfig)
  85. {
  86. app.UseBaseUrlRedirection();
  87. // Wrap rest of configuration so everything only listens on BaseUrl.
  88. app.Map(_serverConfigurationManager.Configuration.BaseUrl, mainApp =>
  89. {
  90. if (env.IsDevelopment())
  91. {
  92. mainApp.UseDeveloperExceptionPage();
  93. }
  94. mainApp.UseMiddleware<ExceptionMiddleware>();
  95. mainApp.UseMiddleware<ResponseTimeMiddleware>();
  96. mainApp.UseWebSockets();
  97. mainApp.UseResponseCompression();
  98. mainApp.UseCors();
  99. if (_serverConfigurationManager.Configuration.RequireHttps
  100. && _serverApplicationHost.ListenWithHttps)
  101. {
  102. mainApp.UseHttpsRedirection();
  103. }
  104. mainApp.UseStaticFiles();
  105. if (appConfig.HostWebClient())
  106. {
  107. mainApp.UseStaticFiles(new StaticFileOptions
  108. {
  109. FileProvider = new PhysicalFileProvider(_serverConfigurationManager.ApplicationPaths.WebPath),
  110. RequestPath = "/web"
  111. });
  112. }
  113. mainApp.UseAuthentication();
  114. mainApp.UseJellyfinApiSwagger(_serverConfigurationManager);
  115. mainApp.UseRouting();
  116. mainApp.UseAuthorization();
  117. mainApp.UseLanFiltering();
  118. mainApp.UseIpBasedAccessValidation();
  119. mainApp.UseWebSocketHandler();
  120. mainApp.UseServerStartupMessage();
  121. if (_serverConfigurationManager.Configuration.EnableMetrics)
  122. {
  123. // Must be registered after any middleware that could change HTTP response codes or the data will be bad
  124. mainApp.UseHttpMetrics();
  125. }
  126. mainApp.UseEndpoints(endpoints =>
  127. {
  128. endpoints.MapControllers();
  129. if (_serverConfigurationManager.Configuration.EnableMetrics)
  130. {
  131. endpoints.MapMetrics("/metrics");
  132. }
  133. endpoints.MapHealthChecks("/health");
  134. });
  135. });
  136. // Add type descriptor for legacy datetime parsing.
  137. TypeDescriptor.AddAttributes(typeof(DateTime?), new TypeConverterAttribute(typeof(DateTimeTypeConverter)));
  138. }
  139. }
  140. }