using System.Security.Cryptography; using System.Text; using System.Xml.Linq; using Application.Abstractions.YouTube; using Microsoft.Extensions.Logging; namespace Infrastructure.YouTube; /// /// YouTube PubSubHubbub(WebSub) 구독 관리 /// - 구독 신청/해지: POST https://pubsubhubbub.appspot.com/subscribe /// - 콜백 검증: hub.challenge GET 요청 응답 /// - Atom Feed 파싱: 새 영상/생방송 알림 수신 /// internal sealed class YouTubePubSubService( IHttpClientFactory httpClientFactory, ILogger logger ) : IYouTubePubSubService { private const string HubUrl = "https://pubsubhubbub.appspot.com/subscribe"; private const string TopicBase = "https://www.youtube.com/feeds/videos.xml?channel_id="; // HMAC 서명 검증용 시크릿 — 실제 운영에서는 Config/환경변수에서 주입 internal string HmacSecret { get; set; } = "dpot-youtube-pubsub-secret"; // 콜백 URL — 서버 시작 시 설정 internal string CallbackUrl { get; set; } = string.Empty; public async Task SubscribeAsync(string channelId, CancellationToken ct) { return await SendHubRequestAsync("subscribe", channelId, ct); } public async Task UnsubscribeAsync(string channelId, CancellationToken ct) { return await SendHubRequestAsync("unsubscribe", channelId, ct); } public YouTubePubSubNotification? ParseNotification(string atomXml) { try { var doc = XDocument.Parse(atomXml); XNamespace atom = "http://www.w3.org/2005/Atom"; XNamespace yt = "http://www.youtube.com/xml/schemas/2015"; var entry = doc.Descendants(atom + "entry").FirstOrDefault(); if (entry is null) { return null; } var videoId = entry.Element(yt + "videoId")?.Value; var channelId = entry.Element(yt + "channelId")?.Value; var title = entry.Element(atom + "title")?.Value ?? ""; var published = entry.Element(atom + "published")?.Value; var updated = entry.Element(atom + "updated")?.Value; if (string.IsNullOrEmpty(videoId) || string.IsNullOrEmpty(channelId)) { logger.LogWarning("[PubSub] Atom feed missing videoId or channelId"); return null; } return new YouTubePubSubNotification( videoId, channelId, title, DateTime.TryParse(published, out var pub) ? pub : DateTime.UtcNow, DateTime.TryParse(updated, out var upd) ? upd : DateTime.UtcNow ); } catch (Exception ex) { logger.LogError(ex, "[PubSub] Failed to parse Atom feed"); return null; } } public bool VerifySignature(string payload, string signature) { if (string.IsNullOrEmpty(signature)) { return false; } // X-Hub-Signature 형식: "sha1=hex_digest" 또는 "sha256=hex_digest" var parts = signature.Split('=', 2); if (parts.Length != 2) { return false; } var algorithm = parts[0]; var expectedHex = parts[1]; byte[] hash; var keyBytes = Encoding.UTF8.GetBytes(HmacSecret); var payloadBytes = Encoding.UTF8.GetBytes(payload); if (algorithm is "sha1") { hash = HMACSHA1.HashData(keyBytes, payloadBytes); } else if (algorithm is "sha256") { hash = HMACSHA256.HashData(keyBytes, payloadBytes); } else { logger.LogWarning("[PubSub] Unsupported HMAC algorithm: {Algorithm}", algorithm); return false; } var computedHex = Convert.ToHexString(hash).ToLowerInvariant(); return string.Equals(computedHex, expectedHex, StringComparison.OrdinalIgnoreCase); } // ── Private ────────────────────────────────────────────────────── private async Task SendHubRequestAsync(string mode, string channelId, CancellationToken ct) { try { var client = httpClientFactory.CreateClient("PubSubHub"); var topicUrl = $"{TopicBase}{channelId}"; var content = new FormUrlEncodedContent(new Dictionary { ["hub.mode"] = mode, ["hub.topic"] = topicUrl, ["hub.callback"] = CallbackUrl, ["hub.secret"] = HmacSecret, ["hub.verify"] = "async" }); var response = await client.PostAsync(HubUrl, content, ct); if (response.IsSuccessStatusCode || (int)response.StatusCode == 202) { logger.LogInformation("[PubSub] {Mode} request accepted for channelId={ChannelId}", mode, channelId); return true; } var body = await response.Content.ReadAsStringAsync(ct); logger.LogWarning("[PubSub] {Mode} failed: {StatusCode} {Body}", mode, response.StatusCode, body); return false; } catch (Exception ex) { logger.LogError(ex, "[PubSub] {Mode} error for channelId={ChannelId}", mode, channelId); return false; } } }