diff --git a/.github/FUNDING.yml b/.github/FUNDING.yml new file mode 100755 index 0000000..d749e5c --- /dev/null +++ b/.github/FUNDING.yml @@ -0,0 +1,6 @@ +patreon: havendv +ko_fi: havendv +custom: [ + "https://www.paypal.me/havendv", + "https://www.buymeacoffee.com/havendv", + "https://www.upwork.com/freelancers/~017b1ad6f6af9cc189"] diff --git a/.github/ISSUE_TEMPLATE/bug-report.yaml b/.github/ISSUE_TEMPLATE/bug-report.yaml new file mode 100755 index 0000000..95303ba --- /dev/null +++ b/.github/ISSUE_TEMPLATE/bug-report.yaml @@ -0,0 +1,41 @@ +name: Bug report +description: File a bug report +title: "Bug title" +labels: [bug] +body: + - type: textarea + validations: + required: true + attributes: + label: Describe the bug + description: Please enter a short, clear description of the bug. + + - type: textarea + validations: + required: true + attributes: + label: Steps to reproduce the bug + description: Please provide any required setup and steps to reproduce the behavior. + placeholder: | + 1. Go to '...' + 2. Click on '....' + + - type: textarea + attributes: + label: Expected behavior + description: Please provide a description of what you expected to happen + + - type: textarea + attributes: + label: Screenshots + description: If applicable, add screenshots here to help explain your problem + + - type: textarea + attributes: + label: NuGet package version + description: Specify the version you're using. + + - type: textarea + attributes: + label: Additional context + description: Enter any other applicable info here diff --git a/.github/workflows/dotnet.yml b/.github/workflows/dotnet.yml new file mode 100755 index 0000000..fe68d5a --- /dev/null +++ b/.github/workflows/dotnet.yml @@ -0,0 +1,13 @@ +name: Build, test and publish +on: [ push ] + +jobs: + build-test-publish: + name: Build, test and publish + uses: HavenDV/workflows/.github/workflows/dotnet_build-test-publish.yml@main + with: + generate-build-number: false + conventional-commits-publish-conditions: false + additional-test-arguments: '--logger GitHubActions' + secrets: + nuget-key: ${{ secrets.NUGET_KEY }} \ No newline at end of file diff --git a/.gitignore b/.gitignore index 8a30d25..370da60 100644 --- a/.gitignore +++ b/.gitignore @@ -396,3 +396,4 @@ FodyWeavers.xsd # JetBrains Rider *.sln.iml +/.idea/ diff --git a/AI21.sln b/AI21.sln new file mode 100755 index 0000000..58a5189 --- /dev/null +++ b/AI21.sln @@ -0,0 +1,53 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio Version 16 +VisualStudioVersion = 16.0.30204.135 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{E793AF18-4371-4EBD-96FC-195EB1798855}" + ProjectSection(SolutionItems) = preProject + .gitignore = .gitignore + src\Directory.Build.props = src\Directory.Build.props + .github\workflows\dotnet.yml = .github\workflows\dotnet.yml + LICENSE = LICENSE + docs\openapi.nswag = docs\openapi.nswag + docs\openapi.yaml = docs\openapi.yaml + README.md = README.md + EndProjectSection +EndProject +Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "libs", "libs", "{61E7E11E-4558-434C-ACE8-06316A3097B3}" + ProjectSection(SolutionItems) = preProject + src\libs\Directory.Build.props = src\libs\Directory.Build.props + EndProjectSection +EndProject +Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "tests", "tests", "{AAA11B78-2764-4520-A97E-46AA7089A588}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AI21", "src\libs\AI21\AI21.csproj", "{0028BC85-0064-4CE8-A21A-C1F5E922BD59}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "AI21.IntegrationTests", "src\tests\AI21.IntegrationTests\AI21.IntegrationTests.csproj", "{4678AC8D-9C8F-415C-8CE4-3F7895A05455}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Any CPU = Debug|Any CPU + Release|Any CPU = Release|Any CPU + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {0028BC85-0064-4CE8-A21A-C1F5E922BD59}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {0028BC85-0064-4CE8-A21A-C1F5E922BD59}.Debug|Any CPU.Build.0 = Debug|Any CPU + {0028BC85-0064-4CE8-A21A-C1F5E922BD59}.Release|Any CPU.ActiveCfg = Release|Any CPU + {0028BC85-0064-4CE8-A21A-C1F5E922BD59}.Release|Any CPU.Build.0 = Release|Any CPU + {4678AC8D-9C8F-415C-8CE4-3F7895A05455}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {4678AC8D-9C8F-415C-8CE4-3F7895A05455}.Debug|Any CPU.Build.0 = Debug|Any CPU + {4678AC8D-9C8F-415C-8CE4-3F7895A05455}.Release|Any CPU.ActiveCfg = Release|Any CPU + {4678AC8D-9C8F-415C-8CE4-3F7895A05455}.Release|Any CPU.Build.0 = Release|Any CPU + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(NestedProjects) = preSolution + {0028BC85-0064-4CE8-A21A-C1F5E922BD59} = {61E7E11E-4558-434C-ACE8-06316A3097B3} + {4678AC8D-9C8F-415C-8CE4-3F7895A05455} = {AAA11B78-2764-4520-A97E-46AA7089A588} + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {CED9A020-DBA5-4BE6-8096-75E528648EC1} + EndGlobalSection +EndGlobal diff --git a/README.md b/README.md index c513944..6a80c39 100644 --- a/README.md +++ b/README.md @@ -1,2 +1,39 @@ # AI21 -Generated C# SDK based on official AI21 OpenAPI specification + +[![Nuget package](https://img.shields.io/nuget/vpre/AI21)](https://www.nuget.org/packages/AI21/) +[![dotnet](https://github.com/tryAGI/AI21/actions/workflows/dotnet.yml/badge.svg?branch=main)](https://github.com/tryAGI/AI21/actions/workflows/dotnet.yml) +[![License: MIT](https://img.shields.io/github/license/tryAGI/AI21)](https://github.com/tryAGI/AI21/blob/main/LICENSE.txt) +[![Discord](https://img.shields.io/discord/1115206893015662663?label=Discord&logo=discord&logoColor=white&color=d82679)](https://discord.gg/Ca2xhfBf3v) + +Generated C# SDK based on AI21 OpenAPI specification using NSwag. +Includes [tokenizer](https://github.com/tryAGI/Tiktoken) and some helper methods. + +### Usage +```csharp +using AI21; + +using var client = new HttpClient(); +var api = new AI21Api(apiKey, client); +var response = await api.GenerateTextAsync( + RecommendedModelIds.Gpt2, + new GenerateTextRequest + { + Inputs = "Hello", + Parameters = new GenerateTextRequestParameters + { + Max_new_tokens = 250, + Return_full_text = false, + }, + Options = new GenerateTextRequestOptions + { + Use_cache = true, + Wait_for_model = false, + }, + }); +``` + +## Support + +Priority place for bugs: https://github.com/tryAGI/AI21/issues +Priority place for ideas and general questions: https://github.com/tryAGI/AI21/discussions +Discord: https://discord.gg/Ca2xhfBf3v \ No newline at end of file diff --git a/assets/nuget_icon.png b/assets/nuget_icon.png new file mode 100644 index 0000000..25574ba Binary files /dev/null and b/assets/nuget_icon.png differ diff --git a/docs/openapi.nswag b/docs/openapi.nswag new file mode 100755 index 0000000..ffa650a --- /dev/null +++ b/docs/openapi.nswag @@ -0,0 +1,100 @@ +{ + "runtime": "Default", + "defaultVariables": null, + "documentGenerator": { + "fromDocument": { + "json": "", + "url": "openapi.yaml", + "output": null, + "newLineBehavior": "Auto" + } + }, + "codeGenerators": { + "openApiToCSharpClient": { + "clientBaseClass": null, + "configurationClass": null, + "generateClientClasses": true, + "generateClientInterfaces": false, + "clientBaseInterface": null, + "injectHttpClient": true, + "disposeHttpClient": true, + "protectedMethods": [], + "generateExceptionClasses": true, + "exceptionClass": "ApiException", + "wrapDtoExceptions": true, + "useHttpClientCreationMethod": false, + "httpClientType": "System.Net.Http.HttpClient", + "useHttpRequestMessageCreationMethod": false, + "useBaseUrl": true, + "generateBaseUrlProperty": true, + "generateSyncMethods": false, + "generatePrepareRequestAndProcessResponseAsAsyncMethods": true, + "exposeJsonSerializerSettings": false, + "clientClassAccessModifier": "public", + "typeAccessModifier": "public", + "generateContractsOutput": false, + "contractsNamespace": null, + "contractsOutputFilePath": null, + "parameterDateTimeFormat": "s", + "parameterDateFormat": "yyyy-MM-dd", + "generateUpdateJsonSerializerSettingsMethod": true, + "useRequestAndResponseSerializationSettings": false, + "serializeTypeInformation": false, + "queryNullValue": "", + "className": "{controller}Ai21Api", + "operationGenerationMode": "MultipleClientsFromOperationId", + "additionalNamespaceUsages": [], + "additionalContractNamespaceUsages": [], + "generateOptionalParameters": true, + "generateJsonMethods": false, + "enforceFlagEnums": false, + "parameterArrayType": "System.Collections.Generic.IEnumerable", + "parameterDictionaryType": "System.Collections.Generic.IDictionary", + "responseArrayType": "System.Collections.Generic.ICollection", + "responseDictionaryType": "System.Collections.Generic.IDictionary", + "wrapResponses": false, + "wrapResponseMethods": [], + "generateResponseClasses": true, + "responseClass": "SwaggerResponse", + "namespace": "AI21", + "requiredPropertiesMustBeDefined": true, + "dateType": "System.DateTimeOffset", + "jsonConverters": null, + "anyType": "object", + "dateTimeType": "System.DateTimeOffset", + "timeType": "System.TimeSpan", + "timeSpanType": "System.TimeSpan", + "arrayType": "System.Collections.Generic.ICollection", + "arrayInstanceType": "System.Collections.ObjectModel.Collection", + "dictionaryType": "System.Collections.Generic.IDictionary", + "dictionaryInstanceType": "System.Collections.Generic.Dictionary", + "arrayBaseType": "System.Collections.ObjectModel.Collection", + "dictionaryBaseType": "System.Collections.Generic.Dictionary", + "classStyle": "Poco", + "jsonLibrary": "SystemTextJson", + "generateDefaultValues": true, + "generateDataAnnotations": true, + "excludedTypeNames": [], + "excludedParameterNames": [], + "handleReferences": false, + "generateImmutableArrayProperties": false, + "generateImmutableDictionaryProperties": false, + "jsonSerializerSettingsTransformationMethod": null, + "inlineNamedArrays": false, + "inlineNamedDictionaries": false, + "inlineNamedTuples": true, + "inlineNamedAny": false, + "generateDtoTypes": true, + "generateOptionalPropertiesAsNullable": true, + "generateNullableReferenceTypes": true, + "templateDirectory": null, + "typeNameGeneratorType": null, + "propertyNameGeneratorType": null, + "enumNameGeneratorType": null, + "serviceHost": null, + "serviceSchemes": null, + "output": null, + "newLineBehavior": "Auto" + } + } +} \ No newline at end of file diff --git a/docs/openapi.yaml b/docs/openapi.yaml new file mode 100644 index 0000000..16e5804 --- /dev/null +++ b/docs/openapi.yaml @@ -0,0 +1,171 @@ +openapi: 3.0.0 +info: + title: AI21 API + description: APIs for sampling from and fine-tuning language models + version: '1.0.0' +servers: + - url: https://api-inference.huggingface.co/ +tags: + - name: AI21 + description: The AI21 REST API +paths: + /models/{modelId}: + post: + operationId: generateText + parameters: + - in: path + name: modelId + schema: + type: string + required: true + description: Model ID of the user to get + tags: + - AI21 + summary: Creates a completion for the provided prompt and parameters. + requestBody: + required: true + content: + application/json: + schema: + $ref: '#/components/schemas/GenerateTextRequest' + responses: + 200: + description: OK + content: + application/json: + schema: + $ref: '#/components/schemas/GenerateTextResponse' + +components: + schemas: + ErrorResponse: + type: object + properties: + error: + anyOf: + - type: string + - type: array + items: + type: string + required: + - error + + GenerateTextRequest: + type: object + properties: + inputs: + description: | + The prompt that you want to complete. + nullable: true + type: string + default: '' + example: "This is a test." + parameters: + $ref: '#/components/schemas/GenerateTextRequestParameters' + options: + $ref: '#/components/schemas/GenerateTextRequestOptions' + required: + - inputs + + GenerateTextRequestParameters: + type: object + properties: + top_k: + type: integer + example: 1 + nullable: true + description: &completions_top_k_description | + Integer to define the top tokens considered within the sample operation to create new text. + top_p: + type: number + example: 1 + nullable: true + description: | + Float to define the tokens that are within the sample operation of text generation. + Add tokens in the sample for more probable to least probable until the sum of the probabilities is greater than top_p. + temperature: + type: number + minimum: 0 + maximum: 100 + default: 1 + example: 1 + nullable: true + description: | + The temperature of the sampling operation. + + 1 means regular sampling, + 0 means always take the highest score, + 100.0 is getting closer to uniform probability. + repetition_penalty: + type: number + minimum: 0 + maximum: 100 + example: 1 + nullable: true + description: | + The more a token is used within generation the more it is penalized to not be picked in successive generation passes. + max_new_tokens: + type: integer + minimum: 1 + maximum: 250 + example: 250 + nullable: true + description: | + The amount of new tokens to be generated, this does not include the input length it is a estimate of the size of generated text you want. + Each new tokens slows down the request, so look for balance between response times and length of text generated. + max_time: + type: number + minimum: 0 + maximum: 120 + example: 1 + nullable: true + description: | + The amount of time in seconds that the query should take maximum. Network can cause some overhead so it will be a soft limit. Use that in combination with max_new_tokens for best results. + return_full_text: + description: > + If set to False, the return results will not contain the original query making it easier for prompting. + default: true + num_return_sequences: + type: integer + default: 1 + example: 1 + nullable: true + description: | + The number of proposition you want to be returned. + do_sample: + description: > + Whether or not to use sampling, use greedy decoding otherwise. + default: true + + GenerateTextRequestOptions: + type: object + properties: + use_cache: + description: > + There is a cache layer on the inference API to speedup requests we have already seen. + Most models can use those results as is as models are deterministic (meaning the results will be the same anyway). + However if you use a non deterministic model, + you can set this parameter to prevent the caching mechanism from being used resulting in a real new query. + default: true + wait_for_model: + description: > + If the model is not ready, wait for it instead of receiving 503. + It limits the number of requests required to get your inference done. + It is advised to only set this flag to true after receiving a 503 error as it will limit hanging + in your application to known places. + default: false + + GenerateTextResponse: + type: array + items: + $ref: '#/components/schemas/GenerateTextResponseValue' + + GenerateTextResponseValue: + type: object + properties: + generated_text: + type: string + description: | + The resulting completion up to and excluding the stop sequences. + required: + - generated_text \ No newline at end of file diff --git a/src/Directory.Build.props b/src/Directory.Build.props new file mode 100755 index 0000000..86f2c1f --- /dev/null +++ b/src/Directory.Build.props @@ -0,0 +1,9 @@ + + + + preview + enable + enable + + + diff --git a/src/key.snk b/src/key.snk new file mode 100755 index 0000000..4205bc6 Binary files /dev/null and b/src/key.snk differ diff --git a/src/libs/AI21/AI21.csproj b/src/libs/AI21/AI21.csproj new file mode 100755 index 0000000..8745777 --- /dev/null +++ b/src/libs/AI21/AI21.csproj @@ -0,0 +1,43 @@ + + + + netstandard2.0;net4.6.2;net6.0;net7.0 + + + + + + + + Generated C# SDK based on AI21 OpenAPI specification. + api;client;sdk;dotnet;swagger;openapi;specification;ai21;generated;nswag + + + + + + + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + + + + + + + + + + + + + + diff --git a/src/libs/AI21/Ai21Api.Authorization.cs b/src/libs/AI21/Ai21Api.Authorization.cs new file mode 100755 index 0000000..9bf673b --- /dev/null +++ b/src/libs/AI21/Ai21Api.Authorization.cs @@ -0,0 +1,36 @@ +using System.Net.Http.Headers; +using System.Text; + +// ReSharper disable UnusedParameter.Local +// ReSharper disable MemberCanBeMadeStatic.Local + +#pragma warning disable CA1822 + +namespace AI21; + +public partial class Ai21Api +{ + private string ApiKey { get; } = string.Empty; + + private Task PrepareRequestAsync(HttpClient client, HttpRequestMessage request, string url, CancellationToken cancellationToken) + { + if (!string.IsNullOrWhiteSpace(ApiKey)) + { + request.Headers.Authorization = new AuthenticationHeaderValue( + scheme: "Bearer", + parameter: ApiKey); + } + + return Task.CompletedTask; + } + + private Task PrepareRequestAsync(HttpClient client, HttpRequestMessage request, StringBuilder builder, CancellationToken cancellationToken) + { + return Task.CompletedTask; + } + + private Task ProcessResponseAsync(HttpClient client, HttpResponseMessage message, CancellationToken cancellationToken) + { + return Task.CompletedTask; + } +} diff --git a/src/libs/AI21/Ai21Api.Constructors.cs b/src/libs/AI21/Ai21Api.Constructors.cs new file mode 100755 index 0000000..e67e84d --- /dev/null +++ b/src/libs/AI21/Ai21Api.Constructors.cs @@ -0,0 +1,17 @@ +namespace AI21; + +/// +/// Class providing methods for API access. +/// +public partial class Ai21Api +{ + /// + /// Sets the selected apiKey as a default header for the HttpClient. + /// + /// + /// + public Ai21Api(string apiKey, HttpClient httpClient) : this(httpClient) + { + ApiKey = apiKey ?? throw new ArgumentNullException(nameof(apiKey)); + } +} diff --git a/src/libs/AI21/Extensions/StringExtensions.cs b/src/libs/AI21/Extensions/StringExtensions.cs new file mode 100644 index 0000000..5ce712a --- /dev/null +++ b/src/libs/AI21/Extensions/StringExtensions.cs @@ -0,0 +1,47 @@ +namespace AI21; + +/// +/// +/// +public static class StringExtensions +{ + /// + /// + /// + /// + /// + public static string AsHumanMessage(this string content) + { + return $"Human: {content}"; + } + + /// + /// + /// + /// + /// + public static string AsAssistantMessage(this string content) + { + return $"Assistant: {content}"; + } + + /// + /// + /// + /// + /// + public static string AsPrompt(this string content) + { + return $"\n\n${content.AsHumanMessage()}\n\nAssistant:"; + } + + /// + /// + /// + /// + /// + public static string AsPrompt(this string[] content) + { + return AsPrompt(string.Join("\n\n", content)); + } +} \ No newline at end of file diff --git a/src/libs/AI21/Helpers/ApiHelpers.cs b/src/libs/AI21/Helpers/ApiHelpers.cs new file mode 100644 index 0000000..0820d21 --- /dev/null +++ b/src/libs/AI21/Helpers/ApiHelpers.cs @@ -0,0 +1,24 @@ +namespace AI21; + +/// +/// +/// +public static class ApiHelpers +{ + /// + /// Calculates the maximum number of tokens possible to generate for a model.
+ /// According https://huggingface.co/docs/api-inference/detailed_parameters#text-generation-task
+ ///
+ /// + /// + /// + public static int CalculateContextLength(string modelId) + { + return modelId switch + { + RecommendedModelIds.Gpt2 => 250, + + _ => throw new NotImplementedException(), + }; + } +} \ No newline at end of file diff --git a/src/libs/AI21/RecommendedModelIds.cs b/src/libs/AI21/RecommendedModelIds.cs new file mode 100644 index 0000000..cb6fdbd --- /dev/null +++ b/src/libs/AI21/RecommendedModelIds.cs @@ -0,0 +1,17 @@ +namespace AI21; + +// ReSharper disable InconsistentNaming +#pragma warning disable CA1707 + +/// +/// Recommended models for AI21. +/// +public static class RecommendedModelIds +{ + /// + /// https://huggingface.co/gpt2
+ /// Pretrained model on English language using a causal language modeling (CLM) objective. + /// It was introduced in this paper and first released at this page.
+ ///
+ public const string Gpt2 = "gpt2"; +} \ No newline at end of file diff --git a/src/libs/Directory.Build.props b/src/libs/Directory.Build.props new file mode 100755 index 0000000..d4b672c --- /dev/null +++ b/src/libs/Directory.Build.props @@ -0,0 +1,49 @@ + + + + + + true + $(MSBuildThisFileDirectory)../key.snk + + + + + <_Parameter1>false + + + + + 0.1.0 + true + true + tryAGI and contributors + MIT + nuget_icon.png + README.md + + + + + + + + + true + $(MSBuildThisFileDirectory)../../../LocalPackages + + + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + + + true + latest + All + + + diff --git a/src/tests/AI21.IntegrationTests/AI21.IntegrationTests.csproj b/src/tests/AI21.IntegrationTests/AI21.IntegrationTests.csproj new file mode 100755 index 0000000..23ca596 --- /dev/null +++ b/src/tests/AI21.IntegrationTests/AI21.IntegrationTests.csproj @@ -0,0 +1,27 @@ + + + + net7.0 + + + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/tests/AI21.IntegrationTests/Tests.cs b/src/tests/AI21.IntegrationTests/Tests.cs new file mode 100755 index 0000000..8c68127 --- /dev/null +++ b/src/tests/AI21.IntegrationTests/Tests.cs @@ -0,0 +1,69 @@ +using AI21; + +namespace tryAGI.OpenAI.IntegrationTests; + +[TestClass] +public class GeneralTests +{ + [TestMethod] + public async Task Generate() + { + var apiKey = + Environment.GetEnvironmentVariable("AI21_API_KEY") ?? + throw new AssertInconclusiveException("AI21_API_KEY environment variable is not found."); + + using var client = new HttpClient(); + var api = new Ai21Api(apiKey, client); + var response = await api.GenerateTextAsync( + RecommendedModelIds.Gpt2, + new GenerateTextRequest + { + Inputs = "Give random 5 words in response", + Parameters = new GenerateTextRequestParameters + { + Max_new_tokens = 250, + Return_full_text = false, + }, + Options = new GenerateTextRequestOptions + { + Use_cache = true, + Wait_for_model = false, + }, + }); + response.Should().NotBeEmpty(); + + foreach (var value in response) + { + Console.WriteLine(value.Generated_text); + } + } + + [TestMethod] + public async Task GenerateError() + { + var apiKey = + Environment.GetEnvironmentVariable("AI21_API_KEY") ?? + throw new AssertInconclusiveException("AI21_API_KEY environment variable is not found."); + + using var client = new HttpClient(); + var api = new Ai21Api(apiKey, client); + Func act = async () => await api.GenerateTextAsync( + RecommendedModelIds.Gpt2, + new GenerateTextRequest + { + Inputs = "Hello", + Parameters = new GenerateTextRequestParameters + { + Max_new_tokens = 2501, + Return_full_text = false, + }, + Options = new GenerateTextRequestOptions + { + Use_cache = true, + Wait_for_model = false, + }, + }); + + await act.Should().ThrowAsync(); + } +}