Aspire is a really great technology for interconnecting all kind of services. It’s so good that you will probably use it in all your old projects.
Ports randomization+proxy is a really good feature to spawn all your services, and maybe launch n instances through tests.
We recently encountered an issue, we wanted to resolve a dynamic TargetPort for an EndPointResource for a TCP service.
In our App « MyAppService » we had a port listening configured like that in appsettings.json :
{
"ListenerConfig":
{
"Port" : 1200
}
}
And Somewhere in the code
var tcpListener = new TcpListener(IPAddress.Loopback, listenerConfig.Port);
tcpListener.Start();
As our service will be launched multiple times with tests thanks to DistributedApplicationTestingBuilder
, we can’t bind multiple times to the same port, and we also don’t want to convert our project to a Docker container (which would solve the problem easily)
In Aspire we add this
builder.AddProject<Projects.MyAppServiceService>("myappservice")
.WithEndpoint("mytcpendpoint", a => a.Port = 1200);
And now we want to inject the TargetPort in the environement variable ListenerConfig__Port
in order to override the port.
var resourceMyApp = builder.AddProject<Projects.MyAppServiceService>("myappservice")
.WithEndpoint("mytcpendpoint", a => a.Port = 1200);
resourceMyApp = resourceMyApp.WithEnvironment((EnvironmentCallbackContext cb) =>
{
var endpointTcp = resourceProxySmtp.GetEndpoint("mytcpendpoint");
// THIS WILL NOT WORK, targetPort will be null
var targetPort = endpointTcp.TargetPort;
cb.EnvironmentVariables.Add("ListenerConfig__Port", targetPort);
}
It would have work but unfortunately, TargetPort will be null here, despite the documentation telling « A callback that allows for deferred execution for computing many environment variables. This runs after resources have been allocated by the orchestrator and allows access to other resources to resolve computed data, e.g. connection strings, ports. »
While digging on the sourcecode of Aspire, I found this code and specially these 2 lines in the ProjectResourceBuilderExtensions.SetKestrelUrlOverrideEnvVariables
method:
var url = ReferenceExpression.Create($"{e.EndpointAnnotation.UriScheme}://{host}:{e.Property(EndpointProperty.TargetPort)}");
// We use special config system environment variables to perform the override.
context.EnvironmentVariables[$"Kestrel__Endpoints__{e.EndpointAnnotation.Name}__Url"] = url;
So, Aspire use a special syntax called ReferenceExpression, which will be called after all WithEnvironment to Resolve endpoints target ports.
Here is the final solution :
var resourceMyApp = builder.AddProject<Projects.MyAppServiceService>("myappservice");
.WithEndpoint("mytcpendpoint", a => a.Port = 1200);
resourceMyApp = resourceMyApp.WithEnvironment((EnvironmentCallbackContext cb) =>
{
var endpointTcp = resourceMyApp.GetEndpoint("mytcpendpoint");
var targetPortExpression = endpointTcp.Property(EndpointProperty.TargetPort);
var targetPortRefExpression = ReferenceExpression.Create($"{targetPortExpression}");
cb.EnvironmentVariables.Add("ListenerConfig__Port", targetPortRefExpression);
}
The solution was hard to find, but I’m happy to found it – in the source code rather than in the documentation, huh – Enjoy !