DashboardController.cs 6.6 KB

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