DashboardController.cs 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Linq;
  5. using Jellyfin.Api.Models;
  6. using MediaBrowser.Common.Plugins;
  7. using MediaBrowser.Controller;
  8. using MediaBrowser.Controller.Configuration;
  9. using MediaBrowser.Controller.Extensions;
  10. using MediaBrowser.Controller.Plugins;
  11. using MediaBrowser.Model.Net;
  12. using MediaBrowser.Model.Plugins;
  13. using Microsoft.AspNetCore.Http;
  14. using Microsoft.AspNetCore.Http.Extensions;
  15. using Microsoft.AspNetCore.Mvc;
  16. using Microsoft.Extensions.Configuration;
  17. using Microsoft.Extensions.Logging;
  18. using Microsoft.Net.Http.Headers;
  19. namespace Jellyfin.Api.Controllers
  20. {
  21. /// <summary>
  22. /// The dashboard controller.
  23. /// </summary>
  24. [Route("")]
  25. public class DashboardController : BaseJellyfinApiController
  26. {
  27. private readonly ILogger<DashboardController> _logger;
  28. private readonly IServerApplicationHost _appHost;
  29. /// <summary>
  30. /// Initializes a new instance of the <see cref="DashboardController"/> class.
  31. /// </summary>
  32. /// <param name="logger">Instance of <see cref="ILogger{DashboardController}"/> interface.</param>
  33. /// <param name="appHost">Instance of <see cref="IServerApplicationHost"/> interface.</param>
  34. public DashboardController(
  35. ILogger<DashboardController> logger,
  36. IServerApplicationHost appHost)
  37. {
  38. _logger = logger;
  39. _appHost = appHost;
  40. }
  41. /// <summary>
  42. /// Gets the configuration pages.
  43. /// </summary>
  44. /// <param name="enableInMainMenu">Whether to enable in the main menu.</param>
  45. /// <param name="pageType">The <see cref="ConfigurationPageInfo"/>.</param>
  46. /// <response code="200">ConfigurationPages returned.</response>
  47. /// <response code="404">Server still loading.</response>
  48. /// <returns>An <see cref="IEnumerable{ConfigurationPageInfo}"/> with infos about the plugins.</returns>
  49. [HttpGet("web/ConfigurationPages")]
  50. [ProducesResponseType(StatusCodes.Status200OK)]
  51. [ProducesResponseType(StatusCodes.Status404NotFound)]
  52. public ActionResult<IEnumerable<ConfigurationPageInfo?>> GetConfigurationPages(
  53. [FromQuery] bool? enableInMainMenu,
  54. [FromQuery] ConfigurationPageType? pageType)
  55. {
  56. const string unavailableMessage = "The server is still loading. Please try again momentarily.";
  57. var pages = _appHost.GetExports<IPluginConfigurationPage>().ToList();
  58. if (pages == null)
  59. {
  60. return NotFound(unavailableMessage);
  61. }
  62. // Don't allow a failing plugin to fail them all
  63. var configPages = pages.Select(p =>
  64. {
  65. try
  66. {
  67. return new ConfigurationPageInfo(p);
  68. }
  69. catch (Exception ex)
  70. {
  71. _logger.LogError(ex, "Error getting plugin information from {Plugin}", p.GetType().Name);
  72. return null;
  73. }
  74. })
  75. .Where(i => i != null)
  76. .ToList();
  77. configPages.AddRange(_appHost.Plugins.SelectMany(GetConfigPages));
  78. if (pageType.HasValue)
  79. {
  80. configPages = configPages.Where(p => p!.ConfigurationPageType == pageType).ToList();
  81. }
  82. if (enableInMainMenu.HasValue)
  83. {
  84. configPages = configPages.Where(p => p!.EnableInMainMenu == enableInMainMenu.Value).ToList();
  85. }
  86. return configPages;
  87. }
  88. /// <summary>
  89. /// Gets a dashboard configuration page.
  90. /// </summary>
  91. /// <param name="name">The name of the page.</param>
  92. /// <response code="200">ConfigurationPage returned.</response>
  93. /// <response code="404">Plugin configuration page not found.</response>
  94. /// <returns>The configuration page.</returns>
  95. [HttpGet("web/ConfigurationPage")]
  96. [ProducesResponseType(StatusCodes.Status200OK)]
  97. [ProducesResponseType(StatusCodes.Status404NotFound)]
  98. public ActionResult GetDashboardConfigurationPage([FromQuery] string? name)
  99. {
  100. IPlugin? plugin = null;
  101. Stream? stream = null;
  102. var isJs = false;
  103. var isTemplate = false;
  104. var page = _appHost.GetExports<IPluginConfigurationPage>().FirstOrDefault(p => string.Equals(p.Name, name, StringComparison.OrdinalIgnoreCase));
  105. if (page != null)
  106. {
  107. plugin = page.Plugin;
  108. stream = page.GetHtmlStream();
  109. }
  110. if (plugin == null)
  111. {
  112. var altPage = GetPluginPages().FirstOrDefault(p => string.Equals(p.Item1.Name, name, StringComparison.OrdinalIgnoreCase));
  113. if (altPage != null)
  114. {
  115. plugin = altPage.Item2;
  116. stream = plugin.GetType().Assembly.GetManifestResourceStream(altPage.Item1.EmbeddedResourcePath);
  117. isJs = string.Equals(Path.GetExtension(altPage.Item1.EmbeddedResourcePath), ".js", StringComparison.OrdinalIgnoreCase);
  118. isTemplate = altPage.Item1.EmbeddedResourcePath.EndsWith(".template.html", StringComparison.Ordinal);
  119. }
  120. }
  121. if (plugin != null && stream != null)
  122. {
  123. if (isJs)
  124. {
  125. return File(stream, MimeTypes.GetMimeType("page.js"));
  126. }
  127. if (isTemplate)
  128. {
  129. return File(stream, MimeTypes.GetMimeType("page.html"));
  130. }
  131. return File(stream, MimeTypes.GetMimeType("page.html"));
  132. }
  133. return NotFound();
  134. }
  135. private IEnumerable<ConfigurationPageInfo> GetConfigPages(IPlugin plugin)
  136. {
  137. return GetPluginPages(plugin).Select(i => new ConfigurationPageInfo(plugin, i.Item1));
  138. }
  139. private IEnumerable<Tuple<PluginPageInfo, IPlugin>> GetPluginPages(IPlugin plugin)
  140. {
  141. if (!(plugin is IHasWebPages hasWebPages))
  142. {
  143. return new List<Tuple<PluginPageInfo, IPlugin>>();
  144. }
  145. return hasWebPages.GetPages().Select(i => new Tuple<PluginPageInfo, IPlugin>(i, plugin));
  146. }
  147. private IEnumerable<Tuple<PluginPageInfo, IPlugin>> GetPluginPages()
  148. {
  149. return _appHost.Plugins.SelectMany(GetPluginPages);
  150. }
  151. }
  152. }