pulumi/sdk/dotnet/Pulumi.Tests/DeploymentTests.cs
Pat Gavlin bd18384038
Await outstanding async work in .NET. (#6993)
The Pulumi .NET SDK does not currently await all outstanding asynchronous
work associated with a Pulumi program. Because all relevant asynchronous
work is created via the Pulumi SDK, we can track this asynchronous work
and ensure that it has all completed prior to returning from
`Deployment.RunAsync`.

The implementation here is simpler than that in #6983, and re-uses the
existing support for tracking outstanding RPCs. If this proves to
negatively impact performance (which is a very real possibility for
programs that create many `Output` instances), we can simplify this
using a semaphore and a counter (essentially Go's `sync.WaitGroup`).

This fixes the .NET portion of #3991.
2021-05-12 13:23:47 -07:00

110 lines
3.9 KiB
C#

// Copyright 2016-2021, Pulumi Corporation
using System;
using System.Collections.Generic;
using System.Threading.Tasks;
using Pulumi.Testing;
using Pulumi.Tests.Mocks;
using Xunit;
namespace Pulumi.Tests
{
public class DeploymentTests
{
[Fact]
public async Task DeploymentInstancePropertyIsProtected()
{
// confirm we cannot retrieve deployment instance early
Assert.Throws<InvalidOperationException>(
() => _ = Deployment.Instance);
// confirm we cannot set deployment instance from downstream execution
var deployment = new Deployment(new MockEngine(), new MockMonitor(new MyMocks()), null);
var task = Deployment.CreateRunnerAndRunAsync(
() => deployment,
_ =>
{
Deployment.Instance = new DeploymentInstance(deployment);
return Task.FromResult(1);
});
// should not throw until awaited
await Assert.ThrowsAsync<InvalidOperationException>(
() => task);
}
[Fact]
public async Task DeploymentInstancesAreSeparate()
{
// this test is more of a sanity check that two separate
// executions have their own deployment instance
DeploymentInstance? instanceOne = null;
DeploymentInstance? instanceTwo = null;
var tcs = new TaskCompletionSource<int>();
var runTaskOne = Deployment.CreateRunnerAndRunAsync(
() => new Deployment(new MockEngine(), new MockMonitor(new MyMocks()), null),
runner =>
{
instanceOne = Deployment.Instance;
return tcs.Task;
});
await Deployment.CreateRunnerAndRunAsync(
() => new Deployment(new MockEngine(), new MockMonitor(new MyMocks()), null),
runner =>
{
instanceTwo = Deployment.Instance;
return Task.FromResult(1);
});
tcs.SetResult(1);
await runTaskOne;
Assert.NotNull(instanceOne);
Assert.NotNull(instanceTwo);
Assert.False(ReferenceEquals(instanceOne, instanceTwo));
}
[Fact]
public async Task DeploymentInstanceIsProtectedFromParallelSynchronousRunAsync()
{
// this test is ensuring that CreateRunnerAndRunAsync method is marked async
var tcs = new TaskCompletionSource<int>();
var runTaskOne = Deployment.CreateRunnerAndRunAsync(
() => new Deployment(new MockEngine(), new MockMonitor(new MyMocks()), null),
runner => tcs.Task);
// this will throw if we didn't protect
// the AsyncLocal scope of Deployment.Instance
// by keeping CreateRunnerAndRunAsync marked async
await Deployment.CreateRunnerAndRunAsync(
() => new Deployment(new MockEngine(), new MockMonitor(new MyMocks()), null),
runner => Task.FromResult(1));
tcs.SetResult(1);
await runTaskOne;
}
[Fact]
public async Task RunWaitsForOrphanedOutput()
{
var result = 0;
var tcs = new TaskCompletionSource<int>();
var runTaskOne = Deployment.CreateRunnerAndRunAsync(
() => new Deployment(new MockEngine(), new MockMonitor(new MyMocks()), null),
runner => runner.RunAsync(() =>
{
Output.Create(tcs.Task).Apply(i => result = i);
return Task.FromResult((IDictionary<string, object?>)new Dictionary<string, object?>());
}, null));
tcs.SetResult(42);
await runTaskOne;
Assert.Equal(42, result);
}
}
}