DlnaEventManager.cs 6.8 KB

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