Skip to content

Commit

Permalink
fixed client factory
Browse files Browse the repository at this point in the history
  • Loading branch information
nicolaparo committed Feb 7, 2024
1 parent 7d0fdd4 commit dd71418
Show file tree
Hide file tree
Showing 7 changed files with 69 additions and 85 deletions.
38 changes: 37 additions & 1 deletion samples/ChatGptBlazor.Wasm/Components/ChatGptStream.razor
Original file line number Diff line number Diff line change
@@ -1,14 +1,33 @@
@using ChatGptNet.Extensions;
@using ChatGptNet.Models;
@using ChatGptNet;
@using ChatGptNet.ServiceConfigurations
@using Markdig;

@inject IChatGptClient chatGptClient
@inject IChatGptClientFactory chatGptClientFactory
@inject IJSRuntime jsRuntime

<div class="h-100" style="position: relative;">
<div class="w-100 h-100" style="position: absolute;">
<div class="h-100 d-flex flex-column" style="max-height: 100%">

<div class="row mt-4">
<div class="col">
<input type="text" class="form-control" @bind="azureResource" placeholder="Azure OpenAI Resource" />
</div>
<div class="col">
<input type="password" class="form-control" @bind="apiKey" placeholder="Api Key" />
</div>
<div class="col">
<input type="text" class="form-control" @bind="model" placeholder="Model Name" />
</div>
<div class="col">
<button type="button" class="btn btn-primary" @onclick="Connect">
Connect
</button>
</div>
</div>

<div class="my-4" style="height:100%; max-height: 100%; overflow-y: auto;">

@foreach (var message in messages)
Expand Down Expand Up @@ -62,6 +81,12 @@
private string? message;
private bool isAsking;

private string azureResource;
private string apiKey;
private string model;

private IChatGptClient? chatGptClient = null;

private bool CanSendMessage()
=> !string.IsNullOrWhiteSpace(message) && !isAsking;

Expand All @@ -73,6 +98,17 @@
}
}

private void Connect()
{
chatGptClient = chatGptClientFactory.CreateClient(options =>
{
options.UseAzure(azureResource, apiKey, authenticationType: AzureAuthenticationType.ApiKey);
options.DefaultModel = model;
options.MessageLimit = 16; // Default: 10
options.MessageExpiration = TimeSpan.FromMinutes(5); // Default: 1 hour
});
}

private async Task AskAsync()
{
if (!isAsking)
Expand Down
24 changes: 2 additions & 22 deletions samples/ChatGptBlazor.Wasm/Program.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
using ChatGptBlazor.Wasm;
using ChatGptNet;
using ChatGptNet.ServiceConfigurations;
using Microsoft.AspNetCore.Components.Web;
using Microsoft.AspNetCore.Components.WebAssembly.Hosting;

Expand All @@ -10,27 +11,6 @@
builder.Services.AddScoped(sp => new HttpClient { BaseAddress = new Uri(builder.HostEnvironment.BaseAddress) });

// Adds ChatGPT service and configure options via code.
builder.Services.AddChatGptClientFactory(options =>
{
// OpenAI.
//options.UseOpenAI(apiKey: "", organization: "");

// Azure OpenAI Service.
options.UseAzure(resourceName: "", apiKey: "", authenticationType: AzureAuthenticationType.ApiKey);

options.DefaultModel = "my-model";
options.MessageLimit = 16; // Default: 10
options.MessageExpiration = TimeSpan.FromMinutes(5); // Default: 1 hour
},
httpClient =>
{
// Configures retry policy on the inner HttpClient using Polly.
httpClient.AddStandardResilienceHandler(options =>
{
options.AttemptTimeout.Timeout = TimeSpan.FromMinutes(1);
options.CircuitBreaker.SamplingDuration = TimeSpan.FromMinutes(3);
options.TotalRequestTimeout.Timeout = TimeSpan.FromMinutes(3);
});
});
builder.Services.AddChatGptClientFactory();

await builder.Build().RunAsync();
40 changes: 16 additions & 24 deletions samples/ChatGptBlazor.Wasm/Properties/launchSettings.json
Original file line number Diff line number Diff line change
@@ -1,40 +1,32 @@
{
"iisSettings": {
"windowsAuthentication": false,
"anonymousAuthentication": true,
"iisExpress": {
"applicationUrl": "http://localhost:8206",
"sslPort": 44385
}
},
"profiles": {
"http": {
"commandName": "Project",
"dotnetRunMessages": true,
"launchBrowser": true,
"inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}",
"applicationUrl": "http://localhost:5297",
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
}
},
"dotnetRunMessages": true,
"inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}",
"applicationUrl": "http://localhost:5297"
},
"https": {
"commandName": "Project",
"dotnetRunMessages": true,
"launchBrowser": true,
"inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}",
"applicationUrl": "https://localhost:7021;http://localhost:5297",
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
}
},
"IIS Express": {
"commandName": "IISExpress",
"launchBrowser": true,
},
"dotnetRunMessages": true,
"inspectUri": "{wsProtocol}://{url.hostname}:{url.port}/_framework/debug/ws-proxy?browser={browserInspectUri}",
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
}
"applicationUrl": "https://localhost:7021;http://localhost:5297"
}
},
"iisSettings": {
"windowsAuthentication": false,
"anonymousAuthentication": true,
"iisExpress": {
"applicationUrl": "http://localhost:8206",
"sslPort": 44385
}
}
}
}
12 changes: 6 additions & 6 deletions src/ChatGptNet/ChatGptClientFactory.cs
Original file line number Diff line number Diff line change
Expand Up @@ -6,25 +6,25 @@ internal class ChatGptClientFactory : IChatGptClientFactory
{
private readonly IServiceProvider services;
private readonly IChatGptCache chatGptCache;
private readonly ChatGptOptions defaultOptions;
private readonly ChatGptOptionsBuilder defaultOptions;

public ChatGptClientFactory(IServiceProvider services, IChatGptCache chatGptCache, ChatGptOptions defaultOptions)
public ChatGptClientFactory(IServiceProvider services, IChatGptCache chatGptCache, ChatGptOptionsBuilder defaultOptions)
{
this.services = services;
this.chatGptCache = chatGptCache;
this.defaultOptions = defaultOptions;
}

public IChatGptClient CreateClient(Action<IServiceProvider, ChatGptOptions>? setupAction)
public IChatGptClient CreateClient(Action<IServiceProvider, ChatGptOptionsBuilder>? setupAction)
{
var options = defaultOptions with { };

if (setupAction is not null)
setupAction(services, options);

return new ChatGptClient(new HttpClient(), chatGptCache, options);
return new ChatGptClient(new HttpClient(), chatGptCache, options.Build());
}
public IChatGptClient CreateClient(Action<ChatGptOptions>? setupAction)
public IChatGptClient CreateClient(Action<ChatGptOptionsBuilder>? setupAction)
{
if (setupAction is null)
return CreateClient();
Expand All @@ -33,6 +33,6 @@ public IChatGptClient CreateClient(Action<ChatGptOptions>? setupAction)
}
public IChatGptClient CreateClient()
{
return CreateClient((Action<IServiceProvider, ChatGptOptions>?)null);
return CreateClient((Action<IServiceProvider, ChatGptOptionsBuilder>?)null);
}
}
34 changes: 5 additions & 29 deletions src/ChatGptNet/ChatGptFactoryServiceCollectionExtensions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -42,44 +42,20 @@ public static IServiceCollection AddChatGptClientFactory(this IServiceCollection
return services.AddChatGptClientFactory(null);

Check warning on line 42 in src/ChatGptNet/ChatGptFactoryServiceCollectionExtensions.cs

View workflow job for this annotation

GitHub Actions / Analyze (csharp)

Cannot convert null literal to non-nullable reference type.

Check warning on line 42 in src/ChatGptNet/ChatGptFactoryServiceCollectionExtensions.cs

View workflow job for this annotation

GitHub Actions / Analyze (csharp)

Cannot convert null literal to non-nullable reference type.

Check warning on line 42 in src/ChatGptNet/ChatGptFactoryServiceCollectionExtensions.cs

View workflow job for this annotation

GitHub Actions / Analyze (csharp)

Cannot convert null literal to non-nullable reference type.
}

/// <summary>
/// Registers a <see cref="ChatGptClientFactory"/> instance reading configuration from the specified <see cref="IConfiguration"/> source.
/// </summary>
/// <param name="services">The <see cref="IServiceCollection"/> to add services to.</param>
/// <param name="configuration">The <see cref="IConfiguration"/> being bound.</param>
/// <param name="sectionName">The name of the configuration section that holds ChatGPT settings (default: ChatGPT).</param>
/// <returns>A reference to this instance after the operation has completed.</returns>
public static IServiceCollection AddChatGptClientFactory(this IServiceCollection services, IConfiguration configuration, string sectionName = "ChatGPT")
{
ArgumentNullException.ThrowIfNull(services);
ArgumentNullException.ThrowIfNull(configuration);

var options = new ChatGptOptions();
var configurationSection = configuration.GetSection(sectionName);
configurationSection.Bind(options);

// Creates the service configuration (OpenAI or Azure) according to the configuration settings.
options.ServiceConfiguration = ChatGptServiceConfiguration.Create(configurationSection);

SetMissingDefaults(options);

services.AddChatGptClientFactoryCore(options);

return services;
}

private static void AddChatGptClientFactoryCore(this IServiceCollection services, ChatGptOptionsBuilder options)
private static IServiceCollection AddChatGptClientFactoryCore(this IServiceCollection services, ChatGptOptionsBuilder deafultOptions)
{
services.AddMemoryCache();
services.AddSingleton<IChatGptCache, ChatGptMemoryCache>();

var httpClientBuilder = services.AddHttpClient<IChatGptClientFactory, ChatGptClientFactory>();
services.AddSingleton<IChatGptClientFactory, ChatGptClientFactory>(
s => new ChatGptClientFactory(s, s.GetRequiredService<IChatGptCache>(), options)
s => new ChatGptClientFactory(s, s.GetRequiredService<IChatGptCache>(), deafultOptions)
);

return services;
}

private static void SetMissingDefaults(ChatGptOptionsBuilder options)
private static void SetMissingDefaults(ChatGptOptions options)
{
// If the provider is OpenAI and no default model has been specified, uses gpt-3.5-turbo by default.
if (options.ServiceConfiguration is OpenAIChatGptServiceConfiguration && string.IsNullOrWhiteSpace(options.DefaultModel))
Expand Down
2 changes: 1 addition & 1 deletion src/ChatGptNet/ChatGptOptionsBuilder.cs
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ namespace ChatGptNet;
/// <summary>
/// Builder class to define settings for configuring ChatGPT.
/// </summary>
public class ChatGptOptionsBuilder
public record ChatGptOptionsBuilder
{
/// <summary>
/// Gets or sets the configuration settings for accessing the service.
Expand Down
4 changes: 2 additions & 2 deletions src/ChatGptNet/IChatGptClientFactory.cs
Original file line number Diff line number Diff line change
Expand Up @@ -10,14 +10,14 @@ public interface IChatGptClientFactory
/// </summary>
/// <param name="setupAction">The <see cref="Action{ChatGptOptions}"/> to configure the provided <see cref="ChatGptOptions"/>.</param>
/// <returns>A new <see cref="IChatGptClient"/></returns>
IChatGptClient CreateClient(Action<IServiceProvider, ChatGptOptions>? setupAction);
IChatGptClient CreateClient(Action<IServiceProvider, ChatGptOptionsBuilder>? setupAction);

/// <summary>
/// Creates a new instance of a ChatGptClient configured with the supplied action.
/// </summary>
/// <param name="setupAction">The <see cref="Action{ChatGptOptions}"/> to configure the provided <see cref="ChatGptOptions"/>.</param>
/// <returns>A new <see cref="IChatGptClient"/></returns>
IChatGptClient CreateClient(Action<ChatGptOptions>? setupAction);
IChatGptClient CreateClient(Action<ChatGptOptionsBuilder>? setupAction);

/// <summary>
/// Creates a new instance of a ChatGptClient.
Expand Down

0 comments on commit dd71418

Please sign in to comment.