jellyfin/Jellyfin.Api/Controllers/DlnaServerController.cs

387 lines
17 KiB
C#
Raw Normal View History

using System;
2020-09-05 16:11:44 -07:00
using System.ComponentModel.DataAnnotations;
2020-07-31 10:01:30 -07:00
using System.Diagnostics.CodeAnalysis;
using System.IO;
using System.Net.Mime;
using System.Threading.Tasks;
using Emby.Dlna;
using Emby.Dlna.Main;
using Jellyfin.Api.Attributes;
using Jellyfin.Api.Constants;
using MediaBrowser.Controller.Dlna;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Mvc;
namespace Jellyfin.Api.Controllers
{
/// <summary>
/// Dlna Server Controller.
/// </summary>
[Route("Dlna")]
[Authorize(Policy = Policies.AnonymousLanAccessPolicy)]
public class DlnaServerController : BaseJellyfinApiController
{
private readonly IDlnaManager _dlnaManager;
private readonly IContentDirectory _contentDirectory;
private readonly IConnectionManager _connectionManager;
private readonly IMediaReceiverRegistrar _mediaReceiverRegistrar;
/// <summary>
/// Initializes a new instance of the <see cref="DlnaServerController"/> class.
/// </summary>
/// <param name="dlnaManager">Instance of the <see cref="IDlnaManager"/> interface.</param>
public DlnaServerController(IDlnaManager dlnaManager)
{
_dlnaManager = dlnaManager;
_contentDirectory = DlnaEntryPoint.Current.ContentDirectory;
_connectionManager = DlnaEntryPoint.Current.ConnectionManager;
_mediaReceiverRegistrar = DlnaEntryPoint.Current.MediaReceiverRegistrar;
}
/// <summary>
/// Get Description Xml.
/// </summary>
2020-07-31 10:01:30 -07:00
/// <param name="serverId">Server UUID.</param>
/// <response code="200">Description xml returned.</response>
2020-12-22 08:33:25 -07:00
/// <response code="503">DLNA is disabled.</response>
2020-07-31 10:01:30 -07:00
/// <returns>An <see cref="OkResult"/> containing the description xml.</returns>
[HttpGet("{serverId}/description")]
2020-08-03 13:38:51 -07:00
[HttpGet("{serverId}/description.xml", Name = "GetDescriptionXml_2")]
2020-09-03 12:48:48 -07:00
[ProducesResponseType(StatusCodes.Status200OK)]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status503ServiceUnavailable)]
[Produces(MediaTypeNames.Text.Xml)]
[ProducesFile(MediaTypeNames.Text.Xml)]
2020-09-06 08:07:27 -07:00
public ActionResult GetDescriptionXml([FromRoute, Required] string serverId)
{
2020-12-18 16:29:21 -07:00
if (DlnaEntryPoint.Enabled)
{
var url = GetAbsoluteUri();
var serverAddress = url.Substring(0, url.IndexOf("/dlna/", StringComparison.OrdinalIgnoreCase));
var xml = _dlnaManager.GetServerDescriptionXml(Request.Headers, serverId, serverAddress);
return Ok(xml);
}
2020-12-22 08:33:25 -07:00
return StatusCode(StatusCodes.Status503ServiceUnavailable);
}
/// <summary>
/// Gets Dlna content directory xml.
/// </summary>
2020-07-31 10:01:30 -07:00
/// <param name="serverId">Server UUID.</param>
/// <response code="200">Dlna content directory returned.</response>
2020-12-22 08:33:25 -07:00
/// <response code="503">DLNA is disabled.</response>
2020-07-31 10:01:30 -07:00
/// <returns>An <see cref="OkResult"/> containing the dlna content directory xml.</returns>
2020-08-16 07:32:13 -07:00
[HttpGet("{serverId}/ContentDirectory")]
2020-08-16 09:53:57 -07:00
[HttpGet("{serverId}/ContentDirectory/ContentDirectory", Name = "GetContentDirectory_2")]
[HttpGet("{serverId}/ContentDirectory/ContentDirectory.xml", Name = "GetContentDirectory_3")]
[ProducesResponseType(StatusCodes.Status200OK)]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status503ServiceUnavailable)]
2020-09-03 12:48:48 -07:00
[Produces(MediaTypeNames.Text.Xml)]
[ProducesFile(MediaTypeNames.Text.Xml)]
2020-07-31 10:01:30 -07:00
[SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
2020-09-06 08:07:27 -07:00
public ActionResult GetContentDirectory([FromRoute, Required] string serverId)
{
2020-12-18 16:29:21 -07:00
if (DlnaEntryPoint.Enabled)
{
return Ok(_contentDirectory.GetServiceXml());
}
2020-12-22 08:33:25 -07:00
return StatusCode(StatusCodes.Status503ServiceUnavailable);
}
/// <summary>
/// Gets Dlna media receiver registrar xml.
/// </summary>
2020-07-31 10:01:30 -07:00
/// <param name="serverId">Server UUID.</param>
2020-11-07 17:30:56 -07:00
/// <response code="200">Dlna media receiver registrar xml returned.</response>
2020-12-22 08:33:25 -07:00
/// <response code="503">DLNA is disabled.</response>
/// <returns>Dlna media receiver registrar xml.</returns>
2020-09-03 12:50:02 -07:00
[HttpGet("{serverId}/MediaReceiverRegistrar")]
2020-08-16 09:53:57 -07:00
[HttpGet("{serverId}/MediaReceiverRegistrar/MediaReceiverRegistrar", Name = "GetMediaReceiverRegistrar_2")]
[HttpGet("{serverId}/MediaReceiverRegistrar/MediaReceiverRegistrar.xml", Name = "GetMediaReceiverRegistrar_3")]
2020-09-03 12:48:48 -07:00
[ProducesResponseType(StatusCodes.Status200OK)]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status503ServiceUnavailable)]
[Produces(MediaTypeNames.Text.Xml)]
[ProducesFile(MediaTypeNames.Text.Xml)]
2020-07-31 10:01:30 -07:00
[SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
2020-09-06 08:07:27 -07:00
public ActionResult GetMediaReceiverRegistrar([FromRoute, Required] string serverId)
{
2020-12-18 16:29:21 -07:00
if (DlnaEntryPoint.Enabled)
{
return Ok(_mediaReceiverRegistrar.GetServiceXml());
}
2020-12-22 08:33:25 -07:00
return StatusCode(StatusCodes.Status503ServiceUnavailable);
}
/// <summary>
/// Gets Dlna media receiver registrar xml.
/// </summary>
2020-07-31 10:01:30 -07:00
/// <param name="serverId">Server UUID.</param>
2020-11-07 17:30:56 -07:00
/// <response code="200">Dlna media receiver registrar xml returned.</response>
2020-12-22 08:33:25 -07:00
/// <response code="503">DLNA is disabled.</response>
/// <returns>Dlna media receiver registrar xml.</returns>
2020-08-16 07:32:13 -07:00
[HttpGet("{serverId}/ConnectionManager")]
2020-08-16 09:53:57 -07:00
[HttpGet("{serverId}/ConnectionManager/ConnectionManager", Name = "GetConnectionManager_2")]
[HttpGet("{serverId}/ConnectionManager/ConnectionManager.xml", Name = "GetConnectionManager_3")]
2020-09-03 12:48:48 -07:00
[ProducesResponseType(StatusCodes.Status200OK)]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status503ServiceUnavailable)]
[Produces(MediaTypeNames.Text.Xml)]
[ProducesFile(MediaTypeNames.Text.Xml)]
2020-07-31 10:01:30 -07:00
[SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
2020-09-06 08:07:27 -07:00
public ActionResult GetConnectionManager([FromRoute, Required] string serverId)
{
2020-12-18 16:29:21 -07:00
if (DlnaEntryPoint.Enabled)
{
return Ok(_connectionManager.GetServiceXml());
}
2020-12-22 08:33:25 -07:00
return StatusCode(StatusCodes.Status503ServiceUnavailable);
}
/// <summary>
/// Process a content directory control request.
/// </summary>
2020-07-31 10:01:30 -07:00
/// <param name="serverId">Server UUID.</param>
2020-11-07 17:30:56 -07:00
/// <response code="200">Request processed.</response>
2020-12-22 08:33:25 -07:00
/// <response code="503">DLNA is disabled.</response>
/// <returns>Control response.</returns>
2020-07-31 10:01:30 -07:00
[HttpPost("{serverId}/ContentDirectory/Control")]
2020-11-07 17:30:56 -07:00
[ProducesResponseType(StatusCodes.Status200OK)]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status503ServiceUnavailable)]
2020-11-07 17:30:56 -07:00
[Produces(MediaTypeNames.Text.Xml)]
[ProducesFile(MediaTypeNames.Text.Xml)]
2020-09-06 08:07:27 -07:00
public async Task<ActionResult<ControlResponse>> ProcessContentDirectoryControlRequest([FromRoute, Required] string serverId)
{
2020-12-19 02:34:04 -07:00
if (DlnaEntryPoint.Enabled)
{
return await ProcessControlRequestInternalAsync(serverId, Request.Body, _contentDirectory).ConfigureAwait(false);
}
2020-12-22 08:33:25 -07:00
return StatusCode(StatusCodes.Status503ServiceUnavailable);
}
/// <summary>
/// Process a connection manager control request.
/// </summary>
2020-07-31 10:01:30 -07:00
/// <param name="serverId">Server UUID.</param>
2020-11-07 17:30:56 -07:00
/// <response code="200">Request processed.</response>
2020-12-22 08:33:25 -07:00
/// <response code="503">DLNA is disabled.</response>
/// <returns>Control response.</returns>
2020-07-31 10:01:30 -07:00
[HttpPost("{serverId}/ConnectionManager/Control")]
2020-11-07 17:30:56 -07:00
[ProducesResponseType(StatusCodes.Status200OK)]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status503ServiceUnavailable)]
2020-11-07 17:30:56 -07:00
[Produces(MediaTypeNames.Text.Xml)]
[ProducesFile(MediaTypeNames.Text.Xml)]
2020-09-06 08:07:27 -07:00
public async Task<ActionResult<ControlResponse>> ProcessConnectionManagerControlRequest([FromRoute, Required] string serverId)
{
2020-12-19 02:34:04 -07:00
if (DlnaEntryPoint.Enabled)
{
return await ProcessControlRequestInternalAsync(serverId, Request.Body, _connectionManager).ConfigureAwait(false);
}
2020-12-22 08:33:25 -07:00
return StatusCode(StatusCodes.Status503ServiceUnavailable);
}
/// <summary>
/// Process a media receiver registrar control request.
/// </summary>
2020-07-31 10:01:30 -07:00
/// <param name="serverId">Server UUID.</param>
2020-11-07 17:30:56 -07:00
/// <response code="200">Request processed.</response>
2020-12-22 08:33:25 -07:00
/// <response code="503">DLNA is disabled.</response>
/// <returns>Control response.</returns>
2020-07-31 10:01:30 -07:00
[HttpPost("{serverId}/MediaReceiverRegistrar/Control")]
2020-11-07 17:30:56 -07:00
[ProducesResponseType(StatusCodes.Status200OK)]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status503ServiceUnavailable)]
2020-11-07 17:30:56 -07:00
[Produces(MediaTypeNames.Text.Xml)]
[ProducesFile(MediaTypeNames.Text.Xml)]
2020-09-06 08:07:27 -07:00
public async Task<ActionResult<ControlResponse>> ProcessMediaReceiverRegistrarControlRequest([FromRoute, Required] string serverId)
{
2020-12-19 02:34:04 -07:00
if (DlnaEntryPoint.Enabled)
{
return await ProcessControlRequestInternalAsync(serverId, Request.Body, _mediaReceiverRegistrar).ConfigureAwait(false);
}
2020-12-22 08:33:25 -07:00
return StatusCode(StatusCodes.Status503ServiceUnavailable);
}
/// <summary>
/// Processes an event subscription request.
/// </summary>
2020-07-31 10:01:30 -07:00
/// <param name="serverId">Server UUID.</param>
2020-11-07 17:30:56 -07:00
/// <response code="200">Request processed.</response>
2020-12-22 08:33:25 -07:00
/// <response code="503">DLNA is disabled.</response>
/// <returns>Event subscription response.</returns>
2020-07-31 10:01:30 -07:00
[HttpSubscribe("{serverId}/MediaReceiverRegistrar/Events")]
[HttpUnsubscribe("{serverId}/MediaReceiverRegistrar/Events")]
[ApiExplorerSettings(IgnoreApi = true)] // Ignore in openapi docs
[SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
2020-11-07 17:30:56 -07:00
[ProducesResponseType(StatusCodes.Status200OK)]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status503ServiceUnavailable)]
2020-11-07 17:30:56 -07:00
[Produces(MediaTypeNames.Text.Xml)]
[ProducesFile(MediaTypeNames.Text.Xml)]
2020-07-31 10:01:30 -07:00
public ActionResult<EventSubscriptionResponse> ProcessMediaReceiverRegistrarEventRequest(string serverId)
{
2020-12-19 02:34:04 -07:00
if (DlnaEntryPoint.Enabled)
{
return ProcessEventRequest(_mediaReceiverRegistrar);
}
2020-12-22 08:33:25 -07:00
return StatusCode(StatusCodes.Status503ServiceUnavailable);
}
/// <summary>
/// Processes an event subscription request.
/// </summary>
2020-07-31 10:01:30 -07:00
/// <param name="serverId">Server UUID.</param>
2020-11-07 17:30:56 -07:00
/// <response code="200">Request processed.</response>
2020-12-22 08:33:25 -07:00
/// <response code="503">DLNA is disabled.</response>
/// <returns>Event subscription response.</returns>
2020-07-31 10:01:30 -07:00
[HttpSubscribe("{serverId}/ContentDirectory/Events")]
[HttpUnsubscribe("{serverId}/ContentDirectory/Events")]
[ApiExplorerSettings(IgnoreApi = true)] // Ignore in openapi docs
[SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
2020-11-07 17:30:56 -07:00
[ProducesResponseType(StatusCodes.Status200OK)]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status503ServiceUnavailable)]
2020-11-07 17:30:56 -07:00
[Produces(MediaTypeNames.Text.Xml)]
[ProducesFile(MediaTypeNames.Text.Xml)]
2020-07-31 10:01:30 -07:00
public ActionResult<EventSubscriptionResponse> ProcessContentDirectoryEventRequest(string serverId)
{
2020-12-19 02:34:04 -07:00
if (DlnaEntryPoint.Enabled)
{
return ProcessEventRequest(_contentDirectory);
}
2020-12-22 08:33:25 -07:00
return StatusCode(StatusCodes.Status503ServiceUnavailable);
}
/// <summary>
/// Processes an event subscription request.
/// </summary>
2020-07-31 10:01:30 -07:00
/// <param name="serverId">Server UUID.</param>
2020-11-07 17:30:56 -07:00
/// <response code="200">Request processed.</response>
2020-12-22 08:33:25 -07:00
/// <response code="503">DLNA is disabled.</response>
/// <returns>Event subscription response.</returns>
2020-07-31 10:01:30 -07:00
[HttpSubscribe("{serverId}/ConnectionManager/Events")]
[HttpUnsubscribe("{serverId}/ConnectionManager/Events")]
[ApiExplorerSettings(IgnoreApi = true)] // Ignore in openapi docs
[SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
2020-11-07 17:30:56 -07:00
[ProducesResponseType(StatusCodes.Status200OK)]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status503ServiceUnavailable)]
2020-11-07 17:30:56 -07:00
[Produces(MediaTypeNames.Text.Xml)]
[ProducesFile(MediaTypeNames.Text.Xml)]
2020-07-31 10:01:30 -07:00
public ActionResult<EventSubscriptionResponse> ProcessConnectionManagerEventRequest(string serverId)
{
2020-12-19 02:34:04 -07:00
if (DlnaEntryPoint.Enabled)
{
return ProcessEventRequest(_connectionManager);
}
2020-12-22 08:33:25 -07:00
return StatusCode(StatusCodes.Status503ServiceUnavailable);
}
/// <summary>
/// Gets a server icon.
/// </summary>
2020-07-31 10:01:30 -07:00
/// <param name="serverId">Server UUID.</param>
/// <param name="fileName">The icon filename.</param>
2020-12-22 08:23:55 -07:00
/// <response code="200">Request processed.</response>
/// <response code="404">Not Found.</response>
2020-12-22 08:33:25 -07:00
/// <response code="503">DLNA is disabled.</response>
/// <returns>Icon stream.</returns>
2020-08-03 13:38:51 -07:00
[HttpGet("{serverId}/icons/{fileName}")]
2020-07-31 10:01:30 -07:00
[SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
[ProducesResponseType(StatusCodes.Status200OK)]
2020-12-19 02:34:04 -07:00
[ProducesResponseType(StatusCodes.Status404NotFound)]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status503ServiceUnavailable)]
[ProducesImageFile]
2020-09-06 08:07:27 -07:00
public ActionResult GetIconId([FromRoute, Required] string serverId, [FromRoute, Required] string fileName)
{
2020-12-19 02:34:04 -07:00
if (DlnaEntryPoint.Enabled)
{
return GetIconInternal(fileName);
}
2020-12-22 08:33:25 -07:00
return StatusCode(StatusCodes.Status503ServiceUnavailable);
}
/// <summary>
/// Gets a server icon.
/// </summary>
/// <param name="fileName">The icon filename.</param>
/// <returns>Icon stream.</returns>
2020-12-22 08:23:55 -07:00
/// <response code="200">Request processed.</response>
/// <response code="404">Not Found.</response>
2020-12-22 08:33:25 -07:00
/// <response code="503">DLNA is disabled.</response>
2020-08-03 13:38:51 -07:00
[HttpGet("icons/{fileName}")]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status200OK)]
2020-12-19 02:34:04 -07:00
[ProducesResponseType(StatusCodes.Status404NotFound)]
2020-12-22 08:23:55 -07:00
[ProducesResponseType(StatusCodes.Status503ServiceUnavailable)]
[ProducesImageFile]
2020-09-06 08:07:27 -07:00
public ActionResult GetIcon([FromRoute, Required] string fileName)
{
2020-12-19 02:34:04 -07:00
if (DlnaEntryPoint.Enabled)
{
return GetIconInternal(fileName);
}
2020-12-22 08:33:25 -07:00
return StatusCode(StatusCodes.Status503ServiceUnavailable);
}
2020-09-02 14:06:13 -07:00
private ActionResult GetIconInternal(string fileName)
{
var icon = _dlnaManager.GetIcon(fileName);
if (icon == null)
{
return NotFound();
}
var contentType = "image/" + Path.GetExtension(fileName)
.TrimStart('.')
.ToLowerInvariant();
2020-07-31 10:01:30 -07:00
return File(icon.Stream, contentType);
}
private string GetAbsoluteUri()
{
2020-11-21 12:50:40 -07:00
return $"{Request.Scheme}://{Request.Host}{Request.PathBase}{Request.Path}";
}
2020-07-31 10:01:30 -07:00
private Task<ControlResponse> ProcessControlRequestInternalAsync(string id, Stream requestStream, IUpnpService service)
{
2020-08-20 12:04:57 -07:00
return service.ProcessControlRequestAsync(new ControlRequest(Request.Headers)
{
InputXml = requestStream,
TargetServerUuId = id,
RequestedUrl = GetAbsoluteUri()
});
}
private EventSubscriptionResponse ProcessEventRequest(IDlnaEventManager dlnaEventManager)
{
var subscriptionId = Request.Headers["SID"];
if (string.Equals(Request.Method, "subscribe", StringComparison.OrdinalIgnoreCase))
{
var notificationType = Request.Headers["NT"];
var callback = Request.Headers["CALLBACK"];
var timeoutString = Request.Headers["TIMEOUT"];
if (string.IsNullOrEmpty(notificationType))
{
return dlnaEventManager.RenewEventSubscription(
subscriptionId,
notificationType,
timeoutString,
callback);
}
return dlnaEventManager.CreateEventSubscription(notificationType, timeoutString, callback);
}
return dlnaEventManager.CancelEventSubscription(subscriptionId);
}
}
}