From maui-skills
Guides .NET MAUI apps consuming .NET Aspire backends: AppHost configuration, mobile service discovery, HttpClient DI, MSAL.NET Entra ID auth, emulator networking, Blazor Hybrid.
npx claudepluginhub davidortinau/maui-skills --plugin maui-skillsThis skill uses the workspace's default tool permissions.
MAUI apps are NOT orchestrated by the AppHost — they run on devices/simulators and connect over the network. This changes everything:
Guides .NET MAUI apps in consuming REST APIs using HttpClient, System.Text.Json, DI services, full CRUD operations, error handling, and platform-specific HTTP configs like clear-text traffic.
Configures dependency injection in .NET MAUI: registers services in MauiProgram.cs, selects lifetimes, enables constructor injection, Shell auto-resolution, and platform-specific implementations.
Orchestrates .NET services with Aspire: AppHost configuration, service defaults, Postgres/Redis/RabbitMQ integrations, service discovery, and dashboard for local development.
Share bugs, ideas, or general feedback.
MAUI apps are NOT orchestrated by the AppHost — they run on devices/simulators and connect over the network. This changes everything:
https+http://apiservice).WithReference() in the AppHostAddServiceDefaults() (not an ASP.NET Core host)// ❌ MAUI doesn't implement IServiceMetadata — this will fail
builder.AddProject<Projects.MyMauiApp>("mauiapp");
// ✅ Only add backend services to the AppHost
var apiService = builder.AddProject<Projects.MyApp_ApiService>("apiservice");
// ❌ This only resolves inside the Aspire AppHost orchestration
client.BaseAddress = new Uri("https+http://apiservice");
// ✅ Use real endpoints
client.BaseAddress = new Uri("https://localhost:7001");
The Android emulator has its own network stack. localhost points to the emulator itself, not the host machine.
// ❌ Fails silently on Android emulator
client.BaseAddress = new Uri("https://localhost:7001");
// ✅ Use 10.0.2.2 for host loopback on Android emulator
#if ANDROID && DEBUG
client.BaseAddress = new Uri("https://10.0.2.2:7001");
#else
client.BaseAddress = new Uri("https://localhost:7001");
#endif
The emulator won't trust your HTTPS dev cert. During local development:
// ⚠️ Development only — never ship this
#if ANDROID && DEBUG
builder.Services.AddHttpClient<IWeatherApiClient, WeatherApiClient>(client =>
{
client.BaseAddress = new Uri("https://10.0.2.2:7001");
})
.ConfigurePrimaryHttpMessageHandler(() => new HttpClientHandler
{
ServerCertificateCustomValidationCallback = (_, _, _, _) => true
});
#endif
MAUI uses PublicClientApplication (MSAL.NET). The Entra Aspire auth skill (entra-id-aspire-authentication) is for the API/Blazor Server side only. For the MAUI side, use the maui-authentication skill.
Ensure your MAUI project also targets net10.0-* TFMs to match the Aspire backend.
In MAUI Blazor Hybrid apps calling Aspire services, authentication happens at the MAUI layer (MSAL.NET), not in the Blazor WebView. Don't use AddMicrosoftIdentityWebApp or server-side OIDC patterns in the MAUI app.
MAUI projects should not use AddServiceDefaults() because:
For telemetry correlation, configure OpenTelemetry separately using System.Diagnostics APIs.
| Scenario | What to do |
|---|---|
| Debugging API issues | Use the Aspire dashboard at https://localhost:17178 |
| Debugging MAUI + API | Run Aspire AppHost in one terminal, MAUI in another |
| VS: simultaneous debug | Open two VS instances — one per project |
| VS Code | Two terminals, or Aspire extension + MAUI extension |
| Platform | localhost works? | HTTPS dev cert trusted? | Special config needed? |
|---|---|---|---|
| iOS Simulator | ✅ Yes | ✅ Yes (macOS keychain) | ATS exception for HTTP |
| Android Emulator | ❌ Use 10.0.2.2 | ❌ No | Network security config |
| Mac Catalyst | ✅ Yes | ✅ Yes | None |
| Windows | ✅ Yes | ✅ Yes | None |
AddProjecthttps+http:// URIs10.0.2.2 instead of localhostPublicClientApplication, not confidential clientnet10.0-*AddServiceDefaults() is NOT called in the MAUI project