Skip to content

Commit

Permalink
Merge branch 'main' into show_btn_with_perm
Browse files Browse the repository at this point in the history
  • Loading branch information
PiemP authored Mar 1, 2024
2 parents 704a6ab + 4847f89 commit 82c566d
Show file tree
Hide file tree
Showing 6 changed files with 213 additions and 2 deletions.
2 changes: 1 addition & 1 deletion src/OrchardCore.Build/Dependencies.props
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@
</PropertyGroup>

<ItemGroup>
<PackageManagement Include="AngleSharp" Version="1.1.0" />
<PackageManagement Include="AngleSharp" Version="1.1.1" />
<PackageManagement Include="AWSSDK.S3" Version="3.7.104.11" />
<PackageManagement Include="AWSSDK.Extensions.NETCore.Setup" Version="3.7.7" />
<PackageManagement Include="AWSSDK.SecurityToken" Version="3.7.101.60" />
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@
using OrchardCore.Users.Handlers;
using OrchardCore.Users.Indexes;
using OrchardCore.Users.Services;
using OrchardCore.Users.Core.Json;
using System.Text.Json;

namespace Microsoft.Extensions.DependencyInjection
{
Expand Down Expand Up @@ -55,6 +57,11 @@ public static IServiceCollection AddUsers(this IServiceCollection services)
services.AddScoped<ITwoFactorAuthenticationHandler, DefaultTwoFactorAuthenticationHandler>();
services.AddScoped<ITwoFactorAuthenticationHandlerCoordinator, DefaultTwoFactorAuthenticationHandlerCoordinator>();

services.Configure<JsonSerializerOptions>(options =>
{
options.Converters.Add(new LoginInfoJsonConverter());
});

return services;
}
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
using System;
using System.Text.Json;
using System.Text.Json.Nodes;
using System.Text.Json.Serialization;
using Microsoft.AspNetCore.Identity;

namespace OrchardCore.Users.Core.Json;

public class LoginInfoJsonConverter : JsonConverter<UserLoginInfo>
{
public static readonly LoginInfoJsonConverter Instance = new();

public override UserLoginInfo Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
{
var loginInfo = new UserLoginInfo(string.Empty, string.Empty, string.Empty);

while (reader.Read())
{
if (reader.TokenType == JsonTokenType.EndObject)
{
break;
}

if (reader.TokenType == JsonTokenType.PropertyName)
{
var propertyName = reader.GetString();
reader.Read();

switch (propertyName)
{
case nameof(UserLoginInfo.LoginProvider):
loginInfo.LoginProvider = reader.GetString();
break;
case nameof(UserLoginInfo.ProviderKey):
loginInfo.ProviderKey = reader.GetString();
break;
case nameof(UserLoginInfo.ProviderDisplayName):
loginInfo.ProviderDisplayName = reader.GetString();
break;
default:
break;
}
}
}

return loginInfo;
}

public override void Write(Utf8JsonWriter writer, UserLoginInfo objectToWrite, JsonSerializerOptions options)
{
writer.WriteStartObject();
writer.WriteString(nameof(UserLoginInfo.LoginProvider), objectToWrite.LoginProvider);
writer.WriteString(nameof(UserLoginInfo.ProviderKey), objectToWrite.ProviderKey);
writer.WriteString(nameof(UserLoginInfo.ProviderDisplayName), objectToWrite.ProviderDisplayName);
writer.WriteEndObject();
}
}
2 changes: 1 addition & 1 deletion src/docs/resources/libraries/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ The below table lists the different .NET libraries used in Orchard Core:

| Library | Usage | Version | License |
|--- | --- | --- | --- |
| [Angle Sharp](https://github.com/AngleSharp/AngleSharp) | Angle brackets parser library. | 1.1.0 |[MIT](https://github.com/AngleSharp/AngleSharp/blob/devel/LICENSE) |
| [Angle Sharp](https://github.com/AngleSharp/AngleSharp) | Angle brackets parser library. | 1.1.1 |[MIT](https://github.com/AngleSharp/AngleSharp/blob/devel/LICENSE) |
| [AWSSDK S3](https://github.com/aws/aws-sdk-net) | AWS SDK for .NET. | 3.7.104.11 |[Apache-2.0 license](https://github.com/aws/aws-sdk-net/blob/master/License.txt) |
| [AWSSDK SecurityToken](https://github.com/aws/aws-sdk-net) | AWS SDK for .NET. | 3.7.101.60 |[Apache-2.0 license](https://github.com/aws/aws-sdk-net/blob/master/License.txt) |
| [AWSSDK Extensions Setup](https://github.com/aws/aws-sdk-net) | AWS SDK for .NET. | 3.7.7 |[Apache-2.0 license](https://github.com/aws/aws-sdk-net/blob/master/License.txt) |
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
using System.Security.Cryptography;
using OrchardCore.ResourceManagement;
using OrchardCore.Resources;

namespace OrchardCore.Tests.Modules.OrchardCore.Resources;

public class SubResourceIntegrityTests
{
[Fact]
public async Task SavedSubResourceIntegritiesShouldMatchCurrentResources()
{
// Arrange
var resourceOptions = Options.Create(new ResourceOptions());
var httpContextAccessorMock = new Mock<IHttpContextAccessor>();
httpContextAccessorMock
.Setup(a => a.HttpContext)
.Returns(new DefaultHttpContext());
var configurationOptions = new ResourceManagementOptionsConfiguration(
resourceOptions,
Mock.Of<IHostEnvironment>(),
httpContextAccessorMock.Object);
var resourceManagementOptions = new ResourceManagementOptions();

// Act
configurationOptions.Configure(resourceManagementOptions);

// Assert
var resourceManifest = resourceManagementOptions.ResourceManifests.First();

using var httpClient = new HttpClient();
await ValidateSubResourceIntegrityAsync("script");
await ValidateSubResourceIntegrityAsync("style");

async Task ValidateSubResourceIntegrityAsync(string resourceType)
{
foreach (var resource in resourceManifest.GetResources(resourceType))
{
foreach (var resourceDefinition in resource.Value)
{
if (!string.IsNullOrEmpty(resourceDefinition.CdnIntegrity) && !string.IsNullOrEmpty(resourceDefinition.UrlCdnDebug))
{
var resourceIntegrity = await GetSubResourceIntegrityAsync(httpClient, resourceDefinition.UrlCdnDebug);

Assert.True(resourceIntegrity.Equals(resourceDefinition.CdnDebugIntegrity),
$"The {resourceType} {resourceDefinition.UrlCdnDebug} has invalid SRI hash, please use '{resourceIntegrity}' instead.");
}

if (!string.IsNullOrEmpty(resourceDefinition.CdnIntegrity) && !string.IsNullOrEmpty(resourceDefinition.UrlCdn))
{
var resourceIntegrity = await GetSubResourceIntegrityAsync(httpClient, resourceDefinition.UrlCdn);

Assert.True(resourceIntegrity.Equals(resourceDefinition.CdnIntegrity),
$"The {resourceType} {resourceDefinition.UrlCdn} has invalid SRI hash, please use '{resourceIntegrity}' instead.");
}
}
}
}
}

private static async Task<string> GetSubResourceIntegrityAsync(HttpClient httpClient, string url)
{
var data = await httpClient.GetByteArrayAsync(url);

using var memoryStream = new MemoryStream(data);
using var sha384Hash = SHA384.Create();
var hash = sha384Hash.ComputeHash(memoryStream);

return "sha384-" + Convert.ToBase64String(hash);
}
}
77 changes: 77 additions & 0 deletions test/OrchardCore.Tests/Serializers/JsonSerializerTests.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
using System.Text.Json;
using OrchardCore.Tests.Apis.Context;
using OrchardCore.Users.Core.Json;
using OrchardCore.Users.Indexes;
using OrchardCore.Users.Models;

namespace OrchardCore.Tests.Serializers;

public class JsonSerializerTests
{
private const string _userLoginInfo = "{\"LoginProvider\":\"OpenIdConnect\",\"ProviderKey\":\"abc\",\"ProviderDisplayName\":\"default\"}";

private readonly JsonSerializerOptions _options;

public JsonSerializerTests()
{
_options = new JsonSerializerOptions();
_options.Converters.Add(new LoginInfoJsonConverter());
}

[Fact]
public void Deserialize_WhenCalled_ReturnValidUserLoginInfo()
{
var obj = JsonSerializer.Deserialize<UserLoginInfo>(_userLoginInfo, _options);

Assert.Equal("OpenIdConnect", obj.LoginProvider);
Assert.Equal("abc", obj.ProviderKey);
Assert.Equal("default", obj.ProviderDisplayName);
}

[Fact]
public void Serialize_WhenCalled_ReturnValidJson()
{
var loginInfo = new UserLoginInfo("OpenIdConnect", "abc", "default");
var json = JsonSerializer.Serialize(loginInfo, _options);

Assert.Equal(_userLoginInfo, json);
}

[Fact]
public async Task DefaultContentSerializer_SerializeAndDeserialize_UserWithUserLoginInfo()
{
using var context = new SiteContext();
await context.InitializeAsync();
await context.UsingTenantScopeAsync(async scope =>
{
var loginInfo = new UserLoginInfo("OpenIdConnect", "abc", "default");

var newUser = new User()
{
UserId = "abc",
UserName = "mike",
Email = "test@test.com",
LoginInfos =
[
loginInfo
]
};

var session = scope.ServiceProvider.GetRequiredService<YesSql.ISession>();

await session.SaveAsync(newUser);
await session.SaveChangesAsync();

var dbUser = await session.Query<User, UserIndex>(x => x.UserId == "abc").FirstOrDefaultAsync();

Assert.NotNull(dbUser);

var userLoginInfo = dbUser.LoginInfos.FirstOrDefault();

Assert.NotNull(userLoginInfo);
Assert.Equal(loginInfo.LoginProvider, userLoginInfo.LoginProvider);
Assert.Equal(loginInfo.ProviderKey, userLoginInfo.ProviderKey);
Assert.Equal(loginInfo.ProviderDisplayName, userLoginInfo.ProviderDisplayName);
});
}
}

0 comments on commit 82c566d

Please sign in to comment.