DlnaServerService.cs 15 KB

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