重构单元测试和集成测试

This commit is contained in:
hello 2025-03-15 17:33:37 +08:00
parent e7e9afe11b
commit 81c8f608dc
34 changed files with 135 additions and 300 deletions

View File

@ -33,8 +33,6 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "HelloShop.FunctionalTests",
EndProject EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "HelloShop.BasketService.UnitTests", "tests\HelloShop.BasketService.UnitTests\HelloShop.BasketService.UnitTests.csproj", "{BE88233A-D6EB-462B-B53C-B588A0BEFAFC}" Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "HelloShop.BasketService.UnitTests", "tests\HelloShop.BasketService.UnitTests\HelloShop.BasketService.UnitTests.csproj", "{BE88233A-D6EB-462B-B53C-B588A0BEFAFC}"
EndProject EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "HelloShop.BasketService.FunctionalTests", "tests\HelloShop.BasketService.FunctionalTests\HelloShop.BasketService.FunctionalTests.csproj", "{A0903D4D-EA4E-433A-AC5B-BE6ED4A5C958}"
EndProject
Global Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU Debug|Any CPU = Debug|Any CPU
@ -95,10 +93,6 @@ Global
{BE88233A-D6EB-462B-B53C-B588A0BEFAFC}.Debug|Any CPU.Build.0 = Debug|Any CPU {BE88233A-D6EB-462B-B53C-B588A0BEFAFC}.Debug|Any CPU.Build.0 = Debug|Any CPU
{BE88233A-D6EB-462B-B53C-B588A0BEFAFC}.Release|Any CPU.ActiveCfg = Release|Any CPU {BE88233A-D6EB-462B-B53C-B588A0BEFAFC}.Release|Any CPU.ActiveCfg = Release|Any CPU
{BE88233A-D6EB-462B-B53C-B588A0BEFAFC}.Release|Any CPU.Build.0 = Release|Any CPU {BE88233A-D6EB-462B-B53C-B588A0BEFAFC}.Release|Any CPU.Build.0 = Release|Any CPU
{A0903D4D-EA4E-433A-AC5B-BE6ED4A5C958}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{A0903D4D-EA4E-433A-AC5B-BE6ED4A5C958}.Debug|Any CPU.Build.0 = Debug|Any CPU
{A0903D4D-EA4E-433A-AC5B-BE6ED4A5C958}.Release|Any CPU.ActiveCfg = Release|Any CPU
{A0903D4D-EA4E-433A-AC5B-BE6ED4A5C958}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection EndGlobalSection
GlobalSection(SolutionProperties) = preSolution GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE HideSolutionNode = FALSE
@ -117,7 +111,6 @@ Global
{E58F82E2-2E48-459B-A40E-497F24FC6DC1} = {1AD03316-A743-4E9D-B3BC-FB9499D15141} {E58F82E2-2E48-459B-A40E-497F24FC6DC1} = {1AD03316-A743-4E9D-B3BC-FB9499D15141}
{6BAA9747-E0D0-41B9-8A1B-88B777498C43} = {29BE158E-825E-48AB-A02D-4E537A5DC502} {6BAA9747-E0D0-41B9-8A1B-88B777498C43} = {29BE158E-825E-48AB-A02D-4E537A5DC502}
{BE88233A-D6EB-462B-B53C-B588A0BEFAFC} = {29BE158E-825E-48AB-A02D-4E537A5DC502} {BE88233A-D6EB-462B-B53C-B588A0BEFAFC} = {29BE158E-825E-48AB-A02D-4E537A5DC502}
{A0903D4D-EA4E-433A-AC5B-BE6ED4A5C958} = {29BE158E-825E-48AB-A02D-4E537A5DC502}
EndGlobalSection EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {845545A8-2006-46C3-ABD7-5BDF63F3858C} SolutionGuid = {845545A8-2006-46C3-ABD7-5BDF63F3858C}

View File

@ -3,11 +3,6 @@
using CommunityToolkit.Aspire.Hosting.Dapr; using CommunityToolkit.Aspire.Hosting.Dapr;
using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.DependencyInjection;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace HelloShop.AppHost.Extensions namespace HelloShop.AppHost.Extensions
{ {

View File

@ -14,7 +14,6 @@
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<PackageReference Include="Aspire.StackExchange.Redis.DistributedCaching" Version="9.1.0" /> <PackageReference Include="Aspire.StackExchange.Redis.DistributedCaching" Version="9.1.0" />
<PackageReference Include="Calzolari.Grpc.AspNetCore.Validation" Version="8.1.0" />
<PackageReference Include="Grpc.AspNetCore" Version="2.70.0" /> <PackageReference Include="Grpc.AspNetCore" Version="2.70.0" />
<PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="9.0.3" /> <PackageReference Include="Microsoft.AspNetCore.Authentication.JwtBearer" Version="9.0.3" />
<PackageReference Include="Microsoft.AspNetCore.Grpc.Swagger" Version="0.9.3" /> <PackageReference Include="Microsoft.AspNetCore.Grpc.Swagger" Version="0.9.3" />

View File

@ -1,7 +1,6 @@
// Copyright (c) HelloShop Corporation. All rights reserved. // Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information. // See the license file in the project root for more information.
using Calzolari.Grpc.AspNetCore.Validation;
using HelloShop.BasketService.DistributedEvents.EventHandling; using HelloShop.BasketService.DistributedEvents.EventHandling;
using HelloShop.BasketService.DistributedEvents.Events; using HelloShop.BasketService.DistributedEvents.Events;
using HelloShop.BasketService.Repositories; using HelloShop.BasketService.Repositories;
@ -31,7 +30,7 @@ builder.Services.AddHttpContextAccessor();
builder.AddRedisDistributedCache("cache"); builder.AddRedisDistributedCache("cache");
builder.Services.AddSingleton<IBasketRepository, DistributedCacheBasketRepository>(); builder.Services.AddSingleton<IBasketRepository, DistributedCacheBasketRepository>();
builder.Services.AddGrpc(options => options.EnableMessageValidation()).AddJsonTranscoding(); builder.Services.AddGrpc().AddJsonTranscoding();
builder.Services.AddGrpcSwagger(); builder.Services.AddGrpcSwagger();
builder.Services.AddOpenApi(); builder.Services.AddOpenApi();
@ -40,7 +39,6 @@ builder.Services.AddCustomLocalization();
builder.Services.AddModelMapper().AddModelValidator(); builder.Services.AddModelMapper().AddModelValidator();
builder.Services.AddLocalization().AddPermissionDefinitions(); builder.Services.AddLocalization().AddPermissionDefinitions();
builder.Services.AddAuthorization().AddRemotePermissionChecker().AddCustomAuthorization(); builder.Services.AddAuthorization().AddRemotePermissionChecker().AddCustomAuthorization();
builder.Services.AddGrpcValidation();
builder.Services.AddCors(); builder.Services.AddCors();
builder.AddDaprDistributedEventBus().AddSubscription<OrderStartedDistributedEvent, OrderStartedDistributedEventHandler>(); builder.AddDaprDistributedEventBus().AddSubscription<OrderStartedDistributedEvent, OrderStartedDistributedEventHandler>();

View File

@ -2,6 +2,7 @@
// See the license file in the project root for more information. // See the license file in the project root for more information.
using AutoMapper; using AutoMapper;
using FluentValidation;
using Google.Protobuf.WellKnownTypes; using Google.Protobuf.WellKnownTypes;
using Grpc.Core; using Grpc.Core;
using HelloShop.BasketService.Entities; using HelloShop.BasketService.Entities;
@ -18,7 +19,7 @@ using System.Security.Claims;
namespace HelloShop.BasketService.Services namespace HelloShop.BasketService.Services
{ {
[Authorize] [Authorize]
public class CustomerBasketService(IBasketRepository repository, ILogger<CustomerBasketService> logger, IMapper mapper) : Basket.BasketBase public class CustomerBasketService(IBasketRepository repository, ILogger<CustomerBasketService> logger, IMapper mapper, IValidator<UpdateBasketRequest> validator) : Basket.BasketBase
{ {
public override async Task<CustomerBasketResponse> GetBasket(Empty request, ServerCallContext context) public override async Task<CustomerBasketResponse> GetBasket(Empty request, ServerCallContext context)
{ {
@ -35,7 +36,7 @@ namespace HelloShop.BasketService.Services
if (basket is not null) if (basket is not null)
{ {
mapper.Map<CustomerBasketResponse>(basket); return mapper.Map<CustomerBasketResponse>(basket);
} }
return new(); return new();
@ -43,6 +44,12 @@ namespace HelloShop.BasketService.Services
public override async Task<CustomerBasketResponse> UpdateBasket(UpdateBasketRequest request, ServerCallContext context) public override async Task<CustomerBasketResponse> UpdateBasket(UpdateBasketRequest request, ServerCallContext context)
{ {
var validationResult = await validator.ValidateAsync(request);
if (!validationResult.IsValid)
{
throw new RpcException(new Status(StatusCode.InvalidArgument, validationResult.ToString()));
}
string? nameIdentifier = context.GetHttpContext().User.FindFirst(ClaimTypes.NameIdentifier)?.Value; string? nameIdentifier = context.GetHttpContext().User.FindFirst(ClaimTypes.NameIdentifier)?.Value;
if (!int.TryParse(nameIdentifier, out int userId)) if (!int.TryParse(nameIdentifier, out int userId))

View File

@ -3,7 +3,6 @@
using Android.App; using Android.App;
using Android.Content.PM; using Android.Content.PM;
using Microsoft.Maui;
namespace HelloShop.HybridApp namespace HelloShop.HybridApp
{ {

View File

@ -2,7 +2,6 @@
// See the license file in the project root for more information. // See the license file in the project root for more information.
using Foundation; using Foundation;
using Microsoft.Maui;
namespace HelloShop.HybridApp namespace HelloShop.HybridApp
{ {

View File

@ -2,7 +2,6 @@
// See the license file in the project root for more information. // See the license file in the project root for more information.
using Foundation; using Foundation;
using Microsoft.Maui;
namespace HelloShop.HybridApp namespace HelloShop.HybridApp
{ {

View File

@ -6,7 +6,6 @@ using HelloShop.IdentityService.Infrastructure;
using HelloShop.ServiceDefaults.Infrastructure; using HelloShop.ServiceDefaults.Infrastructure;
using Microsoft.AspNetCore.Identity; using Microsoft.AspNetCore.Identity;
using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.DependencyInjection;
namespace HelloShop.IdentityService.DataSeeding namespace HelloShop.IdentityService.DataSeeding
{ {

View File

@ -1,4 +1,6 @@
using System; // Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information.
using Microsoft.EntityFrameworkCore.Migrations; using Microsoft.EntityFrameworkCore.Migrations;
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata; using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;

View File

@ -23,7 +23,7 @@ builder.Services.AddControllers();
builder.Services.AddDbContext<IdentityServiceDbContext>(options => builder.Services.AddDbContext<IdentityServiceDbContext>(options =>
{ {
options.UseNpgsql(builder.Configuration.GetConnectionString(DbConstants.ConnectionStringName),x=>x.MigrationsHistoryTable(DbConstants.MigrationsHistoryTableName)); options.UseNpgsql(builder.Configuration.GetConnectionString(DbConstants.ConnectionStringName), x => x.MigrationsHistoryTable(DbConstants.MigrationsHistoryTableName));
}); });
builder.Services.AddIdentity<User, Role>(options => builder.Services.AddIdentity<User, Role>(options =>

View File

@ -8,7 +8,6 @@ using HelloShop.OrderingService.Entities.Orders;
using HelloShop.OrderingService.Infrastructure; using HelloShop.OrderingService.Infrastructure;
using HelloShop.OrderingService.LocalEvents; using HelloShop.OrderingService.LocalEvents;
using HelloShop.OrderingService.Services; using HelloShop.OrderingService.Services;
using HelloShop.ServiceDefaults.DistributedEvents.Abstractions;
using MediatR; using MediatR;
using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore;

View File

@ -2,10 +2,10 @@
// See the license file in the project root for more information. // See the license file in the project root for more information.
using HelloShop.OrderingService.Entities.EventLogs; using HelloShop.OrderingService.Entities.EventLogs;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
using Microsoft.EntityFrameworkCore;
using System.Text.Json;
using HelloShop.ServiceDefaults.DistributedEvents.Abstractions; using HelloShop.ServiceDefaults.DistributedEvents.Abstractions;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
using System.Text.Json;
namespace HelloShop.OrderingService.Infrastructure.EntityConfigurations.EventLogs namespace HelloShop.OrderingService.Infrastructure.EntityConfigurations.EventLogs
{ {

View File

@ -1,4 +1,6 @@
using System; // Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information.
using Microsoft.EntityFrameworkCore.Migrations; using Microsoft.EntityFrameworkCore.Migrations;
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata; using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;

View File

@ -1,4 +1,6 @@
using System; // Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information.
using Microsoft.EntityFrameworkCore.Migrations; using Microsoft.EntityFrameworkCore.Migrations;
using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata; using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;

View File

@ -8,6 +8,8 @@ using HelloShop.ServiceDefaults.DistributedEvents.DaprBuildingBlocks;
using HelloShop.ServiceDefaults.DistributedLocks; using HelloShop.ServiceDefaults.DistributedLocks;
using HelloShop.ServiceDefaults.Extensions; using HelloShop.ServiceDefaults.Extensions;
using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore;
using Microsoft.IdentityModel.Tokens;
using System.Text;
var builder = WebApplication.CreateBuilder(args); var builder = WebApplication.CreateBuilder(args);
@ -17,6 +19,15 @@ builder.AddServiceDefaults();
builder.Services.AddControllers(); builder.Services.AddControllers();
const string issuerSigningKey = HelloShop.ServiceDefaults.Constants.IdentityConstants.IssuerSigningKey;
builder.Services.AddAuthentication().AddJwtBearer(options =>
{
options.TokenValidationParameters.ValidateIssuer = false;
options.TokenValidationParameters.ValidateAudience = false;
options.TokenValidationParameters.IssuerSigningKey = new SymmetricSecurityKey(Encoding.Default.GetBytes(issuerSigningKey));
});
// Add extensions services to the container. // Add extensions services to the container.
builder.Services.AddDbContext<ProductServiceDbContext>(options => builder.Services.AddDbContext<ProductServiceDbContext>(options =>
{ {

View File

@ -2,12 +2,7 @@
// See the license file in the project root for more information. // See the license file in the project root for more information.
using Dapr.Client; using Dapr.Client;
using System;
using System.Collections.Generic;
using System.Diagnostics; using System.Diagnostics;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace HelloShop.ServiceDefaults.DistributedLocks namespace HelloShop.ServiceDefaults.DistributedLocks
{ {

View File

@ -2,11 +2,6 @@
// See the license file in the project root for more information. // See the license file in the project root for more information.
using Dapr.Client; using Dapr.Client;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace HelloShop.ServiceDefaults.DistributedLocks namespace HelloShop.ServiceDefaults.DistributedLocks
{ {

View File

@ -1,12 +1,6 @@
// Copyright (c) HelloShop Corporation. All rights reserved. // Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information. // See the license file in the project root for more information.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace HelloShop.ServiceDefaults.DistributedLocks namespace HelloShop.ServiceDefaults.DistributedLocks
{ {
public interface IDistributedLock public interface IDistributedLock

View File

@ -1,12 +1,6 @@
// Copyright (c) HelloShop Corporation. All rights reserved. // Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information. // See the license file in the project root for more information.
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace HelloShop.ServiceDefaults.DistributedLocks namespace HelloShop.ServiceDefaults.DistributedLocks
{ {
public interface IDistributedLockResult : IAsyncDisposable { } public interface IDistributedLockResult : IAsyncDisposable { }

View File

@ -1,25 +0,0 @@
// Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information.
using Grpc.Core;
using Grpc.Core.Interceptors;
using Microsoft.Net.Http.Headers;
namespace HelloShop.BasketService.FunctionalTests
{
internal class AuthenticatedInterceptor : Interceptor
{
public override AsyncUnaryCall<TResponse> AsyncUnaryCall<TRequest, TResponse>(TRequest request, ClientInterceptorContext<TRequest, TResponse> context, AsyncUnaryCallContinuation<TRequest, TResponse> continuation)
{
const string token = "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJuYW1laWQiOiIxIiwidW5pcXVlX25hbWUiOiJhZG1pbiIsInJvbGVpZCI6IjEiLCJuYmYiOjE3MjA1NzY3NDYsImV4cCI6MTc0MjYwODc0NiwiaWF0IjoxNzIwNTc2NzQ2fQ.ju_D3zeGLKqJYVckbb8Y3yNkp40nOqRAJrdOsISs4d4";
Metadata headers = [new Metadata.Entry(HeaderNames.Authorization, $"Bearer {token}")];
var newOptions = context.Options.WithHeaders(headers);
var newContext = new ClientInterceptorContext<TRequest, TResponse>(context.Method, context.Host, newOptions);
return base.AsyncUnaryCall(request, newContext, continuation);
}
}
}

View File

@ -1,69 +0,0 @@
// Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information.
using Google.Protobuf.WellKnownTypes;
using Grpc.Core;
using Grpc.Core.Interceptors;
using Grpc.Net.Client;
using HelloShop.BasketService.Protos;
namespace HelloShop.BasketService.FunctionalTests
{
public class BasketServiceIntegrationTest
{
private readonly Basket.BasketClient _client;
public BasketServiceIntegrationTest()
{
GrpcChannel channel = GrpcChannel.ForAddress(GrpcConstants.GrpcAddress);
CallInvoker invoker = channel.Intercept(new AuthenticatedInterceptor());
_client = new Basket.BasketClient(invoker);
}
[Fact]
public async Task GetBasketReturnsBasket()
{
// Arrange
var request = new Empty();
// Act
var reply = await _client.GetBasketAsync(request);
// Assert
Assert.NotNull(reply);
}
[Fact]
public async Task UpdateBasketReturnsBasketResponse()
{
// Arrange
var request = new UpdateBasketRequest
{
Items =
{
new BasketListItem { ProductId = 1, Quantity = 2 },
new BasketListItem { ProductId = 2, Quantity = 3 }
}
};
// Act
var reply = await _client.UpdateBasketAsync(request);
// Assert
Assert.Equal(2, reply.Items.Count);
}
[Fact]
public async Task DeleteBasketReturnsEmpty()
{
// Arrange
var request = new Empty();
// Act
var reply = await _client.DeleteBasketAsync(request);
// Assert
Assert.NotNull(reply);
}
}
}

View File

@ -1,25 +0,0 @@
// Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information.
using Grpc.Net.Client;
using HelloShop.BasketService.Protos;
namespace HelloShop.BasketService.FunctionalTests
{
public class GreeterServiceIntegrationTest
{
[Fact]
public async Task SayHelloReturnsHelloMessage()
{
// Arrange
using var channel = GrpcChannel.ForAddress(GrpcConstants.GrpcAddress);
var client = new Greeter.GreeterClient(channel);
// Act
var reply = await client.SayHelloAsync(new HelloRequest { Name = "Greeter" });
// Assert
Assert.Equal("Hello Greeter", reply.Message);
}
}
}

View File

@ -1,10 +0,0 @@
// Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information.
namespace HelloShop.BasketService.FunctionalTests
{
internal class GrpcConstants
{
public const string GrpcAddress = "http://localhost:8004";
}
}

View File

@ -1,39 +0,0 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net8.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
<IsPackable>false</IsPackable>
<IsTestProject>true</IsTestProject>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="coverlet.collector" Version="6.0.0" />
<PackageReference Include="Google.Protobuf" Version="3.21.5" />
<PackageReference Include="Grpc.Net.ClientFactory" Version="2.49.0" />
<PackageReference Include="Grpc.Tools" Version="2.49.0">
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
</PackageReference>
<PackageReference Include="Microsoft.AspNetCore.Grpc.JsonTranscoding" Version="8.0.7" />
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.8.0" />
<PackageReference Include="xunit" Version="2.5.3" />
<PackageReference Include="xunit.runner.visualstudio" Version="2.5.3" />
</ItemGroup>
<ItemGroup>
<Protobuf Include="..\..\src\HelloShop.BasketService\Protos\basket.proto" GrpcServices="Client">
<Link>Protos\basket.proto</Link>
</Protobuf>
<Protobuf Include="..\..\src\HelloShop.BasketService\Protos\greet.proto" GrpcServices="Client">
<Link>Protos\greet.proto</Link>
</Protobuf>
</ItemGroup>
<ItemGroup>
<Using Include="Xunit" />
</ItemGroup>
</Project>

View File

@ -1,40 +0,0 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net9.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
<IsPackable>false</IsPackable>
<IsTestProject>true</IsTestProject>
<IncludeHttpRuleProtos>true</IncludeHttpRuleProtos>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="coverlet.collector" Version="6.0.4">
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
</PackageReference>
<PackageReference Include="Google.Protobuf" Version="3.30.1" />
<PackageReference Include="Grpc.Net.ClientFactory" Version="2.70.0" />
<PackageReference Include="Grpc.Tools" Version="2.71.0">
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
</PackageReference>
<PackageReference Include="Microsoft.AspNetCore.Grpc.JsonTranscoding" Version="9.0.3" />
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.13.0" />
<PackageReference Include="xunit" Version="2.9.3" />
<PackageReference Include="xunit.runner.visualstudio" Version="3.0.2">
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
</PackageReference>
</ItemGroup>
<ItemGroup>
<Protobuf Include="..\..\src\HelloShop.BasketService\Protos\basket.proto" GrpcServices="Client">
<Link>Protos\basket.proto</Link>
</Protobuf>
<Protobuf Include="..\..\src\HelloShop.BasketService\Protos\greet.proto" GrpcServices="Client">
<Link>Protos\greet.proto</Link>
</Protobuf>
</ItemGroup>
<ItemGroup>
<Using Include="Xunit" />
</ItemGroup>
</Project>

View File

@ -2,6 +2,8 @@
// See the license file in the project root for more information. // See the license file in the project root for more information.
using AutoMapper; using AutoMapper;
using FluentValidation;
using FluentValidation.Results;
using Google.Protobuf.WellKnownTypes; using Google.Protobuf.WellKnownTypes;
using HelloShop.BasketService.AutoMapper; using HelloShop.BasketService.AutoMapper;
using HelloShop.BasketService.Entities; using HelloShop.BasketService.Entities;
@ -25,7 +27,8 @@ namespace HelloShop.BasketService.UnitTests.Services
var basketRepositoryMock = new Mock<IBasketRepository>(); var basketRepositoryMock = new Mock<IBasketRepository>();
var loggerMock = NullLogger<CustomerBasketService>.Instance; var loggerMock = NullLogger<CustomerBasketService>.Instance;
var mapperMock = new Mock<IMapper>(); var mapperMock = new Mock<IMapper>();
var service = new CustomerBasketService(basketRepositoryMock.Object, loggerMock, mapperMock.Object); var validatorMock = new Mock<IValidator<UpdateBasketRequest>>();
var service = new CustomerBasketService(basketRepositoryMock.Object, loggerMock, mapperMock.Object, validatorMock.Object);
TestServerCallContext serverCallContext = TestServerCallContext.Create(); TestServerCallContext serverCallContext = TestServerCallContext.Create();
@ -50,11 +53,13 @@ namespace HelloShop.BasketService.UnitTests.Services
var basketRepositoryMock = new Mock<IBasketRepository>(); var basketRepositoryMock = new Mock<IBasketRepository>();
basketRepositoryMock.Setup(x => x.GetBasketAsync(It.IsAny<int>(), It.IsAny<CancellationToken>())).ReturnsAsync(new CustomerBasket() { BuyerId = 1, Items = [new BasketItem { ProductId = 1, Quantity = 1 }] }); basketRepositoryMock.Setup(x => x.GetBasketAsync(It.IsAny<int>(), It.IsAny<CancellationToken>())).ReturnsAsync(new CustomerBasket() { BuyerId = 1, Items = [new BasketItem { ProductId = 1, Quantity = 1 }] });
var validatorMock = new Mock<IValidator<UpdateBasketRequest>>();
var logger = NullLogger<CustomerBasketService>.Instance; var logger = NullLogger<CustomerBasketService>.Instance;
var mapper = new Mapper(new MapperConfiguration(cfg => cfg.AddProfile<BasketsMapConfiguration>())); var mapper = new Mapper(new MapperConfiguration(cfg => cfg.AddProfile<BasketsMapConfiguration>()));
var service = new CustomerBasketService(basketRepositoryMock.Object, logger, mapper); var service = new CustomerBasketService(basketRepositoryMock.Object, logger, mapper, validatorMock.Object);
TestServerCallContext serverCallContext = TestServerCallContext.Create(); TestServerCallContext serverCallContext = TestServerCallContext.Create();
@ -82,7 +87,10 @@ namespace HelloShop.BasketService.UnitTests.Services
var logger = NullLogger<CustomerBasketService>.Instance; var logger = NullLogger<CustomerBasketService>.Instance;
var mapper = new Mapper(new MapperConfiguration(cfg => cfg.AddProfile<BasketsMapConfiguration>())); var mapper = new Mapper(new MapperConfiguration(cfg => cfg.AddProfile<BasketsMapConfiguration>()));
var service = new CustomerBasketService(basketRepositoryMock.Object, logger, mapper); var validatorMock = new Mock<IValidator<UpdateBasketRequest>>();
validatorMock.Setup(x => x.ValidateAsync(It.IsAny<UpdateBasketRequest>(), It.IsAny<CancellationToken>())).ReturnsAsync(new ValidationResult());
var service = new CustomerBasketService(basketRepositoryMock.Object, logger, mapper, validatorMock.Object);
TestServerCallContext serverCallContext = TestServerCallContext.Create(); TestServerCallContext serverCallContext = TestServerCallContext.Create();
@ -112,7 +120,10 @@ namespace HelloShop.BasketService.UnitTests.Services
var basketRepositoryMock = new Mock<IBasketRepository>(); var basketRepositoryMock = new Mock<IBasketRepository>();
var logger = NullLogger<CustomerBasketService>.Instance; var logger = NullLogger<CustomerBasketService>.Instance;
var mapper = new Mapper(new MapperConfiguration(cfg => cfg.AddProfile<BasketsMapConfiguration>())); var mapper = new Mapper(new MapperConfiguration(cfg => cfg.AddProfile<BasketsMapConfiguration>()));
var service = new CustomerBasketService(basketRepositoryMock.Object, logger, mapper);
var validatorMock = new Mock<IValidator<UpdateBasketRequest>>();
var service = new CustomerBasketService(basketRepositoryMock.Object, logger, mapper, validatorMock.Object);
TestServerCallContext serverCallContext = TestServerCallContext.Create(); TestServerCallContext serverCallContext = TestServerCallContext.Create();

View File

@ -1,8 +1,10 @@
// Copyright (c) HelloShop Corporation. All rights reserved. // Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information. // See the license file in the project root for more information.
using Aspire.Hosting; using Aspire.Hosting.ApplicationModel;
using HelloShop.FunctionalTests.Helpers;
using Microsoft.AspNetCore.Authentication.BearerToken; using Microsoft.AspNetCore.Authentication.BearerToken;
using Microsoft.Extensions.DependencyInjection;
using System.Dynamic; using System.Dynamic;
using System.Net; using System.Net;
using System.Net.Http.Headers; using System.Net.Http.Headers;
@ -11,18 +13,24 @@ using System.Text.Json.Nodes;
namespace HelloShop.FunctionalTests namespace HelloShop.FunctionalTests
{ {
public class FirstWebApiIntegrationTest public class FirstWebApiIntegrationTest(TestingAspireAppHost app) : IAsyncLifetime, IClassFixture<TestingAspireAppHost>
{ {
public async Task InitializeAsync()
{
await app.StartAsync();
}
public async Task DisposeAsync() => await Task.CompletedTask;
[Fact] [Fact]
public async Task WebAppRootReturnsOkStatusCode() public async Task WebAppRootReturnsOkStatusCode()
{ {
// Arrange // Arrange
IDistributedApplicationTestingBuilder appHost = await DistributedApplicationTestingBuilder.CreateAsync<Projects.HelloShop_AppHost>(); var resourceNotificationService = app.Services.GetRequiredService<ResourceNotificationService>();
await using DistributedApplication app = await appHost.BuildAsync();
await app.StartAsync();
// Act // Act
HttpClient httpClient = app.CreateHttpClient("webapp"); HttpClient httpClient = app.CreateHttpClient("webapp");
await resourceNotificationService.WaitForResourceAsync("webapp", KnownResourceStates.Running).WaitAsync(TimeSpan.FromSeconds(30));
HttpResponseMessage response = await httpClient.GetAsync("/"); HttpResponseMessage response = await httpClient.GetAsync("/");
// Assert // Assert
@ -33,12 +41,11 @@ namespace HelloShop.FunctionalTests
public async Task IdetityServiceAccountLoginReturnsSuccessStatusCode() public async Task IdetityServiceAccountLoginReturnsSuccessStatusCode()
{ {
// Arrange // Arrange
IDistributedApplicationTestingBuilder appHost = await DistributedApplicationTestingBuilder.CreateAsync<Projects.HelloShop_AppHost>(); var resourceNotificationService = app.Services.GetRequiredService<ResourceNotificationService>();
await using DistributedApplication app = await appHost.BuildAsync();
await app.StartAsync();
// Act // Act
HttpClient httpClient = app.CreateHttpClient("identityservice"); HttpClient httpClient = app.CreateHttpClient("identityservice");
await resourceNotificationService.WaitForResourceHealthyAsync("identityservice").WaitAsync(TimeSpan.FromSeconds(30));
HttpResponseMessage response = await httpClient.PostAsJsonAsync("api/Account/Login", new HttpResponseMessage response = await httpClient.PostAsJsonAsync("api/Account/Login", new
{ {
UserName = "guest", UserName = "guest",
@ -53,12 +60,11 @@ namespace HelloShop.FunctionalTests
public async Task IdetityServiceAccountLoginReturnsAccessToken() public async Task IdetityServiceAccountLoginReturnsAccessToken()
{ {
// Arrange // Arrange
IDistributedApplicationTestingBuilder appHost = await DistributedApplicationTestingBuilder.CreateAsync<Projects.HelloShop_AppHost>(); var resourceNotificationService = app.Services.GetRequiredService<ResourceNotificationService>();
await using DistributedApplication app = await appHost.BuildAsync();
await app.StartAsync();
// Act // Act
HttpClient httpClient = app.CreateHttpClient("identityservice"); HttpClient httpClient = app.CreateHttpClient("identityservice");
await resourceNotificationService.WaitForResourceHealthyAsync("identityservice").WaitAsync(TimeSpan.FromSeconds(30));
HttpResponseMessage response = await httpClient.PostAsJsonAsync("api/Account/Login", new HttpResponseMessage response = await httpClient.PostAsJsonAsync("api/Account/Login", new
{ {
UserName = "guest", UserName = "guest",
@ -74,13 +80,11 @@ namespace HelloShop.FunctionalTests
public async Task IdetityServiceAccountLoginReturnsTokenExpiresInSeconds() public async Task IdetityServiceAccountLoginReturnsTokenExpiresInSeconds()
{ {
// Arrange // Arrange
IDistributedApplicationTestingBuilder appHost = await DistributedApplicationTestingBuilder.CreateAsync<Projects.HelloShop_AppHost>(); var resourceNotificationService = app.Services.GetRequiredService<ResourceNotificationService>();
await using DistributedApplication app = await appHost.BuildAsync();
await app.StartAsync();
// Act // Act
HttpClient httpClient = app.CreateHttpClient("identityservice"); HttpClient httpClient = app.CreateHttpClient("identityservice");
await resourceNotificationService.WaitForResourceHealthyAsync("identityservice").WaitAsync(TimeSpan.FromSeconds(30));
HttpResponseMessage response = await httpClient.PostAsJsonAsync("api/Account/Login", new HttpResponseMessage response = await httpClient.PostAsJsonAsync("api/Account/Login", new
{ {
UserName = "guest", UserName = "guest",
@ -97,13 +101,11 @@ namespace HelloShop.FunctionalTests
public async Task ProductServiceGetProductReturnsProductDetails() public async Task ProductServiceGetProductReturnsProductDetails()
{ {
// Arrange // Arrange
IDistributedApplicationTestingBuilder appHost = await DistributedApplicationTestingBuilder.CreateAsync<Projects.HelloShop_AppHost>(); var resourceNotificationService = app.Services.GetRequiredService<ResourceNotificationService>();
await using DistributedApplication app = await appHost.BuildAsync();
await app.StartAsync();
// Act // Act
HttpClient identityServiceHttpClient = app.CreateHttpClient("identityservice"); HttpClient identityServiceHttpClient = app.CreateHttpClient("identityservice");
await resourceNotificationService.WaitForResourceHealthyAsync("identityservice").WaitAsync(TimeSpan.FromSeconds(30));
HttpResponseMessage loginResponse = await identityServiceHttpClient.PostAsJsonAsync("api/Account/Login", new HttpResponseMessage loginResponse = await identityServiceHttpClient.PostAsJsonAsync("api/Account/Login", new
{ {
UserName = "admin", UserName = "admin",
@ -114,16 +116,17 @@ namespace HelloShop.FunctionalTests
HttpClient productServiceHttpClient = app.CreateHttpClient("productservice"); HttpClient productServiceHttpClient = app.CreateHttpClient("productservice");
productServiceHttpClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", accessTokenResponse?.AccessToken); productServiceHttpClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", accessTokenResponse?.AccessToken);
await resourceNotificationService.WaitForResourceHealthyAsync("productservice").WaitAsync(TimeSpan.FromSeconds(30));
HttpResponseMessage productDetailsResponse = await productServiceHttpClient.GetAsync("api/Products/1"); HttpResponseMessage productDetailsResponse = await productServiceHttpClient.GetAsync("api/Products/1");
JsonNode? result = await productDetailsResponse.Content.ReadFromJsonAsync<JsonNode>(); JsonNode? result = await productDetailsResponse.Content.ReadFromJsonAsync<JsonNode>();
string? productName = result?["Name"]?.GetValue<string>(); int? productId = result?["Id"]?.GetValue<int?>();
// Assert // Assert
Assert.NotNull(productName); Assert.NotNull(productId);
Assert.Equal("Product 1", productName); Assert.Equal(1, productId);
} }
} }
} }

View File

@ -0,0 +1,31 @@
// Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information.
using Aspire.Hosting;
using Microsoft.Extensions.DependencyInjection;
namespace HelloShop.FunctionalTests.Helpers
{
public class TestingAspireAppHost : DistributedApplicationFactory
{
private readonly TaskCompletionSource<IServiceProvider> _serviceTcs = new();
public TestingAspireAppHost() : base(typeof(Projects.HelloShop_AppHost)) { }
protected override void OnBuilderCreated(DistributedApplicationBuilder applicationBuilder)
{
applicationBuilder.Services.ConfigureHttpClientDefaults(clientBuilder =>
{
clientBuilder.AddStandardResilienceHandler();
});
}
protected override void OnBuilt(DistributedApplication application)
{
_serviceTcs.SetResult(application.Services);
}
public IServiceProvider Services => _serviceTcs.Task.GetAwaiter().GetResult();
}
}

View File

@ -1,7 +1,7 @@
// Copyright (c) HelloShop Corporation. All rights reserved. // Copyright (c) HelloShop Corporation. All rights reserved.
// See the license file in the project root for more information. // See the license file in the project root for more information.
using HelloShop.ProductService.FunctionalTests.Utilities; using HelloShop.ProductService.FunctionalTests.Helpers;
using HelloShop.ProductService.Models.Products; using HelloShop.ProductService.Models.Products;
using System.Net; using System.Net;
using System.Net.Http.Json; using System.Net.Http.Json;
@ -31,8 +31,11 @@ namespace HelloShop.ProductService.FunctionalTests
// Act // Act
HttpResponseMessage response = await client.GetAsync("api/Brands/1"); HttpResponseMessage response = await client.GetAsync("api/Brands/1");
string responseContent = await response.Content.ReadAsStringAsync();
BrandDetailsResponse? brandDetails = await response.Content.ReadFromJsonAsync<BrandDetailsResponse>(); BrandDetailsResponse? brandDetails = await response.Content.ReadFromJsonAsync<BrandDetailsResponse>();
// Assert // Assert
Assert.NotNull(brandDetails); Assert.NotNull(brandDetails);
Assert.Equal(1, brandDetails.Id); Assert.Equal(1, brandDetails.Id);

View File

@ -7,40 +7,54 @@ using Microsoft.AspNetCore.Hosting;
using Microsoft.AspNetCore.Mvc.Testing; using Microsoft.AspNetCore.Mvc.Testing;
using Microsoft.Data.Sqlite; using Microsoft.Data.Sqlite;
using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Diagnostics;
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.DependencyInjection.Extensions; using Microsoft.Extensions.DependencyInjection.Extensions;
using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Hosting;
using System.Data.Common; using System.Data.Common;
using System.Net.Http.Headers; using System.Net.Http.Headers;
namespace HelloShop.ProductService.FunctionalTests.Utilities namespace HelloShop.ProductService.FunctionalTests.Helpers
{ {
public class CustomWebApplicationFactory<TProgram> : WebApplicationFactory<TProgram> where TProgram : class public class CustomWebApplicationFactory<TProgram> : WebApplicationFactory<TProgram> where TProgram : class
{ {
protected override void ConfigureWebHost(IWebHostBuilder builder) protected override void ConfigureWebHost(IWebHostBuilder builder)
{ {
builder.ConfigureServices(services => builder.ConfigureServices(static services =>
{ {
ServiceDescriptor? dbContextDescriptor = services.SingleOrDefault(d => d.ServiceType == typeof(DbContextOptions<ProductServiceDbContext>)); var dbContextDescriptor = services.SingleOrDefault(d => d.ServiceType == typeof(DbContextOptions<ProductServiceDbContext>));
if (dbContextDescriptor != null) if (dbContextDescriptor != null)
{ {
services.Remove(dbContextDescriptor); services.Remove(dbContextDescriptor);
} }
// Create open SqliteConnection so EF won't automatically close it. var dbConnectionDescriptor = services.SingleOrDefault(d => d.ServiceType == typeof(DbConnection));
services.AddSingleton(container => if (dbConnectionDescriptor != null)
{ {
DbConnection connection = new SqliteConnection("DataSource=:memory:"); services.Remove(dbConnectionDescriptor);
}
var optionsConfigurationDescriptor = services.SingleOrDefault(d => d.ServiceType == typeof(IDbContextOptionsConfiguration<ProductServiceDbContext>));
if (optionsConfigurationDescriptor != null)
{
services.Remove(optionsConfigurationDescriptor);
}
// Create open SqliteConnection so EF won't automatically close it.
services.AddSingleton<DbConnection>(container =>
{
var connection = new SqliteConnection("DataSource=file::memory:?cache=shared");
connection.Open(); connection.Open();
return connection; return connection;
}); });
services.AddDbContext<ProductServiceDbContext>((container, options) => services.AddDbContextPool<ProductServiceDbContext>((container, options) =>
{ {
var connection = container.GetRequiredService<DbConnection>(); var connection = container.GetRequiredService<DbConnection>();
options.UseSqlite(connection); options.UseSqlite(connection).UseSnakeCaseNamingConvention();
options.ConfigureWarnings(warnings => warnings.Ignore(RelationalEventId.PendingModelChangesWarning));
}); });
services.Replace(ServiceDescriptor.Transient<IPermissionChecker, FakePermissionChecker>()); services.Replace(ServiceDescriptor.Transient<IPermissionChecker, FakePermissionChecker>());
@ -52,7 +66,6 @@ namespace HelloShop.ProductService.FunctionalTests.Utilities
protected override void ConfigureClient(HttpClient client) protected override void ConfigureClient(HttpClient client)
{ {
client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", FakeAccessTokenCreator.Create()); client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", FakeAccessTokenCreator.Create());
base.ConfigureClient(client); base.ConfigureClient(client);
} }
} }

View File

@ -7,7 +7,7 @@ using System.IdentityModel.Tokens.Jwt;
using System.Security.Claims; using System.Security.Claims;
using System.Text; using System.Text;
namespace HelloShop.ProductService.FunctionalTests.Utilities namespace HelloShop.ProductService.FunctionalTests.Helpers
{ {
public class FakeAccessTokenCreator public class FakeAccessTokenCreator
{ {

View File

@ -4,7 +4,7 @@
using HelloShop.ProductService.Infrastructure; using HelloShop.ProductService.Infrastructure;
using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore;
namespace HelloShop.ProductService.UnitTests.Utilities namespace HelloShop.ProductService.UnitTests.Helpers
{ {
public class FakeDbContextFactory : IDbContextFactory<ProductServiceDbContext> public class FakeDbContextFactory : IDbContextFactory<ProductServiceDbContext>
{ {

View File

@ -7,7 +7,7 @@ using HelloShop.ProductService.Controllers;
using HelloShop.ProductService.Entities.Products; using HelloShop.ProductService.Entities.Products;
using HelloShop.ProductService.Infrastructure; using HelloShop.ProductService.Infrastructure;
using HelloShop.ProductService.Models.Products; using HelloShop.ProductService.Models.Products;
using HelloShop.ProductService.UnitTests.Utilities; using HelloShop.ProductService.UnitTests.Helpers;
using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc;
namespace HelloShop.ProductService.UnitTests namespace HelloShop.ProductService.UnitTests