Open GeeWee opened 4 years ago
Thanks for contacting us.
We're moving this issue to the Next sprint planning
milestone for future evaluation / consideration. We will evaluate the request when we are planning the work for the next milestone. To learn more about what to expect next and how this issue will be handled you can read more about our triage process here.
We've moved this issue to the Backlog milestone. This means that it is not going to be worked on for the coming release. We will reassess the backlog following the current release and consider this item at that time. To learn more about our issue management process and to have better expectation regarding different types of issues you can read our Triage Process.
This, and the fact that MVC.Testing inner infrastructure only disposes the host without stopping it first, makes MVC.Testing really hard to use with any service that has lasting effects on infrastructure.
This is a dupe of #29348
@javiercn hello. Sorry, but it does not look like the merged PR fixes this bug. And this bug is not a duplicate of my issue.
This bug describes the test host ignoring IHostApplicationLifetime.StopApplication
.
My issue was the host not being stopped on dispose.
The PR only forces the host to stop when the factory is disposed.
@quixoticaxis My bad, I misunderstood it.
@Tratcher I believe this is a potential issue/enhancement in TestServer, do you have any thoughts?
@GeeWee While I appreciate the minimal repro, what are you really hoping to accomplish with StopApplication? It's not clear why StopApplication is a useful scenario for TestServer to emulate.
@Tratcher I don't know the reasons of this issue's author, but in our code we often encounter the following pattern:
public class ServiceA : IHostedService
{
public Task StartAsync(CancellationToken token)
{
executing = ExecuteAsync();
}
private Task ExecuteAsync()
{
try
{
await serviceLogic.LoopAsync();
}
catch
{
Log("The service A cannot proceed, stopping the application");
lifetime.StopApplication();
}
}
private Task executing;
}
This code enables clean shutdown if any of the services fail, because in the real application StopAsync
would be invoked on all of the services. It does not work in the tests though, which leads to inconveniencies (I described our scenario in #29348).
Similar in case of OutOfMemory
handling described in https://github.com/dotnet/runtime/issues/48157 I would like to perform graceful shutdown once issue is detected and cover such negative path in an integration test.
I guess the best it could do is stop future requests. That seems fine.
We also have a background service does following, would be good have a way to gracefully stop it after test completed (either pass or failed)
public class ServiceA : BackgroundService
{
protected override Task ExecuteAsync(CancellationToken stoppingToken){
while (!stoppingToken.IsCancellationRequested)
{
await pickupMessageFromQueueAndDoSomethingAboutIt();
}
}
}
@jeremy001181 that seems like a different ask than above. That would be handled by calling StopAsync on the host. (Not sure how accessible that is from WebApplicationFactory).
Just want to add another use case. Our application does not migrate the database (this is deferred onto a different application that is guaranteed to run before any other in kubernetes). However in our tests we do want this to run and we want a separate database for all tests as they run in parallel. To that end we inject a startup filter that is ran before all others. This startup filter ties into IHostApplicationLifetime.ApplicationStopping and IHostApplicationLifetime.ApplicationStopped to delete the database it has just created and migrated.
Currently we're having to override WebApplicationFactory.Dispose and through reflection get the _host field and call StopAsync on that to trigger this path. An approach mentioned in https://github.com/dotnet/aspnetcore/issues/29348.
There is other reasons why we want these IHostApplicationLifetime callbacks to be triggered, but this one, which is solely for the tests itself, was not yet mentioned here.
I guess the only question left to ask is: Is this something that you guys would like to see implemented? If so, I would be willing to commit some time to make a PR.
Maybe also a little extra information for when this would be useful.
I've been developing BetterHostedServices which relies on being able to understand when the application is being shut down to do some of its magic. Currently I have to jump through a few hoops because it's not easily possible to integration-test anything that relies on shutting down the TestServer
.
Are there any workarounds for this? It seems that WebApplicationFactory always leave the process running. We are working with a .NET 5 app.
Are there any workarounds for this? It seems that WebApplicationFactory always leave the process running. We are working with a .NET 5 app.
@tapizquent , there was a PR that forces the host to stop on disposing the factory. Disposing the factory should do the trick.
For other cases, you can use the hack I posted in #29348.
Describe the bug
I'm trying to run integration tests, with some things that call
IHostApplicationLifetime.StopApplication
from an IHostedServiceIn a "real project" this works, but it does not when using the TestServer. Test case below
To Reproduce
The IHostedService calls StopApplication. I then don't expect the
GetAsync
call to work then. The logs look like this:So it starts off by writing "Application is shutting down" and then proceeds to start the application.
Further technical details
dotnet --info
Runtime Environment: OS Name: fedora OS Version: 32 OS Platform: Linux RID: fedora.32-x64 Base Path: /usr/share/dotnet/sdk/3.1.401/
Host (useful for support): Version: 3.1.7 Commit: fcfdef8d6b
.NET Core SDKs installed: 2.2.402 [/usr/share/dotnet/sdk] 3.0.103 [/usr/share/dotnet/sdk] 3.1.401 [/usr/share/dotnet/sdk]
.NET Core runtimes installed: Microsoft.AspNetCore.All 2.2.8 [/usr/share/dotnet/shared/Microsoft.AspNetCore.All] Microsoft.AspNetCore.App 2.2.8 [/usr/share/dotnet/shared/Microsoft.AspNetCore.App] Microsoft.AspNetCore.App 3.0.3 [/usr/share/dotnet/shared/Microsoft.AspNetCore.App] Microsoft.AspNetCore.App 3.1.7 [/usr/share/dotnet/shared/Microsoft.AspNetCore.App] Microsoft.NETCore.App 2.2.8 [/usr/share/dotnet/shared/Microsoft.NETCore.App] Microsoft.NETCore.App 3.0.3 [/usr/share/dotnet/shared/Microsoft.NETCore.App] Microsoft.NETCore.App 3.1.7 [/usr/share/dotnet/shared/Microsoft.NETCore.App]
To install additional .NET Core runtimes or SDKs: https://aka.ms/dotnet-download