DlnaServerController.cs 14 KB

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