Al desarrollar aplicaciones ASP.NET Core (APIs, MVC, Razor Pages o Blazor) y ejecutarlas en local, ya sabes que todas ellas comparten el mismo nombre de host: localhost, y lo único que las diferencia es el puerto en el que se encuentran a la escucha.
Esto puede dar lugar a ciertos problemas. Aparte de no ser sencillo identificarlas a partir de la URL mostrada en el navegador, hay ciertos recursos web que pueden mezclarse entre proyectos (por ejemplo, las cookies) y pueden dar lugar a comportamientos inesperados.
En ASP.NET Core 10 se ha puesto solución a esto añadiendo el soporte para el TLD (Top Level Domain) .localhost, lo que permite que cada aplicación pueda tener su propio nombre de host único.
Lo vemos a continuación.
Soporte para TLD .localhost en ASP.NET Core 10
El nombre .localhost es un dominio de nivel superior (TLD, Top Level Domain) reservado por las RFC 2606 y 6761 para ser utilizado en entornos de desarrollo y pruebas, y siempre está asociado a la dirección de loopback, es decir, a la dirección local del propio equipo.
Por tanto, cualquier nombre de host que termine en .localhost (por ejemplo, miapp.localhost, api.localhost, MyApp.dev.localhost, etc.) debería resolverse siempre a la dirección IP 127.0.0.1 o ::1, dependiendo de si se utiliza IPv4 o IPv6.
A partir de ASP.NET Core 10, Kestrel entenderá que todos los nombres de host que terminen en .localhost son alias válidos para localhost, por lo que podemos utilizarlos para configurar el servidor, definir las URLs de nuestras aplicaciones (tanto en launchSettings.json como en variables de entorno), etc. Esto nos permitirá usar distintos nombres de host para cada aplicación, evitando los problemas mencionados anteriormente.
Otro tema que han actualizado es el certificado HTTPS utilizado en desarrollo. Dado que el antiguo certificado solo era válido para localhost, al utilizar otros nombres de host el navegador mostraría advertencias de seguridad. Por esta razón, el certificado que se registra al instalar .NET 10 es un wildcard del dominio *.dev.localhost.
Observad que no han podido hacerlo directamente para soportar
*.localhost. Han tenido que introducir el subdominio.devpor delante porque no es posible crear wildcards sobre top level domains comolocalhost.
En la práctica, esto implica que podremos asignar a nuestras aplicaciones ASP.NET Core nombres de host como miapp.dev.localhost, api.dev.localhost, etc., aunque no es algo que esté habilitado por defecto.
Para activar esta característica, debemos seleccionar la opción "Use the .dev.localhost TLD in the application URL" al crear una nueva aplicación, como se muestra en la siguiente imagen:
Al hacerlo, se configurará el launchsettings.json para que utilice el nombre de proyecto como subdominio. Por ejemplo, a continuación se muestra la configuración generada para el proyecto MyWebApplication, donde podemos ver el uso del host mywebapplication.dev.localhost:
{
"$schema": "https://json.schemastore.org/launchsettings.json",
"profiles": {
"http": {
"commandName": "Project",
"dotnetRunMessages": true,
"launchBrowser": true,
"applicationUrl": "http://mywebapplication.dev.localhost:5244",
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
}
},
"https": {
"commandName": "Project",
"dotnetRunMessages": true,
"launchBrowser": true,
"applicationUrl": "https://mywebapplication.dev.localhost:7279;
http://mywebapplication.dev.localhost:5244",
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
}
}
}
}
También podemos conseguir lo mismo si utilizamos la línea de comandos de .NET para crear los proyectos. Por ejemplo, el siguiente comando crea una aplicación MVC que utiliza el TLD .dev.localhost:
C:\MyProjects\MyWebApp>dotnet new web --localhost-tld
The template "ASP.NET Core Empty" was created successfully.
Processing post-creation actions...
Restoring C:\MyProjects\MyWebApp\MyWebApp.csproj:
Restore succeeded.
C:\MyProjects\MyWebApp>type .\Properties\launchSettings.json
{
"$schema": "https://json.schemastore.org/launchsettings.json",
"profiles": {
"http": {
"commandName": "Project",
"dotnetRunMessages": true,
"launchBrowser": true,
"applicationUrl": "http://mywebapp.dev.localhost:5024",
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
}
},
"https": {
"commandName": "Project",
"dotnetRunMessages": true,
"launchBrowser": true,
"applicationUrl": "https://mywebapp.dev.localhost:7125;
http://mywebapp.dev.localhost:5024",
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
}
}
}
}
C:\MyProjects\MyWebApp>_
Con estas configuraciones, al lanzar el proyecto se utilizarán estos nombres de host exclusivos para el proyecto, evitando los problemas de compartir el mismo localhost entre varias aplicaciones.
C:\MyProjects\MyWebApp> dotnet run
Using launch settings from C:\MyProjects\MyWebApp\Properties\launchSettings.json...
Building...
info: Microsoft.Hosting.Lifetime[14]
Now listening on: http://mywebapp.dev.localhost:5024
info: Microsoft.Hosting.Lifetime[14]
Now listening on: http://localhost:5024
info: Microsoft.Hosting.Lifetime[0]
Application started. Press Ctrl+C to shut down.
info: Microsoft.Hosting.Lifetime[0]
Hosting environment: Development
info: Microsoft.Hosting.Lifetime[0]
Content root path: C:\MyProjects\MyWebApp
_
Un último detalle importante: podremos utilizar los nombres de host personalizados cuando accedamos a las aplicaciones desde el navegador web, pero probablemente no desde otras herramientas como curl, Postman, etc., o al menos de directamente.
Esto se debe a que, aunque el TLD .localhost está reservado para este propósito, no todas las aplicaciones cliente o los sistemas operativos resuelven automáticamente estos nombres a la dirección de loopback. Por esta razón, en estos casos tendríamos que añadir entradas manuales en el archivo hosts de nuestro sistema operativo para que los nombres personalizados funcionen correctamente.
Afortunadamente, los navegadores web modernos implementan esta resolución de forma automática, por lo que funcionará directamente 🙂
Publicado en Variable not found.
Publicado por José M. Aguilar a las 8:05 a. m.
Etiquetas: aspnetcore, novedades
Esta semana me gustaría destacar en primer lugar el detallado análisis de Andrew Lock sobre el proceso de arranque de una aplicación .NET, muy interesante para saber cómo funcionan las cosas por dentro.
Bipin Joshi da un gran repaso a Kestrel, el motor de todas las aplicaciones ASP.NET Core: cómo configurarlo, tunearlo, ejecutarlo y buenas prácticas de uso.
Milan Jovanović nos recuerda que el happy path no es el único camino en nuestras aplicaciones, y nos muestra técnicas para desacoplar servicios para hacerlos más robustos.
Por último, Ricardo Peres nos habla de la relación entre tipos anulables y miembros requeridos en C#, algo que, cuando empezamos a trabajar con ellos, puede resultar algo confuso.
El resto de enlaces, a continuación.
Por si te lo perdiste...
- Soporte para colecciones en parámetros
paramde .NET 9
José M. Aguilar - Establecer textos por defecto y localizados en validaciones de ASP.NET Core MVC
José M. Aguilar
.NET
- TUnit — Why I Spent 2 Years Building a New .NET Testing Framework
Tom Longhurst - C# 14: User-Defined Compound Assignment Operators
Anthony Giretti - Under Pressure: How Queueing Systems Handle Backpressure with Examples in C#
Scott Galloway - Precision Matters in C#: Correctly Handling Money, Time Zones, and Date Ranges
Sudhir Mangla - Exploring the .NET boot process via host tracing
Andrew Lock - .NET Job Scheduling — The Landscape
Martin Stühmer - Why Do You Need To Write Architecture Tests in .NET
Anton Martyniuk - Fetching GitHub content from C#
Thomas Ardal - Using CSX Scripts for Quick C# Testing (English)
Scott Galloway - The Worst Security Vulnerability in Akka.NET
Aaron Stannard - IDistributedCache (Redis)
Josef Ottosson - Adding Intelligence to Blazor with Telerik Smart Components
Héctor Pérez - How to ensure your expert C# knowledge doesn't make you a TypeScript noob
Lewis Cianci - Ix.NET v7.0: .NET 10 and LINQ for IAsyncEnumerable<T>
Ian Griffiths - Real-Time Recommendation Engines in .NET: Hybrid Retrieval, Deep Learning, and Vector Search
Sudhir Mangla - One shot tool execution in .NET 10
Bart Wullems - .NET Job Scheduling — Hangfire and Persistent Reliability
Martin Stühmer - Nullable and Required Types
Ricardo Peres - Unit Testing HttpClient WITHOUT Mocks
Scott Galloway
Años atrás, en los inicios de la informática, se programaba de forma bastante diferente: conectando cables y usando paneles de interruptores. Por eso me ha parecido muy curioso el simulador del Minivac 601, un ordenador de los años 60, donde podemos probar de primera mano cómo era su experiencia de uso.
Y continuando con temas vintage, Microsoft ha anunciado la liberación del código fuente de los juegos Zork I, II y III, la mítica saga de aventuras conversacionales de los años 80. Pura historia del software.
También esta semana encontramos una interesante lectura de Sudhir Mangla donde explica cómo usar patrones modernos y características recientes de C# para construir modelos más expresivos, seguros y mantenibles que los que ofrece la aplicación estricta de SOLID.
El resto de enlaces interesantes recopilados esta semana, entre los que podéis encontrar información sobre .NET 10, ASP.NET Core, IA, desarrollo web y mucho más, los tenéis a continuación.
Por si te lo perdiste...
- Creando GUIDs con orden natural en .NET 9
José M. Aguilar - Localizar errores de validación del binding en ASP.NET Core MVC
José M. Aguilar
.NET
- Introducing C# 14
Uwe Keim - Introducing F# 10
Adam Boniecki - Exploring C# File-based Apps in .NET 10
Milan Jovanović - How to Upgrade to .NET 10 LTS - Complete Guide for .NET Global Tools with Multi-Targeting
John Smith - Post-Quantum Cryptography in .NET
Jeremy Barton - Reinventing how .NET Builds and Ships (Again)
Gábor Szabó - Companies complaining .NET moves too fast should just pay for post-EOL support
Andrew Lock - .NET 10 Testing: Microsoft Finally Fixed the Test Runner (Mostly)
Martin Stühmer - Optimize GUID creation performance in .NET applications
Gérald Barré - How to Update .NET on Wsl or Ubuntu
Steve Smith - No more public partial class Program in .NET 10
Steven Giesel - Streamlining Remote C# Scripts with .NET 10 runfile
Daniel Cazzulino - A Practical Approach to DI Variation in .NET: Introducing Child Service Providers
Fernando Escolar - State of Native AOT in .NET 10
Peter Ritchie
Hace unos días se lanzó .NET 10 y, con él, C# 14, una nueva versión del lenguaje que viene con varias novedades interesantes que mejoran la productividad y la experiencia de desarrollo.
Las más destacables son:
- Miembros extensores
- Asignaciones condicionales
- Propiedades semi-automáticas
- Simplificación del uso de
nameofcon tipos genéricos - Eventos y constructores parciales
- Conversiones implícitas para
Span<T>yReadOnlySpan<T> - Simplificación de parámetros de lambdas
Les damos un vistazo rápido a continuación.
La semana pasada, sin duda el protagonismo se lo ha llevado el lanzamiento de .NET 10, con multitud de artículos que repasan los cambios y características introducidas en esta versión, aunque hay mucho más.
En primer lugar, destacamos el artículo de CampusMVP: .NET 10 - Informe técnico para desarrolladores: novedades y mejoras importantes, imperdible para ponernos rápidamente al día.
También OpenAI ha publicado interesantes artículos, uno sobre los problemas de seguridad que supone el uso de prompt injection, y otro sobre el uso de una nueva técnica que mejora la interpretabilidad de las redes neuronales, algo difícil de conseguir con los modelos tradicionales. Y bueno, ha aprovechado también presentar su nuevo modelo GPT-5.1.
Por último, destacar el artículo de Sudhir Mangla sobre gestión de memoria en .NET, que repasa conceptos clave como stack, heap, Span<T>, Memory<T>, ArrayPool y otros temas que debemos conocer para crear aplicaciones .NET de alto rendimiento.
Por si te lo perdiste...
- Registro y obtención de servicios con nombre (keyed services) en .NET
José M. Aguilar - Cargar automáticamente objetos del modelo en acciones ASP.NET Core MVC
José M. Aguilar
.NET
- Announcing .NET 10
Mark Adamson - .NET 10 - Informe técnico para desarrolladores: novedades y mejoras importantes
CampusMVP - Implementando Azure Service Bus con Clean Architecture
Gerson Azabache Martínez - What′s new in .NET 10
nullGeorgii Tormozov - How to Target Excel Metadata in C# for Smarter Document Management
Mohan Chandran - Scanning .NET Assemblies for References & Isolator with References Scanning
Ricardo Peres - Mocking Statics in .NET - We Made the Impossible Possible
Eli Lopian - Capturing Console Output in xUnit.v3
Steven Giesel - Modern .NET Reflection with UnsafeAccessor
NDepend Team - .NET 10 and the Release Cycle Paradox & .NET 10: Timing Is the New Technical Debt
Martin Stühmer - New Features in .NET 10 and C# 14
Anton Martyniuk - Missing Security Features in .NET 10
Khalid Abuhakmeh - Working with Excel files in .NET: OpenXML vs EPPlus vs ClosedXML
Ali Hamza Ansari - .NET 10: What’s New for Authentication and Authorization
Andrea Chiarelli - C# 14 Extension Members: Enhancing Polyfill Libraries
Gérald Barré - Unlocking Collection Expressions for your own types
Bart Wullems - Syncfusion Delivers Day-1 Support for .NET 10 Across UI Components and Document Libraries
Meena Alagiah - Memory Management Masterclass: Stack vs Heap, Span<T>, Memory<T>, and ArrayPool in High-Performance C#
Sudhir Mangla - Are C# 14's new features worth updating your app to .NET 10?
David Grace - Use TimeSpan Or Specify Units In Duration Properties And Parameters
Steve Smith - .NET 10: Boring by Design, Reliable by Default
Martin Stühmer
La obsesión por los tipos primitivos es un code smell bastante común en el desarrollo de software. Gerson Azabache nos explica cómo los Value Objects en .NET pueden ayudarnos a combatir este problema.
Decidir qué framework de frontend nos conviene aprender siempre es una tarea complicada. Para ayudarnos con esto, José Manuel Alarcón ha preparado una comparativa detallada entre React, Angular, Vue.js y Svelte.
El servidor MCP de Azure permite que agentes o aplicaciones basadas en IA interactúen con los recursos que gestionamos en la nube. Juan Irigoyen da un vistazo a esta herramienta en su artículo.
Daniel Roth nos cuenta cómo Copilot Studio utiliza .NET y WebAssembly para mejorar el rendimiento en este entorno low-code.
Estos, y muchos más enlaces a contenidos interesantes, a continuación.
Por si te lo perdiste...
- Quizás no estés comparando con null de forma totalmente correcta
José M. Aguilar - Novedades de System.Text.Json en .NET 9
José M. Aguilar
.NET
- Value Objects en .NET: la cura contra el code smell Primitive Obsession
Gerson Azabache Martínez - Interesting new C# 14 features coming with .NET 10
Thomas Levesque - Modern Defensive Programming in .NET — Unified Throw-Helpers and Multi-Framework Compatibility
Martin Stühmer - How Copilot Studio uses .NET and WebAssembly for performance and innovation
Daniel Roth - Easier reflection with [UnsafeAccessorType] in .NET 10
Andrew Lock - Define a default startup project in your slnx file
Mark Downie - A Modern Guide to Using OAuth 2.0 with C# and Visual Studio Code
Dean Hume - ArrayPool Benchmarks: We have a problem
Steven Giesel - Stop Breaking Multi-Targeting Builds with String Comparisons
Martin Stühmer - C# 14: The Null-conditional Assignment – Thomas Claudius Huber
Thomas Claudius Huber - .NET 10 Breaking Changes To Keep An Eye On When Upgrading
Khalid Abuhakmeh

