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

Development #5

Merged
merged 20 commits into from
Feb 9, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
The table of contents is too big for display.
Diff view
Diff view
  •  
  •  
  •  
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
<?xml version="1.0" encoding="utf-8"?>
<package>
<id>Intent.Application.MediatR.Behaviours</id>
<version>4.2.10-pre.1</version>
<version>4.2.10-pre.2</version>
<supportedClientVersions>[4.1.0-a, 5.0.0-a)</supportedClientVersions>
<summary>Additional MediatR middleware pipeline behaviours</summary>
<description>Additional MediatR middleware pipeline behaviours</description>
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
<?xml version="1.0" encoding="utf-8"?>
<package>
<id>Intent.AspNetCore.HealthChecks</id>
<version>2.0.2-pre.0</version>
<version>2.0.2-pre.1</version>
<supportedClientVersions>[4.1.0, 5.0.0-a)</supportedClientVersions>
<summary>Monitor various aspects of your application to determine if is responding to requests normally.</summary>
<description>Monitor various aspects of your application to determine if is responding to requests normally.</description>
Expand Down
10 changes: 10 additions & 0 deletions Modules/Intent.Modules.AspNetCore.HealthChecks/NugetPackage.cs
Original file line number Diff line number Diff line change
Expand Up @@ -104,4 +104,14 @@ public static class NugetPackage
(7, 0) => "7.0.0",
_ => "8.0.0"
});

public static INugetPackageInfo AspNetCoreHealthChecksRedis(IOutputTarget outputTarget) => new NugetPackageInfo(
name: "AspNetCore.HealthChecks.Redis",
version: outputTarget.GetMaxNetAppVersion() switch
{
(5, 0) => "5.0.1",
(6, 0) => "6.0.2",
(7, 0) => "7.0.0",
_ => "8.0.0"
});
}
Original file line number Diff line number Diff line change
Expand Up @@ -184,6 +184,13 @@ private IEnumerable<CSharpStatement> GetKnownHealthConfigurationStatements(Infra
expression: "hcBuilder.AddOracle",
connectionStringNameVar: Infrastructure.Oracle.Property.ConnectionStringName,
connectionStringSettingPathVar: Infrastructure.Oracle.Property.ConnectionStringSettingPath);
case Infrastructure.Redis.Name:
AddNugetDependency(NugetPackage.AspNetCoreHealthChecksRedis(OutputTarget));
return GetDatabaseHealthCheckStatements(
@event: @event,
expression: "hcBuilder.AddRedis",
connectionStringNameVar: Infrastructure.Redis.Property.ConnectionStringName,
connectionStringSettingPathVar: Infrastructure.Redis.Property.ConnectionStringSettingPath);
default:
return Enumerable.Empty<CSharpInvocationStatement>();
}
Expand Down
41 changes: 40 additions & 1 deletion Modules/Intent.Modules.AspNetCore.IntegrationTesting/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ This module consumes your `Exposed HTTP Endpoints`, in the `Service Designer` an
- Adds Integration xUnit Testing project.
- Generates service proxies for all service end points, to use to interact with the Application under test.
- Add container support for `MS SQL Server`, `Postgres` and `CosmosDB`
- Generates test classes for each service end point.
- Generates test classes for each modelled service end point.

## Testing Isolation

Expand All @@ -55,3 +55,42 @@ If you are running `Shared Container`, you can set up specific Test Class's to r
...
}
```

## Testing against containerized data stores

If your application is using our `Intent.EntityFrameworkCore` module, the following providers are currently supported for containerization

- SQL Server
- Postgres

If your application is using our `Intent.CosmosDB` module, the tests will be run against a dockerized CosmosDB Emulator.

## Adding Tests

You can then simply add your integration tests to the test classes as required.
Our `Intent.AspNetCore.IntegrationTesting.CRUD` module can be used to generate integration test implementations for CRUD orientated services.

```csharp

[IntentManaged(Mode.Merge, Signature = Mode.Fully)]
[Collection("SharedContainer")]
public class MyCustomEndpointTests : BaseIntegrationTest
{
public MyCustomEndpointTests(IntegrationTestWebAppFactory factory) : base(factory)
{
}

[Fact]
public async Task MyCustomEndpoint_ShouldDoX()
{
//Arrange
var client = new MyCustomServiceHttpClient(CreateClient());

//Act
client.InvokeMyCustomEndpoint();

//Assert
...
}
}
```
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
<authors>Intent Architect</authors>
<tags>csharp dotnet crud integration testing asp.net xunit</tags>
<releaseNotes>release-notes.md</releaseNotes>
<projectUrl>https://github.com/IntentArchitect/Intent.Modules.NET/blob/development/Modules/Intent.Modules.AspNetCore.IntegrationTesting.CRUD/README.md</projectUrl>
<iconUrl></iconUrl>
<templates>
<template id="Intent.AspNetCore.IntegrationTests.CRUD.PopulateIdsSpecimenBuilder" externalReference="6cc9daf4-9b9d-45ef-bdac-912919be0654">
Expand Down
95 changes: 95 additions & 0 deletions Modules/Intent.Modules.AspNetCore.IntegrationTests.CRUD/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
# Intent.AspNetCore.IntegrationTesting.CRUD

This module provides Integration Test implementation for Services which are CRUD orientated.

## What's in this module?

This module consumes your `Services` both `ServiceModel` and `CQRS`, in the `Service Designer` and generates an Integration test for each Endpoint.

For a service to be eligible is must implement at least

- Create{Entity} - returning either the `Primary Key` or a `DTO` with a mapped `Primary Key`.
- Get{Entity}ById - returning an `Entity` based `DTO` taking a single parameter of the `Entitiy`s primary key.

## Sample Tests

```csharp
[Fact]
public async Task CreateCustomer_ShouldCreateCustomer()
{
//Arrange
var client = new CustomersServiceHttpClient(CreateClient());

var dataFactory = new TestDataFactory(WebAppFactory);
await dataFactory.CreateCustomerDependencies();

var command = dataFactory.CreateCommand<CustomerCreateDto>();

//Act
var customerId = await client.CreateCustomerAsync(command);

//Assert
var customer = await client.FindCustomerByIdAsync(customerId);
Assert.NotNull(customer);
}
```

```csharp
[Fact]
public async Task FindCustomerById_ShouldFindCustomerById()
{
//Arrange
var client = new CustomersServiceHttpClient(CreateClient());

var dataFactory = new TestDataFactory(WebAppFactory);
var customerId = await dataFactory.CreateCustomer();

//Act
var customer = await client.FindCustomerByIdAsync(customerId);

//Assert
Assert.NotNull(customer);
}
```

```csharp
[Fact]
public async Task UpdateCustomer_ShouldUpdateCustomer()
{
//Arrange
var client = new CustomersServiceHttpClient(CreateClient());

var dataFactory = new TestDataFactory(WebAppFactory);
var customerId = await dataFactory.CreateCustomer();

var command = dataFactory.CreateCommand<CustomerUpdateDto>();
command.Id = customerId;

//Act
await client.UpdateCustomerAsync(customerId, command);

//Assert
var customer = await client.FindCustomerByIdAsync(customerId);
Assert.NotNull(customer);
Assert.Equal(command.Name, customer.Name);
}
```

```csharp
[Fact]
public async Task DeleteCustomer_ShouldDeleteCustomer()
{
//Arrange
var client = new CustomersServiceHttpClient(CreateClient());

var dataFactory = new TestDataFactory(WebAppFactory);
var customerId = await dataFactory.CreateCustomer();

//Act
await client.DeleteCustomerAsync(customerId);

//Assert
var exception = await Assert.ThrowsAsync<HttpClientRequestException>(() => client.FindCustomerByIdAsync(customerId));
Assert.Equal(HttpStatusCode.NotFound, exception.StatusCode);
}
```
11 changes: 11 additions & 0 deletions Modules/Intent.Modules.Constants/Infrastructure.cs
Original file line number Diff line number Diff line change
Expand Up @@ -71,4 +71,15 @@ public static class Property
public const string ConnectionStringSettingPath = nameof(ConnectionStringSettingPath);
}
}

public static class Redis
{
public const string Name = nameof(Redis);

public static class Property
{
public const string ConnectionStringName = nameof(ConnectionStringName);
public const string ConnectionStringSettingPath = nameof(ConnectionStringSettingPath);
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
using Intent.Modelers.Domain.Api;
using Intent.Modules.Common.CSharp.Builder;
using Intent.Modules.Common.CSharp.Templates;
using Intent.Modules.DocumentDB.Shared;

namespace Intent.Modules.CosmosDB.FactoryExtensions;

public class CosmosDbPrimaryKeyInitStrategy : IPrimaryKeyInitStrategy
{
public bool ShouldInsertPkInitializationCode(ClassModel targetClass)
{
return true;
}

public string GetGetterInitExpression(ICSharpTemplate template, ClassModel targetClass, string fieldName, string fieldTypeName)
{
return fieldTypeName switch
{
"string" => $"{fieldName} ??= {template.UseType("System.Guid")}.NewGuid().ToString()",
"guid" => $"{fieldName} ??= {template.UseType("System.Guid")}.NewGuid()",
"int" or "long" => $"{fieldName} ?? throw new {template.UseType("System.NullReferenceException")}(\"{fieldName} has not been set\")",
_ => fieldName
};
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -34,10 +34,9 @@ protected override void OnAfterTemplateRegistrations(IApplication application)
EntityFactoryExtensionHelper.Execute(
application: application,
dbProviderApplies: CosmosDbProvider.FilterDbProvider,
initializePrimaryKeyOnAggregateRoots: true,
primaryKeyInitStrategy: new CosmosDbPrimaryKeyInitStrategy(),
makeNonPersistentPropertiesVirtual: false);
RegisterServices(application);

}

private static void RegisterServices(IApplication application)
Expand Down
Loading