2
0

Startup.cs 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  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(), _serverConfigurationManager.Configuration.KnownProxies);
  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.UseForwardedHeaders();
  95. mainApp.UseMiddleware<ExceptionMiddleware>();
  96. mainApp.UseMiddleware<ResponseTimeMiddleware>();
  97. mainApp.UseWebSockets();
  98. mainApp.UseResponseCompression();
  99. mainApp.UseCors();
  100. if (_serverConfigurationManager.Configuration.RequireHttps
  101. && _serverApplicationHost.ListenWithHttps)
  102. {
  103. mainApp.UseHttpsRedirection();
  104. }
  105. mainApp.UseStaticFiles();
  106. if (appConfig.HostWebClient())
  107. {
  108. mainApp.UseStaticFiles(new StaticFileOptions
  109. {
  110. FileProvider = new PhysicalFileProvider(_serverConfigurationManager.ApplicationPaths.WebPath),
  111. RequestPath = "/web"
  112. });
  113. }
  114. mainApp.UseAuthentication();
  115. mainApp.UseJellyfinApiSwagger(_serverConfigurationManager);
  116. mainApp.UseRouting();
  117. mainApp.UseAuthorization();
  118. mainApp.UseLanFiltering();
  119. mainApp.UseIpBasedAccessValidation();
  120. mainApp.UseWebSocketHandler();
  121. mainApp.UseServerStartupMessage();
  122. if (_serverConfigurationManager.Configuration.EnableMetrics)
  123. {
  124. // Must be registered after any middleware that could change HTTP response codes or the data will be bad
  125. mainApp.UseHttpMetrics();
  126. }
  127. mainApp.UseEndpoints(endpoints =>
  128. {
  129. endpoints.MapControllers();
  130. if (_serverConfigurationManager.Configuration.EnableMetrics)
  131. {
  132. endpoints.MapMetrics("/metrics");
  133. }
  134. endpoints.MapHealthChecks("/health");
  135. });
  136. });
  137. // Add type descriptor for legacy datetime parsing.
  138. TypeDescriptor.AddAttributes(typeof(DateTime?), new TypeConverterAttribute(typeof(DateTimeTypeConverter)));
  139. }
  140. }
  141. }