Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added HttpClient Injection #83

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 7 additions & 0 deletions OpenAI_API/Chat/ChatEndpoint.cs
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,13 @@ public class ChatEndpoint : EndpointBase, IChatEndpoint
/// <param name="api"></param>
internal ChatEndpoint(OpenAIAPI api) : base(api) { }

/// <summary>
/// Constructor of the api endpoint. Rather than instantiating this yourself, access it through an instance of <see cref="OpenAIAPI"/> as <see cref="OpenAIAPI.Completions"/>.
/// </summary>
/// <param name="httpClient"></param>
/// <param name="api"></param>
internal ChatEndpoint(HttpClient httpClient, OpenAIAPI api) : base(api) { }

/// <summary>
/// Creates an ongoing chat which can easily encapsulate the conversation. This is the simplest way to use the Chat endpoint.
/// </summary>
Expand Down
7 changes: 7 additions & 0 deletions OpenAI_API/Completions/CompletionEndpoint.cs
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,13 @@ public class CompletionEndpoint : EndpointBase, ICompletionEndpoint
/// <param name="api"></param>
internal CompletionEndpoint(OpenAIAPI api) : base(api) { }

/// <summary>
/// Constructor of the api endpoint. Rather than instantiating this yourself, access it through an instance of <see cref="OpenAIAPI"/> as <see cref="OpenAIAPI.Completions"/>.
/// </summary>
/// <param name="httpClient"></param>
/// <param name="api"></param>
internal CompletionEndpoint(HttpClient httpClient, OpenAIAPI api) : base(httpClient, api) { }

#region Non-streaming

/// <summary>
Expand Down
10 changes: 9 additions & 1 deletion OpenAI_API/Embedding/EmbeddingEndpoint.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
using OpenAI_API.Models;
using System.Net.Http;
using OpenAI_API.Models;
using System.Threading.Tasks;

namespace OpenAI_API.Embedding
Expand All @@ -24,6 +25,13 @@ public class EmbeddingEndpoint : EndpointBase, IEmbeddingEndpoint
/// <param name="api"></param>
internal EmbeddingEndpoint(OpenAIAPI api) : base(api) { }

/// <summary>
/// Constructor of the api endpoint. Rather than instantiating this yourself, access it through an instance of <see cref="OpenAIAPI"/> as <see cref="OpenAIAPI.Embeddings"/>.
/// </summary>
/// <param name="httpClient"></param>
/// <param name="api"></param>
internal EmbeddingEndpoint(HttpClient httpClient, OpenAIAPI api) : base(httpClient, api) { }

/// <summary>
/// Ask the API to embedd text using the default embedding model <see cref="Model.AdaTextEmbedding"/>
/// </summary>
Expand Down
30 changes: 23 additions & 7 deletions OpenAI_API/EndpointBase.cs
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,8 @@ public abstract class EndpointBase
{
private const string UserAgent = "okgodoit/dotnet_openai_api";

private HttpClient _httpClient;

/// <summary>
/// The internal reference to the API, mostly used for authentication
/// </summary>
Expand All @@ -32,6 +34,18 @@ internal EndpointBase(OpenAIAPI api)
this._Api = api;
}

/// <summary>
/// Constructor of the api endpoint base, to be called from the contructor of any devived classes.
/// Rather than instantiating any endpoint yourself, access it through an instance of <see cref="OpenAIAPI"/>.
/// </summary>
/// <param name="httpClient"></param>
/// <param name="api"></param>
internal EndpointBase(HttpClient httpClient, OpenAIAPI api)
{
this._Api = api;
this._httpClient = httpClient;
}

/// <summary>
/// The name of the endpoint, which is the final path segment in the API URL. Must be overriden in a derived class.
/// </summary>
Expand Down Expand Up @@ -67,15 +81,17 @@ protected HttpClient GetClient()
_Api.SharedHttpClient.
}
*/

HttpClient client = new HttpClient();
client.DefaultRequestHeaders.Authorization = new System.Net.Http.Headers.AuthenticationHeaderValue("Bearer", _Api.Auth.ApiKey);

_httpClient ??= new HttpClient();

_httpClient.DefaultRequestHeaders.Authorization = new System.Net.Http.Headers.AuthenticationHeaderValue("Bearer", _Api.Auth.ApiKey);
// Further authentication-header used for Azure openAI service
client.DefaultRequestHeaders.Add("api-key", _Api.Auth.ApiKey);
client.DefaultRequestHeaders.Add("User-Agent", UserAgent);
if (!string.IsNullOrEmpty(_Api.Auth.OpenAIOrganization)) client.DefaultRequestHeaders.Add("OpenAI-Organization", _Api.Auth.OpenAIOrganization);
_httpClient.DefaultRequestHeaders.Add("api-key", _Api.Auth.ApiKey);
_httpClient.DefaultRequestHeaders.Add("User-Agent", UserAgent);
if (!string.IsNullOrEmpty(_Api.Auth.OpenAIOrganization))
_httpClient.DefaultRequestHeaders.Add("OpenAI-Organization", _Api.Auth.OpenAIOrganization);

return client;
return _httpClient;
}

/// <summary>
Expand Down
7 changes: 7 additions & 0 deletions OpenAI_API/Files/FilesEndpoint.cs
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,13 @@ public class FilesEndpoint : EndpointBase, IFilesEndpoint
/// <param name="api"></param>
internal FilesEndpoint(OpenAIAPI api) : base(api) { }

/// <summary>
/// Constructor of the api endpoint. Rather than instantiating this yourself, access it through an instance of <see cref="OpenAIAPI"/> as <see cref="OpenAIAPI.Files"/>.
/// </summary>
/// <param name="httpClient"></param>
/// <param name="api"></param>
internal FilesEndpoint(HttpClient httpClient, OpenAIAPI api) : base(httpClient, api) { }

/// <summary>
/// The name of the endpoint, which is the final path segment in the API URL. For example, "files".
/// </summary>
Expand Down
8 changes: 8 additions & 0 deletions OpenAI_API/Images/ImageGenerationEndpoint.cs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
using OpenAI_API.Models;
using System;
using System.Collections.Generic;
using System.Net.Http;
using System.Text;
using System.Threading.Tasks;

Expand All @@ -22,6 +23,13 @@ public class ImageGenerationEndpoint : EndpointBase, IImageGenerationEndpoint
/// <param name="api"></param>
internal ImageGenerationEndpoint(OpenAIAPI api) : base(api) { }

/// <summary>
/// Constructor of the api endpoint. Rather than instantiating this yourself, access it through an instance of <see cref="OpenAIAPI"/> as <see cref="OpenAIAPI.ImageGenerations"/>.
/// </summary>
/// <param name="httpClient"></param>
/// <param name="api"></param>
internal ImageGenerationEndpoint(HttpClient httpClient, OpenAIAPI api) : base(httpClient, api) { }

/// <summary>
/// Ask the API to Creates an image given a prompt.
/// </summary>
Expand Down
8 changes: 8 additions & 0 deletions OpenAI_API/Model/ModelsEndpoint.cs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
using Newtonsoft.Json;
using System;
using System.Collections.Generic;
using System.Net.Http;
using System.Threading.Tasks;

namespace OpenAI_API.Models
Expand All @@ -21,6 +22,13 @@ public class ModelsEndpoint : EndpointBase, IModelsEndpoint
/// <param name="api"></param>
internal ModelsEndpoint(OpenAIAPI api) : base(api) { }

/// <summary>
/// Constructor of the api endpoint. Rather than instantiating this yourself, access it through an instance of <see cref="OpenAIAPI"/> as <see cref="OpenAIAPI.Models"/>.
/// </summary>
/// <param name="httpClient"></param>
/// <param name="api"></param>
internal ModelsEndpoint(HttpClient httpClient, OpenAIAPI api) : base(httpClient, api) { }

/// <summary>
/// Get details about a particular Model from the API, specifically properties such as <see cref="Model.OwnedBy"/> and permissions.
/// </summary>
Expand Down
8 changes: 8 additions & 0 deletions OpenAI_API/Moderation/ModerationEndpoint.cs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
using OpenAI_API.Models;
using System;
using System.Collections.Generic;
using System.Net.Http;
using System.Text;
using System.Threading.Tasks;

Expand All @@ -27,6 +28,13 @@ public class ModerationEndpoint : EndpointBase, IModerationEndpoint
/// <param name="api"></param>
internal ModerationEndpoint(OpenAIAPI api) : base(api) { }

/// <summary>
/// Constructor of the api endpoint. Rather than instantiating this yourself, access it through an instance of <see cref="OpenAIAPI"/> as <see cref="OpenAIAPI.Moderation"/>.
/// </summary>
/// <param name="httpClient"></param>
/// <param name="api"></param>
internal ModerationEndpoint(HttpClient httpClient, OpenAIAPI api) : base(httpClient, api) { }

/// <summary>
/// Ask the API to classify the text using the default model.
/// </summary>
Expand Down
36 changes: 35 additions & 1 deletion OpenAI_API/OpenAIAPI.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
using OpenAI_API.Chat;
using System.Net.Http;
using OpenAI_API.Chat;
using OpenAI_API.Completions;
using OpenAI_API.Embedding;
using OpenAI_API.Files;
Expand Down Expand Up @@ -47,6 +48,23 @@ public OpenAIAPI(APIAuthentication apiKeys = null)
ImageGenerations = new ImageGenerationEndpoint(this);
}

/// <summary>
/// Creates a new entry point to the OpenAPI API, handling auth and allowing access to the various API endpoints
/// </summary>
/// <param name="httpClient">The HttpClient to use while calling OpenAI's API. By injecting it, you can customize its behavior.</param>
/// <param name="apiKeys">The API authentication information to use for API calls, or <see langword="null"/> to attempt to use the <see cref="APIAuthentication.Default"/>, potentially loading from environment vars or from a config file.</param>
public OpenAIAPI(HttpClient httpClient, APIAuthentication apiKeys = null)
{
this.Auth = apiKeys.ThisOrDefault();
Completions = new CompletionEndpoint(httpClient, this);
Models = new ModelsEndpoint(httpClient, this);
Files = new FilesEndpoint(httpClient, this);
Embeddings = new EmbeddingEndpoint(httpClient,this);
Chat = new ChatEndpoint(httpClient, this);
Moderation = new ModerationEndpoint(httpClient, this);
ImageGenerations = new ImageGenerationEndpoint(httpClient, this);
}

/// <summary>
/// Instantiates a version of the API for connecting to the Azure OpenAI endpoint instead of the main OpenAI endpoint.
/// </summary>
Expand All @@ -62,6 +80,22 @@ public static OpenAIAPI ForAzure(string YourResourceName, string deploymentId, A
return api;
}

/// <summary>
/// Instantiates a version of the API for connecting to the Azure OpenAI endpoint instead of the main OpenAI endpoint.
/// </summary>
/// <param name="httpClient">The HttpClient to use while calling OpenAI's API. By injecting it, you can customize its behavior.</param>
/// <param name="YourResourceName">The name of your Azure OpenAI Resource</param>
/// <param name="deploymentId">The name of your model deployment. You're required to first deploy a model before you can make calls.</param>
/// <param name="apiKey">The API authentication information to use for API calls, or <see langword="null"/> to attempt to use the <see cref="APIAuthentication.Default"/>, potentially loading from environment vars or from a config file. Currently this library only supports the api-key flow, not the AD-Flow.</param>
/// <returns></returns>
public static OpenAIAPI ForAzure(HttpClient httpClient, string YourResourceName, string deploymentId, APIAuthentication apiKey = null)
{
OpenAIAPI api = new OpenAIAPI(httpClient, apiKey);
api.ApiVersion = "2022-12-01";
api.ApiUrlFormat = $"https://{YourResourceName}.openai.azure.com/openai/deployments/{deploymentId}/" + "{1}?api-version={0}";
return api;
}

/// <summary>
/// Text generation is the core function of the API. You give the API a prompt, and it generates a completion. The way you “program” the API to do a task is by simply describing the task in plain english or providing a few written examples. This simple approach works for a wide range of use cases, including summarization, translation, grammar correction, question answering, chatbots, composing emails, and much more (see the prompt library for inspiration).
/// </summary>
Expand Down
24 changes: 24 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ Console.WriteLine(result);
* [Files API](#files-for-fine-tuning)
* [Image APIs (DALL-E)](#images)
* [Azure](#azure)
* [Customize the HTTP Client](#customize-the-http-client)
* [Additonal Documentation](#documentation)
* [License](#license)

Expand Down Expand Up @@ -276,6 +277,29 @@ OpenAIAPI api = OpenAIAPI.ForAzure("YourResourceName", "deploymentId", "api-key"

You may then use the `api` object like normal. You may also specify the `APIAuthentication` is any of the other ways listed in the [Authentication](#authentication) section above. Currently this library only supports the api-key flow, not the AD-Flow.

## Customize the HTTP Client

You can customize the HttpClient used in the wrapper by specifing it in the constructor or in the `ForAzure` static method.
```csharp
builder.Services.AddHttpClient<OpenAiBusinessService>()
.SetHandlerLifetime(TimeSpan.FromMinutes(30))
.AddPolicyHandler(RetryPolicy.GetTooManyRequestRetryPolicy())
```
And then in your class
```csharp
public class OpenAiBusinessService
{
private readonly OpenAIAPI _azureOpenAiApiWrapper;
private readonly OpenAIAPI _openAiWrapper;

public OpenAiBusinessService(HttpClient httpClient)
{
_azureOpenAiApiWrapper = OpenAIAPI.ForAzure("resource", "deploymentId", "apiKey");
_openAiWrapper = new OpenAIAPI(httpClient, new APIAuthentication("apiKey"));

}
}
```
## Documentation

Every single class, method, and property has extensive XML documentation, so it should show up automatically in IntelliSense. That combined with the official OpenAI documentation should be enough to get started. Feel free to open an issue here if you have any questions. Better documentation may come later.
Expand Down