From maui-skills
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.
npx claudepluginhub davidortinau/maui-skills --plugin maui-skillsThis skill uses the workspace's default tool permissions.
```csharp
Guides .NET MAUI apps consuming .NET Aspire backends: AppHost configuration, mobile service discovery, HttpClient DI, MSAL.NET Entra ID auth, emulator networking, Blazor Hybrid.
Configures dependency injection in .NET MAUI: registers services in MauiProgram.cs, selects lifetimes, enables constructor injection, Shell auto-resolution, and platform-specific implementations.
Searches, retrieves, and installs Agent Skills from prompts.chat registry using MCP tools like search_skills and get_skill. Activates for finding skills, browsing catalogs, or extending Claude.
Share bugs, ideas, or general feedback.
// ❌ Creates socket exhaustion — each instance opens a new connection
public async Task<List<Item>> GetItemsAsync()
{
using var client = new HttpClient();
var response = await client.GetAsync("https://api.example.com/items");
// ...
}
// ✅ Register once in DI, inject everywhere
builder.Services.AddSingleton(sp => new HttpClient
{
BaseAddress = new Uri("https://api.example.com")
});
// ❌ Deadlocks on the UI thread
var items = _apiService.GetItemsAsync().Result;
// ✅ Always use async/await
var items = await _apiService.GetItemsAsync();
// ❌ Tries to deserialize error HTML/JSON as your model
var content = await response.Content.ReadAsStringAsync();
var items = JsonSerializer.Deserialize<List<Item>>(content, _jsonOptions);
// ✅ Check status first
response.EnsureSuccessStatusCode();
var content = await response.Content.ReadAsStringAsync();
var items = JsonSerializer.Deserialize<List<Item>>(content, _jsonOptions) ?? [];
// ❌ Absolute URIs in every method — hard to change, easy to typo
await _httpClient.GetAsync("https://api.example.com/api/items");
// ✅ Set BaseAddress in DI, use relative URIs in methods
await _httpClient.GetAsync("api/items");
// ❌ Crashes on network timeout, DNS failure, etc.
var items = await _apiService.GetItemsAsync();
// ✅ Catch both network and deserialization errors
try
{
var items = await _apiService.GetItemsAsync();
}
catch (HttpRequestException ex) { /* network or HTTP error */ }
catch (JsonException ex) { /* malformed response */ }
Local dev servers on http:// are blocked by default. Configure exceptions:
network_security_config.xml with cleartextTrafficPermitted="true" for 10.0.2.2NSAllowsLocalNetworking in Info.plistThe Android emulator maps 10.0.2.2 to the host machine. localhost refers to the emulator itself.
// ❌ On Android emulator, this hits the emulator, not your dev machine
new Uri("http://localhost:5000")
// ✅ Use the emulator's host loopback address
new Uri("http://10.0.2.2:5000")
iOS simulators use localhost directly.
APIs typically use camelCase; C# properties are PascalCase. Without JsonSerializerOptions, deserialization silently returns default values.
// ❌ Properties stay null/default — no error thrown
JsonSerializer.Deserialize<Item>(content);
// ✅ Configure casing policy
private static readonly JsonSerializerOptions _jsonOptions = new()
{
PropertyNamingPolicy = JsonNamingPolicy.CamelCase,
PropertyNameCaseInsensitive = true
};
| Scenario | Error handling approach |
|---|---|
| Failure is unexpected (auth'd endpoints) | EnsureSuccessStatusCode() — throws HttpRequestException |
| Need to branch on status codes | Check IsSuccessStatusCode or response.StatusCode |
| Network may be unreliable (mobile) | Wrap in try/catch for HttpRequestException |
| Response format may vary | Also catch JsonException |
HttpClient registered as singleton or via IHttpClientFactory — never created per-requestBaseAddress set in DI; service methods use relative URIsJsonSerializerOptions with CamelCase policy applied consistentlyIsSuccessStatusCode or EnsureSuccessStatusCode() checked before deserializingtry/catch for HttpRequestException and JsonException in ViewModel callsasync/await — no .Result or .Wait()10.0.2.2)NSAllowsLocalNetworking for local dev