DlnaServerController.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  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. /// <response code="200">Dlna media receiver registrar xml returned.</response>
  76. /// <returns>Dlna media receiver registrar xml.</returns>
  77. [HttpGet("{serverId}/MediaReceiverRegistrar")]
  78. [HttpGet("{serverId}/MediaReceiverRegistrar/MediaReceiverRegistrar", Name = "GetMediaReceiverRegistrar_2")]
  79. [HttpGet("{serverId}/MediaReceiverRegistrar/MediaReceiverRegistrar.xml", Name = "GetMediaReceiverRegistrar_3")]
  80. [ProducesResponseType(StatusCodes.Status200OK)]
  81. [Produces(MediaTypeNames.Text.Xml)]
  82. [ProducesFile(MediaTypeNames.Text.Xml)]
  83. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  84. public ActionResult GetMediaReceiverRegistrar([FromRoute, Required] string serverId)
  85. {
  86. return Ok(_mediaReceiverRegistrar.GetServiceXml());
  87. }
  88. /// <summary>
  89. /// Gets Dlna media receiver registrar xml.
  90. /// </summary>
  91. /// <param name="serverId">Server UUID.</param>
  92. /// <response code="200">Dlna media receiver registrar xml returned.</response>
  93. /// <returns>Dlna media receiver registrar xml.</returns>
  94. [HttpGet("{serverId}/ConnectionManager")]
  95. [HttpGet("{serverId}/ConnectionManager/ConnectionManager", Name = "GetConnectionManager_2")]
  96. [HttpGet("{serverId}/ConnectionManager/ConnectionManager.xml", Name = "GetConnectionManager_3")]
  97. [ProducesResponseType(StatusCodes.Status200OK)]
  98. [Produces(MediaTypeNames.Text.Xml)]
  99. [ProducesFile(MediaTypeNames.Text.Xml)]
  100. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  101. public ActionResult GetConnectionManager([FromRoute, Required] string serverId)
  102. {
  103. return Ok(_connectionManager.GetServiceXml());
  104. }
  105. /// <summary>
  106. /// Process a content directory control request.
  107. /// </summary>
  108. /// <param name="serverId">Server UUID.</param>
  109. /// <response code="200">Request processed.</response>
  110. /// <returns>Control response.</returns>
  111. [HttpPost("{serverId}/ContentDirectory/Control")]
  112. [ProducesResponseType(StatusCodes.Status200OK)]
  113. [Produces(MediaTypeNames.Text.Xml)]
  114. [ProducesFile(MediaTypeNames.Text.Xml)]
  115. public async Task<ActionResult<ControlResponse>> ProcessContentDirectoryControlRequest([FromRoute, Required] string serverId)
  116. {
  117. return await ProcessControlRequestInternalAsync(serverId, Request.Body, _contentDirectory).ConfigureAwait(false);
  118. }
  119. /// <summary>
  120. /// Process a connection manager control request.
  121. /// </summary>
  122. /// <param name="serverId">Server UUID.</param>
  123. /// <response code="200">Request processed.</response>
  124. /// <returns>Control response.</returns>
  125. [HttpPost("{serverId}/ConnectionManager/Control")]
  126. [ProducesResponseType(StatusCodes.Status200OK)]
  127. [Produces(MediaTypeNames.Text.Xml)]
  128. [ProducesFile(MediaTypeNames.Text.Xml)]
  129. public async Task<ActionResult<ControlResponse>> ProcessConnectionManagerControlRequest([FromRoute, Required] string serverId)
  130. {
  131. return await ProcessControlRequestInternalAsync(serverId, Request.Body, _connectionManager).ConfigureAwait(false);
  132. }
  133. /// <summary>
  134. /// Process a media receiver registrar control request.
  135. /// </summary>
  136. /// <param name="serverId">Server UUID.</param>
  137. /// <response code="200">Request processed.</response>
  138. /// <returns>Control response.</returns>
  139. [HttpPost("{serverId}/MediaReceiverRegistrar/Control")]
  140. [ProducesResponseType(StatusCodes.Status200OK)]
  141. [Produces(MediaTypeNames.Text.Xml)]
  142. [ProducesFile(MediaTypeNames.Text.Xml)]
  143. public async Task<ActionResult<ControlResponse>> ProcessMediaReceiverRegistrarControlRequest([FromRoute, Required] string serverId)
  144. {
  145. return await ProcessControlRequestInternalAsync(serverId, Request.Body, _mediaReceiverRegistrar).ConfigureAwait(false);
  146. }
  147. /// <summary>
  148. /// Processes an event subscription request.
  149. /// </summary>
  150. /// <param name="serverId">Server UUID.</param>
  151. /// <response code="200">Request processed.</response>
  152. /// <returns>Event subscription response.</returns>
  153. [HttpSubscribe("{serverId}/MediaReceiverRegistrar/Events")]
  154. [HttpUnsubscribe("{serverId}/MediaReceiverRegistrar/Events")]
  155. [ApiExplorerSettings(IgnoreApi = true)] // Ignore in openapi docs
  156. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  157. [ProducesResponseType(StatusCodes.Status200OK)]
  158. [Produces(MediaTypeNames.Text.Xml)]
  159. [ProducesFile(MediaTypeNames.Text.Xml)]
  160. public ActionResult<EventSubscriptionResponse> ProcessMediaReceiverRegistrarEventRequest(string serverId)
  161. {
  162. return ProcessEventRequest(_mediaReceiverRegistrar);
  163. }
  164. /// <summary>
  165. /// Processes an event subscription request.
  166. /// </summary>
  167. /// <param name="serverId">Server UUID.</param>
  168. /// <response code="200">Request processed.</response>
  169. /// <returns>Event subscription response.</returns>
  170. [HttpSubscribe("{serverId}/ContentDirectory/Events")]
  171. [HttpUnsubscribe("{serverId}/ContentDirectory/Events")]
  172. [ApiExplorerSettings(IgnoreApi = true)] // Ignore in openapi docs
  173. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  174. [ProducesResponseType(StatusCodes.Status200OK)]
  175. [Produces(MediaTypeNames.Text.Xml)]
  176. [ProducesFile(MediaTypeNames.Text.Xml)]
  177. public ActionResult<EventSubscriptionResponse> ProcessContentDirectoryEventRequest(string serverId)
  178. {
  179. return ProcessEventRequest(_contentDirectory);
  180. }
  181. /// <summary>
  182. /// Processes an event subscription request.
  183. /// </summary>
  184. /// <param name="serverId">Server UUID.</param>
  185. /// <response code="200">Request processed.</response>
  186. /// <returns>Event subscription response.</returns>
  187. [HttpSubscribe("{serverId}/ConnectionManager/Events")]
  188. [HttpUnsubscribe("{serverId}/ConnectionManager/Events")]
  189. [ApiExplorerSettings(IgnoreApi = true)] // Ignore in openapi docs
  190. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  191. [ProducesResponseType(StatusCodes.Status200OK)]
  192. [Produces(MediaTypeNames.Text.Xml)]
  193. [ProducesFile(MediaTypeNames.Text.Xml)]
  194. public ActionResult<EventSubscriptionResponse> ProcessConnectionManagerEventRequest(string serverId)
  195. {
  196. return ProcessEventRequest(_connectionManager);
  197. }
  198. /// <summary>
  199. /// Gets a server icon.
  200. /// </summary>
  201. /// <param name="serverId">Server UUID.</param>
  202. /// <param name="fileName">The icon filename.</param>
  203. /// <returns>Icon stream.</returns>
  204. [HttpGet("{serverId}/icons/{fileName}")]
  205. [SuppressMessage("Microsoft.Performance", "CA1801:ReviewUnusedParameters", MessageId = "serverId", Justification = "Required for DLNA")]
  206. [ProducesResponseType(StatusCodes.Status200OK)]
  207. [ProducesImageFile]
  208. public ActionResult GetIconId([FromRoute, Required] string serverId, [FromRoute, Required] string fileName)
  209. {
  210. return GetIconInternal(fileName);
  211. }
  212. /// <summary>
  213. /// Gets a server icon.
  214. /// </summary>
  215. /// <param name="fileName">The icon filename.</param>
  216. /// <returns>Icon stream.</returns>
  217. [HttpGet("icons/{fileName}")]
  218. [ProducesImageFile]
  219. public ActionResult GetIcon([FromRoute, Required] string fileName)
  220. {
  221. return GetIconInternal(fileName);
  222. }
  223. private ActionResult GetIconInternal(string fileName)
  224. {
  225. var icon = _dlnaManager.GetIcon(fileName);
  226. if (icon == null)
  227. {
  228. return NotFound();
  229. }
  230. var contentType = "image/" + Path.GetExtension(fileName)
  231. .TrimStart('.')
  232. .ToLowerInvariant();
  233. return File(icon.Stream, contentType);
  234. }
  235. private string GetAbsoluteUri()
  236. {
  237. return $"{Request.Scheme}://{Request.Host}{Request.Path}";
  238. }
  239. private Task<ControlResponse> ProcessControlRequestInternalAsync(string id, Stream requestStream, IUpnpService service)
  240. {
  241. return service.ProcessControlRequestAsync(new ControlRequest(Request.Headers)
  242. {
  243. InputXml = requestStream,
  244. TargetServerUuId = id,
  245. RequestedUrl = GetAbsoluteUri()
  246. });
  247. }
  248. private EventSubscriptionResponse ProcessEventRequest(IDlnaEventManager dlnaEventManager)
  249. {
  250. var subscriptionId = Request.Headers["SID"];
  251. if (string.Equals(Request.Method, "subscribe", StringComparison.OrdinalIgnoreCase))
  252. {
  253. var notificationType = Request.Headers["NT"];
  254. var callback = Request.Headers["CALLBACK"];
  255. var timeoutString = Request.Headers["TIMEOUT"];
  256. if (string.IsNullOrEmpty(notificationType))
  257. {
  258. return dlnaEventManager.RenewEventSubscription(
  259. subscriptionId,
  260. notificationType,
  261. timeoutString,
  262. callback);
  263. }
  264. return dlnaEventManager.CreateEventSubscription(notificationType, timeoutString, callback);
  265. }
  266. return dlnaEventManager.CancelEventSubscription(subscriptionId);
  267. }
  268. }
  269. }