로그인 STA 스레드화·SSO 로그아웃·UserInfo 역직렬화 수정 및 AuthTest 추가
- LoginWindow: 로그인/로그아웃/쿠키삭제를 전용 STA 스레드(자체 Dispatcher 펌프)에서 실행 → SignIn/SignInAsync가 MTA/STA 어느 스레드에서 호출돼도 동작 (RunOnDedicatedUiThreadAsync, AuthenticateAsync) - SsoClient: 로그인/로그아웃 창을 AuthenticateAsync로 호출, 크로스스레드 Owner 제거 - BaronSSO: SignIn/SignOut을 Task.Run(...).GetAwaiter().GetResult()로 정리, SignInAsync 데드락 주석 추가 - UserInfo: private set 프로퍼티에 [JsonInclude] 적용 → FromSsoFile 역직렬화 복원 정상화, LastAuthTime [JsonIgnore] - AuthTest 샘플 프로젝트 추가 Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
@@ -4,7 +4,7 @@
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||
xmlns:wv2="clr-namespace:Microsoft.Web.WebView2.Wpf;assembly=Microsoft.Web.WebView2.Wpf"
|
||||
Title="BARON 로그인" Height="760" Width="520"
|
||||
WindowStartupLocation="CenterOwner">
|
||||
WindowStartupLocation="CenterScreen">
|
||||
<Grid>
|
||||
<wv2:WebView2 x:Name="webview" />
|
||||
</Grid>
|
||||
|
||||
@@ -1,5 +1,7 @@
|
||||
using Microsoft.Web.WebView2.Core;
|
||||
using System.Threading;
|
||||
using System.Windows;
|
||||
using System.Windows.Threading;
|
||||
using WebView2 = Microsoft.Web.WebView2.Wpf.WebView2;
|
||||
|
||||
namespace BaronSoftware.SSO
|
||||
@@ -58,41 +60,96 @@ namespace BaronSoftware.SSO
|
||||
return _tcs.Task;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// 호출 스레드가 MTA/STA 무엇이든, 전용 STA 스레드(자체 Dispatcher 펌프)에서
|
||||
/// 로그인 창을 띄워 콜백 URL(code 또는 logout 복귀)을 비동기로 반환한다.
|
||||
/// </summary>
|
||||
internal static Task<string> AuthenticateAsync(string authorizeUrl, string redirectUri)
|
||||
=> RunOnDedicatedUiThreadAsync(async () =>
|
||||
{
|
||||
var window = new LoginWindow(authorizeUrl, redirectUri);
|
||||
return await window.ShowAndGetRedirectAsync();
|
||||
});
|
||||
|
||||
/// <summary>
|
||||
/// 로그인 WebView가 사용하는 프로필의 모든 쿠키(=SSO 세션 쿠키)를 삭제합니다.
|
||||
/// 화면에 보이지 않는 오프스크린 WebView2를 잠깐 띄워 동일 프로필의 쿠키를 비웁니다.
|
||||
/// 전용 STA 스레드에서 오프스크린 WebView2를 잠깐 띄워 동일 프로필의 쿠키를 비웁니다.
|
||||
/// 이후 다음 로그인 시 세션이 없어 로그인 폼이 다시 표시됩니다.
|
||||
/// </summary>
|
||||
internal static async Task ClearSessionCookiesAsync()
|
||||
internal static Task ClearSessionCookiesAsync()
|
||||
=> RunOnDedicatedUiThreadAsync(async () =>
|
||||
{
|
||||
var holder = new Window
|
||||
{
|
||||
Width = 1,
|
||||
Height = 1,
|
||||
Left = -32000,
|
||||
Top = -32000,
|
||||
WindowStyle = WindowStyle.None,
|
||||
ShowInTaskbar = false,
|
||||
ShowActivated = false,
|
||||
Title = string.Empty,
|
||||
};
|
||||
var web = new WebView2();
|
||||
holder.Content = web;
|
||||
holder.Show(); // WebView2 초기화에 필요한 HWND 확보(화면 밖)
|
||||
try
|
||||
{
|
||||
await web.EnsureCoreWebView2Async();
|
||||
// DeleteAllCookies()는 즉시 반환이라, 디스크 반영 전에 Dispose되면 쿠키가 남아
|
||||
// 다음 로그인이 SSO로 조용히 통과될 수 있다. 완료까지 await하는 ClearBrowsingDataAsync로
|
||||
// SSO 세션 쿠키/사이트 데이터를 확실히 제거한다.
|
||||
await web.CoreWebView2.Profile.ClearBrowsingDataAsync(
|
||||
CoreWebView2BrowsingDataKinds.AllSite);
|
||||
}
|
||||
finally
|
||||
{
|
||||
web.Dispose();
|
||||
holder.Close();
|
||||
}
|
||||
});
|
||||
|
||||
/// <summary>
|
||||
/// 호출 스레드(MTA/STA)와 무관하게, 전용 STA 스레드에서 자체 Dispatcher 메시지펌프를 돌려
|
||||
/// WPF/WebView2 UI 작업을 실행하고 그 결과를 Task로 반환한다.
|
||||
/// (WPF 창은 STA + 메시지펌프가 필요하므로, MTA/Task.Run/콘솔에서도 안전하게 동작)
|
||||
/// </summary>
|
||||
internal static Task<T> RunOnDedicatedUiThreadAsync<T>(Func<Task<T>> work)
|
||||
{
|
||||
var holder = new Window
|
||||
var tcs = new TaskCompletionSource<T>(TaskCreationOptions.RunContinuationsAsynchronously);
|
||||
|
||||
var thread = new Thread(() =>
|
||||
{
|
||||
Width = 1,
|
||||
Height = 1,
|
||||
Left = -32000,
|
||||
Top = -32000,
|
||||
WindowStyle = WindowStyle.None,
|
||||
ShowInTaskbar = false,
|
||||
ShowActivated = false,
|
||||
Title = string.Empty,
|
||||
};
|
||||
var web = new WebView2();
|
||||
holder.Content = web;
|
||||
holder.Show(); // WebView2 초기화에 필요한 HWND 확보(화면 밖)
|
||||
try
|
||||
{
|
||||
await web.EnsureCoreWebView2Async();
|
||||
// DeleteAllCookies()는 즉시 반환이라, 디스크 반영 전에 Dispose되면 쿠키가 남아
|
||||
// 다음 로그인이 SSO로 조용히 통과될 수 있다. 완료까지 await하는 ClearBrowsingDataAsync로
|
||||
// SSO 세션 쿠키/사이트 데이터를 확실히 제거한다.
|
||||
await web.CoreWebView2.Profile.ClearBrowsingDataAsync(
|
||||
CoreWebView2BrowsingDataKinds.AllSite);
|
||||
}
|
||||
finally
|
||||
{
|
||||
web.Dispose();
|
||||
holder.Close();
|
||||
}
|
||||
try
|
||||
{
|
||||
// 이 전용 스레드에 Dispatcher 기반 SynchronizationContext를 설치 → await 연속이 같은 스레드로 복귀
|
||||
SynchronizationContext.SetSynchronizationContext(
|
||||
new DispatcherSynchronizationContext(Dispatcher.CurrentDispatcher));
|
||||
|
||||
work().ContinueWith(t =>
|
||||
{
|
||||
if (t.IsFaulted) tcs.TrySetException(t.Exception!.InnerExceptions);
|
||||
else if (t.IsCanceled) tcs.TrySetCanceled();
|
||||
else tcs.TrySetResult(t.Result);
|
||||
|
||||
Dispatcher.CurrentDispatcher.BeginInvokeShutdown(DispatcherPriority.Background);
|
||||
}, TaskScheduler.FromCurrentSynchronizationContext());
|
||||
|
||||
Dispatcher.Run(); // 메시지펌프 시작 (작업 완료 시 InvokeShutdown으로 종료)
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
tcs.TrySetException(ex);
|
||||
}
|
||||
});
|
||||
|
||||
thread.IsBackground = true;
|
||||
thread.SetApartmentState(ApartmentState.STA);
|
||||
thread.Start();
|
||||
return tcs.Task;
|
||||
}
|
||||
|
||||
internal static Task RunOnDedicatedUiThreadAsync(Func<Task> work)
|
||||
=> RunOnDedicatedUiThreadAsync<bool>(async () => { await work(); return true; });
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user