2
0

DlnaEventManager.cs 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. #pragma warning disable CS1591
  2. using System;
  3. using System.Collections.Concurrent;
  4. using System.Collections.Generic;
  5. using System.Globalization;
  6. using System.Linq;
  7. using System.Net.Http;
  8. using System.Net.Mime;
  9. using System.Text;
  10. using System.Threading.Tasks;
  11. using MediaBrowser.Common.Extensions;
  12. using MediaBrowser.Common.Net;
  13. using Microsoft.Extensions.Logging;
  14. namespace Emby.Dlna.Eventing
  15. {
  16. public class DlnaEventManager : IDlnaEventManager
  17. {
  18. private readonly ConcurrentDictionary<string, EventSubscription> _subscriptions =
  19. new ConcurrentDictionary<string, EventSubscription>(StringComparer.OrdinalIgnoreCase);
  20. private readonly ILogger _logger;
  21. private readonly IHttpClientFactory _httpClientFactory;
  22. private readonly CultureInfo _usCulture = new CultureInfo("en-US");
  23. public DlnaEventManager(ILogger logger, IHttpClientFactory httpClientFactory)
  24. {
  25. _httpClientFactory = httpClientFactory;
  26. _logger = logger;
  27. }
  28. public EventSubscriptionResponse RenewEventSubscription(string subscriptionId, string notificationType, string requestedTimeoutString, string callbackUrl)
  29. {
  30. var subscription = GetSubscription(subscriptionId, false);
  31. if (subscription != null)
  32. {
  33. subscription.TimeoutSeconds = ParseTimeout(requestedTimeoutString) ?? 300;
  34. int timeoutSeconds = subscription.TimeoutSeconds;
  35. subscription.SubscriptionTime = DateTime.UtcNow;
  36. _logger.LogDebug(
  37. "Renewing event subscription for {0} with timeout of {1} to {2}",
  38. subscription.NotificationType,
  39. timeoutSeconds,
  40. subscription.CallbackUrl);
  41. return GetEventSubscriptionResponse(subscriptionId, requestedTimeoutString, timeoutSeconds);
  42. }
  43. return new EventSubscriptionResponse
  44. {
  45. Content = string.Empty,
  46. ContentType = "text/plain"
  47. };
  48. }
  49. public EventSubscriptionResponse CreateEventSubscription(string notificationType, string requestedTimeoutString, string callbackUrl)
  50. {
  51. var timeout = ParseTimeout(requestedTimeoutString) ?? 300;
  52. var id = "uuid:" + Guid.NewGuid().ToString("N", CultureInfo.InvariantCulture);
  53. _logger.LogDebug(
  54. "Creating event subscription for {0} with timeout of {1} to {2}",
  55. notificationType,
  56. timeout,
  57. callbackUrl);
  58. _subscriptions.TryAdd(id, new EventSubscription
  59. {
  60. Id = id,
  61. CallbackUrl = callbackUrl,
  62. SubscriptionTime = DateTime.UtcNow,
  63. TimeoutSeconds = timeout
  64. });
  65. return GetEventSubscriptionResponse(id, requestedTimeoutString, timeout);
  66. }
  67. private int? ParseTimeout(string header)
  68. {
  69. if (!string.IsNullOrEmpty(header))
  70. {
  71. // Starts with SECOND-
  72. header = header.Split('-').Last();
  73. if (int.TryParse(header, NumberStyles.Integer, _usCulture, out var val))
  74. {
  75. return val;
  76. }
  77. }
  78. return null;
  79. }
  80. public EventSubscriptionResponse CancelEventSubscription(string subscriptionId)
  81. {
  82. _logger.LogDebug("Cancelling event subscription {0}", subscriptionId);
  83. _subscriptions.TryRemove(subscriptionId, out _);
  84. return new EventSubscriptionResponse
  85. {
  86. Content = string.Empty,
  87. ContentType = "text/plain"
  88. };
  89. }
  90. private EventSubscriptionResponse GetEventSubscriptionResponse(string subscriptionId, string requestedTimeoutString, int timeoutSeconds)
  91. {
  92. var response = new EventSubscriptionResponse
  93. {
  94. Content = string.Empty,
  95. ContentType = "text/plain"
  96. };
  97. response.Headers["SID"] = subscriptionId;
  98. response.Headers["TIMEOUT"] = string.IsNullOrEmpty(requestedTimeoutString) ? ("SECOND-" + timeoutSeconds.ToString(_usCulture)) : requestedTimeoutString;
  99. return response;
  100. }
  101. public EventSubscription GetSubscription(string id)
  102. {
  103. return GetSubscription(id, false);
  104. }
  105. private EventSubscription GetSubscription(string id, bool throwOnMissing)
  106. {
  107. if (!_subscriptions.TryGetValue(id, out EventSubscription e) && throwOnMissing)
  108. {
  109. throw new ResourceNotFoundException("Event with Id " + id + " not found.");
  110. }
  111. return e;
  112. }
  113. public Task TriggerEvent(string notificationType, IDictionary<string, string> stateVariables)
  114. {
  115. var subs = _subscriptions.Values
  116. .Where(i => !i.IsExpired && string.Equals(notificationType, i.NotificationType, StringComparison.OrdinalIgnoreCase))
  117. .ToList();
  118. var tasks = subs.Select(i => TriggerEvent(i, stateVariables));
  119. return Task.WhenAll(tasks);
  120. }
  121. private async Task TriggerEvent(EventSubscription subscription, IDictionary<string, string> stateVariables)
  122. {
  123. var builder = new StringBuilder();
  124. builder.Append("<?xml version=\"1.0\"?>");
  125. builder.Append("<e:propertyset xmlns:e=\"urn:schemas-upnp-org:event-1-0\">");
  126. foreach (var key in stateVariables.Keys)
  127. {
  128. builder.Append("<e:property>")
  129. .Append('<')
  130. .Append(key)
  131. .Append('>')
  132. .Append(stateVariables[key])
  133. .Append("</")
  134. .Append(key)
  135. .Append('>')
  136. .Append("</e:property>");
  137. }
  138. builder.Append("</e:propertyset>");
  139. using var options = new HttpRequestMessage(new HttpMethod("NOTIFY"), subscription.CallbackUrl);
  140. options.Content = new StringContent(builder.ToString(), Encoding.UTF8, MediaTypeNames.Text.Xml);
  141. options.Headers.TryAddWithoutValidation("NT", subscription.NotificationType);
  142. options.Headers.TryAddWithoutValidation("NTS", "upnp:propchange");
  143. options.Headers.TryAddWithoutValidation("SID", subscription.Id);
  144. options.Headers.TryAddWithoutValidation("SEQ", subscription.TriggerCount.ToString(_usCulture));
  145. try
  146. {
  147. using var response = await _httpClientFactory.CreateClient(NamedClient.Default)
  148. .SendAsync(options, HttpCompletionOption.ResponseHeadersRead).ConfigureAwait(false);
  149. }
  150. catch (OperationCanceledException)
  151. {
  152. }
  153. catch
  154. {
  155. // Already logged at lower levels
  156. }
  157. finally
  158. {
  159. subscription.IncrementTriggerCount();
  160. }
  161. }
  162. }
  163. }