Skip to content

Commit f55adeb

Browse files
committed
Disable proxy for persistent container endpoints
1 parent ef8ddb1 commit f55adeb

5 files changed

+54
-36
lines changed

src/Aspire.Hosting/ApplicationModel/EndpointAnnotation.cs

+5
Original file line numberDiff line numberDiff line change
@@ -83,6 +83,11 @@ public int? Port
8383
}
8484
}
8585

86+
/// <summary>
87+
/// Indicates the the port is set with an explicit value.
88+
/// </summary>
89+
public bool IsPortSet => _port != null;
90+
8691
/// <summary>
8792
/// This is the port the resource is listening on. If the endpoint is used for the container, it is the container port.
8893
/// </summary>

src/Aspire.Hosting/ApplicationModel/ResourceExtensions.cs

+11-1
Original file line numberDiff line numberDiff line change
@@ -320,7 +320,7 @@ public static int GetReplicaCount(this IResource resource)
320320
/// Gets the lifetime type of the container for the specified resource.
321321
/// Defaults to <see cref="ContainerLifetime.Session"/> if no <see cref="ContainerLifetimeAnnotation"/> is found.
322322
/// </summary>
323-
/// <param name="resource">The resource to the get the ContainerLifetimeType for.</param>
323+
/// <param name="resource">The resource to get the ContainerLifetimeType for.</param>
324324
/// <returns>
325325
/// The <see cref="ContainerLifetime"/> from the <see cref="ContainerLifetimeAnnotation"/> for the resource (if the annotation exists).
326326
/// Defaults to <see cref="ContainerLifetime.Session"/> if the annotation is not set.
@@ -335,6 +335,16 @@ internal static ContainerLifetime GetContainerLifetimeType(this IResource resour
335335
return ContainerLifetime.Session;
336336
}
337337

338+
/// <summary>
339+
/// Determines whether a resource supports proxied endpoints/services. Returns true for non-container resources container resources with a lifetime other than <see cref="ContainerLifetime.Persistent"/>.
340+
/// </summary>
341+
/// <param name="resource">The resource to get proxy support for.</param>
342+
/// <returns>True if the resource supports proxied endpoints/services, false otherwise.</returns>
343+
internal static bool SupportsProxy(this IResource resource)
344+
{
345+
return !resource.IsContainer() || resource.GetContainerLifetimeType() != ContainerLifetime.Persistent;
346+
}
347+
338348
/// <summary>
339349
/// Get the top resource in the resource hierarchy.
340350
/// e.g. for a AzureBlobStorageResource, the top resource is the AzureStorageResource.

src/Aspire.Hosting/Dcp/ApplicationExecutor.cs

+23-21
Original file line numberDiff line numberDiff line change
@@ -1015,7 +1015,7 @@ private void AddAllocatedEndpointInfo(IEnumerable<AppResource> resources)
10151015
throw new InvalidDataException($"Service {svc.Metadata.Name} should have valid address at this point");
10161016
}
10171017

1018-
if (!sp.EndpointAnnotation.IsProxied && svc.AllocatedPort is null)
1018+
if (!sp.EndpointAnnotation.IsProxied && svc.AllocatedPort == null)
10191019
{
10201020
throw new InvalidOperationException($"Service '{svc.Metadata.Name}' needs to specify a port for endpoint '{sp.EndpointAnnotation.Name}' since it isn't using a proxy.");
10211021
}
@@ -1040,12 +1040,25 @@ private void PrepareServices()
10401040
// services produced by different resources).
10411041
HashSet<string> serviceNames = [];
10421042

1043+
List<(IResource, EndpointAnnotation)> endpointsWithHostUnset = [];
1044+
10431045
foreach (var sp in serviceProducers)
10441046
{
10451047
var endpoints = sp.Endpoints.ToArray();
10461048

10471049
foreach (var endpoint in endpoints)
10481050
{
1051+
if (!sp.ModelResource.SupportsProxy())
1052+
{
1053+
// If the resource can't be proxied, we need to enforce that on the annotation
1054+
endpoint.IsProxied = false;
1055+
}
1056+
1057+
if (sp.ModelResource.IsContainer() && !endpoint.IsProxied && !endpoint.IsPortSet && endpoint.TargetPort is int)
1058+
{
1059+
endpointsWithHostUnset.Add((sp.ModelResource, endpoint));
1060+
}
1061+
10491062
var candidateServiceName = endpoints.Length == 1
10501063
? GetObjectNameForResource(sp.ModelResource)
10511064
: GetObjectNameForResource(sp.ModelResource, endpoint.Name);
@@ -1070,6 +1083,15 @@ private void PrepareServices()
10701083
_appResources.Add(new ServiceAppResource(sp.ModelResource, svc, endpoint));
10711084
}
10721085
}
1086+
1087+
if (endpointsWithHostUnset.Any())
1088+
{
1089+
_logger.LogWarning("You have unproxied container endpoints without an explicit host port set. By default these endpoints will attempt to bind a host port that matches the container target port. This can lead to port conflicts if multiple containers are using the same target port(s). For containers running with a persistent lifetime or container endpoints with IsProxied disabled, we recommend specifying explicit host ports. For more information on container networking in .NET Aspire see: https://aka.ms/dotnet/aspire/container-networking");
1090+
foreach (var (resource, endpoint) in endpointsWithHostUnset)
1091+
{
1092+
_logger.LogInformation("'{EndpointName}' endpoint for '{ResourceName}' doesn't have a host port set. Attempting to bind host port '{TargetPort}'.", endpoint.Name, resource.Name, endpoint.TargetPort);
1093+
}
1094+
}
10731095
}
10741096

10751097
private void PrepareExecutables()
@@ -1626,10 +1648,6 @@ private async Task CreateContainerAsync(AppResource cr, ILogger resourceLogger,
16261648

16271649
dcpContainerResource.Spec.Env = [];
16281650

1629-
var proxiedPorts = new List<EndpointAnnotation>();
1630-
1631-
var isPersistentContainer = modelContainerResource.GetContainerLifetimeType() == ContainerLifetime.Persistent;
1632-
16331651
if (cr.ServicesProduced.Count > 0)
16341652
{
16351653
dcpContainerResource.Spec.Ports = new();
@@ -1657,25 +1675,9 @@ private async Task CreateContainerAsync(AppResource cr, ILogger resourceLogger,
16571675
}
16581676

16591677
dcpContainerResource.Spec.Ports.Add(portSpec);
1660-
1661-
if (isPersistentContainer)
1662-
{
1663-
if (ea.IsProxied)
1664-
{
1665-
proxiedPorts.Add(ea);
1666-
}
1667-
}
16681678
}
16691679
}
16701680

1671-
if (proxiedPorts.Count > 0)
1672-
{
1673-
_logger.LogWarning(
1674-
"'{ResourceName}' is configured with a persistent lifetime, but has proxied endpoints. The target port(s) ({TargetPorts}) will be exposed via a proxy while your App Host project is running, but may not otherwise be accessible between App Host runs. It is recommended to configure fixed unproxied endpoint ports for persistent containers. For more information on networking in .NET Aspire see: https://aka.ms/dotnet/aspire/networking",
1675-
modelContainerResource.Name,
1676-
string.Join(", ", proxiedPorts.Select(p => p.TargetPort)));
1677-
}
1678-
16791681
if (modelContainerResource.TryGetEnvironmentVariables(out var containerEnvironmentVariables))
16801682
{
16811683
var context = new EnvironmentCallbackContext(_executionContext, config, cancellationToken);

src/Aspire.Hosting/ProjectResourceBuilderExtensions.cs

+14-14
Original file line numberDiff line numberDiff line change
@@ -52,9 +52,9 @@ public static class ProjectResourceBuilderExtensions
5252
/// Example of adding a project to the application model.
5353
/// <code lang="csharp">
5454
/// var builder = DistributedApplication.CreateBuilder(args);
55-
///
55+
///
5656
/// builder.AddProject&lt;Projects.InventoryService&gt;("inventoryservice");
57-
///
57+
///
5858
/// builder.Build().Run();
5959
/// </code>
6060
/// </example>
@@ -67,7 +67,7 @@ public static class ProjectResourceBuilderExtensions
6767
}
6868

6969
/// <summary>
70-
/// Adds a .NET project to the application model.
70+
/// Adds a .NET project to the application model.
7171
/// </summary>
7272
/// <param name="builder">The <see cref="IDistributedApplicationBuilder"/>.</param>
7373
/// <param name="name">The name of the resource. This name will be used for service discovery when referenced in a dependency.</param>
@@ -85,9 +85,9 @@ public static class ProjectResourceBuilderExtensions
8585
/// Add a project to the app model via a project path.
8686
/// <code lang="csharp">
8787
/// var builder = DistributedApplication.CreateBuilder(args);
88-
///
88+
///
8989
/// builder.AddProject("inventoryservice", @"..\InventoryService\InventoryService.csproj");
90-
///
90+
///
9191
/// builder.Build().Run();
9292
/// </code>
9393
/// </example>
@@ -132,9 +132,9 @@ public static IResourceBuilder<ProjectResource> AddProject(this IDistributedAppl
132132
/// Example of adding a project to the application model.
133133
/// <code lang="csharp">
134134
/// var builder = DistributedApplication.CreateBuilder(args);
135-
///
135+
///
136136
/// builder.AddProject&lt;Projects.InventoryService&gt;("inventoryservice", launchProfileName: "otherLaunchProfile");
137-
///
137+
///
138138
/// builder.Build().Run();
139139
/// </code>
140140
/// </example>
@@ -170,9 +170,9 @@ public static IResourceBuilder<ProjectResource> AddProject(this IDistributedAppl
170170
/// Add a project to the app model via a project path.
171171
/// <code lang="csharp">
172172
/// var builder = DistributedApplication.CreateBuilder(args);
173-
///
173+
///
174174
/// builder.AddProject("inventoryservice", @"..\InventoryService\InventoryService.csproj", launchProfileName: "otherLaunchProfile");
175-
///
175+
///
176176
/// builder.Build().Run();
177177
/// </code>
178178
/// </example>
@@ -219,9 +219,9 @@ public static IResourceBuilder<ProjectResource> AddProject(this IDistributedAppl
219219
/// Example of adding a project to the application model.
220220
/// <code lang="csharp">
221221
/// var builder = DistributedApplication.CreateBuilder(args);
222-
///
222+
///
223223
/// builder.AddProject&lt;Projects.InventoryService&gt;("inventoryservice", options => { options.LaunchProfileName = "otherLaunchProfile"; });
224-
///
224+
///
225225
/// builder.Build().Run();
226226
/// </code>
227227
/// </example>
@@ -259,9 +259,9 @@ public static IResourceBuilder<ProjectResource> AddProject(this IDistributedAppl
259259
/// Add a project to the app model via a project path.
260260
/// <code lang="csharp">
261261
/// var builder = DistributedApplication.CreateBuilder(args);
262-
///
262+
///
263263
/// builder.AddProject("inventoryservice", @"..\InventoryService\InventoryService.csproj", options => { options.LaunchProfileName = "otherLaunchProfile"; });
264-
///
264+
///
265265
/// builder.Build().Run();
266266
/// </code>
267267
/// </example>
@@ -724,7 +724,7 @@ private static void SetAspNetCoreUrls(this IResourceBuilder<ProjectResource> bui
724724
}
725725

726726
// If the endpoint is proxied, we will use localhost as the target host since DCP will be forwarding the traffic
727-
var targetHost = e.EndpointAnnotation.IsProxied ? "localhost" : e.EndpointAnnotation.TargetHost;
727+
var targetHost = e.EndpointAnnotation.IsProxied && builder.Resource.SupportsProxy() ? "localhost" : e.EndpointAnnotation.TargetHost;
728728

729729
aspnetCoreUrls.Append($"{e.Property(EndpointProperty.Scheme)}://{targetHost}:{e.Property(EndpointProperty.TargetPort)}");
730730
first = false;

src/Aspire.Hosting/PublicAPI.Unshipped.txt

+1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ Aspire.Hosting.ApplicationModel.DockerfileBuildAnnotation.ContextPath.get -> str
1010
Aspire.Hosting.ApplicationModel.DockerfileBuildAnnotation.DockerfileBuildAnnotation(string! contextPath, string! dockerfilePath, string? stage) -> void
1111
Aspire.Hosting.ApplicationModel.DockerfileBuildAnnotation.DockerfilePath.get -> string!
1212
Aspire.Hosting.ApplicationModel.DockerfileBuildAnnotation.Stage.get -> string?
13+
Aspire.Hosting.ApplicationModel.EndpointAnnotation.IsPortSet.get -> bool
1314
Aspire.Hosting.ApplicationModel.EndpointNameAttribute
1415
Aspire.Hosting.ApplicationModel.EndpointNameAttribute.EndpointNameAttribute() -> void
1516
Aspire.Hosting.ApplicationModel.HealthReportSnapshot

0 commit comments

Comments
 (0)