[CHANGE] Cookie login fixed
This commit is contained in:
@@ -71,14 +71,12 @@ namespace Manager.App.Components.Dialogs
|
|||||||
var name = parts[0].Trim();
|
var name = parts[0].Trim();
|
||||||
var value = parts[1].Trim();
|
var value = parts[1].Trim();
|
||||||
|
|
||||||
// Escape invalid characters
|
var cookie = new Cookie(name, value);
|
||||||
var safeName = Uri.EscapeDataString(name);
|
|
||||||
var safeValue = Uri.EscapeDataString(value);
|
|
||||||
|
|
||||||
var cookie = new Cookie(safeName, safeValue);
|
|
||||||
|
|
||||||
if (!string.IsNullOrEmpty(domain))
|
if (!string.IsNullOrEmpty(domain))
|
||||||
cookie.Domain = domain;
|
cookie.Domain = domain;
|
||||||
|
cookie.Expires = DateTime.Now.AddDays(1);
|
||||||
|
cookie.Path = "/";
|
||||||
|
|
||||||
collection.Add(cookie);
|
collection.Add(cookie);
|
||||||
}
|
}
|
||||||
@@ -105,7 +103,7 @@ namespace Manager.App.Components.Dialogs
|
|||||||
private async Task ValidateAccount()
|
private async Task ValidateAccount()
|
||||||
{
|
{
|
||||||
_isLoading = true;
|
_isLoading = true;
|
||||||
await Client.GetStateAsync();
|
await Client.BuildClientAsync();
|
||||||
_isLoading = false;
|
_isLoading = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -8,6 +8,7 @@ var builder = WebApplication.CreateBuilder(args);
|
|||||||
builder.Services.AddRazorComponents()
|
builder.Services.AddRazorComponents()
|
||||||
.AddInteractiveServerComponents();
|
.AddInteractiveServerComponents();
|
||||||
|
|
||||||
|
AppContext.SetSwitch("System.Net.Http.EnableActivityPropagation", false);
|
||||||
|
|
||||||
/* Manager */
|
/* Manager */
|
||||||
builder.SetupLogging();
|
builder.SetupLogging();
|
||||||
|
@@ -44,7 +44,7 @@ public class ClientManager : BackgroundService
|
|||||||
var ytClient = new YouTubeClient();
|
var ytClient = new YouTubeClient();
|
||||||
//ytClient.CookieContainer = container;
|
//ytClient.CookieContainer = container;
|
||||||
ytClient.UserAgent = accountEntity.UserAgent;
|
ytClient.UserAgent = accountEntity.UserAgent;
|
||||||
await ytClient.GetStateAsync();
|
await ytClient.BuildClientAsync();
|
||||||
|
|
||||||
return ytClient;
|
return ytClient;
|
||||||
}
|
}
|
||||||
|
8
Manager.YouTube/Models/AccountMenuInfo.cs
Normal file
8
Manager.YouTube/Models/AccountMenuInfo.cs
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
namespace Manager.YouTube.Models;
|
||||||
|
|
||||||
|
public class AccountMenuInfo
|
||||||
|
{
|
||||||
|
public string? AccountId { get; set; }
|
||||||
|
public string? AccountHandle { get; set; }
|
||||||
|
public string? ImageUrl { get; set; }
|
||||||
|
}
|
@@ -1,8 +1,10 @@
|
|||||||
|
using System.Net;
|
||||||
using System.Net.Mime;
|
using System.Net.Mime;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using System.Text.Json;
|
using System.Text.Json;
|
||||||
using System.Text.Json.Nodes;
|
using System.Text.Json.Nodes;
|
||||||
using DotBased.Monads;
|
using DotBased.Monads;
|
||||||
|
using Manager.YouTube.Models;
|
||||||
using Manager.YouTube.Models.Innertube;
|
using Manager.YouTube.Models.Innertube;
|
||||||
using Manager.YouTube.Parsers;
|
using Manager.YouTube.Parsers;
|
||||||
using Manager.YouTube.Util;
|
using Manager.YouTube.Util;
|
||||||
@@ -20,12 +22,8 @@ public static class NetworkService
|
|||||||
Method = HttpMethod.Get,
|
Method = HttpMethod.Get,
|
||||||
RequestUri = new Uri(Origin)
|
RequestUri = new Uri(Origin)
|
||||||
};
|
};
|
||||||
httpRequest.Headers.IfModifiedSince = new DateTimeOffset(DateTime.UtcNow);
|
httpRequest.Headers.Clear();
|
||||||
httpRequest.Headers.UserAgent.ParseAdd(client.UserAgent);
|
httpRequest.Headers.UserAgent.ParseAdd(client.UserAgent);
|
||||||
httpRequest.Headers.Accept.ParseAdd("text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8");
|
|
||||||
httpRequest.Headers.Connection.Add("keep-alive");
|
|
||||||
httpRequest.Headers.Add("DNT", "1");
|
|
||||||
httpRequest.Headers.Add("Upgrade-Insecure-Requests", "1");
|
|
||||||
|
|
||||||
var http = client.GetHttpClient();
|
var http = client.GetHttpClient();
|
||||||
if (http == null)
|
if (http == null)
|
||||||
@@ -59,7 +57,48 @@ public static class NetworkService
|
|||||||
return clientState == null ? ResultError.Fail("Unable to parse client state!") : clientState;
|
return clientState == null ? ResultError.Fail("Unable to parse client state!") : clientState;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async Task<Result> GetCurrentAccountInfoAsync(YouTubeClient client)
|
public static async Task<Result<string[]>> GetDatasyncIds(YouTubeClient client)
|
||||||
|
{
|
||||||
|
if (client.ClientState is not { LoggedIn: true } || client.CookieContainer.Count == 0)
|
||||||
|
{
|
||||||
|
return ResultError.Fail("Client is not logged in, requires logged in client for this endpoint (/getDatasyncIdsEndpoint).");
|
||||||
|
}
|
||||||
|
|
||||||
|
var httpClient = client.GetHttpClient();
|
||||||
|
if (httpClient == null)
|
||||||
|
{
|
||||||
|
return ResultError.Fail("Unable to get http client!");
|
||||||
|
}
|
||||||
|
|
||||||
|
var httpRequest = new HttpRequestMessage
|
||||||
|
{
|
||||||
|
Method = HttpMethod.Get,
|
||||||
|
RequestUri = new Uri($"{Origin}/getDatasyncIdsEndpoint")
|
||||||
|
};
|
||||||
|
httpRequest.Headers.UserAgent.ParseAdd(client.UserAgent);
|
||||||
|
httpRequest.Headers.Add("Origin", Origin);
|
||||||
|
|
||||||
|
var response = await httpClient.SendAsync(httpRequest);
|
||||||
|
if (!response.IsSuccessStatusCode)
|
||||||
|
{
|
||||||
|
var responseResult = await response.Content.ReadAsStringAsync();
|
||||||
|
return ResultError.Fail(responseResult);
|
||||||
|
}
|
||||||
|
|
||||||
|
var responseContent = await response.Content.ReadAsStringAsync();
|
||||||
|
var datasyncIdsJson = JsonNode.Parse(responseContent.Replace(")]}'", ""));
|
||||||
|
|
||||||
|
var isLoggedOut = datasyncIdsJson?["responseContext"]?["mainAppWebResponseContext"]?["loggedOut"]
|
||||||
|
.Deserialize<bool>() ?? true;
|
||||||
|
if (!isLoggedOut)
|
||||||
|
{
|
||||||
|
return datasyncIdsJson?["datasyncIds"].Deserialize<string[]>() ?? [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return ResultError.Fail("Failed to get datasyncIds!");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async Task<Result<AccountMenuInfo>> GetCurrentAccountInfoAsync(YouTubeClient client)
|
||||||
{
|
{
|
||||||
if (client.ClientState is not { LoggedIn: true })
|
if (client.ClientState is not { LoggedIn: true })
|
||||||
{
|
{
|
||||||
|
@@ -9,11 +9,11 @@ public static class AuthenticationUtilities
|
|||||||
{
|
{
|
||||||
private const string HeaderScheme = "SAPISIDHASH";
|
private const string HeaderScheme = "SAPISIDHASH";
|
||||||
|
|
||||||
// Dave Thomas @ https://stackoverflow.com/a/32065323/9948300
|
// Dave Thomas & windy for updated answer @ https://stackoverflow.com/a/32065323/9948300
|
||||||
public static AuthenticationHeaderValue? GetSapisidHashHeader(string datasyncId, string sapisid, string origin)
|
public static AuthenticationHeaderValue? GetSapisidHashHeader(string datasyncId, string sapisid, string origin)
|
||||||
{
|
{
|
||||||
var strHash = GetSapisidHash(datasyncId, sapisid, origin);
|
var strHash = GetSapisidHash(datasyncId, sapisid, origin);
|
||||||
return new AuthenticationHeaderValue(HeaderScheme, strHash);
|
return strHash == null ? null : new AuthenticationHeaderValue(HeaderScheme, strHash);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static string? GetSapisidHash(string datasyncId, string sapisid, string origin, string? time = null)
|
public static string? GetSapisidHash(string datasyncId, string sapisid, string origin, string? time = null)
|
||||||
|
@@ -8,9 +8,10 @@ public sealed class YouTubeClient : IDisposable
|
|||||||
public string Id { get; private set; } = "";
|
public string Id { get; private set; } = "";
|
||||||
public string AccountName => ClientState?.UserAccountName ?? "";
|
public string AccountName => ClientState?.UserAccountName ?? "";
|
||||||
public string? UserAgent { get; set; }
|
public string? UserAgent { get; set; }
|
||||||
public CookieContainer CookieContainer { get; } = new();
|
public CookieContainer CookieContainer { get; } = new() { Capacity = 100, PerDomainCapacity = 50 };
|
||||||
public ClientState? ClientState { get; private set; }
|
public ClientState? ClientState { get; private set; }
|
||||||
public Cookie? SapisidCookie => CookieContainer.GetAllCookies()["SAPISID"] ?? CookieContainer.GetAllCookies()["__Secure-3PAPISID"];
|
public List<string> DatasyncIds { get; set; } = [];
|
||||||
|
public Cookie? SapisidCookie => CookieContainer.GetAllCookies()["SAPISID"];
|
||||||
public HttpClient? GetHttpClient() => _httpClient;
|
public HttpClient? GetHttpClient() => _httpClient;
|
||||||
|
|
||||||
private HttpClient? _httpClient;
|
private HttpClient? _httpClient;
|
||||||
@@ -31,9 +32,10 @@ public sealed class YouTubeClient : IDisposable
|
|||||||
CookieContainer = CookieContainer
|
CookieContainer = CookieContainer
|
||||||
};
|
};
|
||||||
_httpClient = new HttpClient(clientHandler);
|
_httpClient = new HttpClient(clientHandler);
|
||||||
|
_httpClient.DefaultRequestHeaders.Clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task GetStateAsync()
|
public async Task BuildClientAsync()
|
||||||
{
|
{
|
||||||
if (ClientState == null || !ClientState.LoggedIn)
|
if (ClientState == null || !ClientState.LoggedIn)
|
||||||
{
|
{
|
||||||
@@ -44,7 +46,23 @@ public sealed class YouTubeClient : IDisposable
|
|||||||
}
|
}
|
||||||
ClientState = state.Value;
|
ClientState = state.Value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (string.IsNullOrWhiteSpace(ClientState.WebPlayerContextConfig?.WebPlayerContext?.DatasyncId))
|
||||||
|
{
|
||||||
|
var datasyncResult = await NetworkService.GetDatasyncIds(this);
|
||||||
|
if (!datasyncResult.IsSuccess)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (var id in datasyncResult.Value)
|
||||||
|
{
|
||||||
|
if (DatasyncIds.Contains(id))
|
||||||
|
continue;
|
||||||
|
DatasyncIds.Add(id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var accountInfo = await NetworkService.GetCurrentAccountInfoAsync(this);
|
var accountInfo = await NetworkService.GetCurrentAccountInfoAsync(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user