-
Notifications
You must be signed in to change notification settings - Fork 581
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
DisposeAsync
hangs within DistributedApplicationTestingBuilder
#7139
Comments
The testing host waits for the AppHost's thread to exit before DisposeAsync will complete, so if the AppHost's thread is stuck (eg, on StartAsync) then it will hang. There is an internal race in lifecycle which can prevent container notifications from DCP reaching the AppHost once shutdown has begun. We have some non-mutually-exclusive options to address this:
That being said, does this arise in real-world scenarios? If so, could you elaborate for me? |
@afscrome what do you think? |
This error was uncovered by a dev on another team who asked for my help. So the tests they have likely better represent what a "real" developer would do, not the shienagans I've been known to get up to. This minimal repo was derived from that real world use case. Report from my colleague that I was asked to help investigate: What they were experiencing was tests passing, but Visual Studio / Rider's test runners were hanging for minutes after the test passed/failed before you could run another test.
Let me try and pull the code version up from when this was reported and more fully remind myself of the scenario. |
Much appreciated, @afscrome! |
Was a while ago now so I can't remember exactly which project and version it was in and I'm mixing up the various different issues we've encountered with the test host. I'll keep digging to remind myself of that case. I have found one of the failure cases, although this behaves slightly differently in that it hangs forever: Tests: private DistributedApplication? _app;
private ResourceNotificationService? _resourceNotificationService;
[SetUp]
public async Task SetUp()
{
// Arrange
var appHost = await DistributedApplicationTestingBuilder.CreateAsync<Projects.My_AppHost>();
appHost.Services.ConfigureHttpClientDefaults(clientBuilder => { clientBuilder.AddStandardResilienceHandler(); });
appHost.Services.AddLogging(x => x
.AddNUnit()
.AddFilter("Default", LogLevel.Information)
.AddFilter("Microsoft.AspNetCore", LogLevel.Warning)
.AddFilter("Aspire.Hosting.Dcp", LogLevel.Warning)
);
_app = await appHost.BuildAsync();
_resourceNotificationService = _app.Services.GetRequiredService<ResourceNotificationService>();
await _app.StartAsync()
.WaitAsync(TimeSpan.FromMinutes(1));
}
[TearDown]
public async Task TearDown()
{
if (_app != null)
{
await _app.StopAsync();
await _app.DisposeAsync();
}
}
[TestCase("resource")]
[CancelAfter(120_000)]
public async Task HttpResourceRootReturnsOkStatusCode(string resourceName, CancellationToken ct)
{
// Frankly contents of this test are irrelevant - it can be a no-op and the SetUp / TearDown combination will still hang foever
// Arrange
await _resourceNotificationService!
.WaitForResourceAsync(resourceName, KnownResourceStates.Running, ct)
.WaitAsync(TimeSpan.FromSeconds(30), ct);
using var httpClient = _app!.CreateHttpClient(resourceName);
// Act
var response = await httpClient.GetAsync("/", ct);
// Assert
Assert.That(response.StatusCode, Is.EqualTo(HttpStatusCode.OK));
} App: I can't really share my app in a way that is usable externally, but essentially it boiled down to
The following roughly simulates that: var failsToGoHealthy = builder.AddRedis("failsToGoHealthy")
.WithHttpEndpoint(targetPort: 1234)
.WithHttpHealthCheck();
var dependency = builder.AddExecutable("dependency", "pwsh.exe", ".")
.WaitFor(failsToGoHealthy); |
I've been encountering a number of cases where our Tests appear to be done, yet the tests are still running in the test runner. Further investigation suggests this hanging is happening when disposing the app built through
DistributedApplicationTestingBuilder
. I've sometimes seen delays of several minutes here.The criteria seem to be
DistributedApplicationTestingBuilder
(doesn't replicate using plainDistributedApplication.CreateBuilder
)A minimal repro for this is below . I expect the time between the two beeps to be negligible, however it is 10+ seconds.
AppHost:
Test:
Looking at console output from the test and you can see there is a ~13 second gap between
Dependency resource 'initial' failed to start.
andFailed to create resource app
. Given the 1 second timeout in the above code, I'd expect this to be no more than that.Interestingly this only happens through the test host - replace the
app.Run()
call inProgram.cs
with the following, andDisposeAsync
is near instantMay also be some relationship with #7009 as they both share the criteria of mixing health checks with waitfor, although this issue does only reproduce under the test host.
The text was updated successfully, but these errors were encountered: