DlnaServerController.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. using System;
  2. using System.ComponentModel.DataAnnotations;
  3. using System.Diagnostics.CodeAnalysis;
  4. using System.IO;
  5. using System.Net.Mime;
  6. using System.Threading.Tasks;
  7. using Emby.Dlna;
  8. using Emby.Dlna.Main;
  9. using Jellyfin.Api.Attributes;
  10. using MediaBrowser.Controller.Dlna;
  11. using Microsoft.AspNetCore.Http;
  12. using Microsoft.AspNetCore.Mvc;
  13. namespace Jellyfin.Api.Controllers
  14. {
  15. /// <summary>
  16. /// Dlna Server Controller.
  17. /// </summary>
  18. [Route("Dlna")]
  19. public class DlnaServerController : BaseJellyfinApiController
  20. {
  21. private readonly IDlnaManager _dlnaManager;
  22. private readonly IContentDirectory _contentDirectory;
  23. private readonly IConnectionManager _connectionManager;
  24. private readonly IMediaReceiverRegistrar _mediaReceiverRegistrar;
  25. /// <summary>
  26. /// Initializes a new instance of the <see cref="DlnaServerController"/> class.
  27. /// </summary>
  28. /// <param name="dlnaManager">Instance of the <see cref="IDlnaManager"/> interface.</param>
  29. public DlnaServerController(IDlnaManager dlnaManager)
  30. {
  31. _dlnaManager = dlnaManager;
  32. _contentDirectory = DlnaEntryPoint.Current.ContentDirectory;
  33. _connectionManager = DlnaEntryPoint.Current.ConnectionManager;
  34. _mediaReceiverRegistrar = DlnaEntryPoint.Current.MediaReceiverRegistrar;
  35. }
  36. /// <summary>
  37. /// Get Description Xml.
  38. /// </summary>
  39. /// <param name="serverId">Server UUID.</param>
  40. /// <response code="200">Description xml returned.</response>
  41. /// <returns>An <see cref="OkResult"/> containing the description xml.</returns>
  42. [HttpGet("{serverId}/description")]
  43. [HttpGet("{serverId}/description.xml", Name = "GetDescriptionXml_2")]
  44. [ProducesResponseType(StatusCodes.Status200OK)]
  45. [Produces(MediaTypeNames.Text.Xml)]
  46. [ProducesFile(MediaTypeNames.Text.Xml)]
  47. public ActionResult GetDescriptionXml([FromRoute, Required] string serverId)
  48. {
  49. var url = GetAbsoluteUri();
  50. var serverAddress = url.Substring(0, url.IndexOf("/dlna/", StringComparison.OrdinalIgnoreCase));
  51. var xml = _dlnaManager.GetServerDescriptionXml(Request.Headers, serverId, serverAddress);
  52. return Ok(xml);
  53. }
  54. /// <summary>
  55. /// Gets Dlna content directory xml.
  56. /// </summary>
  57. /// <param name="serverId">Server UUID.</param>
  58. /// <response code="200">Dlna content directory returned.</response>
  59. /// <returns>An <see cref="OkResult"/> containing the dlna content directory xml.</returns>
  60. [HttpGet("{serverId}/ContentDirectory")]
  61. [HttpGet("{serverId}/ContentDirectory/ContentDirectory", Name = "GetContentDirectory_2")]
  62. [HttpGet("{serverId}/ContentDirectory/ContentDirectory.xml", Name = "GetContentDirectory_3")]
  63. [ProducesResponseType(StatusCodes.Status200OK)]
  64. [Produces(MediaTypeNames.Text.Xml)]
  65. [ProducesFile(MediaTypeNames.Text.Xml)]
  66. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  67. public ActionResult GetContentDirectory([FromRoute, Required] string serverId)
  68. {
  69. return Ok(_contentDirectory.GetServiceXml());
  70. }
  71. /// <summary>
  72. /// Gets Dlna media receiver registrar xml.
  73. /// </summary>
  74. /// <param name="serverId">Server UUID.</param>
  75. /// <returns>Dlna media receiver registrar xml.</returns>
  76. [HttpGet("{serverId}/MediaReceiverRegistrar")]
  77. [HttpGet("{serverId}/MediaReceiverRegistrar/MediaReceiverRegistrar", Name = "GetMediaReceiverRegistrar_2")]
  78. [HttpGet("{serverId}/MediaReceiverRegistrar/MediaReceiverRegistrar.xml", Name = "GetMediaReceiverRegistrar_3")]
  79. [ProducesResponseType(StatusCodes.Status200OK)]
  80. [Produces(MediaTypeNames.Text.Xml)]
  81. [ProducesFile(MediaTypeNames.Text.Xml)]
  82. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  83. public ActionResult GetMediaReceiverRegistrar([FromRoute, Required] string serverId)
  84. {
  85. return Ok(_mediaReceiverRegistrar.GetServiceXml());
  86. }
  87. /// <summary>
  88. /// Gets Dlna media receiver registrar xml.
  89. /// </summary>
  90. /// <param name="serverId">Server UUID.</param>
  91. /// <returns>Dlna media receiver registrar xml.</returns>
  92. [HttpGet("{serverId}/ConnectionManager")]
  93. [HttpGet("{serverId}/ConnectionManager/ConnectionManager", Name = "GetConnectionManager_2")]
  94. [HttpGet("{serverId}/ConnectionManager/ConnectionManager.xml", Name = "GetConnectionManager_3")]
  95. [ProducesResponseType(StatusCodes.Status200OK)]
  96. [Produces(MediaTypeNames.Text.Xml)]
  97. [ProducesFile(MediaTypeNames.Text.Xml)]
  98. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  99. public ActionResult GetConnectionManager([FromRoute, Required] string serverId)
  100. {
  101. return Ok(_connectionManager.GetServiceXml());
  102. }
  103. /// <summary>
  104. /// Process a content directory control request.
  105. /// </summary>
  106. /// <param name="serverId">Server UUID.</param>
  107. /// <returns>Control response.</returns>
  108. [HttpPost("{serverId}/ContentDirectory/Control")]
  109. public async Task<ActionResult<ControlResponse>> ProcessContentDirectoryControlRequest([FromRoute, Required] string serverId)
  110. {
  111. return await ProcessControlRequestInternalAsync(serverId, Request.Body, _contentDirectory).ConfigureAwait(false);
  112. }
  113. /// <summary>
  114. /// Process a connection manager control request.
  115. /// </summary>
  116. /// <param name="serverId">Server UUID.</param>
  117. /// <returns>Control response.</returns>
  118. [HttpPost("{serverId}/ConnectionManager/Control")]
  119. public async Task<ActionResult<ControlResponse>> ProcessConnectionManagerControlRequest([FromRoute, Required] string serverId)
  120. {
  121. return await ProcessControlRequestInternalAsync(serverId, Request.Body, _connectionManager).ConfigureAwait(false);
  122. }
  123. /// <summary>
  124. /// Process a media receiver registrar control request.
  125. /// </summary>
  126. /// <param name="serverId">Server UUID.</param>
  127. /// <returns>Control response.</returns>
  128. [HttpPost("{serverId}/MediaReceiverRegistrar/Control")]
  129. public async Task<ActionResult<ControlResponse>> ProcessMediaReceiverRegistrarControlRequest([FromRoute, Required] string serverId)
  130. {
  131. return await ProcessControlRequestInternalAsync(serverId, Request.Body, _mediaReceiverRegistrar).ConfigureAwait(false);
  132. }
  133. /// <summary>
  134. /// Processes an event subscription request.
  135. /// </summary>
  136. /// <param name="serverId">Server UUID.</param>
  137. /// <returns>Event subscription response.</returns>
  138. [HttpSubscribe("{serverId}/MediaReceiverRegistrar/Events")]
  139. [HttpUnsubscribe("{serverId}/MediaReceiverRegistrar/Events")]
  140. [ApiExplorerSettings(IgnoreApi = true)] // Ignore in openapi docs
  141. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  142. public ActionResult<EventSubscriptionResponse> ProcessMediaReceiverRegistrarEventRequest(string serverId)
  143. {
  144. return ProcessEventRequest(_mediaReceiverRegistrar);
  145. }
  146. /// <summary>
  147. /// Processes an event subscription request.
  148. /// </summary>
  149. /// <param name="serverId">Server UUID.</param>
  150. /// <returns>Event subscription response.</returns>
  151. [HttpSubscribe("{serverId}/ContentDirectory/Events")]
  152. [HttpUnsubscribe("{serverId}/ContentDirectory/Events")]
  153. [ApiExplorerSettings(IgnoreApi = true)] // Ignore in openapi docs
  154. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  155. public ActionResult<EventSubscriptionResponse> ProcessContentDirectoryEventRequest(string serverId)
  156. {
  157. return ProcessEventRequest(_contentDirectory);
  158. }
  159. /// <summary>
  160. /// Processes an event subscription request.
  161. /// </summary>
  162. /// <param name="serverId">Server UUID.</param>
  163. /// <returns>Event subscription response.</returns>
  164. [HttpSubscribe("{serverId}/ConnectionManager/Events")]
  165. [HttpUnsubscribe("{serverId}/ConnectionManager/Events")]
  166. [ApiExplorerSettings(IgnoreApi = true)] // Ignore in openapi docs
  167. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  168. public ActionResult<EventSubscriptionResponse> ProcessConnectionManagerEventRequest(string serverId)
  169. {
  170. return ProcessEventRequest(_connectionManager);
  171. }
  172. /// <summary>
  173. /// Gets a server icon.
  174. /// </summary>
  175. /// <param name="serverId">Server UUID.</param>
  176. /// <param name="fileName">The icon filename.</param>
  177. /// <returns>Icon stream.</returns>
  178. [HttpGet("{serverId}/icons/{fileName}")]
  179. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  180. [ProducesResponseType(StatusCodes.Status200OK)]
  181. [ProducesImageFile]
  182. public ActionResult GetIconId([FromRoute, Required] string serverId, [FromRoute, Required] string fileName)
  183. {
  184. return GetIconInternal(fileName);
  185. }
  186. /// <summary>
  187. /// Gets a server icon.
  188. /// </summary>
  189. /// <param name="fileName">The icon filename.</param>
  190. /// <returns>Icon stream.</returns>
  191. [HttpGet("icons/{fileName}")]
  192. [ProducesImageFile]
  193. public ActionResult GetIcon([FromRoute, Required] string fileName)
  194. {
  195. return GetIconInternal(fileName);
  196. }
  197. private ActionResult GetIconInternal(string fileName)
  198. {
  199. var icon = _dlnaManager.GetIcon(fileName);
  200. if (icon == null)
  201. {
  202. return NotFound();
  203. }
  204. var contentType = "image/" + Path.GetExtension(fileName)
  205. .TrimStart('.')
  206. .ToLowerInvariant();
  207. return File(icon.Stream, contentType);
  208. }
  209. private string GetAbsoluteUri()
  210. {
  211. return $"{Request.Scheme}://{Request.Host}{Request.Path}";
  212. }
  213. private Task<ControlResponse> ProcessControlRequestInternalAsync(string id, Stream requestStream, IUpnpService service)
  214. {
  215. return service.ProcessControlRequestAsync(new ControlRequest(Request.Headers)
  216. {
  217. InputXml = requestStream,
  218. TargetServerUuId = id,
  219. RequestedUrl = GetAbsoluteUri()
  220. });
  221. }
  222. private EventSubscriptionResponse ProcessEventRequest(IDlnaEventManager dlnaEventManager)
  223. {
  224. var subscriptionId = Request.Headers["SID"];
  225. if (string.Equals(Request.Method, "subscribe", StringComparison.OrdinalIgnoreCase))
  226. {
  227. var notificationType = Request.Headers["NT"];
  228. var callback = Request.Headers["CALLBACK"];
  229. var timeoutString = Request.Headers["TIMEOUT"];
  230. if (string.IsNullOrEmpty(notificationType))
  231. {
  232. return dlnaEventManager.RenewEventSubscription(
  233. subscriptionId,
  234. notificationType,
  235. timeoutString,
  236. callback);
  237. }
  238. return dlnaEventManager.CreateEventSubscription(notificationType, timeoutString, callback);
  239. }
  240. return dlnaEventManager.CancelEventSubscription(subscriptionId);
  241. }
  242. }
  243. }