DashboardController.cs 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  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.Model.Net;
  10. using MediaBrowser.Model.Plugins;
  11. using Microsoft.AspNetCore.Authorization;
  12. using Microsoft.AspNetCore.Http;
  13. using Microsoft.AspNetCore.Mvc;
  14. using Microsoft.Extensions.Logging;
  15. namespace Jellyfin.Api.Controllers;
  16. /// <summary>
  17. /// The dashboard controller.
  18. /// </summary>
  19. [Route("")]
  20. public class DashboardController : BaseJellyfinApiController
  21. {
  22. private readonly ILogger<DashboardController> _logger;
  23. private readonly IPluginManager _pluginManager;
  24. /// <summary>
  25. /// Initializes a new instance of the <see cref="DashboardController"/> class.
  26. /// </summary>
  27. /// <param name="logger">Instance of <see cref="ILogger{DashboardController}"/> interface.</param>
  28. /// <param name="pluginManager">Instance of <see cref="IPluginManager"/> interface.</param>
  29. public DashboardController(
  30. ILogger<DashboardController> logger,
  31. IPluginManager pluginManager)
  32. {
  33. _logger = logger;
  34. _pluginManager = pluginManager;
  35. }
  36. /// <summary>
  37. /// Gets the configuration pages.
  38. /// </summary>
  39. /// <param name="enableInMainMenu">Whether to enable in the main menu.</param>
  40. /// <response code="200">ConfigurationPages returned.</response>
  41. /// <response code="404">Server still loading.</response>
  42. /// <returns>An <see cref="IEnumerable{ConfigurationPageInfo}"/> with infos about the plugins.</returns>
  43. [HttpGet("web/ConfigurationPages")]
  44. [ProducesResponseType(StatusCodes.Status200OK)]
  45. [ProducesResponseType(StatusCodes.Status404NotFound)]
  46. [Authorize]
  47. public ActionResult<IEnumerable<ConfigurationPageInfo>> GetConfigurationPages(
  48. [FromQuery] bool? enableInMainMenu)
  49. {
  50. var configPages = _pluginManager.Plugins.SelectMany(GetConfigPages).ToList();
  51. if (enableInMainMenu.HasValue)
  52. {
  53. configPages = configPages.Where(p => p.EnableInMainMenu == enableInMainMenu.Value).ToList();
  54. }
  55. return configPages;
  56. }
  57. /// <summary>
  58. /// Gets a dashboard configuration page.
  59. /// </summary>
  60. /// <param name="name">The name of the page.</param>
  61. /// <response code="200">ConfigurationPage returned.</response>
  62. /// <response code="404">Plugin configuration page not found.</response>
  63. /// <returns>The configuration page.</returns>
  64. [HttpGet("web/ConfigurationPage")]
  65. [ProducesResponseType(StatusCodes.Status200OK)]
  66. [ProducesResponseType(StatusCodes.Status404NotFound)]
  67. [ProducesFile(MediaTypeNames.Text.Html, "application/x-javascript")]
  68. public ActionResult GetDashboardConfigurationPage([FromQuery] string? name)
  69. {
  70. var altPage = GetPluginPages().FirstOrDefault(p => string.Equals(p.Item1.Name, name, StringComparison.OrdinalIgnoreCase));
  71. if (altPage is null)
  72. {
  73. return NotFound();
  74. }
  75. IPlugin plugin = altPage.Item2;
  76. string resourcePath = altPage.Item1.EmbeddedResourcePath;
  77. Stream? stream = plugin.GetType().Assembly.GetManifestResourceStream(resourcePath);
  78. if (stream is null)
  79. {
  80. _logger.LogError("Failed to get resource {Resource} from plugin {Plugin}", resourcePath, plugin.Name);
  81. return NotFound();
  82. }
  83. return File(stream, MimeTypes.GetMimeType(resourcePath));
  84. }
  85. private IEnumerable<ConfigurationPageInfo> GetConfigPages(LocalPlugin plugin)
  86. {
  87. return GetPluginPages(plugin).Select(i => new ConfigurationPageInfo(plugin.Instance, i.Item1));
  88. }
  89. private IEnumerable<Tuple<PluginPageInfo, IPlugin>> GetPluginPages(LocalPlugin plugin)
  90. {
  91. if (plugin.Instance is not IHasWebPages hasWebPages)
  92. {
  93. return Enumerable.Empty<Tuple<PluginPageInfo, IPlugin>>();
  94. }
  95. return hasWebPages.GetPages().Select(i => new Tuple<PluginPageInfo, IPlugin>(i, plugin.Instance));
  96. }
  97. private IEnumerable<Tuple<PluginPageInfo, IPlugin>> GetPluginPages()
  98. {
  99. return _pluginManager.Plugins.SelectMany(GetPluginPages);
  100. }
  101. }