DlnaServerService.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. #pragma warning disable CS1591
  2. using System;
  3. using System.IO;
  4. using System.Text;
  5. using System.Threading.Tasks;
  6. using Emby.Dlna.Main;
  7. using MediaBrowser.Common.Extensions;
  8. using MediaBrowser.Controller.Configuration;
  9. using MediaBrowser.Controller.Dlna;
  10. using MediaBrowser.Controller.Net;
  11. using MediaBrowser.Model.Services;
  12. namespace Emby.Dlna.Api
  13. {
  14. [Route("/Dlna/{UuId}/description.xml", "GET", Summary = "Gets dlna server info")]
  15. [Route("/Dlna/{UuId}/description", "GET", Summary = "Gets dlna server info")]
  16. public class GetDescriptionXml
  17. {
  18. [ApiMember(Name = "UuId", Description = "Server UuId", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "GET")]
  19. public string UuId { get; set; }
  20. }
  21. [Route("/Dlna/{UuId}/contentdirectory/contentdirectory.xml", "GET", Summary = "Gets dlna content directory xml")]
  22. [Route("/Dlna/{UuId}/contentdirectory/contentdirectory", "GET", Summary = "Gets dlna content directory xml")]
  23. public class GetContentDirectory
  24. {
  25. [ApiMember(Name = "UuId", Description = "Server UuId", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "GET")]
  26. public string UuId { get; set; }
  27. }
  28. [Route("/Dlna/{UuId}/connectionmanager/connectionmanager.xml", "GET", Summary = "Gets dlna connection manager xml")]
  29. [Route("/Dlna/{UuId}/connectionmanager/connectionmanager", "GET", Summary = "Gets dlna connection manager xml")]
  30. public class GetConnnectionManager
  31. {
  32. [ApiMember(Name = "UuId", Description = "Server UuId", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "GET")]
  33. public string UuId { get; set; }
  34. }
  35. [Route("/Dlna/{UuId}/mediareceiverregistrar/mediareceiverregistrar.xml", "GET", Summary = "Gets dlna mediareceiverregistrar xml")]
  36. [Route("/Dlna/{UuId}/mediareceiverregistrar/mediareceiverregistrar", "GET", Summary = "Gets dlna mediareceiverregistrar xml")]
  37. public class GetMediaReceiverRegistrar
  38. {
  39. [ApiMember(Name = "UuId", Description = "Server UuId", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "GET")]
  40. public string UuId { get; set; }
  41. }
  42. [Route("/Dlna/{UuId}/contentdirectory/control", "POST", Summary = "Processes a control request")]
  43. public class ProcessContentDirectoryControlRequest : IRequiresRequestStream
  44. {
  45. [ApiMember(Name = "UuId", Description = "Server UuId", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "GET")]
  46. public string UuId { get; set; }
  47. public Stream RequestStream { get; set; }
  48. }
  49. [Route("/Dlna/{UuId}/connectionmanager/control", "POST", Summary = "Processes a control request")]
  50. public class ProcessConnectionManagerControlRequest : IRequiresRequestStream
  51. {
  52. [ApiMember(Name = "UuId", Description = "Server UuId", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "GET")]
  53. public string UuId { get; set; }
  54. public Stream RequestStream { get; set; }
  55. }
  56. [Route("/Dlna/{UuId}/mediareceiverregistrar/control", "POST", Summary = "Processes a control request")]
  57. public class ProcessMediaReceiverRegistrarControlRequest : IRequiresRequestStream
  58. {
  59. [ApiMember(Name = "UuId", Description = "Server UuId", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "GET")]
  60. public string UuId { get; set; }
  61. public Stream RequestStream { get; set; }
  62. }
  63. [Route("/Dlna/{UuId}/mediareceiverregistrar/events", "SUBSCRIBE", Summary = "Processes an event subscription request")]
  64. [Route("/Dlna/{UuId}/mediareceiverregistrar/events", "UNSUBSCRIBE", Summary = "Processes an event subscription request")]
  65. public class ProcessMediaReceiverRegistrarEventRequest
  66. {
  67. [ApiMember(Name = "UuId", Description = "Server UuId", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "SUBSCRIBE,UNSUBSCRIBE")]
  68. public string UuId { get; set; }
  69. }
  70. [Route("/Dlna/{UuId}/contentdirectory/events", "SUBSCRIBE", Summary = "Processes an event subscription request")]
  71. [Route("/Dlna/{UuId}/contentdirectory/events", "UNSUBSCRIBE", Summary = "Processes an event subscription request")]
  72. public class ProcessContentDirectoryEventRequest
  73. {
  74. [ApiMember(Name = "UuId", Description = "Server UuId", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "SUBSCRIBE,UNSUBSCRIBE")]
  75. public string UuId { get; set; }
  76. }
  77. [Route("/Dlna/{UuId}/connectionmanager/events", "SUBSCRIBE", Summary = "Processes an event subscription request")]
  78. [Route("/Dlna/{UuId}/connectionmanager/events", "UNSUBSCRIBE", Summary = "Processes an event subscription request")]
  79. public class ProcessConnectionManagerEventRequest
  80. {
  81. [ApiMember(Name = "UuId", Description = "Server UuId", IsRequired = false, DataType = "string", ParameterType = "path", Verb = "SUBSCRIBE,UNSUBSCRIBE")]
  82. public string UuId { get; set; }
  83. }
  84. [Route("/Dlna/{UuId}/icons/{Filename}", "GET", Summary = "Gets a server icon")]
  85. [Route("/Dlna/icons/{Filename}", "GET", Summary = "Gets a server icon")]
  86. public class GetIcon
  87. {
  88. [ApiMember(Name = "UuId", Description = "Server UuId", IsRequired = false, DataType = "string", ParameterType = "query", Verb = "GET")]
  89. public string UuId { get; set; }
  90. [ApiMember(Name = "Filename", Description = "The icon filename", IsRequired = true, DataType = "string", ParameterType = "path", Verb = "GET")]
  91. public string Filename { get; set; }
  92. }
  93. public class DlnaServerService : IService, IRequiresRequest
  94. {
  95. private const string XMLContentType = "text/xml; charset=UTF-8";
  96. private readonly IDlnaManager _dlnaManager;
  97. private readonly IHttpResultFactory _resultFactory;
  98. private readonly IServerConfigurationManager _configurationManager;
  99. public IRequest Request { get; set; }
  100. private IContentDirectory ContentDirectory => DlnaEntryPoint.Current.ContentDirectory;
  101. private IConnectionManager ConnectionManager => DlnaEntryPoint.Current.ConnectionManager;
  102. private IMediaReceiverRegistrar MediaReceiverRegistrar => DlnaEntryPoint.Current.MediaReceiverRegistrar;
  103. public DlnaServerService(
  104. IDlnaManager dlnaManager,
  105. IHttpResultFactory httpResultFactory,
  106. IServerConfigurationManager configurationManager)
  107. {
  108. _dlnaManager = dlnaManager;
  109. _resultFactory = httpResultFactory;
  110. _configurationManager = configurationManager;
  111. }
  112. private string GetHeader(string name)
  113. {
  114. return Request.Headers[name];
  115. }
  116. public object Get(GetDescriptionXml request)
  117. {
  118. var url = Request.AbsoluteUri;
  119. var serverAddress = url.Substring(0, url.IndexOf("/dlna/", StringComparison.OrdinalIgnoreCase));
  120. var xml = _dlnaManager.GetServerDescriptionXml(Request.Headers, request.UuId, serverAddress);
  121. var cacheLength = TimeSpan.FromDays(1);
  122. var cacheKey = Request.RawUrl.GetMD5();
  123. var bytes = Encoding.UTF8.GetBytes(xml);
  124. return _resultFactory.GetStaticResult(Request, cacheKey, null, cacheLength, XMLContentType, () => Task.FromResult<Stream>(new MemoryStream(bytes)));
  125. }
  126. public object Get(GetContentDirectory request)
  127. {
  128. var xml = ContentDirectory.GetServiceXml();
  129. return _resultFactory.GetResult(Request, xml, XMLContentType);
  130. }
  131. public object Get(GetMediaReceiverRegistrar request)
  132. {
  133. var xml = MediaReceiverRegistrar.GetServiceXml();
  134. return _resultFactory.GetResult(Request, xml, XMLContentType);
  135. }
  136. public object Get(GetConnnectionManager request)
  137. {
  138. var xml = ConnectionManager.GetServiceXml();
  139. return _resultFactory.GetResult(Request, xml, XMLContentType);
  140. }
  141. public async Task<object> Post(ProcessMediaReceiverRegistrarControlRequest request)
  142. {
  143. var response = await PostAsync(request.RequestStream, MediaReceiverRegistrar).ConfigureAwait(false);
  144. return _resultFactory.GetResult(Request, response.Xml, XMLContentType);
  145. }
  146. public async Task<object> Post(ProcessContentDirectoryControlRequest request)
  147. {
  148. var response = await PostAsync(request.RequestStream, ContentDirectory).ConfigureAwait(false);
  149. return _resultFactory.GetResult(Request, response.Xml, XMLContentType);
  150. }
  151. public async Task<object> Post(ProcessConnectionManagerControlRequest request)
  152. {
  153. var response = await PostAsync(request.RequestStream, ConnectionManager).ConfigureAwait(false);
  154. return _resultFactory.GetResult(Request, response.Xml, XMLContentType);
  155. }
  156. private Task<ControlResponse> PostAsync(Stream requestStream, IUpnpService service)
  157. {
  158. var id = GetPathValue(2).ToString();
  159. return service.ProcessControlRequestAsync(new ControlRequest
  160. {
  161. Headers = Request.Headers,
  162. InputXml = requestStream,
  163. TargetServerUuId = id,
  164. RequestedUrl = Request.AbsoluteUri
  165. });
  166. }
  167. // Copied from MediaBrowser.Api/BaseApiService.cs
  168. // TODO: Remove code duplication
  169. /// <summary>
  170. /// Gets the path segment at the specified index.
  171. /// </summary>
  172. /// <param name="index">The index of the path segment.</param>
  173. /// <returns>The path segment at the specified index.</returns>
  174. /// <exception cref="IndexOutOfRangeException" >Path doesn't contain enough segments.</exception>
  175. /// <exception cref="InvalidDataException" >Path doesn't start with the base url.</exception>
  176. protected internal ReadOnlySpan<char> GetPathValue(int index)
  177. {
  178. static void ThrowIndexOutOfRangeException()
  179. => throw new IndexOutOfRangeException("Path doesn't contain enough segments.");
  180. static void ThrowInvalidDataException()
  181. => throw new InvalidDataException("Path doesn't start with the base url.");
  182. ReadOnlySpan<char> path = Request.PathInfo;
  183. // Remove the protocol part from the url
  184. int pos = path.LastIndexOf("://");
  185. if (pos != -1)
  186. {
  187. path = path.Slice(pos + 3);
  188. }
  189. // Remove the query string
  190. pos = path.LastIndexOf('?');
  191. if (pos != -1)
  192. {
  193. path = path.Slice(0, pos);
  194. }
  195. // Remove the domain
  196. pos = path.IndexOf('/');
  197. if (pos != -1)
  198. {
  199. path = path.Slice(pos);
  200. }
  201. // Remove base url
  202. string baseUrl = _configurationManager.Configuration.BaseUrl;
  203. int baseUrlLen = baseUrl.Length;
  204. if (baseUrlLen != 0)
  205. {
  206. if (path.StartsWith(baseUrl, StringComparison.OrdinalIgnoreCase))
  207. {
  208. path = path.Slice(baseUrlLen);
  209. }
  210. else
  211. {
  212. // The path doesn't start with the base url,
  213. // how did we get here?
  214. ThrowInvalidDataException();
  215. }
  216. }
  217. // Remove leading /
  218. path = path.Slice(1);
  219. // Backwards compatibility
  220. const string Emby = "emby/";
  221. if (path.StartsWith(Emby, StringComparison.OrdinalIgnoreCase))
  222. {
  223. path = path.Slice(Emby.Length);
  224. }
  225. const string MediaBrowser = "mediabrowser/";
  226. if (path.StartsWith(MediaBrowser, StringComparison.OrdinalIgnoreCase))
  227. {
  228. path = path.Slice(MediaBrowser.Length);
  229. }
  230. // Skip segments until we are at the right index
  231. for (int i = 0; i < index; i++)
  232. {
  233. pos = path.IndexOf('/');
  234. if (pos == -1)
  235. {
  236. ThrowIndexOutOfRangeException();
  237. }
  238. path = path.Slice(pos + 1);
  239. }
  240. // Remove the rest
  241. pos = path.IndexOf('/');
  242. if (pos != -1)
  243. {
  244. path = path.Slice(0, pos);
  245. }
  246. return path;
  247. }
  248. public object Get(GetIcon request)
  249. {
  250. var contentType = "image/" + Path.GetExtension(request.Filename)
  251. .TrimStart('.')
  252. .ToLowerInvariant();
  253. var cacheLength = TimeSpan.FromDays(365);
  254. var cacheKey = Request.RawUrl.GetMD5();
  255. return _resultFactory.GetStaticResult(Request, cacheKey, null, cacheLength, contentType, () => Task.FromResult(_dlnaManager.GetIcon(request.Filename).Stream));
  256. }
  257. public object Subscribe(ProcessContentDirectoryEventRequest request)
  258. {
  259. return ProcessEventRequest(ContentDirectory);
  260. }
  261. public object Subscribe(ProcessConnectionManagerEventRequest request)
  262. {
  263. return ProcessEventRequest(ConnectionManager);
  264. }
  265. public object Subscribe(ProcessMediaReceiverRegistrarEventRequest request)
  266. {
  267. return ProcessEventRequest(MediaReceiverRegistrar);
  268. }
  269. public object Unsubscribe(ProcessContentDirectoryEventRequest request)
  270. {
  271. return ProcessEventRequest(ContentDirectory);
  272. }
  273. public object Unsubscribe(ProcessConnectionManagerEventRequest request)
  274. {
  275. return ProcessEventRequest(ConnectionManager);
  276. }
  277. public object Unsubscribe(ProcessMediaReceiverRegistrarEventRequest request)
  278. {
  279. return ProcessEventRequest(MediaReceiverRegistrar);
  280. }
  281. private object ProcessEventRequest(IEventManager eventManager)
  282. {
  283. var subscriptionId = GetHeader("SID");
  284. if (string.Equals(Request.Verb, "SUBSCRIBE", StringComparison.OrdinalIgnoreCase))
  285. {
  286. var notificationType = GetHeader("NT");
  287. var callback = GetHeader("CALLBACK");
  288. var timeoutString = GetHeader("TIMEOUT");
  289. if (string.IsNullOrEmpty(notificationType))
  290. {
  291. return GetSubscriptionResponse(eventManager.RenewEventSubscription(subscriptionId, notificationType, timeoutString, callback));
  292. }
  293. return GetSubscriptionResponse(eventManager.CreateEventSubscription(notificationType, timeoutString, callback));
  294. }
  295. return GetSubscriptionResponse(eventManager.CancelEventSubscription(subscriptionId));
  296. }
  297. private object GetSubscriptionResponse(EventSubscriptionResponse response)
  298. {
  299. return _resultFactory.GetResult(Request, response.Content, response.ContentType, response.Headers);
  300. }
  301. }
  302. }