Crunchy-Downloader/CRD/Utils/Http/HttpClientReq.cs
Elwador aca28a4e17 Add - Added option to **mux fonts into the MKV**
Add - Added **completion sound** when downloads finish, allowing a specified sound to be played
Add - Added **changelog** for easier tracking of changes and updates
Add - Added **Retry button** to reset all failed downloads, allowing "Auto Download" to restart them
Add - Added **dub/sub info** to the Upcoming tab
Chg - Changed **history table view** to include missing features from the poster view
Chg - Changed and **adjusted error messages**
Chg - Changed **chapters formatting** for consistent output across locales
Chg - Changed **Clear Queue** button to icon-only
Chg - Changed **update button** behavior
Chg - Changed some **error messages** for better debugging
Chg - Changed **history series access**, now allowing it to open while others are refreshing
Chg - Changed **device ID reuse** to fix continuous login emails
Chg - Changed **authentication log** to write the full message for better debugging
Chg - Updated **dependencies**
Fix - Temporary fix for **authentication issues**
Fix - Fixed **unable to download movies** [#237](https://github.com/Crunchy-DL/Crunchy-Downloader/issues/237)
Fix - Fixed **buggy queue behavior** during active downloads
Fix - Fixed **duplicate seasons in history** when adding multiple episodes from the calendar
Fix - Fixed crash if  **all** subtitles option is selected
Fix - Fixed "**Cannot set download directory to Drive**" https://github.com/Crunchy-DL/Crunchy-Downloader/issues/220
Fix - Fixed missing **subtitles and none subs** for some series
2025-04-04 20:12:29 +02:00

275 lines
No EOL
12 KiB
C#

using System;
using System.Net;
using System.Net.Http;
using System.Collections.Generic;
using System.Collections.Specialized;
using System.Linq;
using System.Net.Http.Headers;
using System.Net.Sockets;
using System.Text;
using System.Threading.Tasks;
using CRD.Downloader.Crunchyroll;
namespace CRD.Utils;
public class HttpClientReq{
#region Singelton
private static HttpClientReq? instance;
private static readonly object padlock = new object();
public static HttpClientReq Instance{
get{
if (instance == null){
lock (padlock){
if (instance == null){
instance = new HttpClientReq();
}
}
}
return instance;
}
}
#endregion
private HttpClient client;
private Dictionary<string, CookieCollection> cookieStore;
private HttpClientHandler handler;
public HttpClientReq(){
cookieStore = new Dictionary<string, CookieCollection>();
IWebProxy systemProxy = WebRequest.DefaultWebProxy;
HttpClientHandler handler = new HttpClientHandler();
if (CrunchyrollManager.Instance.CrunOptions.ProxyEnabled && !string.IsNullOrEmpty(CrunchyrollManager.Instance.CrunOptions.ProxyHost)){
handler = CreateHandler(true, CrunchyrollManager.Instance.CrunOptions.ProxySocks, CrunchyrollManager.Instance.CrunOptions.ProxyHost, CrunchyrollManager.Instance.CrunOptions.ProxyPort,
CrunchyrollManager.Instance.CrunOptions.ProxyUsername, CrunchyrollManager.Instance.CrunOptions.ProxyPassword);
string scheme = CrunchyrollManager.Instance.CrunOptions.ProxySocks ? "socks5" : "http";
Console.Error.WriteLine($"Proxy is set: {scheme}://{CrunchyrollManager.Instance.CrunOptions.ProxyHost}:{CrunchyrollManager.Instance.CrunOptions.ProxyPort}");
client = new HttpClient(handler);
} else if (systemProxy != null){
Uri testUri = new Uri("https://icanhazip.com");
Uri? proxyUri = systemProxy.GetProxy(testUri);
if (proxyUri != null && proxyUri != testUri){
if (proxyUri is{ Host: "127.0.0.1", Port: 7890 }){
Console.Error.WriteLine($"Proxy is set: {proxyUri}");
handler = CreateHandler(true);
} else{
Console.Error.WriteLine("No proxy will be used.");
handler = CreateHandler(false);
}
client = new HttpClient(handler);
} else{
Console.Error.WriteLine("No proxy is being used.");
client = new HttpClient(CreateHttpClientHandler());
}
} else{
Console.Error.WriteLine("No proxy is being used.");
client = new HttpClient(CreateHttpClientHandler());
}
client.Timeout = TimeSpan.FromSeconds(100);
// client.DefaultRequestHeaders.UserAgent.ParseAdd("Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:126.0) Gecko/20100101 Firefox/126.0");
client.DefaultRequestHeaders.UserAgent.ParseAdd("Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/129.0.0.0 Safari/537.36");
// client.DefaultRequestHeaders.UserAgent.ParseAdd("Crunchyroll/1.9.0 Nintendo Switch/18.1.0.0 UE4/4.27");
// client.DefaultRequestHeaders.UserAgent.ParseAdd("Crunchyroll/3.60.0 Android/9 okhttp/4.12.0");
client.DefaultRequestHeaders.Accept.ParseAdd("text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8");
client.DefaultRequestHeaders.AcceptEncoding.ParseAdd("gzip, deflate, br");
client.DefaultRequestHeaders.AcceptLanguage.ParseAdd("en-US,en;q=0.5");
client.DefaultRequestHeaders.Connection.ParseAdd("keep-alive");
}
private HttpMessageHandler CreateHttpClientHandler(){
return new SocketsHttpHandler(){
AutomaticDecompression = DecompressionMethods.GZip | DecompressionMethods.Deflate | DecompressionMethods.Brotli,
ConnectCallback = async (context, cancellationToken) => {
// Resolve IPv4 addresses only
var entry = await Dns.GetHostEntryAsync(context.DnsEndPoint.Host, AddressFamily.InterNetwork, cancellationToken).ConfigureAwait(false);
// Create an IPv4 socket
var socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
socket.NoDelay = true;
try{
await socket.ConnectAsync(entry.AddressList, context.DnsEndPoint.Port, cancellationToken).ConfigureAwait(false);
return new NetworkStream(socket, ownsSocket: true);
} catch{
socket.Dispose();
throw;
}
}
};
}
private HttpClientHandler CreateHandler(bool useProxy, bool useSocks = false, string? proxyHost = null, int proxyPort = 0, string? proxyUsername = "", string? proxyPassword = ""){
var handler = new HttpClientHandler{
CookieContainer = new CookieContainer(),
UseCookies = true,
AutomaticDecompression = DecompressionMethods.GZip | DecompressionMethods.Deflate | DecompressionMethods.Brotli,
UseProxy = useProxy
};
if (useProxy && proxyHost != null){
string scheme = useSocks ? "socks5" : "http";
handler.Proxy = new WebProxy($"{scheme}://{proxyHost}:{proxyPort}");
if (!string.IsNullOrEmpty(proxyUsername) && !string.IsNullOrEmpty(proxyPassword)){
handler.Proxy.Credentials = new NetworkCredential(proxyUsername, proxyPassword);
}
}
return handler;
}
public void SetETPCookie(string refreshToken){
// var cookie = new Cookie("etp_rt", refreshToken){
// Domain = "crunchyroll.com",
// Path = "/",
// };
//
// var cookie2 = new Cookie("c_locale", "en-US"){
// Domain = "crunchyroll.com",
// Path = "/",
// };
//
// handler.CookieContainer.Add(cookie);
// handler.CookieContainer.Add(cookie2);
AddCookie(".crunchyroll.com", new Cookie("etp_rt", refreshToken));
AddCookie(".crunchyroll.com", new Cookie("c_locale", "en-US"));
}
private void AddCookie(string domain, Cookie cookie){
if (!cookieStore.ContainsKey(domain)){
cookieStore[domain] = new CookieCollection();
}
var existingCookie = cookieStore[domain].FirstOrDefault(c => c.Name == cookie.Name);
if (existingCookie != null){
cookieStore[domain].Remove(existingCookie);
}
cookieStore[domain].Add(cookie);
}
public async Task<(bool IsOk, string ResponseContent)> SendHttpRequest(HttpRequestMessage request, bool suppressError = false){
string content = string.Empty;
try{
AttachCookies(request);
HttpResponseMessage response = await client.SendAsync(request);
if (ChallengeDetector.IsClearanceRequired(response)){
Console.Error.WriteLine($" Cloudflare Challenge detected");
}
content = await response.Content.ReadAsStringAsync();
response.EnsureSuccessStatusCode();
return (IsOk: true, ResponseContent: content);
} catch (Exception e){
// Console.Error.WriteLine($"Error: {e} \n Response: {(content.Length < 500 ? content : "error to long")}");
if (!suppressError){
Console.Error.WriteLine($"Error: {e} \n Response: {(content.Length < 500 ? content : "error to long")}");
}
return (IsOk: false, ResponseContent: content);
}
}
private void AttachCookies(HttpRequestMessage request){
var cookieHeader = new StringBuilder();
if (request.Headers.TryGetValues("Cookie", out var existingCookies)){
cookieHeader.Append(string.Join("; ", existingCookies));
}
foreach (var cookie in cookieStore.SelectMany(keyValuePair => keyValuePair.Value)){
string cookieString = $"{cookie.Name}={cookie.Value}";
if (!cookieHeader.ToString().Contains(cookieString)){
if (cookieHeader.Length > 0){
cookieHeader.Append("; ");
}
cookieHeader.Append(cookieString);
}
}
if (cookieHeader.Length > 0){
request.Headers.Remove("Cookie");
request.Headers.Add("Cookie", cookieHeader.ToString());
}
}
public static HttpRequestMessage CreateRequestMessage(string uri, HttpMethod requestMethod, bool authHeader, bool disableDrmHeader, NameValueCollection? query){
if (string.IsNullOrEmpty(uri)){
Console.Error.WriteLine($" Request URI is empty");
return new HttpRequestMessage(HttpMethod.Get, "about:blank");
}
UriBuilder uriBuilder = new UriBuilder(uri);
if (query != null){
uriBuilder.Query = query.ToString();
}
var request = new HttpRequestMessage(requestMethod, uriBuilder.ToString());
if (authHeader){
request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", CrunchyrollManager.Instance.Token?.access_token);
}
if (disableDrmHeader){
}
return request;
}
public static async Task DeAuthVideo(string currentMediaId, string token){
var deauthVideoToken = HttpClientReq.CreateRequestMessage($"https://cr-play-service.prd.crunchyrollsvc.com/v1/token/{currentMediaId}/{token}/inactive", HttpMethod.Patch, true, false, null);
var deauthVideoTokenResponse = await HttpClientReq.Instance.SendHttpRequest(deauthVideoToken);
}
public HttpClient GetHttpClient(){
return client;
}
}
public static class ApiUrls{
public static readonly string ApiBeta = "https://beta-api.crunchyroll.com";
public static readonly string ApiN = "https://www.crunchyroll.com";
public static readonly string Anilist = "https://graphql.anilist.co";
public static string Auth => (CrunchyrollManager.Instance.CrunOptions.UseCrBetaApi ? ApiBeta : ApiN) + "/auth/v1/token";
public static string Profile => (CrunchyrollManager.Instance.CrunOptions.UseCrBetaApi ? ApiBeta : ApiN) + "/accounts/v1/me/profile";
public static string CmsToken => (CrunchyrollManager.Instance.CrunOptions.UseCrBetaApi ? ApiBeta : ApiN) + "/index/v2";
public static string Search => (CrunchyrollManager.Instance.CrunOptions.UseCrBetaApi ? ApiBeta : ApiN) + "/content/v2/discover/search";
public static string Browse => (CrunchyrollManager.Instance.CrunOptions.UseCrBetaApi ? ApiBeta : ApiN) + "/content/v2/discover/browse";
public static string Cms => (CrunchyrollManager.Instance.CrunOptions.UseCrBetaApi ? ApiBeta : ApiN) + "/content/v2/cms";
public static string Content => (CrunchyrollManager.Instance.CrunOptions.UseCrBetaApi ? ApiBeta : ApiN) + "/content/v2";
public static string Subscription => (CrunchyrollManager.Instance.CrunOptions.UseCrBetaApi ? ApiBeta : ApiN) + "/subs/v3/subscriptions/";
public static readonly string BetaBrowse = ApiBeta + "/content/v1/browse";
public static readonly string BetaCms = ApiBeta + "/cms/v2";
public static readonly string DRM = ApiBeta + "/drm/v1/auth";
public static string authBasicMob = "Basic eHVuaWh2ZWRidDNtYmlzdWhldnQ6MWtJUzVkeVR2akUwX3JxYUEzWWVBaDBiVVhVbXhXMTE=";
public static readonly string MobileUserAgent = "Crunchyroll/3.78.3 Android/15 okhttp/4.12.0";
}