DashboardController.cs 6.9 KB

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