로그인 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:
10
AuthTest/App.xaml
Normal file
10
AuthTest/App.xaml
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
<Application x:Class="AuthTest.App"
|
||||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
||||||
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||||
|
xmlns:local="clr-namespace:AuthTest"
|
||||||
|
ShutdownMode="OnMainWindowClose"
|
||||||
|
StartupUri="MainWindow.xaml">
|
||||||
|
<Application.Resources>
|
||||||
|
|
||||||
|
</Application.Resources>
|
||||||
|
</Application>
|
||||||
31
AuthTest/App.xaml.cs
Normal file
31
AuthTest/App.xaml.cs
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
using BaronSoftware.SSO;
|
||||||
|
using System.Configuration;
|
||||||
|
using System.Data;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using System.Windows;
|
||||||
|
|
||||||
|
namespace AuthTest
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Interaction logic for App.xaml
|
||||||
|
/// </summary>
|
||||||
|
public partial class App : Application
|
||||||
|
{
|
||||||
|
protected override void OnStartup(StartupEventArgs e)
|
||||||
|
{
|
||||||
|
base.OnStartup(e);
|
||||||
|
|
||||||
|
BaronSSOOption option = new()
|
||||||
|
{
|
||||||
|
Authority = "https://sso.hmac.kr/oidc",
|
||||||
|
ClientId = "aca44872-8280-40c3-9a80-3aefafdf722a",
|
||||||
|
RedirectUri = "http://localhost:9090/eg-bim/auth/callback",
|
||||||
|
PostLogoutRedirectUri = "http://localhost:9090/eg-bim/logout/callback",
|
||||||
|
ExtraUserValidator = new ExtraUserInvalidator()
|
||||||
|
};
|
||||||
|
var auth = new BaronSSO(option);
|
||||||
|
auth.SignIn();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
10
AuthTest/AssemblyInfo.cs
Normal file
10
AuthTest/AssemblyInfo.cs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
using System.Windows;
|
||||||
|
|
||||||
|
[assembly: ThemeInfo(
|
||||||
|
ResourceDictionaryLocation.None, //where theme specific resource dictionaries are located
|
||||||
|
//(used if a resource is not found in the page,
|
||||||
|
// or application resource dictionaries)
|
||||||
|
ResourceDictionaryLocation.SourceAssembly //where the generic resource dictionary is located
|
||||||
|
//(used if a resource is not found in the page,
|
||||||
|
// app, or any theme specific resource dictionaries)
|
||||||
|
)]
|
||||||
15
AuthTest/AuthTest.csproj
Normal file
15
AuthTest/AuthTest.csproj
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<OutputType>WinExe</OutputType>
|
||||||
|
<TargetFramework>net8.0-windows</TargetFramework>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<UseWPF>true</UseWPF>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\BaronSoftware.SSO\BaronSoftware.SSO.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
16
AuthTest/ExtraUserInvalidator.cs
Normal file
16
AuthTest/ExtraUserInvalidator.cs
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
using BaronSoftware.SSO;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace AuthTest
|
||||||
|
{
|
||||||
|
public class ExtraUserInvalidator : IUserValidator
|
||||||
|
{
|
||||||
|
public void Validate(UserInfo user)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
12
AuthTest/MainWindow.xaml
Normal file
12
AuthTest/MainWindow.xaml
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
<Window x:Class="AuthTest.MainWindow"
|
||||||
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
||||||
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||||
|
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
|
||||||
|
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
|
||||||
|
xmlns:local="clr-namespace:AuthTest"
|
||||||
|
mc:Ignorable="d"
|
||||||
|
Title="MainWindow" Height="450" Width="800">
|
||||||
|
<Grid>
|
||||||
|
|
||||||
|
</Grid>
|
||||||
|
</Window>
|
||||||
24
AuthTest/MainWindow.xaml.cs
Normal file
24
AuthTest/MainWindow.xaml.cs
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
using System.Text;
|
||||||
|
using System.Windows;
|
||||||
|
using System.Windows.Controls;
|
||||||
|
using System.Windows.Data;
|
||||||
|
using System.Windows.Documents;
|
||||||
|
using System.Windows.Input;
|
||||||
|
using System.Windows.Media;
|
||||||
|
using System.Windows.Media.Imaging;
|
||||||
|
using System.Windows.Navigation;
|
||||||
|
using System.Windows.Shapes;
|
||||||
|
|
||||||
|
namespace AuthTest
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Interaction logic for MainWindow.xaml
|
||||||
|
/// </summary>
|
||||||
|
public partial class MainWindow : Window
|
||||||
|
{
|
||||||
|
public MainWindow()
|
||||||
|
{
|
||||||
|
InitializeComponent();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -7,6 +7,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "BaronSoftware.SSO", "BaronS
|
|||||||
EndProject
|
EndProject
|
||||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "BaronSoftware.SSO.Sample", "BaronSoftware.SSO.Sample\BaronSoftware.SSO.Sample.csproj", "{6F80A264-7099-3F9E-64A3-89E569A3E7A4}"
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "BaronSoftware.SSO.Sample", "BaronSoftware.SSO.Sample\BaronSoftware.SSO.Sample.csproj", "{6F80A264-7099-3F9E-64A3-89E569A3E7A4}"
|
||||||
EndProject
|
EndProject
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AuthTest", "AuthTest\AuthTest.csproj", "{CC253CA9-0F0E-43D9-80D5-CEC2A181E889}"
|
||||||
|
EndProject
|
||||||
Global
|
Global
|
||||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
Debug|Any CPU = Debug|Any CPU
|
Debug|Any CPU = Debug|Any CPU
|
||||||
@@ -21,6 +23,10 @@ Global
|
|||||||
{6F80A264-7099-3F9E-64A3-89E569A3E7A4}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
{6F80A264-7099-3F9E-64A3-89E569A3E7A4}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
{6F80A264-7099-3F9E-64A3-89E569A3E7A4}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
{6F80A264-7099-3F9E-64A3-89E569A3E7A4}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
{6F80A264-7099-3F9E-64A3-89E569A3E7A4}.Release|Any CPU.Build.0 = Release|Any CPU
|
{6F80A264-7099-3F9E-64A3-89E569A3E7A4}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{CC253CA9-0F0E-43D9-80D5-CEC2A181E889}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{CC253CA9-0F0E-43D9-80D5-CEC2A181E889}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{CC253CA9-0F0E-43D9-80D5-CEC2A181E889}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{CC253CA9-0F0E-43D9-80D5-CEC2A181E889}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
GlobalSection(SolutionProperties) = preSolution
|
GlobalSection(SolutionProperties) = preSolution
|
||||||
HideSolutionNode = FALSE
|
HideSolutionNode = FALSE
|
||||||
|
|||||||
@@ -24,16 +24,42 @@ namespace BaronSoftware.SSO
|
|||||||
|
|
||||||
public void SignIn()
|
public void SignIn()
|
||||||
{
|
{
|
||||||
// STA Thread 이슈로 인해, Task.Run() 으로 감싸서 동기화 함수 구현.
|
// UI 스레드에서 호출돼도 데드락이 없도록 Task.Run(스레드풀)에서 실행한다.
|
||||||
Task.Run(async () => await SignInAsync()).Wait();
|
// (실제 WebView 창은 SsoClient가 전용 STA 스레드에서 띄우므로 STA/MTA 무관하게 동작)
|
||||||
|
// .GetAwaiter().GetResult()로 원본 예외를 그대로 전파한다(AggregateException 래핑 방지).
|
||||||
|
Task.Run(() => SignInAsync()).GetAwaiter().GetResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void SignOut()
|
public void SignOut()
|
||||||
{
|
{
|
||||||
Task.Run(async () => await SignOutAsync()).Wait();
|
Task.Run(() => SignOutAsync()).GetAwaiter().GetResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>웹뷰 로그인 창을 띄워 인증합니다.</summary>
|
/// <summary>
|
||||||
|
/// 웹뷰 로그인 창을 띄워 인증합니다. (비동기)
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// ⚠️ 절대 UI 스레드에서 <c>.Wait()</c> / <c>.Result</c> 로 블로킹하지 마세요. 데드락이 발생합니다.
|
||||||
|
///
|
||||||
|
/// [데드락이 나는 이유 — sync-over-async]
|
||||||
|
/// 1) WPF UI 스레드에는 DispatcherSynchronizationContext 가 설치돼 있습니다.
|
||||||
|
/// 2) 이 메서드를 UI 스레드에서 호출하면, 내부의 첫 <c>await</c>
|
||||||
|
/// (CheckConnection → GetDiscoveryAsync → httpclient.GetStringAsync 등)에서
|
||||||
|
/// ConfigureAwait(false) 를 쓰지 않으므로 "연속(continuation)을 UI 스레드로 되돌려" 실행하도록
|
||||||
|
/// 컨텍스트를 캡처합니다.
|
||||||
|
/// 3) 그런데 호출 측이 <c>.Wait()</c> 로 UI 스레드를 막으면 메시지펌프가 멈춥니다.
|
||||||
|
/// 4) await 대상(HTTP 등)이 스레드풀에서 끝나고 연속을 UI 디스패처 큐에 post 하지만,
|
||||||
|
/// UI 스레드가 막혀 있어 그 큐를 영원히 처리하지 못합니다 → Task 미완료 → .Wait() 무한 대기 → 데드락.
|
||||||
|
/// (로그인 창에 도달하기 전, 첫 HTTP await 에서 이미 멈춥니다.)
|
||||||
|
///
|
||||||
|
/// [올바른 사용법]
|
||||||
|
/// • UI 스레드라면 블로킹하지 말고 await 하세요: <c>await auth.SignInAsync();</c>
|
||||||
|
/// • 동기적으로 써야 하면 동기 진입점 <see cref="SignIn"/> 을 쓰세요.
|
||||||
|
/// SignIn() 은 <c>Task.Run(() => SignInAsync()).GetAwaiter().GetResult()</c> 로,
|
||||||
|
/// SignInAsync 를 SynchronizationContext 가 없는 스레드풀 스레드에서 실행하므로
|
||||||
|
/// 연속이 UI 스레드로 되돌아올 필요가 없어 데드락이 발생하지 않습니다.
|
||||||
|
/// (실제 WebView 로그인 창은 SsoClient 가 전용 STA 스레드에서 띄우므로 STA/MTA 무관하게 동작합니다.)
|
||||||
|
/// </remarks>
|
||||||
public async Task SignInAsync()
|
public async Task SignInAsync()
|
||||||
{
|
{
|
||||||
UserInfo? user = null;
|
UserInfo? user = null;
|
||||||
@@ -43,6 +69,13 @@ namespace BaronSoftware.SSO
|
|||||||
await SignInAsync(user?.RefreshToken);
|
await SignInAsync(user?.RefreshToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// refresh_token 으로 무창 인증을 시도하고, 불가하면 웹뷰 로그인으로 진행합니다. (비동기)
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// UI 스레드에서 <c>.Wait()</c>/<c>.Result</c> 로 블로킹하면 데드락입니다. 자세한 이유는
|
||||||
|
/// <see cref="SignInAsync()"/> 의 설명을 참고하세요. (UI 스레드면 await, 동기 호출은 <see cref="SignIn"/>)
|
||||||
|
/// </remarks>
|
||||||
public async Task SignInAsync(string refreshToken)
|
public async Task SignInAsync(string refreshToken)
|
||||||
{
|
{
|
||||||
UserInfo user = null;
|
UserInfo user = null;
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||||
xmlns:wv2="clr-namespace:Microsoft.Web.WebView2.Wpf;assembly=Microsoft.Web.WebView2.Wpf"
|
xmlns:wv2="clr-namespace:Microsoft.Web.WebView2.Wpf;assembly=Microsoft.Web.WebView2.Wpf"
|
||||||
Title="BARON 로그인" Height="760" Width="520"
|
Title="BARON 로그인" Height="760" Width="520"
|
||||||
WindowStartupLocation="CenterOwner">
|
WindowStartupLocation="CenterScreen">
|
||||||
<Grid>
|
<Grid>
|
||||||
<wv2:WebView2 x:Name="webview" />
|
<wv2:WebView2 x:Name="webview" />
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
using Microsoft.Web.WebView2.Core;
|
using Microsoft.Web.WebView2.Core;
|
||||||
|
using System.Threading;
|
||||||
using System.Windows;
|
using System.Windows;
|
||||||
|
using System.Windows.Threading;
|
||||||
using WebView2 = Microsoft.Web.WebView2.Wpf.WebView2;
|
using WebView2 = Microsoft.Web.WebView2.Wpf.WebView2;
|
||||||
|
|
||||||
namespace BaronSoftware.SSO
|
namespace BaronSoftware.SSO
|
||||||
@@ -58,12 +60,24 @@ namespace BaronSoftware.SSO
|
|||||||
return _tcs.Task;
|
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>
|
/// <summary>
|
||||||
/// 로그인 WebView가 사용하는 프로필의 모든 쿠키(=SSO 세션 쿠키)를 삭제합니다.
|
/// 로그인 WebView가 사용하는 프로필의 모든 쿠키(=SSO 세션 쿠키)를 삭제합니다.
|
||||||
/// 화면에 보이지 않는 오프스크린 WebView2를 잠깐 띄워 동일 프로필의 쿠키를 비웁니다.
|
/// 전용 STA 스레드에서 오프스크린 WebView2를 잠깐 띄워 동일 프로필의 쿠키를 비웁니다.
|
||||||
/// 이후 다음 로그인 시 세션이 없어 로그인 폼이 다시 표시됩니다.
|
/// 이후 다음 로그인 시 세션이 없어 로그인 폼이 다시 표시됩니다.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
internal static async Task ClearSessionCookiesAsync()
|
internal static Task ClearSessionCookiesAsync()
|
||||||
|
=> RunOnDedicatedUiThreadAsync(async () =>
|
||||||
{
|
{
|
||||||
var holder = new Window
|
var holder = new Window
|
||||||
{
|
{
|
||||||
@@ -93,6 +107,49 @@ namespace BaronSoftware.SSO
|
|||||||
web.Dispose();
|
web.Dispose();
|
||||||
holder.Close();
|
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 tcs = new TaskCompletionSource<T>(TaskCreationOptions.RunContinuationsAsynchronously);
|
||||||
|
|
||||||
|
var thread = new Thread(() =>
|
||||||
|
{
|
||||||
|
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; });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ namespace BaronSoftware.SSO
|
|||||||
{
|
{
|
||||||
internal static class GlobalConfigs
|
internal static class GlobalConfigs
|
||||||
{
|
{
|
||||||
|
|
||||||
internal static readonly string CenterTanant_UUID = "5530ca6e-c5e6-4bf0-84d6-76c6a8fb70ee";
|
internal static readonly string CenterTanant_UUID = "5530ca6e-c5e6-4bf0-84d6-76c6a8fb70ee";
|
||||||
internal static readonly string FamilyTanant_UUID = "038326b6-954a-48a7-a85f-efd83f62b82a";
|
internal static readonly string FamilyTanant_UUID = "038326b6-954a-48a7-a85f-efd83f62b82a";
|
||||||
|
|
||||||
|
|||||||
@@ -7,28 +7,53 @@ using System.Security.Cryptography;
|
|||||||
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 System.Text.Json.Serialization;
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
|
|
||||||
namespace BaronSoftware.SSO
|
namespace BaronSoftware.SSO
|
||||||
{
|
{
|
||||||
public class UserInfo
|
public class UserInfo
|
||||||
{
|
{
|
||||||
|
// System.Text.Json은 기본적으로 private setter에 역직렬화하지 못한다.
|
||||||
|
// [JsonInclude]를 붙여야 FromSsoFile()의 Deserialize가 private set 프로퍼티를 복원한다.
|
||||||
|
[JsonInclude]
|
||||||
public string UUID { get; private set; }
|
public string UUID { get; private set; }
|
||||||
|
|
||||||
|
[JsonInclude]
|
||||||
public string Name { get; private set; }
|
public string Name { get; private set; }
|
||||||
|
|
||||||
|
[JsonInclude]
|
||||||
public string Email { get; private set; }
|
public string Email { get; private set; }
|
||||||
|
|
||||||
|
[JsonInclude]
|
||||||
public string[] SubEmails { get; private set; }
|
public string[] SubEmails { get; private set; }
|
||||||
|
|
||||||
|
[JsonInclude]
|
||||||
public string RefreshToken { get; private set; }
|
public string RefreshToken { get; private set; }
|
||||||
|
|
||||||
|
[JsonInclude]
|
||||||
public string TenantId { get; private set; }
|
public string TenantId { get; private set; }
|
||||||
|
|
||||||
/// <summary>'tenants' 클레임 안에 등장하는 모든 테넌트 id 목록(상위/조상 테넌트 포함, 중복 제거).</summary>
|
/// <summary>'tenants' 클레임 안에 등장하는 모든 테넌트 id 목록(상위/조상 테넌트 포함, 중복 제거).</summary>
|
||||||
|
[JsonInclude]
|
||||||
public string[] AllTenantIds { get; private set; }
|
public string[] AllTenantIds { get; private set; }
|
||||||
|
|
||||||
|
[JsonInclude]
|
||||||
public long LastAuthUnixTimeStamp { get; private set; }
|
public long LastAuthUnixTimeStamp { get; private set; }
|
||||||
|
|
||||||
|
// 계산형 프로퍼티(setter 없음) — 파일에 저장/복원할 필요 없으므로 직렬화 제외
|
||||||
|
[JsonIgnore]
|
||||||
public DateTime LastAuthTime => DateTimeOffset.FromUnixTimeSeconds(LastAuthUnixTimeStamp).LocalDateTime;
|
public DateTime LastAuthTime => DateTimeOffset.FromUnixTimeSeconds(LastAuthUnixTimeStamp).LocalDateTime;
|
||||||
|
|
||||||
|
[JsonInclude]
|
||||||
public string IdToken { get; private set; }
|
public string IdToken { get; private set; }
|
||||||
|
|
||||||
|
[JsonInclude]
|
||||||
public string Raw { get; private set; }
|
public string Raw { get; private set; }
|
||||||
|
[JsonInclude]
|
||||||
public string RawTokenResponse { get; private set; }
|
public string RawTokenResponse { get; private set; }
|
||||||
|
|
||||||
|
[JsonInclude]
|
||||||
public Dictionary<string, object> Claims { get; private set; }
|
public Dictionary<string, object> Claims { get; private set; }
|
||||||
|
|
||||||
internal static UserInfo FromJson(string rawjson, TokenResponse reposeToken)
|
internal static UserInfo FromJson(string rawjson, TokenResponse reposeToken)
|
||||||
|
|||||||
@@ -41,11 +41,8 @@ namespace BaronSoftware.SSO
|
|||||||
$"&code_challenge={PKCEUtil.Challenge(verifier)}&code_challenge_method=S256" +
|
$"&code_challenge={PKCEUtil.Challenge(verifier)}&code_challenge_method=S256" +
|
||||||
$"&state={state}&nonce={nonce}";
|
$"&state={state}&nonce={nonce}";
|
||||||
|
|
||||||
var window = new LoginWindow(authorizeUrl, options.RedirectUri);
|
// 전용 STA 스레드에서 로그인 창을 띄운다(호출 스레드가 MTA/STA 무엇이든 동작).
|
||||||
if (Application.Current?.MainWindow is { } owner && !ReferenceEquals(owner, window))
|
var redirected = await LoginWindow.AuthenticateAsync(authorizeUrl, options.RedirectUri);
|
||||||
window.Owner = owner;
|
|
||||||
|
|
||||||
var redirected = await window.ShowAndGetRedirectAsync();
|
|
||||||
|
|
||||||
var q = PKCEUtil.ParseQuery(redirected);
|
var q = PKCEUtil.ParseQuery(redirected);
|
||||||
if (q.TryGetValue("error", out var error))
|
if (q.TryGetValue("error", out var error))
|
||||||
@@ -124,13 +121,10 @@ namespace BaronSoftware.SSO
|
|||||||
$"&post_logout_redirect_uri={Uri.EscapeDataString(options.PostLogoutRedirectUri)}" +
|
$"&post_logout_redirect_uri={Uri.EscapeDataString(options.PostLogoutRedirectUri)}" +
|
||||||
$"&state={PKCEUtil.RandomToken()}";
|
$"&state={PKCEUtil.RandomToken()}";
|
||||||
|
|
||||||
var window = new LoginWindow(url, options.PostLogoutRedirectUri);
|
|
||||||
if (Application.Current?.MainWindow is { } owner && !ReferenceEquals(owner, window))
|
|
||||||
window.Owner = owner;
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await window.ShowAndGetRedirectAsync();
|
// 전용 STA 스레드에서 로그아웃(end_session) 창을 띄운다.
|
||||||
|
await LoginWindow.AuthenticateAsync(url, options.PostLogoutRedirectUri);
|
||||||
}
|
}
|
||||||
catch (OperationCanceledException)
|
catch (OperationCanceledException)
|
||||||
{
|
{
|
||||||
|
|||||||
Reference in New Issue
Block a user