wabbajack/Wabbajack/View Models/UserInterventionHandlers.cs

212 lines
7.8 KiB
C#
Raw Normal View History

2019-12-08 17:00:22 +00:00
using System;
using System.Collections.Generic;
using System.Linq;
2020-02-11 00:30:38 +00:00
using System.Net.Http;
2019-12-08 17:00:22 +00:00
using System.Text;
using System.Threading;
2019-12-08 17:00:22 +00:00
using System.Threading.Tasks;
2021-06-14 03:31:09 +00:00
using System.Web;
2019-12-08 17:00:22 +00:00
using System.Windows;
using System.Windows.Threading;
2020-02-06 05:30:31 +00:00
using CefSharp;
2019-12-08 17:00:22 +00:00
using ReactiveUI;
using Wabbajack.Common;
using Wabbajack.Common.StatusFeed;
using Wabbajack.Lib;
2019-12-08 17:00:22 +00:00
using Wabbajack.Lib.Downloaders;
2020-02-11 00:30:38 +00:00
using Wabbajack.Lib.LibCefHelpers;
2019-12-08 17:00:22 +00:00
using Wabbajack.Lib.NexusApi;
using Wabbajack.Lib.WebAutomation;
2021-06-14 03:31:09 +00:00
using WebSocketSharp;
2019-12-08 17:00:22 +00:00
namespace Wabbajack
{
public class UserInterventionHandlers
{
public MainWindowVM MainWindow { get; }
2021-06-30 21:57:35 +00:00
private AsyncLock _browserLock = new();
2019-12-08 17:00:22 +00:00
public UserInterventionHandlers(MainWindowVM mvm)
{
MainWindow = mvm;
2019-12-08 17:00:22 +00:00
}
private async Task WrapBrowserJob(IUserIntervention intervention, Func<WebBrowserVM, CancellationTokenSource, Task> toDo)
2019-12-08 17:00:22 +00:00
{
2021-06-30 21:57:35 +00:00
var wait = await _browserLock.WaitAsync();
2020-02-11 00:30:38 +00:00
var cancel = new CancellationTokenSource();
var oldPane = MainWindow.ActivePane;
2020-04-20 22:36:11 +00:00
using var vm = await WebBrowserVM.GetNew();
MainWindow.NavigateTo(vm);
vm.BackCommand = ReactiveCommand.Create(() =>
2019-12-08 17:00:22 +00:00
{
cancel.Cancel();
MainWindow.NavigateTo(oldPane);
intervention.Cancel();
2019-12-08 17:00:22 +00:00
});
try
{
await toDo(vm, cancel);
}
catch (TaskCanceledException)
{
intervention.Cancel();
}
catch (Exception ex)
{
Utils.Error(ex);
intervention.Cancel();
}
2021-06-30 21:57:35 +00:00
finally
{
wait.Dispose();
}
MainWindow.NavigateTo(oldPane);
2019-12-08 17:00:22 +00:00
}
public async Task Handle(IStatusMessage msg)
2019-12-08 17:00:22 +00:00
{
switch (msg)
{
case RequestNexusAuthorization c:
await WrapBrowserJob(c, async (vm, cancel) =>
{
await vm.Driver.WaitForInitialized();
var key = await NexusApiClient.SetupNexusLogin(new CefSharpWrapper(vm.Browser), m => vm.Instructions = m, cancel.Token);
c.Resume(key);
});
2019-12-08 17:00:22 +00:00
break;
2020-02-06 05:30:31 +00:00
case ManuallyDownloadNexusFile c:
await WrapBrowserJob(c, (vm, cancel) => HandleManualNexusDownload(vm, cancel, c));
2020-02-06 05:30:31 +00:00
break;
2020-02-11 00:30:38 +00:00
case ManuallyDownloadFile c:
await WrapBrowserJob(c, (vm, cancel) => HandleManualDownload(vm, cancel, c));
2020-02-11 00:30:38 +00:00
break;
case AbstractNeedsLoginDownloader.RequestSiteLogin c:
await WrapBrowserJob(c, async (vm, cancel) =>
{
await vm.Driver.WaitForInitialized();
var data = await c.Downloader.GetAndCacheCookies(new CefSharpWrapper(vm.Browser), m => vm.Instructions = m, cancel.Token);
c.Resume(data);
});
2021-06-14 03:31:09 +00:00
break;
case RequestOAuthLogin oa:
await WrapBrowserJob(oa, async (vm, cancel) =>
{
await OAuthLogin(oa, vm, cancel);
});
2019-12-08 17:00:22 +00:00
break;
case CriticalFailureIntervention c:
MessageBox.Show(c.ExtendedDescription, c.ShortDescription, MessageBoxButton.OK,
MessageBoxImage.Error);
c.Cancel();
if (c.ExitApplication) await MainWindow.ShutdownApplication();
break;
case ConfirmationIntervention c:
break;
2019-12-08 17:00:22 +00:00
default:
throw new NotImplementedException($"No handler for {msg}");
}
}
2019-12-20 20:51:10 +00:00
2021-06-14 03:31:09 +00:00
private async Task OAuthLogin(RequestOAuthLogin oa, WebBrowserVM vm, CancellationTokenSource cancel)
{
await vm.Driver.WaitForInitialized();
2021-06-19 21:19:29 +00:00
vm.Instructions = $"Please log in and allow Wabbajack to access your {oa.SiteName} account";
2021-06-14 03:31:09 +00:00
var wrapper = new CefSharpWrapper(vm.Browser);
var scopes = string.Join(" ", oa.Scopes);
var state = Guid.NewGuid().ToString();
2021-06-19 21:19:29 +00:00
2021-06-14 03:31:09 +00:00
2021-06-30 02:16:48 +00:00
var oldHandler = Helpers.SchemeHandler;
2021-06-14 03:31:09 +00:00
Helpers.SchemeHandler = (browser, frame, _, request) =>
{
var req = new Uri(request.Url);
2021-06-30 21:57:35 +00:00
Utils.LogStraightToFile($"Got Scheme callback {req}");
2021-06-14 03:31:09 +00:00
var parsed = HttpUtility.ParseQueryString(req.Query);
if (parsed.Contains("state"))
{
if (parsed.Get("state") != state)
{
Utils.Log("Bad OAuth state, state, this shouldn't happen");
oa.Cancel();
return new ResourceHandler();
}
}
2021-06-14 03:31:09 +00:00
if (parsed.Contains("code"))
{
2021-06-30 21:57:35 +00:00
Helpers.SchemeHandler = oldHandler;
2021-06-14 03:31:09 +00:00
oa.Resume(parsed.Get("code"));
}
else
{
oa.Cancel();
}
return new ResourceHandler();
};
2021-06-19 21:19:29 +00:00
await wrapper.NavigateTo(new Uri(oa.AuthorizationEndpoint + $"?response_type=code&client_id={oa.ClientID}&state={state}&scope={scopes}"));
2021-06-14 03:31:09 +00:00
while (!oa.Task.IsCanceled && !oa.Task.IsCompleted && !cancel.IsCancellationRequested)
await Task.Delay(250);
}
2020-02-11 00:30:38 +00:00
private async Task HandleManualDownload(WebBrowserVM vm, CancellationTokenSource cancel, ManuallyDownloadFile manuallyDownloadFile)
{
var browser = new CefSharpWrapper(vm.Browser);
vm.Instructions = $"Please locate and download {manuallyDownloadFile.State.Url}";
var result = new TaskCompletionSource<Uri>();
browser.DownloadHandler = uri =>
{
//var client = Helpers.GetClient(browser.GetCookies("").Result, browser.Location);
result.SetResult(uri);
};
await vm.Driver.WaitForInitialized();
await browser.NavigateTo(new Uri(manuallyDownloadFile.State.Url));
while (!cancel.IsCancellationRequested)
{
if (result.Task.IsCompleted)
{
var cookies = await Helpers.GetCookies();
var referer = browser.Location;
var client = Helpers.GetClient(cookies, referer);
manuallyDownloadFile.Resume(result.Task.Result, client);
break;
}
await Task.Delay(100);
}
}
2020-02-06 05:30:31 +00:00
private async Task HandleManualNexusDownload(WebBrowserVM vm, CancellationTokenSource cancel, ManuallyDownloadNexusFile manuallyDownloadNexusFile)
{
var state = manuallyDownloadNexusFile.State;
var game = state.Game.MetaData();
2020-02-06 05:30:31 +00:00
await vm.Driver.WaitForInitialized();
IWebDriver browser = new CefSharpWrapper(vm.Browser);
2021-06-20 22:36:56 +00:00
vm.Instructions = $"Click the download button to continue (get a NexusMods.com Premium account to automate this)";
2020-02-06 05:30:31 +00:00
browser.DownloadHandler = uri =>
{
manuallyDownloadNexusFile.Resume(uri);
2020-02-11 00:30:38 +00:00
browser.DownloadHandler = null;
2020-02-06 05:30:31 +00:00
};
2021-06-20 22:36:56 +00:00
var url = new Uri(@$"https://www.nexusmods.com/{game.NexusName}/mods/{state.ModID}?tab=files&file_id={state.FileID}");
await browser.NavigateTo(url);
2020-02-06 05:30:31 +00:00
while (!cancel.IsCancellationRequested && !manuallyDownloadNexusFile.Task.IsCompleted) {
await Task.Delay(250);
}
}
2019-12-08 17:00:22 +00:00
}
}