diff --git a/ApiToolKit.cs b/ApiToolKit.cs index 9e6f0cf..cd0c90d 100644 --- a/ApiToolKit.cs +++ b/ApiToolKit.cs @@ -9,6 +9,7 @@ using Newtonsoft.Json.Linq; using Newtonsoft.Json; using System.Diagnostics; +using Microsoft.Extensions.DependencyInjection; using System.Text.Json; using static System.Web.HttpUtility; @@ -424,6 +425,31 @@ public class ATError public string StackTrace { get; set; } } +public interface IApiToolkitClientFactory +{ + HttpClient CreateClient(ATOptions options); +} + +public class ApiToolkitClientFactory : IApiToolkitClientFactory +{ + private readonly IHttpClientFactory _httpClientFactory; + private readonly IServiceProvider _serviceProvider; + + public ApiToolkitClientFactory(IHttpClientFactory httpClientFactory, IServiceProvider serviceProvider) + { + _httpClientFactory = httpClientFactory; + _serviceProvider = serviceProvider; + } + + public HttpClient CreateClient(ATOptions options) + { + var client = _httpClientFactory.CreateClient(); + + var handler = _serviceProvider.GetRequiredService(); + handler.SetOptions(options); + return new HttpClient(handler); + } +} public class ATOptions { @@ -436,9 +462,13 @@ public class ObservingHandler : DelegatingHandler { private readonly HttpContext _context; private readonly Func _publishMessageAsync; - private readonly ATOptions _options; + private ATOptions _options; private readonly string _project_id; private readonly string? _msg_id; + public void SetOptions(ATOptions options) + { + _options = options; + } public ObservingHandler(Func publishMessage, string project_id, HttpContext? httpContext = null, ATOptions? options = null) : base(new HttpClientHandler()) { _context = httpContext ?? throw new ArgumentNullException(nameof(httpContext)); diff --git a/ApiToolkitTests.cs b/ApiToolkitTests.cs index b5c593e..144dd1f 100644 --- a/ApiToolkitTests.cs +++ b/ApiToolkitTests.cs @@ -10,7 +10,9 @@ using Microsoft.AspNetCore.Hosting; using Microsoft.Extensions.Hosting; using Microsoft.AspNetCore.Builder; - +using Moq; +using System.Net.Http; +using Microsoft.Extensions.DependencyInjection; public class RedactTests { [Test] @@ -208,3 +210,74 @@ public async Task MiddlewareTest_ReturnsNotFoundForRequest() } } + + public class ApiToolkitClientFactoryTests + { + private Mock _httpClientFactoryMock; + private Mock _serviceProviderMock; + private Mock _observingHandlerMock; + private IApiToolkitClientFactory _apiToolkitClientFactory; + + [SetUp] + public void SetUp() + { + _httpClientFactoryMock = new Mock(); + _serviceProviderMock = new Mock(); + _observingHandlerMock = new Mock(); + + // Set up the service provider to return the observing handler mock + _serviceProviderMock + .Setup(sp => sp.GetService(typeof(ObservingHandler))) + .Returns(_observingHandlerMock.Object); + + // Initialize the ApiToolkitClientFactory with mocks + _apiToolkitClientFactory = new ApiToolkitClientFactory(_httpClientFactoryMock.Object, _serviceProviderMock.Object); + } + + [Test] + public void CreateClient_ShouldReturnHttpClient_WithConfiguredHandler() + { + // Arrange + var options = new ATOptions + { + PathWildCard = "/posts/{id}", + RedactHeaders = new List { "User-Agent" }, + RedactRequestBody = new List { "$.user.password" }, + RedactResponseBody = new List { "$.user.data.email" } + }; + + // Act + var client = _apiToolkitClientFactory.CreateClient(options); + + // Assert + Assert.NotNull(client); + _observingHandlerMock.Verify(h => h.SetOptions(options), Times.Once); + } + + [Test] + public void CreateClient_ShouldUseObservingHandler_FromServiceProvider() + { + // Act + var client = _apiToolkitClientFactory.CreateClient(new ATOptions()); + + // Assert + _serviceProviderMock.Verify(sp => sp.GetService(typeof(ObservingHandler)), Times.Once); + } + + [Test] + public void CreateClient_ShouldThrowException_WhenHandlerNotRegistered() + { + // Arrange + var invalidServiceProvider = new Mock(); + invalidServiceProvider.Setup(sp => sp.GetService(typeof(ObservingHandler))) + .Returns(null); // Simulate missing handler registration + + var factory = new ApiToolkitClientFactory(_httpClientFactoryMock.Object, invalidServiceProvider.Object); + + // Act & Assert + Assert.Throws(() => factory.CreateClient(new ATOptions())); + } + } + + + diff --git a/README.md b/README.md index 118b593..c03f7d9 100644 --- a/README.md +++ b/README.md @@ -38,6 +38,11 @@ Next, initialize APItoolkit in your application's entry point (e.g., `Program.cs using ApiToolkit.Net; // Initialize the APItoolkit client +builder.Services.AddTransient(); + +// Register the custom API Toolkit Client Factory +builder.Services.AddSingleton(); + var config = new Config { ApiKey = "{ENTER_YOUR_API_KEY_HERE}", @@ -60,6 +65,66 @@ app.Use(async (context, next) => # ... ``` +## Usage + +You can now use the IApiToolKitClientFactory Interface to directly make your Http requests + +```csharp +public class MyService +{ + private readonly IApiToolkitClientFactory _apiToolkitClientFactory; + + public MyService(IApiToolkitClientFactory apiToolkitClientFactory) + { + _apiToolkitClientFactory = apiToolkitClientFactory; + } + + public async Task GetPostAsync() + { + var options = new ATOptions + { + PathWildCard = "/posts/{id}", + RedactHeaders = new[] { "User-Agent" }, + RedactRequestBody = new[] { "$.user.password" }, + RedactResponseBody = new[] { "$.user.data.email" } + }; + + var client = _apiToolkitClientFactory.CreateClient(options); + var response = await client.GetAsync("https://jsonplaceholder.typicode.com/posts/1"); + return await response.Content.ReadAsStringAsync(); + } +} +``` + +Traditional Middleware Setup +If you prefer to set up the middleware traditionally, here's how you can initialize APItoolkit in your application's entry point (e.g., Program.cs): + +```csharp +using ApiToolkit.Net; + +// Initialize the APItoolkit client +var config = new Config +{ + ApiKey = "{ENTER_YOUR_API_KEY_HERE}", + Debug = false, + Tags = new List { "environment: production", "region: us-east-1" }, + ServiceVersion: "v2.0", +}; +var client = await APIToolkit.NewClientAsync(config); + +// Register the middleware to use the initialized client +app.Use(async (context, next) => +{ + var apiToolkit = new APIToolkit(next, client); + await apiToolkit.InvokeAsync(context); +}); + +// app.UseEndpoint(..) +// other middleware and logic +// ... +``` + + > [!NOTE] > > - Please make sure the APItoolkit middleware is added before `UseEndpoint` and other middleware are initialized. diff --git a/apitoolkit-dotnet.csproj b/apitoolkit-dotnet.csproj index 573ccf1..6c6e0ae 100644 --- a/apitoolkit-dotnet.csproj +++ b/apitoolkit-dotnet.csproj @@ -28,6 +28,7 @@ +