diff --git a/.vscode/launch.json b/.vscode/launch.json index d747f10bb4..e2e3dbdd89 100644 --- a/.vscode/launch.json +++ b/.vscode/launch.json @@ -404,6 +404,25 @@ "type": "coreclr", "request": "attach", "processId": "${command:pickProcess}" + }, + { + "name": "Launch Http", + "type": "coreclr", + "request": "launch", + "preLaunchTask": "build", + "program": "${workspaceFolder}/src/kiota/bin/Debug/net8.0/kiota.dll", + "args": [ + "generate", + "--openapi", + "https://raw.githubusercontent.com/microsoftgraph/msgraph-sdk-powershell/dev/openApiDocs/v1.0/Mail.yml", + "--language", + "http", + "--output", + "${workspaceFolder}/samples/msgraph-mail/http", + ], + "cwd": "${workspaceFolder}/src/kiota", + "stopAtEntry": false, + "console": "internalConsole" } ] } diff --git a/src/Kiota.Builder/GenerationLanguage.cs b/src/Kiota.Builder/GenerationLanguage.cs index 278bdcd8b2..99dfd831f2 100644 --- a/src/Kiota.Builder/GenerationLanguage.cs +++ b/src/Kiota.Builder/GenerationLanguage.cs @@ -11,4 +11,5 @@ public enum GenerationLanguage Ruby, CLI, Dart, + HTTP } diff --git a/src/Kiota.Builder/KiotaBuilder.cs b/src/Kiota.Builder/KiotaBuilder.cs index de3d7adac6..8025360676 100644 --- a/src/Kiota.Builder/KiotaBuilder.cs +++ b/src/Kiota.Builder/KiotaBuilder.cs @@ -26,6 +26,7 @@ using Kiota.Builder.OpenApiExtensions; using Kiota.Builder.Plugins; using Kiota.Builder.Refiners; +using Kiota.Builder.Settings; using Kiota.Builder.WorkspaceManagement; using Kiota.Builder.Writers; using Microsoft.Extensions.Logging; @@ -46,9 +47,10 @@ public partial class KiotaBuilder private readonly ParallelOptions parallelOptions; private readonly HttpClient httpClient; private OpenApiDocument? openApiDocument; + private readonly ISettingsManagementService settingsFileManagementService; internal void SetOpenApiDocument(OpenApiDocument document) => openApiDocument = document ?? throw new ArgumentNullException(nameof(document)); - public KiotaBuilder(ILogger logger, GenerationConfiguration config, HttpClient client, bool useKiotaConfig = false) + public KiotaBuilder(ILogger logger, GenerationConfiguration config, HttpClient client, bool useKiotaConfig = false, ISettingsManagementService? settingsManagementService = null) { ArgumentNullException.ThrowIfNull(logger); ArgumentNullException.ThrowIfNull(config); @@ -64,6 +66,7 @@ public KiotaBuilder(ILogger logger, GenerationConfiguration config workspaceManagementService = new WorkspaceManagementService(logger, client, useKiotaConfig, workingDirectory); this.useKiotaConfig = useKiotaConfig; openApiDocumentDownloadService = new OpenApiDocumentDownloadService(client, logger); + settingsFileManagementService = settingsManagementService ?? new SettingsFileManagementService(); } private readonly OpenApiDocumentDownloadService openApiDocumentDownloadService; private readonly bool useKiotaConfig; @@ -285,6 +288,13 @@ public async Task GenerateClientAsync(CancellationToken cancellationToken) sw.Start(); await CreateLanguageSourceFilesAsync(config.Language, generatedCode, cancellationToken).ConfigureAwait(false); StopLogAndReset(sw, $"step {++stepId} - writing files - took"); + + if (config.Language == GenerationLanguage.HTTP && openApiDocument is not null) + { + sw.Start(); + await settingsFileManagementService.WriteSettingsFileAsync(config.OutputPath, openApiDocument, cancellationToken).ConfigureAwait(false); + StopLogAndReset(sw, $"step {++stepId} - generating settings file for HTTP authentication - took"); + } return stepId; }, cancellationToken).ConfigureAwait(false); } @@ -554,6 +564,41 @@ public CodeNamespace CreateSourceModel(OpenApiUrlTreeNode? root) return rootNamespace; } + private void AddOperationSecurityRequirementToDOM(OpenApiOperation operation, CodeClass codeClass) + { + if (openApiDocument is null) + { + logger.LogWarning("OpenAPI document is null"); + return; + } + + if (operation.Security == null || !operation.Security.Any()) + return; + + var securitySchemes = openApiDocument.Components.SecuritySchemes; + foreach (var securityRequirement in operation.Security) + { + foreach (var scheme in securityRequirement.Keys) + { + if (securitySchemes.TryGetValue(scheme.Reference.Id, out var securityScheme)) + { + AddSecurity(codeClass, securityScheme); + } + } + } + } + + private void AddSecurity(CodeClass codeClass, OpenApiSecurityScheme openApiSecurityScheme) + { + codeClass.AddProperty( + new CodeProperty + { + Type = new CodeType { Name = openApiSecurityScheme.Type.ToString(), IsExternal = true }, + Kind = CodePropertyKind.Headers + } + ); + } + /// /// Manipulate CodeDOM for language specific issues /// @@ -671,7 +716,14 @@ private void CreateRequestBuilderClass(CodeNamespace currentNamespace, OpenApiUr foreach (var operation in currentNode .PathItems[Constants.DefaultOpenApiLabel] .Operations) + { + CreateOperationMethods(currentNode, operation.Key, operation.Value, codeClass); + if (config.Language == GenerationLanguage.HTTP) + { + AddOperationSecurityRequirementToDOM(operation.Value, codeClass); + } + } } if (rootNamespace != null) diff --git a/src/Kiota.Builder/PathSegmenters/HttpPathSegmenter.cs b/src/Kiota.Builder/PathSegmenters/HttpPathSegmenter.cs new file mode 100644 index 0000000000..052da6654f --- /dev/null +++ b/src/Kiota.Builder/PathSegmenters/HttpPathSegmenter.cs @@ -0,0 +1,13 @@ +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Extensions; + +namespace Kiota.Builder.PathSegmenters; +public class HttpPathSegmenter(string rootPath, string clientNamespaceName) : CommonPathSegmenter(rootPath, clientNamespaceName) +{ + public override string FileSuffix => ".http"; + public override string NormalizeNamespaceSegment(string segmentName) => segmentName.ToFirstCharacterUpperCase(); + public override string NormalizeFileName(CodeElement currentElement) + { + return GetLastFileNameSegment(currentElement).ToFirstCharacterUpperCase(); + } +} diff --git a/src/Kiota.Builder/Refiners/HttpRefiner.cs b/src/Kiota.Builder/Refiners/HttpRefiner.cs new file mode 100644 index 0000000000..e023cade8d --- /dev/null +++ b/src/Kiota.Builder/Refiners/HttpRefiner.cs @@ -0,0 +1,148 @@ +using System; +using System.Collections.Generic; +using System.Diagnostics; +using System.Linq; +using System.Threading; +using System.Threading.Tasks; +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Configuration; +using Kiota.Builder.Extensions; + +namespace Kiota.Builder.Refiners; +public class HttpRefiner(GenerationConfiguration configuration) : CommonLanguageRefiner(configuration) +{ + private const string BaseUrl = "BaseUrl"; + private const string BaseUrlName = "string"; + public override Task RefineAsync(CodeNamespace generatedCode, CancellationToken cancellationToken) + { + return Task.Run(() => + { + cancellationToken.ThrowIfCancellationRequested(); + CapitalizeNamespacesFirstLetters(generatedCode); + ReplaceIndexersByMethodsWithParameter( + generatedCode, + false, + static x => $"By{x.ToFirstCharacterUpperCase()}", + static x => x.ToFirstCharacterUpperCase(), + GenerationLanguage.HTTP); + cancellationToken.ThrowIfCancellationRequested(); + ReplaceReservedNames( + generatedCode, + new HttpReservedNamesProvider(), + x => $"{x}_escaped"); + RemoveCancellationParameter(generatedCode); + ConvertUnionTypesToWrapper( + generatedCode, + _configuration.UsesBackingStore, + static s => s + ); + cancellationToken.ThrowIfCancellationRequested(); + SetBaseUrlForRequestBuilderMethods(generatedCode, GetBaseUrl(generatedCode)); + AddPathParameters(generatedCode); + // Remove unused code from the DOM e.g Models, BarrelInitializers, e.t.c + RemoveUnusedCodeElements(generatedCode); + }, cancellationToken); + } + + private string? GetBaseUrl(CodeElement element) + { + return element.GetImmediateParentOfType() + .GetRootNamespace()? + .FindChildByName(_configuration.ClientClassName)? + .Methods? + .FirstOrDefault(static x => x.IsOfKind(CodeMethodKind.ClientConstructor))? + .BaseUrl; + } + + private static void CapitalizeNamespacesFirstLetters(CodeElement current) + { + if (current is CodeNamespace currentNamespace) + currentNamespace.Name = currentNamespace.Name.Split('.').Select(static x => x.ToFirstCharacterUpperCase()).Aggregate(static (x, y) => $"{x}.{y}"); + CrawlTree(current, CapitalizeNamespacesFirstLetters); + } + + private static void SetBaseUrlForRequestBuilderMethods(CodeElement current, string? baseUrl) + { + if (baseUrl is not null && current is CodeClass codeClass && codeClass.IsOfKind(CodeClassKind.RequestBuilder)) + { + // Add a new property named BaseUrl and set its value to the baseUrl string + var baseUrlProperty = new CodeProperty + { + Name = BaseUrl, + Kind = CodePropertyKind.Custom, + Access = AccessModifier.Private, + DefaultValue = baseUrl, + Type = new CodeType { Name = BaseUrlName, IsExternal = true } + }; + codeClass.AddProperty(baseUrlProperty); + } + CrawlTree(current, (element) => SetBaseUrlForRequestBuilderMethods(element, baseUrl)); + } + + private void RemoveUnusedCodeElements(CodeElement element) + { + if (!IsRequestBuilderClass(element) || IsBaseRequestBuilder(element) || IsRequestBuilderClassWithoutAnyHttpOperations(element)) + { + var parentNameSpace = element.GetImmediateParentOfType(); + parentNameSpace?.RemoveChildElement(element); + } + CrawlTree(element, RemoveUnusedCodeElements); + } + + private void AddPathParameters(CodeElement element) + { + var parent = element.GetImmediateParentOfType().Parent; + while (parent is not null) + { + var codeIndexer = parent.GetChildElements(false) + .OfType() + .FirstOrDefault()? + .GetChildElements(false) + .OfType() + .FirstOrDefault(static x => x.IsOfKind(CodeMethodKind.IndexerBackwardCompatibility)); + + if (codeIndexer is not null && element is CodeClass codeClass) + { + // Retrieve all the parameters of kind CodeParameterKind.Custom + var customProperties = codeIndexer.Parameters + .Where(static x => x.IsOfKind(CodeParameterKind.Custom)) + .Select(x => new CodeProperty + { + Name = x.Name, + Kind = CodePropertyKind.PathParameters, + Type = x.Type, + Access = AccessModifier.Public, + DefaultValue = x.DefaultValue, + SerializationName = x.SerializationName, + Documentation = x.Documentation + }) + .ToArray(); + + if (customProperties.Length > 0) + { + codeClass.AddProperty(customProperties); + } + } + + parent = parent.Parent?.GetImmediateParentOfType(); + } + CrawlTree(element, AddPathParameters); + } + + private static bool IsRequestBuilderClass(CodeElement element) + { + return element is CodeClass code && code.IsOfKind(CodeClassKind.RequestBuilder); + } + + private bool IsBaseRequestBuilder(CodeElement element) + { + return element is CodeClass codeClass && + codeClass.Name.Equals(_configuration.ClientClassName, StringComparison.Ordinal); + } + + private static bool IsRequestBuilderClassWithoutAnyHttpOperations(CodeElement element) + { + return element is CodeClass codeClass && codeClass.IsOfKind(CodeClassKind.RequestBuilder) && + !codeClass.Methods.Any(static method => method.IsOfKind(CodeMethodKind.RequestExecutor)); + } +} diff --git a/src/Kiota.Builder/Refiners/HttpReservedNamesProvider.cs b/src/Kiota.Builder/Refiners/HttpReservedNamesProvider.cs new file mode 100644 index 0000000000..df60e5ee20 --- /dev/null +++ b/src/Kiota.Builder/Refiners/HttpReservedNamesProvider.cs @@ -0,0 +1,11 @@ +using System; +using System.Collections.Generic; + +namespace Kiota.Builder.Refiners; +public class HttpReservedNamesProvider : IReservedNamesProvider +{ + private readonly Lazy> _reservedNames = new(() => new HashSet(StringComparer.OrdinalIgnoreCase) + { + }); + public HashSet ReservedNames => _reservedNames.Value; +} diff --git a/src/Kiota.Builder/Refiners/ILanguageRefiner.cs b/src/Kiota.Builder/Refiners/ILanguageRefiner.cs index 37b42e51cb..3573e98671 100644 --- a/src/Kiota.Builder/Refiners/ILanguageRefiner.cs +++ b/src/Kiota.Builder/Refiners/ILanguageRefiner.cs @@ -37,6 +37,9 @@ public static async Task RefineAsync(GenerationConfiguration config, CodeNamespa case GenerationLanguage.Swift: await new SwiftRefiner(config).RefineAsync(generatedCode, cancellationToken).ConfigureAwait(false); break; + case GenerationLanguage.HTTP: + await new HttpRefiner(config).RefineAsync(generatedCode, cancellationToken).ConfigureAwait(false); + break; case GenerationLanguage.Python: await new PythonRefiner(config).RefineAsync(generatedCode, cancellationToken).ConfigureAwait(false); break; diff --git a/src/Kiota.Builder/Settings/ISettingsManagementService.cs b/src/Kiota.Builder/Settings/ISettingsManagementService.cs new file mode 100644 index 0000000000..a7b719240b --- /dev/null +++ b/src/Kiota.Builder/Settings/ISettingsManagementService.cs @@ -0,0 +1,28 @@ +using System.IO; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.OpenApi.Models; +using Microsoft.OpenApi.Services; + +namespace Kiota.Builder.Settings; +/// +/// A service that manages the settings file for http language snippets. +/// +public interface ISettingsManagementService +{ + /// + /// Gets the settings file for a Kiota project by crawling the directory tree. + /// + /// + /// + string? GetDirectoryContainingSettingsFile(string searchDirectory); + + /// + /// Writes the settings file to a directory. + /// + /// + /// OpenApi document + /// + /// + Task WriteSettingsFileAsync(string directoryPath, OpenApiDocument openApiDocument, CancellationToken cancellationToken); +} diff --git a/src/Kiota.Builder/Settings/SettingsFile.cs b/src/Kiota.Builder/Settings/SettingsFile.cs new file mode 100644 index 0000000000..7a9dbe2926 --- /dev/null +++ b/src/Kiota.Builder/Settings/SettingsFile.cs @@ -0,0 +1,85 @@ +using System.Text.Json.Serialization; +using Kiota.Builder.Configuration; + +namespace Kiota.Builder.Settings; +public class SettingsFile +{ + [JsonPropertyName("rest-client.environmentVariables")] + public EnvironmentVariables EnvironmentVariables + { + get; set; + } + + public SettingsFile() + { + EnvironmentVariables = new EnvironmentVariables(); + } +} + +public class EnvironmentVariables +{ + [JsonPropertyName("$shared")] + public SharedAuth Shared + { + get; set; + } + + [JsonPropertyName("remote")] + public AuthenticationSettings Remote + { + get; set; + } + + [JsonPropertyName("development")] + public AuthenticationSettings Development + { + get; set; + } + + public EnvironmentVariables() + { + Shared = new SharedAuth(); + Remote = new AuthenticationSettings(); + Development = new AuthenticationSettings(); + } +} + +public class SharedAuth +{ + +} + +public class AuthenticationSettings +{ + [JsonPropertyName("hostAddress")] + public string HostAddress + { + get; set; + } + + [JsonPropertyName("basicAuth")] + public string BasicAuth + { + get; set; + } + + [JsonPropertyName("bearerAuth")] + public string BearerAuth + { + get; set; + } + + [JsonPropertyName("apiKeyAuth")] + public string ApiKey + { + get; set; + } + + public AuthenticationSettings() + { + HostAddress = string.Empty; + BasicAuth = string.Empty; + BearerAuth = string.Empty; + ApiKey = string.Empty; + } +} diff --git a/src/Kiota.Builder/Settings/SettingsFileGenerationContext.cs b/src/Kiota.Builder/Settings/SettingsFileGenerationContext.cs new file mode 100644 index 0000000000..8b4ea3ba01 --- /dev/null +++ b/src/Kiota.Builder/Settings/SettingsFileGenerationContext.cs @@ -0,0 +1,14 @@ +using System; +using System.Collections.Generic; +using System.Text.Json; +using System.Text.Json.Serialization; + +namespace Kiota.Builder.Settings; + +[JsonSerializable(typeof(SettingsFile))] +[JsonSourceGenerationOptions(WriteIndented = true)] +[JsonSerializable(typeof(Dictionary))] +[JsonSerializable(typeof(Dictionary))] +internal partial class SettingsFileGenerationContext : JsonSerializerContext +{ +} diff --git a/src/Kiota.Builder/Settings/SettingsFileManagementService.cs b/src/Kiota.Builder/Settings/SettingsFileManagementService.cs new file mode 100644 index 0000000000..c519c3538a --- /dev/null +++ b/src/Kiota.Builder/Settings/SettingsFileManagementService.cs @@ -0,0 +1,56 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Runtime.Serialization.Formatters.Binary; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.OpenApi.Models; + +namespace Kiota.Builder.Settings; + +public class SettingsFileManagementService : ISettingsManagementService +{ + internal const string SettingsFileName = "settings.json"; + internal const string EnvironmentVariablesKey = "rest-client.environmentVariables"; + internal const string VsCodeDirectoryName = ".vscode"; + public string GetDirectoryContainingSettingsFile(string searchDirectory) + { + var currentDirectory = new DirectoryInfo(searchDirectory); + var vscodeDirectoryPath = Path.Combine(currentDirectory.FullName, VsCodeDirectoryName); + if (Directory.Exists(vscodeDirectoryPath)) + { + return vscodeDirectoryPath; + } + var pathToWrite = Path.Combine(searchDirectory, VsCodeDirectoryName); + return Directory.CreateDirectory(pathToWrite).FullName; + } + + public Task WriteSettingsFileAsync(string directoryPath, OpenApiDocument openApiDocument, CancellationToken cancellationToken) + { + ArgumentException.ThrowIfNullOrEmpty(directoryPath); + ArgumentNullException.ThrowIfNull(openApiDocument); + var settings = GenerateSettingsFile(openApiDocument); + return WriteSettingsFileInternalAsync(directoryPath, settings, cancellationToken); + } + + private static SettingsFile GenerateSettingsFile(OpenApiDocument openApiDocument) + { + var settings = new SettingsFile(); + if (openApiDocument.Servers?.Count > 0) + { + settings.EnvironmentVariables.Development.HostAddress = openApiDocument.Servers[0].Url; + settings.EnvironmentVariables.Remote.HostAddress = openApiDocument.Servers[0].Url; + } + return settings; + } + + private async Task WriteSettingsFileInternalAsync(string directoryPath, SettingsFile settings, CancellationToken cancellationToken) + { + var parentDirectoryPath = Path.GetDirectoryName(directoryPath); + var vscodeDirectoryPath = GetDirectoryContainingSettingsFile(parentDirectoryPath!); + var settingsObjectString = JsonSerializer.Serialize(settings, SettingsFileGenerationContext.Default.SettingsFile); + var fileUpdatePath = Path.Combine(vscodeDirectoryPath, SettingsFileName); + await VsCodeSettingsManager.UpdateFileAsync(settingsObjectString, fileUpdatePath, EnvironmentVariablesKey, cancellationToken).ConfigureAwait(false); + } +} diff --git a/src/Kiota.Builder/Settings/VsCodeFileManagement.cs b/src/Kiota.Builder/Settings/VsCodeFileManagement.cs new file mode 100644 index 0000000000..eef4690580 --- /dev/null +++ b/src/Kiota.Builder/Settings/VsCodeFileManagement.cs @@ -0,0 +1,53 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Text.Json; +using System.Threading; +using System.Threading.Tasks; + +namespace Kiota.Builder.Settings; + +public static class VsCodeSettingsManager +{ + private static readonly JsonSerializerOptions options = new() + { + PropertyNamingPolicy = JsonNamingPolicy.CamelCase, + WriteIndented = true, + }; + private static readonly SettingsFileGenerationContext context = new(options); + public static async Task UpdateFileAsync(string fileUpdate, string fileUpdatePath, string fileUpdateKey, CancellationToken cancellationToken) + { + ArgumentException.ThrowIfNullOrEmpty(fileUpdate); + ArgumentException.ThrowIfNullOrEmpty(fileUpdatePath); + ArgumentException.ThrowIfNullOrEmpty(fileUpdateKey); + Dictionary settings = []; + + // Read existing settings or create new if file doesn't exist + if (File.Exists(fileUpdatePath)) + { + using var stream = File.OpenRead(fileUpdatePath); + settings = await JsonSerializer.DeserializeAsync( + stream, + context.DictionaryStringObject, + cancellationToken + ).ConfigureAwait(false) ?? []; + } + + var fileUpdateDictionary = JsonSerializer.Deserialize(fileUpdate, context.DictionaryStringObject); + if (fileUpdateDictionary is not null) + { + if (fileUpdateDictionary.TryGetValue(fileUpdateKey, out var environmentVariables)) + { + settings[fileUpdateKey] = environmentVariables; + } + else + { + settings[fileUpdateKey] = fileUpdateDictionary; + } + } + +#pragma warning disable CA2007 + await using var fileStream = File.Open(fileUpdatePath, FileMode.Create); + await JsonSerializer.SerializeAsync(fileStream, settings, context.DictionaryStringObject, cancellationToken).ConfigureAwait(false); + } +} diff --git a/src/Kiota.Builder/Writers/GenericWriters/GenericElementWriter.cs b/src/Kiota.Builder/Writers/GenericWriters/GenericElementWriter.cs new file mode 100644 index 0000000000..82b6831aa7 --- /dev/null +++ b/src/Kiota.Builder/Writers/GenericWriters/GenericElementWriter.cs @@ -0,0 +1,32 @@ +using System; +using Kiota.Builder.CodeDOM; + +namespace Kiota.Builder.Writers; + +// Generic base class for writing code elements +public class GenericWriter(ILanguageConventionService conventionService) + : BaseElementWriter(conventionService) + where TElement : CodeElement +{ + public override void WriteCodeElement(TElement codeElement, LanguageWriter writer) + { + ArgumentNullException.ThrowIfNull(codeElement); + ArgumentNullException.ThrowIfNull(writer); + } +} + +// Specific writers inheriting from the generic class +public class GenericCodePropertyWriter(ILanguageConventionService conventionService) + : GenericWriter(conventionService) +{ +} + +public class GenericCodeMethodWriter(ILanguageConventionService conventionService) + : GenericWriter(conventionService) +{ +} + +public class GenericCodeElementWriter(ILanguageConventionService conventionService) + : GenericWriter(conventionService) +{ +} diff --git a/src/Kiota.Builder/Writers/HTTP/CodeClassDeclarationWriter.cs b/src/Kiota.Builder/Writers/HTTP/CodeClassDeclarationWriter.cs new file mode 100644 index 0000000000..1299a0ad65 --- /dev/null +++ b/src/Kiota.Builder/Writers/HTTP/CodeClassDeclarationWriter.cs @@ -0,0 +1,346 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Extensions; +using Microsoft.Kiota.Abstractions; +using Microsoft.OpenApi.Models; + +namespace Kiota.Builder.Writers.Http; +public class CodeClassDeclarationWriter(HttpConventionService conventionService) : CodeProprietableBlockDeclarationWriter(conventionService) +{ + private static class Constants + { + internal const string BaseUrlPropertyName = "hostAddress"; + internal const string PathParameters = "pathParameters"; + internal const string BaseUrl = "baseUrl"; + internal const string ApiKeyAuth = "apiKeyAuth"; + internal const string BearerAuth = "bearerAuth"; + internal const string HttpVersion = "HTTP/1.1"; + internal const string LocalHostUrl = "http://localhost/"; + + internal static Dictionary SchemeTypeMapping = new() + { + { SecuritySchemeType.ApiKey.ToString().ToLowerInvariant(), ApiKeyAuth }, + { SecuritySchemeType.Http.ToString().ToLowerInvariant(), BearerAuth }, + { SecuritySchemeType.OAuth2.ToString().ToLowerInvariant(), BearerAuth }, + { SecuritySchemeType.OpenIdConnect.ToString().ToLowerInvariant(), BearerAuth } + }; + } + + protected override void WriteTypeDeclaration(ClassDeclaration codeElement, LanguageWriter writer) + { + ArgumentNullException.ThrowIfNull(codeElement); + ArgumentNullException.ThrowIfNull(writer); + + if (codeElement.Parent is CodeClass requestBuilderClass && requestBuilderClass.IsOfKind(CodeClassKind.RequestBuilder) && GetUrlTemplateProperty(requestBuilderClass) is CodeProperty urlTemplateProperty) + { + // Write short description + conventions.WriteShortDescription(requestBuilderClass, writer); + writer.WriteLine(); + + // Retrieve all query parameters + var queryParameters = GetAllQueryParameters(requestBuilderClass); + + // Retrieve all path parameters + var pathParameters = GetPathParameters(requestBuilderClass); + + var baseUrl = GetBaseUrl(requestBuilderClass); + + // Write path parameters + WritePathParameters(pathParameters, writer); + + // Write all query parameter variables + WriteQueryParameters(queryParameters, writer); + + // Write all HTTP methods GET, POST, PUT, DELETE e.t.c + WriteHttpMethods(requestBuilderClass, writer, queryParameters, pathParameters, urlTemplateProperty, baseUrl); + } + } + + /// + /// Retrieves all the query parameters for the given request builder class. + /// + /// The request builder class containing the query parameters. + /// An array of all query parameters. + private static CodeProperty[] GetAllQueryParameters(CodeClass requestBuilderClass) + { + // Retrieve all the query parameter classes + return requestBuilderClass + .GetChildElements(true) + .OfType() + .Where(static element => element.IsOfKind(CodeClassKind.QueryParameters)) + .SelectMany(paramCodeClass => paramCodeClass.Properties) + .Where(static property => property.IsOfKind(CodePropertyKind.QueryParameter)) + .ToArray(); + } + + /// + /// Retrieves all the path parameters for the given request builder class. + /// + /// The request builder class containing the path parameters. + /// An array of all path parameters, or an empty array if none are found. + private static CodeProperty[] GetPathParameters(CodeClass requestBuilderClass) + { + // Retrieve all the path variables except the generic path parameter named "pathParameters" + var pathParameters = requestBuilderClass + .GetChildElements(true) + .OfType() + .Where(property => property.IsOfKind(CodePropertyKind.PathParameters) && !property.Name.Equals(Constants.PathParameters, StringComparison.OrdinalIgnoreCase)) + .ToArray(); + + return pathParameters; + } + + /// + /// Retrieves the base URL for the given request builder class. + /// + /// The request builder class containing the base URL property. + /// The base URL as a string, or null if not found. + private static string? GetBaseUrl(CodeClass requestBuilderClass) + { + // Retrieve the base URL property from the request builder class + return requestBuilderClass.Properties + .FirstOrDefault(property => property.Name.Equals(Constants.BaseUrl, StringComparison.OrdinalIgnoreCase))?.DefaultValue; + } + + /// + /// Retrieves the URL template property for the given request builder class. + /// + /// The request builder class containing the URL template property. + /// The URL template property, or null if not found. + private static CodeProperty? GetUrlTemplateProperty(CodeClass requestBuilderClass) + { + // Retrieve the URL template property from the request builder class + return requestBuilderClass + .GetChildElements(true) + .OfType() + .FirstOrDefault(static property => property.IsOfKind(CodePropertyKind.UrlTemplate)); + } + + + /// + /// Writes the path parameters for the given request builder class to the writer. + /// + /// The array of path parameters to write. + /// The language writer to write the path parameters to. + private static void WritePathParameters(CodeProperty[] pathParameters, LanguageWriter writer) + { + // Write each path parameter property + foreach (var pathParameter in pathParameters) + { + WriteHttpParameterProperty(pathParameter, writer); + } + } + + /// + /// Writes the query parameters for the given request builder class to the writer. + /// + /// The array of query parameters to write. + /// The language writer to write the query parameters to. + private static void WriteQueryParameters(CodeProperty[] queryParameters, LanguageWriter writer) + { + // Write each query parameter property + foreach (var queryParameter in queryParameters) + { + WriteHttpParameterProperty(queryParameter, writer); + } + } + + /// + /// Writes the HTTP parameter property to the writer. + /// + /// The property to write. + /// The language writer to write the property to. + private static void WriteHttpParameterProperty(CodeProperty property, LanguageWriter writer) + { + if (!string.IsNullOrEmpty(property.Name)) + { + // Write the property documentation as a comment + writer.WriteLine($"# {property.Documentation.DescriptionTemplate}"); + + // Write the property name and an assignment placeholder + writer.WriteLine($"@{property.Name.ToFirstCharacterLowerCase()} = "); + + // Write an empty line for separation + writer.WriteLine(); + } + } + + /// + /// Writes the HTTP methods (GET, POST, PATCH, DELETE, etc.) for the given request builder class to the writer. + /// + /// The request builder class containing the HTTP methods. + /// The language writer to write the HTTP methods to. + /// The array of query parameters. + /// The array of path parameters. + /// The URL template property containing the URL template. + /// The base URL. + private static void WriteHttpMethods( + CodeClass requestBuilderClass, + LanguageWriter writer, + CodeProperty[] queryParameters, + CodeProperty[] pathParameters, + CodeProperty urlTemplateProperty, + string? baseUrl) + { + // Retrieve all the HTTP methods of kind RequestExecutor + var httpMethods = GetHttpMethods(requestBuilderClass); + + var methodCount = httpMethods.Length; + var currentIndex = 0; + + foreach (var method in httpMethods) + { + // Write the method documentation as a comment + writer.WriteLine($"# {method.Documentation.DescriptionTemplate}"); + + // Build the actual URL string and replace all required fields (path and query) with placeholder variables + var url = BuildUrlStringFromTemplate( + urlTemplateProperty.DefaultValue, + queryParameters, + pathParameters, + baseUrl + ); + + // Write the HTTP operation (e.g., GET, POST, PATCH, etc.) + writer.WriteLine($"{method.Name.ToUpperInvariant()} {url} {Constants.HttpVersion}"); + + var authenticationMethod = requestBuilderClass + .Properties + .FirstOrDefault(static prop => prop.IsOfKind(CodePropertyKind.Headers)); + + if (authenticationMethod != null + && Enum.TryParse(typeof(SecuritySchemeType), authenticationMethod.Type.Name, true, out var schemeTypeObj) + && schemeTypeObj is SecuritySchemeType schemeType + && Constants.SchemeTypeMapping.TryGetValue(schemeType.ToString().ToLowerInvariant(), out var mappedSchemeType)) + { + writer.WriteLine($"Authorization: {{{{{mappedSchemeType}}}}}"); + } + + // Write the request body if present + WriteRequestBody(method, writer); + + // Write a separator if there are more items that follow + if (++currentIndex < methodCount) + { + writer.WriteLine(); + writer.WriteLine("###"); + writer.WriteLine(); + } + } + } + + /// + /// Retrieves all the HTTP methods of kind RequestExecutor for the given request builder class. + /// + /// The request builder class containing the HTTP methods. + /// An array of HTTP methods of kind RequestExecutor. + private static CodeMethod[] GetHttpMethods(CodeClass requestBuilderClass) + { + return [.. requestBuilderClass + .GetChildElements(true) + .OfType() + .Where(static element => element.IsOfKind(CodeMethodKind.RequestExecutor))]; + } + + /// + /// Writes the request body for the given method to the writer. + /// + /// The method containing the request body. + /// The language writer to write the request body to. + private static void WriteRequestBody(CodeMethod method, LanguageWriter writer) + { + // If there is a request body, write it + var requestBody = method.Parameters.FirstOrDefault(static param => param.IsOfKind(CodeParameterKind.RequestBody)); + if (requestBody is null) return; + + writer.WriteLine($"Content-Type: {method.RequestBodyContentType}"); + + // Empty line before body content + writer.WriteLine(); + + // Loop through the properties of the request body and write a JSON object + if (requestBody.Type is CodeType ct && ct.TypeDefinition is CodeClass requestBodyClass) + { + writer.StartBlock(); + WriteProperties(requestBodyClass, writer); + writer.CloseBlock(); + } + } + + /// + /// Writes the properties of the given request body class to the writer. + /// + /// The request body class containing the properties. + /// The language writer to write the properties to. + private static void WriteProperties(CodeClass requestBodyClass, LanguageWriter writer, HashSet? processedClasses = null, int depth = 0) + { + + processedClasses ??= []; + + // Add the current class to the set of processed classes + if (!processedClasses.Add(requestBodyClass)) + { + // If the class is already processed, write its properties again up to a certain depth + if (depth >= 3) + { + return; + } + } + + var properties = requestBodyClass.Properties + .Where(static prop => prop.IsOfKind(CodePropertyKind.Custom)) + .ToArray(); + + foreach (var prop in properties) + { + // Add a trailing comma if there are more properties to be written + var separator = ","; + var propName = $"\"{prop.Name}\""; + writer.Write($"{propName}: "); + + if (prop.Type is CodeType propType && propType.TypeDefinition is CodeClass propClass) + { + // If the property is an object, write a JSON representation recursively + writer.WriteLine("{", includeIndent: false); + writer.IncreaseIndent(); + WriteProperties(propClass, writer, processedClasses, depth + 1); + writer.CloseBlock($"}}{separator}"); + } + else + { + writer.WriteLine($"{HttpConventionService.GetDefaultValueForProperty(prop)}{separator}", includeIndent: false); + } + } + + // Remove the current class from the set of processed classes after processing + processedClasses.Remove(requestBodyClass); + + // If the class extends another class, write properties of the base class + if (requestBodyClass.StartBlock.Inherits?.TypeDefinition is CodeClass baseClass) + { + WriteProperties(baseClass, writer, processedClasses, depth + 1); + } + } + + private static string BuildUrlStringFromTemplate(string urlTemplateString, CodeProperty[] queryParameters, CodeProperty[] pathParameters, string? baseUrl) + { + // Use the provided baseUrl or default to "http://localhost/" + baseUrl ??= Constants.LocalHostUrl; + + // unquote the urlTemplate string and replace the {+baseurl} with the actual base url string + urlTemplateString = urlTemplateString.Trim('"').Replace("{+baseurl}", baseUrl, StringComparison.InvariantCultureIgnoreCase); + + // Build RequestInformation using the URL + var requestInformation = new RequestInformation() + { + UrlTemplate = urlTemplateString, + QueryParameters = queryParameters.ToDictionary(item => item.WireName, item => $"{{{{{item.Name.ToFirstCharacterLowerCase()}}}}}" as object), + PathParameters = pathParameters.ToDictionary(item => item.WireName, item => $"{{{{{item.Name.ToFirstCharacterLowerCase()}}}}}" as object), + }; + + // Erase baseUrl and use the placeholder variable {baseUrl} already defined in the snippet + return requestInformation.URI.ToString().Replace(baseUrl, $"{{{{{Constants.BaseUrlPropertyName}}}}}", StringComparison.InvariantCultureIgnoreCase); + } +} diff --git a/src/Kiota.Builder/Writers/HTTP/CodeProprietableBlockDeclarationWriter.cs b/src/Kiota.Builder/Writers/HTTP/CodeProprietableBlockDeclarationWriter.cs new file mode 100644 index 0000000000..39dfd0d731 --- /dev/null +++ b/src/Kiota.Builder/Writers/HTTP/CodeProprietableBlockDeclarationWriter.cs @@ -0,0 +1,18 @@ +using System; +using System.Linq; + +using Kiota.Builder.CodeDOM; + +namespace Kiota.Builder.Writers.Http; + +public abstract class CodeProprietableBlockDeclarationWriter(HttpConventionService conventionService) : BaseElementWriter(conventionService) + where T : ProprietableBlockDeclaration +{ + public override void WriteCodeElement(T codeElement, LanguageWriter writer) + { + ArgumentNullException.ThrowIfNull(codeElement); + ArgumentNullException.ThrowIfNull(writer); + WriteTypeDeclaration(codeElement, writer); + } + protected abstract void WriteTypeDeclaration(T codeElement, LanguageWriter writer); +} diff --git a/src/Kiota.Builder/Writers/HTTP/HttpConventionService.cs b/src/Kiota.Builder/Writers/HTTP/HttpConventionService.cs new file mode 100644 index 0000000000..72e1c43fbe --- /dev/null +++ b/src/Kiota.Builder/Writers/HTTP/HttpConventionService.cs @@ -0,0 +1,104 @@ +using System; +using System.Linq; +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Extensions; + +namespace Kiota.Builder.Writers.Http; +public class HttpConventionService : CommonLanguageConventionService +{ + public HttpConventionService() + { + } + public override string StreamTypeName => "stream"; + public override string VoidTypeName => "void"; + public override string DocCommentPrefix => "###"; + public static string NullableMarkerAsString => "?"; + public override string ParseNodeInterfaceName => "ParseNode"; + public override bool WriteShortDescription(IDocumentedElement element, LanguageWriter writer, string prefix = "", string suffix = "") + { + ArgumentNullException.ThrowIfNull(writer); + ArgumentNullException.ThrowIfNull(element); + if (!element.Documentation.DescriptionAvailable) return false; + if (element is not CodeElement codeElement) return false; + + var description = element.Documentation.GetDescription(type => GetTypeString(type, codeElement)); + writer.WriteLine($"{DocCommentPrefix} {prefix}{description}{prefix}"); + + return true; + } + public override string GetAccessModifier(AccessModifier access) + { + return access switch + { + AccessModifier.Public => "public", + AccessModifier.Protected => "internal", + _ => "private", + }; + } + public override string TempDictionaryVarName => "urlTplParams"; + public override string GetTypeString(CodeTypeBase code, CodeElement targetElement, bool includeCollectionInformation = true, LanguageWriter? writer = null) + { + if (code is CodeType currentType) + { + var typeName = TranslateType(currentType); + var nullableSuffix = code.IsNullable ? NullableMarkerAsString : string.Empty; + var collectionPrefix = currentType.IsCollection && includeCollectionInformation ? "[" : string.Empty; + var collectionSuffix = currentType.IsCollection && includeCollectionInformation ? $"]{nullableSuffix}" : string.Empty; + if (currentType.IsCollection && !string.IsNullOrEmpty(nullableSuffix)) + nullableSuffix = string.Empty; + + if (currentType.ActionOf) + return $"({collectionPrefix}{typeName}{nullableSuffix}{collectionSuffix}>) -> Void"; + return $"{collectionPrefix}{typeName}{nullableSuffix}{collectionSuffix}"; + } + + throw new InvalidOperationException($"type of type {code?.GetType()} is unknown"); + } + public override string TranslateType(CodeType type) + { + return type?.Name switch + { + "integer" => "Int32", + "boolean" => "Bool", + "float" => "Float32", + "long" => "Int64", + "double" or "decimal" => "Float64", + "guid" => "UUID", + "void" or "uint8" or "int8" or "int32" or "int64" or "float32" or "float64" or "string" => type.Name.ToFirstCharacterUpperCase(), + "binary" or "base64" or "base64url" => "[UInt8]", + "DateTimeOffset" => "Date", + null => "object", + _ => type.Name.ToFirstCharacterUpperCase() is string typeName && !string.IsNullOrEmpty(typeName) ? typeName : "object", + }; + } + public override string GetParameterSignature(CodeParameter parameter, CodeElement targetElement, LanguageWriter? writer = null) + { + ArgumentNullException.ThrowIfNull(parameter); + var parameterType = GetTypeString(parameter.Type, targetElement); + var defaultValue = parameter switch + { + _ when !string.IsNullOrEmpty(parameter.DefaultValue) => $" = {parameter.DefaultValue}", + _ when parameter.Optional => " = default", + _ => string.Empty, + }; + return $"{parameter.Name.ToFirstCharacterLowerCase()} : {parameterType}{defaultValue}"; + } + + /// + /// Gets the default value for the given property. + /// + /// The property to get the default value for. + /// The default value as a string. + public static string GetDefaultValueForProperty(CodeProperty codeProperty) + { + return codeProperty?.Type.Name switch + { + "int" or "integer" => "0", + "string" => "\"string\"", + "bool" or "boolean" => "false", + _ when codeProperty?.Type is CodeType enumType && enumType.TypeDefinition is CodeEnum enumDefinition => + enumDefinition.Options.FirstOrDefault()?.Name is string enumName ? $"\"{enumName}\"" : "null", + _ => "null" + }; + } +} diff --git a/src/Kiota.Builder/Writers/HTTP/HttpWriter.cs b/src/Kiota.Builder/Writers/HTTP/HttpWriter.cs new file mode 100644 index 0000000000..6e97831c74 --- /dev/null +++ b/src/Kiota.Builder/Writers/HTTP/HttpWriter.cs @@ -0,0 +1,16 @@ +using Kiota.Builder.PathSegmenters; + +namespace Kiota.Builder.Writers.Http; + +public class HttpWriter : LanguageWriter +{ + public HttpWriter(string rootPath, string clientNamespaceName) + { + PathSegmenter = new HttpPathSegmenter(rootPath, clientNamespaceName); + var conventionService = new HttpConventionService(); + AddOrReplaceCodeElementWriter(new CodeClassDeclarationWriter(conventionService)); + AddOrReplaceCodeElementWriter(new GenericCodePropertyWriter(conventionService)); + AddOrReplaceCodeElementWriter(new GenericCodeMethodWriter(conventionService)); + AddOrReplaceCodeElementWriter(new GenericCodeElementWriter(conventionService)); + } +} diff --git a/src/Kiota.Builder/Writers/LanguageWriter.cs b/src/Kiota.Builder/Writers/LanguageWriter.cs index 9ccf0652ab..d181eb5ba1 100644 --- a/src/Kiota.Builder/Writers/LanguageWriter.cs +++ b/src/Kiota.Builder/Writers/LanguageWriter.cs @@ -3,13 +3,13 @@ using System.ComponentModel; using System.IO; using System.Linq; - using Kiota.Builder.CodeDOM; using Kiota.Builder.PathSegmenters; using Kiota.Builder.Writers.Cli; using Kiota.Builder.Writers.CSharp; using Kiota.Builder.Writers.Dart; using Kiota.Builder.Writers.Go; +using Kiota.Builder.Writers.Http; using Kiota.Builder.Writers.Java; using Kiota.Builder.Writers.Php; using Kiota.Builder.Writers.Python; @@ -194,6 +194,7 @@ public static LanguageWriter GetLanguageWriter(GenerationLanguage language, stri GenerationLanguage.CLI => new CliWriter(outputPath, clientNamespaceName), GenerationLanguage.Swift => new SwiftWriter(outputPath, clientNamespaceName), GenerationLanguage.Dart => new DartWriter(outputPath, clientNamespaceName), + GenerationLanguage.HTTP => new HttpWriter(outputPath, clientNamespaceName), _ => throw new InvalidEnumArgumentException($"{language} language currently not supported."), }; } diff --git a/tests/Kiota.Builder.Tests/PathSegmenters/HttpPathSegmenterTests.cs b/tests/Kiota.Builder.Tests/PathSegmenters/HttpPathSegmenterTests.cs new file mode 100644 index 0000000000..915ec3c01a --- /dev/null +++ b/tests/Kiota.Builder.Tests/PathSegmenters/HttpPathSegmenterTests.cs @@ -0,0 +1,36 @@ +using System.IO; +using Kiota.Builder.CodeDOM; +using Kiota.Builder.PathSegmenters; +using Xunit; + +namespace Kiota.Builder.Tests.PathSegmenters +{ + public class HttpPathSegmenterTests + { + private readonly HttpPathSegmenter segmenter; + + public HttpPathSegmenterTests() + { + var tempFilePath = Path.Combine(Path.GetTempPath(), Path.GetTempFileName()); + segmenter = new HttpPathSegmenter(tempFilePath, "client"); + } + + [Fact] + public void HttpPathSegmenterGeneratesCorrectFileName() + { + var fileName = segmenter.NormalizeFileName(new CodeClass + { + Name = "testClass" + }); + Assert.Equal("TestClass", fileName);// the file name should be Proper case + } + + [Fact] + public void HttpPathSegmenterGeneratesNamespaceFolderName() + { + var namespaceName = "microsoft.Graph"; + var normalizedNamespace = segmenter.NormalizeNamespaceSegment(namespaceName); + Assert.Equal("Microsoft.Graph", normalizedNamespace);// the first character is upper case + } + } +} diff --git a/tests/Kiota.Builder.Tests/Settings/SettingsManagementTests.cs b/tests/Kiota.Builder.Tests/Settings/SettingsManagementTests.cs new file mode 100644 index 0000000000..a6e779915a --- /dev/null +++ b/tests/Kiota.Builder.Tests/Settings/SettingsManagementTests.cs @@ -0,0 +1,86 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.OpenApi.Models; +using Xunit; + +namespace Kiota.Builder.Settings.Tests +{ + public class SettingsFileManagementServiceTest + { + [Fact] + public void GetDirectoryContainingSettingsFile_ShouldCreateTheDirectory_If_It_Doesnt_Exist() + { + // Arrange + var service = new SettingsFileManagementService(); + var tempDirectory = Path.Combine(Path.GetTempPath(), Guid.NewGuid().ToString()); + Directory.CreateDirectory(tempDirectory); + + // Act + var result = service.GetDirectoryContainingSettingsFile(tempDirectory); + tempDirectory = Path.Combine(tempDirectory, ".vscode"); + // Assert + Assert.Equal(tempDirectory, result); + + // Cleanup + try + { + Directory.Delete(tempDirectory, true); + } + catch (IOException) + { + // Handle the case where the directory is not empty + Directory.Delete(tempDirectory, true); + } + } + + [Fact] + public void GetDirectoryContainingSettingsFile_ShouldReturnVscodeDirectory_WhenExists() + { + // Arrange + var service = new SettingsFileManagementService(); + var tempDirectory = Path.Combine(Path.GetTempPath(), Guid.NewGuid().ToString()); + var vscodeDirectory = Path.Combine(tempDirectory, ".vscode"); + Directory.CreateDirectory(vscodeDirectory); + + // Act + var result = service.GetDirectoryContainingSettingsFile(tempDirectory); + + // Assert + Assert.Equal(vscodeDirectory, result); + + // Cleanup + Directory.Delete(tempDirectory, true); + } + + [Fact] + public async Task WriteSettingsFileAsync_ShouldThrowArgumentException_WhenDirectoryPathIsNullOrEmpty() + { + // Arrange + var service = new SettingsFileManagementService(); + var openApiDocument = new OpenApiDocument(); + var cancellationToken = CancellationToken.None; + + // Act & Assert + await Assert.ThrowsAsync(() => service.WriteSettingsFileAsync(string.Empty, openApiDocument, cancellationToken)); + } + + [Fact] + public async Task WriteSettingsFileAsync_ShouldThrowArgumentNullException_WhenOpenApiDocumentIsNull() + { + // Arrange + var service = new SettingsFileManagementService(); + var tempDirectory = Path.Combine(Path.GetTempPath(), Guid.NewGuid().ToString()); + Directory.CreateDirectory(tempDirectory); + var cancellationToken = CancellationToken.None; + + // Act & Assert + await Assert.ThrowsAsync(() => service.WriteSettingsFileAsync(tempDirectory, null, cancellationToken)); + + // Cleanup + Directory.Delete(tempDirectory); + } + } +} diff --git a/tests/Kiota.Builder.Tests/Writers/HTTP/CodeClassDeclarationWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/HTTP/CodeClassDeclarationWriterTests.cs new file mode 100644 index 0000000000..5e4dbd3457 --- /dev/null +++ b/tests/Kiota.Builder.Tests/Writers/HTTP/CodeClassDeclarationWriterTests.cs @@ -0,0 +1,174 @@ +using System; +using System.Collections.Generic; +using System.Globalization; +using System.IO; +using System.Linq; +using System.Net.Http; +using System.Threading.Tasks; +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Configuration; +using Kiota.Builder.Extensions; +using Kiota.Builder.Refiners; +using Kiota.Builder.Tests.OpenApiSampleFiles; +using Kiota.Builder.Writers; +using Microsoft.Extensions.Logging; +using Moq; +using Xunit; +using static Kiota.Builder.Refiners.HttpRefiner; + +namespace Kiota.Builder.Tests.Writers.Http; +public sealed class CodeClassDeclarationWriterTests : IDisposable +{ + private const string DefaultPath = "./"; + private const string DefaultName = "name"; + private readonly StringWriter tw; + private readonly LanguageWriter writer; + private readonly CodeNamespace root; + + public CodeClassDeclarationWriterTests() + { + writer = LanguageWriter.GetLanguageWriter(GenerationLanguage.HTTP, DefaultPath, DefaultName); + tw = new StringWriter(); + writer.SetTextWriter(tw); + root = CodeNamespace.InitRootNamespace(); + } + public void Dispose() + { + tw?.Dispose(); + GC.SuppressFinalize(this); + } + + [Fact] + public async Task WritesRequestExecutorMethods() + { + var codeClass = new CodeClass + { + Name = "TestClass", + Kind = CodeClassKind.RequestBuilder + }; + var urlTemplateProperty = new CodeProperty + { + Name = "urlTemplate", + Kind = CodePropertyKind.UrlTemplate, + DefaultValue = "\"{+baseurl}/posts\"", + Type = new CodeType + { + Name = "string", + IsExternal = true + }, + Documentation = new CodeDocumentation + { + DescriptionTemplate = "The URL template for the request." + } + }; + codeClass.AddProperty(urlTemplateProperty); + + // Add base url property + var baseUrlProperty = new CodeProperty + { + Name = "BaseUrl", + Kind = CodePropertyKind.Custom, + Access = AccessModifier.Private, + DefaultValue = "https://example.com", + Type = new CodeType { Name = "string", IsExternal = true } + }; + codeClass.AddProperty(baseUrlProperty); + + var method = new CodeMethod + { + Name = "get", + Kind = CodeMethodKind.RequestExecutor, + Documentation = new CodeDocumentation { DescriptionTemplate = "GET method" }, + ReturnType = new CodeType { Name = "void" } + }; + codeClass.AddMethod(method); + + var postMethod = new CodeMethod + { + Name = "post", + Kind = CodeMethodKind.RequestExecutor, + Documentation = new CodeDocumentation { DescriptionTemplate = "Post method" }, + ReturnType = new CodeType { Name = "void" }, + RequestBodyContentType = "application/json" + }; + + + var typeDefinition = new CodeClass + { + Name = "PostParameter", + }; + + var properties = new List + { + new() { + Name = "body", + Kind = CodePropertyKind.Custom, + Type = new CodeType { Name = "string", IsExternal = true } + }, + new() { + Name = "id", + Kind = CodePropertyKind.Custom, + Type = new CodeType { Name = "int", IsExternal = true } + }, + new() { + Name = "title", + Kind = CodePropertyKind.Custom, + Type = new CodeType { Name = "string", IsExternal = true } + }, + new() { + Name = "userId", + Kind = CodePropertyKind.Custom, + Type = new CodeType { Name = "int", IsExternal = true } + } + }; + + typeDefinition.AddProperty(properties.ToArray()); + + // Define the parameter with the specified properties + var postParameter = new CodeParameter + { + Name = "postParameter", + Kind = CodeParameterKind.RequestBody, + Type = new CodeType + { + Name = "PostParameter", + TypeDefinition = typeDefinition + } + }; + + // Add the parameter to the post method + postMethod.AddParameter(postParameter); + + codeClass.AddMethod(postMethod); + + var patchMethod = new CodeMethod + { + Name = "patch", + Kind = CodeMethodKind.RequestExecutor, + Documentation = new CodeDocumentation { DescriptionTemplate = "Patch method" }, + ReturnType = new CodeType { Name = "void" } + }; + codeClass.AddMethod(patchMethod); + + root.AddClass(codeClass); + + await ILanguageRefiner.RefineAsync(new GenerationConfiguration { Language = GenerationLanguage.HTTP }, root); + + writer.Write(codeClass.StartBlock); + var result = tw.ToString(); + + // Check HTTP operations + Assert.Contains("GET {{hostAddress}}/posts HTTP/1.1", result); + Assert.Contains("PATCH {{hostAddress}}/posts HTTP/1.1", result); + Assert.Contains("POST {{hostAddress}}/posts HTTP/1.1", result); + + // Check content type + Assert.Contains("Content-Type: application/json", result); + + // check the request body + Assert.Contains("\"body\": \"string\"", result); + Assert.Contains("\"id\": 0", result); + Assert.Contains("\"title\": \"string\"", result); + Assert.Contains("\"userId\": 0", result); + } +} diff --git a/tests/Kiota.Builder.Tests/Writers/HTTP/CodeEnumWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/HTTP/CodeEnumWriterTests.cs new file mode 100644 index 0000000000..a1631daa52 --- /dev/null +++ b/tests/Kiota.Builder.Tests/Writers/HTTP/CodeEnumWriterTests.cs @@ -0,0 +1,62 @@ +using System; +using System.IO; +using System.Linq; + +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Writers; +using Kiota.Builder.Writers.Http; +using Xunit; + +namespace Kiota.Builder.Tests.Writers.Http; +public sealed class CodeEnumWriterTests : IDisposable +{ + private const string DefaultPath = "./"; + private const string DefaultName = "name"; + private readonly StringWriter tw; + private readonly LanguageWriter writer; + private readonly CodeEnum currentEnum; + private const string EnumName = "someEnum"; + private readonly GenericCodeElementWriter codeEnumWriter; + public CodeEnumWriterTests() + { + writer = LanguageWriter.GetLanguageWriter(GenerationLanguage.HTTP, DefaultPath, DefaultName); + codeEnumWriter = new GenericCodeElementWriter(new HttpConventionService()); + tw = new StringWriter(); + writer.SetTextWriter(tw); + var root = CodeNamespace.InitRootNamespace(); + currentEnum = root.AddEnum(new CodeEnum + { + Name = EnumName, + }).First(); + if (CodeConstant.FromCodeEnum(currentEnum) is CodeConstant constant) + { + currentEnum.CodeEnumObject = constant; + root.AddConstant(constant); + } + } + public void Dispose() + { + tw?.Dispose(); + GC.SuppressFinalize(this); + } + [Fact] + public void WriteCodeElement_ThrowsException_WhenCodeElementIsNull() + { + Assert.Throws(() => codeEnumWriter.WriteCodeElement(null, writer)); + } + [Fact] + public void WriteCodeElement_ThrowsException_WhenWriterIsNull() + { + var codeElement = new CodeEnum(); + Assert.Throws(() => codeEnumWriter.WriteCodeElement(codeElement, null)); + } + [Fact] + public void SkipsEnum() + { + const string optionName = "option1"; + currentEnum.AddOption(new CodeEnumOption { Name = optionName }); + codeEnumWriter.WriteCodeElement(currentEnum, writer); + var result = tw.ToString(); + Assert.True(string.IsNullOrEmpty(result)); + } +} diff --git a/tests/Kiota.Builder.Tests/Writers/HTTP/HttpConventionServiceTests.cs b/tests/Kiota.Builder.Tests/Writers/HTTP/HttpConventionServiceTests.cs new file mode 100644 index 0000000000..23efdcfd4f --- /dev/null +++ b/tests/Kiota.Builder.Tests/Writers/HTTP/HttpConventionServiceTests.cs @@ -0,0 +1,78 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; + +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Writers; +using Kiota.Builder.Writers.Http; +using Xunit; + +namespace Kiota.Builder.Tests.Writers.Http; +public sealed class HttpConventionServiceTest +{ + + [Fact] + public void TestGetDefaultValueForProperty_Int() + { + // Arrange + var codeProperty = new CodeProperty + { + Type = new CodeType { Name = "int" } + }; + + // Act + var result = HttpConventionService.GetDefaultValueForProperty(codeProperty); + + // Assert + Assert.Equal("0", result); + } + + [Fact] + public void TestGetDefaultValueForProperty_String() + { + // Arrange + var codeProperty = new CodeProperty + { + Type = new CodeType { Name = "string" } + }; + + // Act + var result = HttpConventionService.GetDefaultValueForProperty(codeProperty); + + // Assert + Assert.Equal("\"string\"", result); + } + + [Fact] + public void TestGetDefaultValueForProperty_Bool() + { + // Arrange + var codeProperty = new CodeProperty + { + Type = new CodeType { Name = "bool" } + }; + + // Act + var result = HttpConventionService.GetDefaultValueForProperty(codeProperty); + + // Assert + Assert.Equal("false", result); + } + + [Fact] + public void TestGetDefaultValueForProperty_Null() + { + // Arrange + var codeProperty = new CodeProperty + { + Type = new CodeType { Name = "unknown" } + }; + + // Act + var result = HttpConventionService.GetDefaultValueForProperty(codeProperty); + + // Assert + Assert.Equal("null", result); + } +}