MX.Api.Web.Extensions 2.1.4

There is a newer version of this package available.
See the version list below for details.
dotnet add package MX.Api.Web.Extensions --version 2.1.4
                    
NuGet\Install-Package MX.Api.Web.Extensions -Version 2.1.4
                    
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="MX.Api.Web.Extensions" Version="2.1.4" />
                    
For projects that support PackageReference, copy this XML node into the project file to reference the package.
<PackageVersion Include="MX.Api.Web.Extensions" Version="2.1.4" />
                    
Directory.Packages.props
<PackageReference Include="MX.Api.Web.Extensions" />
                    
Project file
For projects that support Central Package Management (CPM), copy this XML node into the solution Directory.Packages.props file to version the package.
paket add MX.Api.Web.Extensions --version 2.1.4
                    
#r "nuget: MX.Api.Web.Extensions, 2.1.4"
                    
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
#:package MX.Api.Web.Extensions@2.1.4
                    
#:package directive can be used in C# file-based apps starting in .NET 10 preview 4. Copy this into a .cs file before any lines of code to reference the package.
#addin nuget:?package=MX.Api.Web.Extensions&version=2.1.4
                    
Install as a Cake Addin
#tool nuget:?package=MX.Api.Web.Extensions&version=2.1.4
                    
Install as a Cake Tool

MX.Api.Web.Extensions

ASP.NET Core integration library providing extension methods for converting API response objects to HTTP results. Seamlessly bridges API clients and web applications following the API design pattern.

Installation

dotnet add package MX.Api.Web.Extensions

Key Features

  • 🔄 Response Conversion - Convert ApiResponse<T> to IActionResult with proper status codes
  • 📊 Smart Status Mapping - Automatic HTTP status code assignment based on response content
  • 🚀 Controller Simplification - Reduce boilerplate code in API controllers
  • 🔗 Client Integration - Seamless integration between API clients and web applications
  • 📋 Error Preservation - Maintain error details and metadata in HTTP responses

Quick Start

In API Controllers

[ApiController]
[Route("api/[controller]")]
public class UsersController : ControllerBase
{
    private readonly IUserService _userService;

    public UsersController(IUserService userService)
    {
        _userService = userService;
    }

    [HttpGet("{id}")]
    public async Task<IActionResult> GetUser(int id)
    {
        var user = await _userService.GetUserAsync(id);
        
        if (user == null)
        {
            var response = new ApiResponse<User>(
                new ApiError("USER_NOT_FOUND", $"User {id} not found"));
            return response.ToApiResult(HttpStatusCode.NotFound).ToHttpResult();
        }
        
        var successResponse = new ApiResponse<User>(user);
        return successResponse.ToApiResult().ToHttpResult();
    }

    [HttpPost]
    public async Task<IActionResult> CreateUser([FromBody] CreateUserRequest request)
    {
        var user = await _userService.CreateUserAsync(request);
        var response = new ApiResponse<User>(user);
        return response.ToApiResult(HttpStatusCode.Created).ToHttpResult();
    }

    [HttpDelete("{id}")]
    public async Task<IActionResult> DeleteUser(int id)
    {
        await _userService.DeleteUserAsync(id);
        var response = new ApiResponse(); // Success with no data
        return response.ToApiResult(HttpStatusCode.NoContent).ToHttpResult();
    }
}

With API Client Integration

[ApiController]
[Route("api/[controller]")]
public class ProxyController : ControllerBase
{
    private readonly IExternalApiClient _apiClient;

    public ProxyController(IExternalApiClient apiClient)
    {
        _apiClient = apiClient;
    }

    [HttpGet("users/{id}")]
    public async Task<IActionResult> GetUserFromExternalApi(int id)
    {
        // API client returns ApiResult<User>
        var apiResult = await _apiClient.GetUserAsync(id);
        
        // Convert directly to HTTP result
        return apiResult.ToHttpResult();
    }

    [HttpGet("products")]
    public async Task<IActionResult> GetProducts([FromQuery] int page = 1, [FromQuery] int pageSize = 10)
    {
        var apiResult = await _apiClient.GetProductsAsync(page, pageSize);
        return apiResult.ToHttpResult();
    }
}

Extension Methods

ApiResponse to ApiResult

Convert API response models to HTTP-aware results:

// Success response (200 OK)
var response = new ApiResponse<Product>(product);
var result = response.ToApiResult();

// Success with custom status code (201 Created)
var result = response.ToApiResult(HttpStatusCode.Created);

// Error response (400 Bad Request)
var errorResponse = new ApiResponse<Product>(
    new ApiError("VALIDATION_ERROR", "Invalid product data"));
var result = errorResponse.ToApiResult(HttpStatusCode.BadRequest);

ApiResult to IActionResult

Convert API results to ASP.NET Core action results:

public async Task<IActionResult> GetProduct(int id)
{
    var apiResult = await _productService.GetProductAsync(id);
    
    // Automatically maps status codes to appropriate IActionResult types
    return apiResult.ToHttpResult();
    
    // Results in:
    // - 200 OK -> Ok(response)
    // - 404 Not Found -> NotFound(response)  
    // - 400 Bad Request -> BadRequest(response)
    // - 500 Internal Server Error -> StatusCode(500, response)
}

Error Handling with Smart Mapping

public async Task<IActionResult> UpdateProduct(int id, [FromBody] UpdateProductRequest request)
{
    try
    {
        var product = await _productService.UpdateProductAsync(id, request);
        var response = new ApiResponse<Product>(product);
        return response.ToApiResult().ToHttpResult();
    }
    catch (ValidationException ex)
    {
        var errorResponse = new ApiResponse<Product>(
            new ApiError("VALIDATION_ERROR", ex.Message));
        return errorResponse.ToApiResult(HttpStatusCode.BadRequest).ToHttpResult();
    }
    catch (NotFoundException ex)
    {
        var errorResponse = new ApiResponse<Product>(
            new ApiError("PRODUCT_NOT_FOUND", ex.Message));
        return errorResponse.ToApiResult(HttpStatusCode.NotFound).ToHttpResult();
    }
}

Advanced Scenarios

Collection Responses with Pagination

[HttpGet]
public async Task<IActionResult> GetProducts(
    [FromQuery] int page = 1,
    [FromQuery] int pageSize = 10,
    [FromQuery] string? category = null)
{
    var products = await _productService.GetProductsAsync(page, pageSize, category);
    
    var collection = new CollectionModel<Product>
    {
        Items = products.Items
    };

    var response = new ApiResponse<CollectionModel<Product>>(collection)
    {
        Pagination = new ApiPagination
        {
            TotalCount = products.TotalCount,
            FilteredCount = products.FilteredCount,
            Skip = (page - 1) * pageSize,
            Top = pageSize,
            HasMore = products.FilteredCount > page * pageSize
        }
    };
    return response.ToApiResult().ToHttpResult();
}

Custom Status Code Mapping

public async Task<IActionResult> ProcessPayment([FromBody] PaymentRequest request)
{
    var result = await _paymentService.ProcessPaymentAsync(request);
    
    return result.Status switch
    {
        PaymentStatus.Success => new ApiResponse<PaymentResult>(result)
            .ToApiResult(HttpStatusCode.OK).ToHttpResult(),
            
        PaymentStatus.Pending => new ApiResponse<PaymentResult>(result)
            .ToApiResult(HttpStatusCode.Accepted).ToHttpResult(),
            
        PaymentStatus.InsufficientFunds => new ApiResponse<PaymentResult>(
            new ApiError("INSUFFICIENT_FUNDS", "Payment declined due to insufficient funds"))
            .ToApiResult(HttpStatusCode.PaymentRequired).ToHttpResult(),
            
        PaymentStatus.Declined => new ApiResponse<PaymentResult>(
            new ApiError("PAYMENT_DECLINED", "Payment was declined"))
            .ToApiResult(HttpStatusCode.BadRequest).ToHttpResult(),
            
        _ => new ApiResponse<PaymentResult>(
            new ApiError("PAYMENT_ERROR", "An error occurred processing the payment"))
            .ToApiResult(HttpStatusCode.InternalServerError).ToHttpResult()
    };
}

Metadata and Headers

public async Task<IActionResult> GetUserWithMetadata(int id)
{
    var user = await _userService.GetUserAsync(id);
    
    var response = new ApiResponse<User>(user)
    {
        Metadata = new Dictionary<string, string>
        {
            ["Cache-Control"] = "max-age=300",
            ["Last-Modified"] = DateTime.UtcNow.ToString("R"),
            ["X-User-Role"] = user.Role
        }
    };

    var result = response.ToApiResult();
    
    // Add metadata as response headers
    if (result.Result?.Metadata != null)
    {
        foreach (var (key, value) in result.Result.Metadata)
        {
            Response.Headers.Add(key, value);
        }
    }

    return result.ToHttpResult();
}

Integration Patterns

API Gateway/Proxy Pattern

[ApiController]
[Route("api/gateway/[controller]")]
public class GatewayController : ControllerBase
{
    private readonly IUserApiClient _userClient;
    private readonly IOrderApiClient _orderClient;

    public GatewayController(IUserApiClient userClient, IOrderApiClient orderClient)
    {
        _userClient = userClient;
        _orderClient = orderClient;
    }

    [HttpGet("users/{id}")]
    public async Task<IActionResult> GetUser(int id)
    {
        var result = await _userClient.GetUserAsync(id);
        return result.ToHttpResult(); // Direct passthrough
    }

    [HttpGet("users/{userId}/orders")]
    public async Task<IActionResult> GetUserOrders(int userId)
    {
        // Aggregate multiple API calls
        var userResult = await _userClient.GetUserAsync(userId);
        if (!userResult.IsSuccess)
            return userResult.ToHttpResult();

        var ordersResult = await _orderClient.GetOrdersByUserAsync(userId);
        return ordersResult.ToHttpResult();
    }
}

Backend for Frontend (BFF) Pattern

[ApiController]
[Route("api/bff/[controller]")]
public class DashboardController : ControllerBase
{
    private readonly IUserApiClient _userClient;
    private readonly IOrderApiClient _orderClient;
    private readonly IAnalyticsApiClient _analyticsClient;

    [HttpGet("summary/{userId}")]
    public async Task<IActionResult> GetDashboardSummary(int userId)
    {
        // Aggregate data from multiple APIs
        var userTask = _userClient.GetUserAsync(userId);
        var ordersTask = _orderClient.GetRecentOrdersAsync(userId, 5);
        var analyticsTask = _analyticsClient.GetUserAnalyticsAsync(userId);

        await Task.WhenAll(userTask, ordersTask, analyticsTask);

        // Compose custom response
        var summary = new DashboardSummary
        {
            User = userTask.Result.IsSuccess ? userTask.Result.Result?.Data : null,
            RecentOrders = ordersTask.Result.IsSuccess ? ordersTask.Result.Result?.Data : null,
            Analytics = analyticsTask.Result.IsSuccess ? analyticsTask.Result.Result?.Data : null
        };

        var response = new ApiResponse<DashboardSummary>(summary);
        return response.ToApiResult().ToHttpResult();
    }
}

Status Code Mappings

The extension methods automatically map common scenarios:

Condition HTTP Status Code IActionResult Type
ApiResponse.Errors == null 200 OK Ok()
Custom status provided As specified StatusCode()
ApiResult.IsNotFound 404 Not Found NotFound()
ApiResult.IsBadRequest 400 Bad Request BadRequest()
ApiResult.IsUnauthorized 401 Unauthorized Unauthorized()
Other error status codes As specified StatusCode()

Dependencies

This package depends on:

  • MX.Api.Abstractions - Core response models and interfaces
  • Microsoft.AspNetCore.Mvc.Core - ASP.NET Core MVC abstractions
  • Microsoft.Extensions.Logging - Logging abstractions

Documentation

  • 📖 Implementation Guide - API Providers - Building APIs with these extensions

  • 📖 API Design Patterns - Understanding the design principles

    [HttpPost] public async Task<IActionResult> CreateUser([FromBody] CreateUserRequest request) { var response = await _apiClient.CreateUserAsync(request);

      // Convert to Created result (HTTP 201)
      var result = response.ToCreatedResult();
    
      return result.ToHttpResult();
    

    }

    [HttpPut("{id}")] public async Task<IActionResult> UpdateUser(int id, [FromBody] UpdateUserRequest request) { var response = await _apiClient.UpdateUserAsync(id, request);

      // Convert to Accepted result (HTTP 202)
      var result = response.ToAcceptedResult();
    
      return result.ToHttpResult();
    

    } }


### Available Extension Methods

#### Basic Conversion Methods

- `ToApiResult(HttpStatusCode statusCode = HttpStatusCode.OK)` - Convert with specific status code
- `ToCreatedResult()` - Convert with HTTP 201 Created status
- `ToAcceptedResult()` - Convert with HTTP 202 Accepted status
- `ToNotFoundResult()` - Convert with HTTP 404 Not Found status
- `ToBadRequestResult()` - Convert with HTTP 400 Bad Request status
- `ToConflictResult()` - Convert with HTTP 409 Conflict status

#### Smart Error Handling Methods

- `ToApiResultWithErrorHandling()` - Automatically determines status based on errors and data:
  - Returns HTTP 200 OK if no errors and data is present
  - Returns HTTP 404 Not Found if no errors but data is null (for generic responses)
  - Returns HTTP 400 Bad Request if errors exist

### Basic Usage in Controllers

```csharp
[ApiController]
[Route("api/[controller]")]
public class UsersController : ControllerBase
{
    private readonly IUserApiClient _apiClient;

    public UsersController(IUserApiClient apiClient)
    {
        _apiClient = apiClient;
    }

    [HttpGet("{id}")]
    public async Task<IActionResult> GetUser(string id)
    {
        // Get data from your API client
        var response = await _apiClient.GetUserAsync(id);
        
        // Convert the API response to an appropriate HTTP response
        return response.ToHttpResult();
    }

    [HttpGet]
    public async Task<IActionResult> GetUsers([FromQuery] FilterOptions filter)
    {
        // Get collection from your API client
        var response = await _apiClient.GetUsersAsync(filter);
        
        // Convert the API response to an HTTP response with pagination headers
        return response.ToHttpResult(HttpContext);
    }
}

Working with Different Response Types

// Converting ApiResponse<T> to IActionResult
public async Task<IActionResult> GetResource(string id)
{
    ApiResponse<ResourceDto> response = await _apiClient.GetResourceAsync(id);
    return response.ToHttpResult();
}

// Converting ApiResult<T> to IActionResult
public async Task<IActionResult> GetResource(string id)
{
    ApiResult<ResourceDto> wrapper = await _apiClient.GetResourceWithWrapperAsync(id);
    return wrapper.ToHttpResult();
}

Adding Pagination Headers

When returning collections, you can add pagination headers to the response:

public async Task<IActionResult> GetResources([FromQuery] FilterOptions filter)
{
    var response = await _apiClient.GetResourcesAsync(filter);
    
    // Adds pagination headers to the HTTP response
    return response.ToHttpResult(HttpContext);
}

Error Handling

The extension methods automatically handle error responses:

public async Task<IActionResult> CreateResource([FromBody] ResourceCreateDto dto)
{
    var response = await _apiClient.CreateResourceAsync(dto);
    
    // Will return appropriate status code and error details if creation fails
    return response.ToHttpResult();
}

Advanced Usage

Custom Response Formatting

You can customize how responses are formatted:

public async Task<IActionResult> GetCustomFormattedResponse(string id)
{
    var response = await _apiClient.GetResourceAsync(id);
    
    return response.ToHttpResult(formatResponse: result => 
    {
        // Custom response formatting
        return new
        {
            resource = result,
            timestamp = DateTime.UtcNow,
            version = "1.0"
        };
    });
}

License

GPL-3.0-only

Product Compatible and additional computed target framework versions.
.NET net9.0 is compatible.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed.  net10.0 is compatible.  net10.0-android was computed.  net10.0-browser was computed.  net10.0-ios was computed.  net10.0-maccatalyst was computed.  net10.0-macos was computed.  net10.0-tvos was computed.  net10.0-windows was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last Updated
2.2.40 275 2/1/2026
2.2.34 101 1/31/2026
2.2.31 235 1/24/2026
2.2.9 508 11/30/2025
2.2.8 123 11/29/2025
2.2.7 121 11/29/2025
2.2.6 127 11/29/2025
2.2.5 134 11/29/2025
2.2.4 127 11/29/2025
2.2.3 123 11/29/2025
2.2.2 134 11/29/2025
2.2.1 131 11/29/2025
2.1.5.1-preview 107 11/29/2025
2.1.4 125 11/29/2025
2.1.0-preview 103 11/29/2025
2.0.200.1 121 11/29/2025
2.0.196.1 299 11/13/2025
2.0.195.1 946 11/6/2025
2.0.194.1 195 10/30/2025
2.0.193.1 192 10/23/2025
Loading failed