using System; using System.Collections.Generic; using System.Collections.Specialized; using System.Linq; using System.Text.RegularExpressions; using CefSharp; using TweetLib.Core.Utils; namespace TweetDuck.Browser.Handling { class ResourceRequestHandlerBase : ResourceRequestHandler { private static readonly Regex TweetDeckResourceUrl = new Regex(@"/dist/(.*?)\.(.*?)\.(css|js)$"); private static readonly SortedList<string, string> TweetDeckHashes = new SortedList<string, string>(4); public static void LoadResourceRewriteRules(string rules) { if (string.IsNullOrEmpty(rules)) { return; } TweetDeckHashes.Clear(); foreach (string rule in rules.Replace(" ", "").ToLower().Split(',')) { var (key, hash) = StringUtils.SplitInTwo(rule, '=') ?? throw new ArgumentException("A rule must have one '=' character: " + rule); if (hash.All(chr => char.IsDigit(chr) || (chr >= 'a' && chr <= 'f'))) { TweetDeckHashes.Add(key, hash); } else { throw new ArgumentException("Invalid hash characters: " + rule); } } } protected override CefReturnValue OnBeforeResourceLoad(IWebBrowser browserControl, IBrowser browser, IFrame frame, IRequest request, IRequestCallback callback) { if (request.ResourceType == ResourceType.CspReport) { callback.Dispose(); return CefReturnValue.Cancel; } NameValueCollection headers = request.Headers; headers.Remove("x-devtools-emulate-network-conditions-client-id"); request.Headers = headers; return base.OnBeforeResourceLoad(browserControl, browser, frame, request, callback); } protected override bool OnResourceResponse(IWebBrowser browserControl, IBrowser browser, IFrame frame, IRequest request, IResponse response) { if ((request.ResourceType == ResourceType.Script || request.ResourceType == ResourceType.Stylesheet) && TweetDeckHashes.Count > 0) { string url = request.Url; Match match = TweetDeckResourceUrl.Match(url); if (match.Success && TweetDeckHashes.TryGetValue($"{match.Groups[1]}.{match.Groups[3]}", out string hash)) { if (match.Groups[2].Value == hash) { Program.Reporter.LogVerbose("[RequestHandlerBase] Accepting " + url); } else { Program.Reporter.LogVerbose("[RequestHandlerBase] Replacing " + url + " hash with " + hash); request.Url = TweetDeckResourceUrl.Replace(url, $"/dist/$1.{hash}.$3"); return true; } } } return base.OnResourceResponse(browserControl, browser, frame, request, response); } } }