Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Steeltoe Security Updates #1311

Merged
merged 28 commits into from
Jun 18, 2024
Merged
Show file tree
Hide file tree
Changes from 16 commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
6e36ed1
depend more heavily on Microsoft libraries for auth
TimHess Jun 11, 2024
9b6c3b7
cleanup, more validation and tests
TimHess Jun 11, 2024
faab613
adjust tests to call Steeltoe configure methods before AddAuth
TimHess Jun 11, 2024
363aabd
skip .NET 6 for app security component build
TimHess Jun 11, 2024
9a4b4f9
use default HeaderConverter for X-Client-Cert processing
TimHess Jun 11, 2024
681cbcb
formatting
TimHess Jun 11, 2024
2154543
Apply suggestions from code review
TimHess Jun 12, 2024
de03386
Rename Steeltoe.Common.Security to Steeltoe.Common.Certificate
TimHess Jun 12, 2024
9427171
PR feedback, more simplification
TimHess Jun 12, 2024
573d476
Apply suggestions from code review
TimHess Jun 13, 2024
3a81fa9
PR feedback
TimHess Jun 13, 2024
cad92ae
Apply suggestions from code review
TimHess Jun 14, 2024
89b9184
small fixes
TimHess Jun 14, 2024
efd3d3b
reduce sso product name specificity
TimHess Jun 14, 2024
60e4ae5
certificate subject parsing method signature change
TimHess Jun 14, 2024
dc67946
PR suggestions, move OIDC and JWT extensions to AuthenticationBuilder
TimHess Jun 14, 2024
23f72b7
Apply suggestions from code review
TimHess Jun 17, 2024
862510f
build fixes, revise s3900 again, disposable httpclient
TimHess Jun 17, 2024
0231cdc
drop certificates from autoconfiguration,
TimHess Jun 17, 2024
5b0db2c
certificateName explicitly nullable, fix Security.slnf
TimHess Jun 17, 2024
50454ee
rename test class
TimHess Jun 17, 2024
94ab5bf
revert more s3900
TimHess Jun 17, 2024
5cae4a5
cleanup
TimHess Jun 17, 2024
97e57f0
Apply suggestions from code review
TimHess Jun 18, 2024
34f04c6
pr feedback
TimHess Jun 18, 2024
208b1e3
style
TimHess Jun 18, 2024
5c8605e
update method description, don't run cert auth tests in parallel
TimHess Jun 18, 2024
cd01711
add missing word
TimHess Jun 18, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions build/security.yml
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,9 @@ jobs:
parameters:
component: Security
skipFilter: --filter "Category!=SkipOnLinux"
skipNET6: true
- template: templates/component-build.yaml
parameters:
component: Security
OS: windows
skipNET6: true
3 changes: 3 additions & 0 deletions build/templates/component-build.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ parameters:
component: ''
runConfigServer: false
runRabbitMQ: false
skipNET6: false
skipFilter: ''
OS: ubuntu

Expand All @@ -27,6 +28,7 @@ jobs:
- checkout: self
fetchDepth: 0
- task: UseDotNet@2
condition: ne(${{parameters.skipNET6}}, 'true')
displayName: Install .NET 6
inputs:
version: 6.0.x
Expand Down Expand Up @@ -68,6 +70,7 @@ jobs:
condition: eq(${{parameters.runConfigServer}}, 'true')
displayName: Start Config Server
- task: DotNetCoreCLI@2
condition: ne(${{parameters.skipNET6}}, 'true')
displayName: dotnet test 6.0
inputs:
command: test
Expand Down
10 changes: 7 additions & 3 deletions shared-package.props
Original file line number Diff line number Diff line change
Expand Up @@ -54,16 +54,20 @@
</Target>

<ItemGroup
Condition="$(MSBuildProjectName.StartsWith('Steeltoe.Configuration')) Or $(MSBuildProjectName.StartsWith('Steeltoe.Management')) Or $(MSBuildProjectName.StartsWith('Steeltoe.Connectors')) Or $(MSBuildProjectName.StartsWith('Steeltoe.Logging')) Or $(MSBuildProjectName.StartsWith('Steeltoe.Bootstrap')) Or $(MSBuildProjectName.StartsWith('Steeltoe.Discovery')) Or $(MSBuildProjectName.StartsWith('Steeltoe.Security.DataProtection'))">
Condition="$(MSBuildProjectName.StartsWith('Steeltoe.Bootstrap')) Or $(MSBuildProjectName.StartsWith('Steeltoe.Common.Certificates')) Or $(MSBuildProjectName.StartsWith('Steeltoe.Configuration')) Or
$(MSBuildProjectName.StartsWith('Steeltoe.Connectors')) Or $(MSBuildProjectName.StartsWith('Steeltoe.Discovery')) Or $(MSBuildProjectName.StartsWith('Steeltoe.Logging')) Or
$(MSBuildProjectName.StartsWith('Steeltoe.Management')) Or $(MSBuildProjectName.StartsWith('Steeltoe.Security'))">
bart-vmware marked this conversation as resolved.
Show resolved Hide resolved

<!-- Widen the condition above as we're completing more public API reviews -->
<PackageReference Include="Microsoft.CodeAnalysis.PublicApiAnalyzers" Version="$(PublicApiAnalyzersVersion)" PrivateAssets="All" />
</ItemGroup>

<PropertyGroup
Condition="!$(MSBuildProjectName.StartsWith('Steeltoe.Configuration')) And !$(MSBuildProjectName.StartsWith('Steeltoe.Management')) And !$(MSBuildProjectName.StartsWith('Steeltoe.Connectors')) And !$(MSBuildProjectName.StartsWith('Steeltoe.Logging')) And !$(MSBuildProjectName.StartsWith('Steeltoe.Bootstrap')) And !$(MSBuildProjectName.StartsWith('Steeltoe.Discovery')) And !$(MSBuildProjectName.StartsWith('Steeltoe.Security.DataProtection'))">
Condition="!$(MSBuildProjectName.StartsWith('Steeltoe.Bootstrap')) And !$(MSBuildProjectName.StartsWith('Steeltoe.Common.Certificates')) And !$(MSBuildProjectName.StartsWith('Steeltoe.Configuration')) And
!$(MSBuildProjectName.StartsWith('Steeltoe.Connectors')) And !$(MSBuildProjectName.StartsWith('Steeltoe.Discovery')) And !$(MSBuildProjectName.StartsWith('Steeltoe.Logging')) And
!$(MSBuildProjectName.StartsWith('Steeltoe.Management')) And !$(MSBuildProjectName.StartsWith('Steeltoe.Security'))">

<!-- Narrow the condition above as we're completing more public API reviews -->
<NoWarn>$(NoWarn);SA1401;S1168;S2360;S3900;S3956;S4004;S4023</NoWarn>
<NoWarn>$(NoWarn);SA1401;S1168;S2360;S3956;S4004;S4023</NoWarn>
</PropertyGroup>
</Project>
2 changes: 1 addition & 1 deletion shared-test.props
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
<Project>
<PropertyGroup>
<NoWarn>$(NoWarn);S2094;NU5104</NoWarn>
<NoWarn>$(NoWarn);S2094;S3900;NU5104</NoWarn>
</PropertyGroup>

<ItemGroup>
Expand Down
12 changes: 6 additions & 6 deletions src/Bootstrap/src/AutoConfiguration/BootstrapScanner.cs
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.Logging;
using Steeltoe.Common;
using Steeltoe.Common.Certificates;
using Steeltoe.Common.DynamicTypeAccess;
using Steeltoe.Common.Hosting;
using Steeltoe.Common.Logging;
Expand Down Expand Up @@ -35,7 +36,6 @@
using Steeltoe.Management.Tracing;
using Steeltoe.Management.Wavefront;
using Steeltoe.Management.Wavefront.Exporters;
using Steeltoe.Security.Authentication.CloudFoundry;

namespace Steeltoe.Bootstrap.AutoConfiguration;

Expand Down Expand Up @@ -81,7 +81,7 @@ public void ConfigureSteeltoe()
WireIfLoaded(WirePrometheus, SteeltoeAssemblyNames.ManagementPrometheus);
WireIfLoaded(WireWavefrontMetrics, SteeltoeAssemblyNames.ManagementWavefront);
WireIfLoaded(WireDistributedTracing, SteeltoeAssemblyNames.ManagementTracing);
WireIfLoaded(WireCloudFoundryContainerIdentity, SteeltoeAssemblyNames.SecurityAuthenticationCloudFoundry);
WireIfLoaded(WireAppInstanceIdentity, SteeltoeAssemblyNames.CommonCertificates);
}

private void WireConfigServer()
Expand Down Expand Up @@ -256,12 +256,12 @@ private void WireDistributedTracing()
_logger.LogInformation("Configured distributed tracing");
}

private void WireCloudFoundryContainerIdentity()
private void WireAppInstanceIdentity()
{
_wrapper.ConfigureAppConfiguration(configurationBuilder => configurationBuilder.AddCloudFoundryContainerIdentity());
_wrapper.ConfigureServices(services => services.AddCloudFoundryCertificateAuth());
_wrapper.ConfigureAppConfiguration(configurationBuilder => configurationBuilder.AddAppInstanceIdentityCertificate());
_wrapper.ConfigureServices(services => services.ConfigureCertificateOptions("AppInstanceIdentity", null));
bart-vmware marked this conversation as resolved.
Show resolved Hide resolved

_logger.LogInformation("Configured Cloud Foundry mTLS security");
_logger.LogInformation("Configured application instance identity certificate");
}

private bool WireIfLoaded(Action wireAction, string assemblyName)
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
#nullable enable
const Steeltoe.Bootstrap.AutoConfiguration.SteeltoeAssemblyNames.CommonCertificates = "Steeltoe.Common.Certificates" -> string!
const Steeltoe.Bootstrap.AutoConfiguration.SteeltoeAssemblyNames.ConfigurationCloudFoundry = "Steeltoe.Configuration.CloudFoundry" -> string!
const Steeltoe.Bootstrap.AutoConfiguration.SteeltoeAssemblyNames.ConfigurationConfigServer = "Steeltoe.Configuration.ConfigServer" -> string!
const Steeltoe.Bootstrap.AutoConfiguration.SteeltoeAssemblyNames.ConfigurationPlaceholder = "Steeltoe.Configuration.Placeholder" -> string!
Expand All @@ -12,7 +13,6 @@ const Steeltoe.Bootstrap.AutoConfiguration.SteeltoeAssemblyNames.ManagementEndpo
const Steeltoe.Bootstrap.AutoConfiguration.SteeltoeAssemblyNames.ManagementPrometheus = "Steeltoe.Management.Prometheus" -> string!
const Steeltoe.Bootstrap.AutoConfiguration.SteeltoeAssemblyNames.ManagementTracing = "Steeltoe.Management.Tracing" -> string!
const Steeltoe.Bootstrap.AutoConfiguration.SteeltoeAssemblyNames.ManagementWavefront = "Steeltoe.Management.Wavefront" -> string!
const Steeltoe.Bootstrap.AutoConfiguration.SteeltoeAssemblyNames.SecurityAuthenticationCloudFoundry = "Steeltoe.Security.Authentication.CloudFoundry" -> string!
static Steeltoe.Bootstrap.AutoConfiguration.HostBuilderExtensions.AddSteeltoe(this Microsoft.Extensions.Hosting.IHostBuilder! builder) -> Microsoft.Extensions.Hosting.IHostBuilder!
static Steeltoe.Bootstrap.AutoConfiguration.HostBuilderExtensions.AddSteeltoe(this Microsoft.Extensions.Hosting.IHostBuilder! builder, Microsoft.Extensions.Logging.ILoggerFactory! loggerFactory) -> Microsoft.Extensions.Hosting.IHostBuilder!
static Steeltoe.Bootstrap.AutoConfiguration.HostBuilderExtensions.AddSteeltoe(this Microsoft.Extensions.Hosting.IHostBuilder! builder, System.Collections.Generic.IReadOnlySet<string!>! assemblyNamesToExclude) -> Microsoft.Extensions.Hosting.IHostBuilder!
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -14,8 +14,9 @@
</ItemGroup>

<ItemGroup>
<ProjectReference Include="..\..\..\Common\src\Common.Hosting\Steeltoe.Common.Hosting.csproj" />
<ProjectReference Include="..\..\..\Common\src\Common\Steeltoe.Common.csproj" />
<ProjectReference Include="..\..\..\Common\src\Common.Hosting\Steeltoe.Common.Hosting.csproj" />
<ProjectReference Include="..\..\..\Common\src\Common.Certificates\Steeltoe.Common.Certificates.csproj" PrivateAssets="All" />
<ProjectReference Include="..\..\..\Configuration\src\CloudFoundry\Steeltoe.Configuration.CloudFoundry.csproj" PrivateAssets="All" />
<ProjectReference Include="..\..\..\Configuration\src\ConfigServer\Steeltoe.Configuration.ConfigServer.csproj" PrivateAssets="All" />
<ProjectReference Include="..\..\..\Configuration\src\RandomValue\Steeltoe.Configuration.RandomValue.csproj" PrivateAssets="All" />
Expand All @@ -30,10 +31,7 @@
<ProjectReference Include="..\..\..\Management\src\Task\Steeltoe.Management.Task.csproj" PrivateAssets="All" />
<ProjectReference Include="..\..\..\Management\src\Tracing\Steeltoe.Management.Tracing.csproj" PrivateAssets="All" />
<ProjectReference Include="..\..\..\Management\src\Wavefront\Steeltoe.Management.Wavefront.csproj" PrivateAssets="All" />
<ProjectReference Include="..\..\..\Security\src\Authentication.CloudFoundry\Steeltoe.Security.Authentication.CloudFoundry.csproj" PrivateAssets="All" />
<ProjectReference Include="..\..\..\Security\src\Authentication.Mtls\Steeltoe.Security.Authentication.Mtls.csproj" PrivateAssets="All" />
<ProjectReference Include="..\..\..\Security\src\DataProtection.Redis\Steeltoe.Security.DataProtection.Redis.csproj" PrivateAssets="All" />
<ProjectReference Include="..\..\..\Connectors\src\CloudFoundry\Steeltoe.Connectors.CloudFoundry.csproj" PrivateAssets="All" />
<ProjectReference Include="..\..\..\Logging\src\DynamicLogger\Steeltoe.Logging.DynamicLogger.csproj" PrivateAssets="All" />
</ItemGroup>

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ namespace Steeltoe.Bootstrap.AutoConfiguration;
/// </summary>
public static class SteeltoeAssemblyNames
{
public const string CommonCertificates = "Steeltoe.Common.Certificates";
public const string ConfigurationCloudFoundry = "Steeltoe.Configuration.CloudFoundry";
public const string ConfigurationConfigServer = "Steeltoe.Configuration.ConfigServer";
public const string ConfigurationRandomValue = "Steeltoe.Configuration.RandomValue";
Expand All @@ -24,7 +25,6 @@ public static class SteeltoeAssemblyNames
public const string ManagementPrometheus = "Steeltoe.Management.Prometheus";
public const string ManagementTracing = "Steeltoe.Management.Tracing";
public const string ManagementWavefront = "Steeltoe.Management.Wavefront";
public const string SecurityAuthenticationCloudFoundry = "Steeltoe.Security.Authentication.CloudFoundry";

internal static readonly IReadOnlySet<string> All = typeof(SteeltoeAssemblyNames).GetFields().Where(field => field.FieldType == typeof(string))
.Select(field => field.GetValue(null)).Cast<string>().ToHashSet();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@

using System.Reflection;
using FluentAssertions;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.AspNetCore.TestHost;
Expand Down Expand Up @@ -227,17 +226,16 @@ public void Tracing_IsAutowired()
}

[Fact]
public void CloudFoundryContainerSecurity_IsAutowired()
public void AppInstanceIdentityCertificate_IsAutowired()
{
using IHost host = GetHostForOnly(SteeltoeAssemblyNames.SecurityAuthenticationCloudFoundry);
using IHost host = GetHostForOnly(SteeltoeAssemblyNames.CommonCertificates);
var configuration = host.Services.GetRequiredService<IConfiguration>();

configuration[$"{CertificateOptions.ConfigurationKeyPrefix}:AppInstanceIdentity:CertificateFilePath"].Should().NotBeNull();
configuration[$"{CertificateOptions.ConfigurationKeyPrefix}:AppInstanceIdentity:PrivateKeyFilePath"].Should().NotBeNull();

host.Services.GetService<IOptionsMonitor<CertificateOptions>>()?.Get("AppInstanceIdentity").Certificate.Should().NotBeNull();
host.Services.GetService<IOptionsChangeTokenSource<CertificateOptions>>().Should().NotBeNull();
host.Services.GetService<IAuthorizationHandler>().Should().NotBeNull();
}

private static IHost GetHostForOnly(string assemblyNameToInclude)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,15 +7,16 @@
<Import Project="..\..\..\..\shared.props" />

<ItemGroup>
<ProjectReference Include="..\..\..\Common\src\Common.Certificates\Steeltoe.Common.Certificates.csproj" />
bart-vmware marked this conversation as resolved.
Show resolved Hide resolved
<ProjectReference Include="..\..\..\Configuration\src\ConfigServer\Steeltoe.Configuration.ConfigServer.csproj" />
<ProjectReference Include="..\..\..\Configuration\src\RandomValue\Steeltoe.Configuration.RandomValue.csproj" />
<ProjectReference Include="..\..\..\Connectors\src\Connectors\Steeltoe.Connectors.csproj" />
<ProjectReference Include="..\..\..\Discovery\src\Configuration\Steeltoe.Discovery.Configuration.csproj" />
<ProjectReference Include="..\..\..\Discovery\src\Consul\Steeltoe.Discovery.Consul.csproj" />
<ProjectReference Include="..\..\..\Discovery\src\Eureka\Steeltoe.Discovery.Eureka.csproj" />
<ProjectReference Include="..\..\..\Logging\src\DynamicSerilog\Steeltoe.Logging.DynamicSerilog.csproj" />
<ProjectReference Include="..\..\..\Management\src\Endpoint\Steeltoe.Management.Endpoint.csproj" />
<ProjectReference Include="..\..\..\Management\src\Tracing\Steeltoe.Management.Tracing.csproj" />
<ProjectReference Include="..\..\..\Security\src\Authentication.CloudFoundry\Steeltoe.Security.Authentication.CloudFoundry.csproj" />
<ProjectReference Include="..\..\src\AutoConfiguration\Steeltoe.Bootstrap.AutoConfiguration.csproj" />
</ItemGroup>

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@
using System.Net;
using System.Reflection;
using FluentAssertions;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.AspNetCore.TestHost;
Expand Down Expand Up @@ -228,17 +227,16 @@ public void Tracing_IsAutowired()
}

[Fact]
public void CloudFoundryContainerSecurity_IsAutowired()
public void AppInstanceIdentityCertificate_IsAutowired()
{
using WebApplication host = GetWebApplicationForOnly(SteeltoeAssemblyNames.SecurityAuthenticationCloudFoundry);
using WebApplication host = GetWebApplicationForOnly(SteeltoeAssemblyNames.CommonCertificates);
var configuration = host.Services.GetRequiredService<IConfiguration>();

configuration[$"{CertificateOptions.ConfigurationKeyPrefix}:AppInstanceIdentity:CertificateFilePath"].Should().NotBeNull();
configuration[$"{CertificateOptions.ConfigurationKeyPrefix}:AppInstanceIdentity:PrivateKeyFilePath"].Should().NotBeNull();

host.Services.GetService<IOptionsMonitor<CertificateOptions>>()?.Get("AppInstanceIdentity").Certificate.Should().NotBeNull();
host.Services.GetService<IOptionsChangeTokenSource<CertificateOptions>>().Should().NotBeNull();
host.Services.GetService<IAuthorizationHandler>().Should().NotBeNull();
}

private static WebApplication GetWebApplicationForOnly(string assemblyNameToInclude)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@
using System.Reflection;
using FluentAssertions;
using Microsoft.AspNetCore;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.AspNetCore.TestHost;
Expand Down Expand Up @@ -228,17 +227,16 @@ public void Tracing_IsAutowired()
}

[Fact]
public void CloudFoundryContainerSecurity_IsAutowired()
public void AppInstanceIdentityCertificate_IsAutowired()
{
using IWebHost host = GetWebHostForOnly(SteeltoeAssemblyNames.SecurityAuthenticationCloudFoundry);
using IWebHost host = GetWebHostForOnly(SteeltoeAssemblyNames.CommonCertificates);
var configuration = host.Services.GetRequiredService<IConfiguration>();

configuration[$"{CertificateOptions.ConfigurationKeyPrefix}:AppInstanceIdentity:CertificateFilePath"].Should().NotBeNull();
configuration[$"{CertificateOptions.ConfigurationKeyPrefix}:AppInstanceIdentity:PrivateKeyFilePath"].Should().NotBeNull();

host.Services.GetService<IOptionsMonitor<CertificateOptions>>()?.Get("AppInstanceIdentity").Certificate.Should().NotBeNull();
host.Services.GetService<IOptionsChangeTokenSource<CertificateOptions>>().Should().NotBeNull();
host.Services.GetService<IAuthorizationHandler>().Should().NotBeNull();
}

private static IWebHost GetWebHostForOnly(string assemblyNameToInclude)
Expand Down
4 changes: 2 additions & 2 deletions src/Common/src/Abstractions/Properties/AssemblyInfo.cs
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,8 @@

[assembly: InternalsVisibleTo("Steeltoe.Bootstrap.AutoConfiguration")]
[assembly: InternalsVisibleTo("Steeltoe.Bootstrap.AutoConfiguration.Test")]
[assembly: InternalsVisibleTo("Steeltoe.Common.Security")]
[assembly: InternalsVisibleTo("Steeltoe.Common.Security.Test")]
[assembly: InternalsVisibleTo("Steeltoe.Common.Certificates")]
[assembly: InternalsVisibleTo("Steeltoe.Common.Certificates.Test")]
[assembly: InternalsVisibleTo("Steeltoe.Configuration.ConfigServer")]
[assembly: InternalsVisibleTo("Steeltoe.Connectors")]
[assembly: InternalsVisibleTo("Steeltoe.Connectors.EntityFrameworkCore")]
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
// Licensed to the .NET Foundation under one or more agreements.
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the Apache 2.0 License.
// See the LICENSE file in the project root for more information.

Expand All @@ -20,6 +20,8 @@ public sealed class SnakeCaseAllCapsEnumMemberJsonConverter : JsonConverterFacto
/// <inheritdoc />
public override bool CanConvert(Type typeToConvert)
{
ArgumentGuard.NotNull(typeToConvert);

return typeToConvert.IsEnum;
}

Expand Down

This file was deleted.

Loading
Loading