From 60c48ce95215f9065f8b6196d0d5d54ba334e1f3 Mon Sep 17 00:00:00 2001 From: James Chapman Date: Sun, 12 Nov 2023 01:36:32 +0000 Subject: [PATCH] Fix form submissions failing when using OAuth --- Groceries/Program.cs | 26 ++++++++++++++++++++++++++ Groceries/config.ini | 2 ++ 2 files changed, 28 insertions(+) diff --git a/Groceries/Program.cs b/Groceries/Program.cs index 10d5c60..706bde2 100644 --- a/Groceries/Program.cs +++ b/Groceries/Program.cs @@ -8,6 +8,9 @@ using Microsoft.AspNetCore.Identity; using Microsoft.AspNetCore.Mvc.Infrastructure; using Microsoft.AspNetCore.Mvc.Razor; using Microsoft.EntityFrameworkCore; +using System.Net.Http.Headers; +using System.Security.Claims; +using System.Text.Json; var builder = WebApplication.CreateBuilder(args); var env = builder.Environment; @@ -63,6 +66,29 @@ if (oauthConfig.Exists()) { options.SignInScheme = IdentityConstants.ExternalScheme; options.CallbackPath = "/signin"; + + foreach (string scope in (oauthConfig["Scopes"] ?? string.Empty).Split(',', StringSplitOptions.RemoveEmptyEntries)) + { + options.Scope.Add(scope.Trim()); + } + + // Antiforgery token generation requires authenticated identities to have claims + // and the default OAuth authentication handler does not fetch user info. + options.Events.OnCreatingTicket = async context => + { + using var request = new HttpRequestMessage(HttpMethod.Get, context.Options.UserInformationEndpoint); + request.Headers.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json")); + request.Headers.Authorization = new AuthenticationHeaderValue(context.TokenType ?? "Bearer", context.AccessToken); + + using var response = await context.Backchannel.SendAsync(request, HttpCompletionOption.ResponseHeadersRead, context.HttpContext.RequestAborted); + response.EnsureSuccessStatusCode(); + + using var payload = JsonDocument.Parse(await response.Content.ReadAsStreamAsync(context.HttpContext.RequestAborted)); + context.Identity!.AddClaims( + payload.RootElement.EnumerateObject() + .Where(property => property.Value.ValueKind is JsonValueKind.String or JsonValueKind.Number) + .Select(property => new Claim(property.Name, property.Value.ToString()))); + }; }) .AddExternalCookie(); diff --git a/Groceries/config.ini b/Groceries/config.ini index a83ec50..8f3a9bb 100644 --- a/Groceries/config.ini +++ b/Groceries/config.ini @@ -7,6 +7,8 @@ Microsoft = Warning [OAuth] ;AuthorizationEndpoint = ;TokenEndpoint = +;UserInformationEndpoint = ;ClientId = ;ClientSecret = ;UsePkce = false +;Scopes = scope1, scope2