From e4d0c383694dcb2c1e81177298f5a502842b7dc6 Mon Sep 17 00:00:00 2001 From: Silas Kenneth Date: Wed, 29 May 2024 15:26:06 +0300 Subject: [PATCH 01/33] Fix for scalar member composed types. --- src/Kiota.Builder/Refiners/PhpRefiner.cs | 38 ++++++++++--------- .../Writers/Php/CodeMethodWriter.cs | 4 +- 2 files changed, 23 insertions(+), 19 deletions(-) diff --git a/src/Kiota.Builder/Refiners/PhpRefiner.cs b/src/Kiota.Builder/Refiners/PhpRefiner.cs index c861b2405e..19ccb59918 100644 --- a/src/Kiota.Builder/Refiners/PhpRefiner.cs +++ b/src/Kiota.Builder/Refiners/PhpRefiner.cs @@ -14,6 +14,7 @@ public class PhpRefiner : CommonLanguageRefiner private static readonly CodeUsingDeclarationNameComparer usingComparer = new(); public PhpRefiner(GenerationConfiguration configuration) : base(configuration) { } + private const string AbstractionsNamespaceName = @"Microsoft\Kiota\Abstractions"; public override Task Refine(CodeNamespace generatedCode, CancellationToken cancellationToken) { @@ -31,7 +32,7 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance Name = "BaseRequestBuilder", Declaration = new CodeType { - Name = "Microsoft\\Kiota\\Abstractions", + Name = $@"{AbstractionsNamespaceName}\Serialization", IsExternal = true } }, AccessModifier.Public); @@ -45,7 +46,10 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance ConvertUnionTypesToWrapper(generatedCode, _configuration.UsesBackingStore, static s => s, - false); + false, + AbstractionsNamespaceName, + "ComposedTypeWrapper" + ); ReplaceReservedNames(generatedCode, new PhpReservedNamesProvider(), reservedWord => $"Escaped{reservedWord.ToFirstCharacterUpperCase()}", new HashSet { typeof(CodeEnumOption) }); AddQueryParameterFactoryMethod(generatedCode); AddPrimaryErrorMessage(generatedCode, "getPrimaryErrorMessage", () => new CodeType { IsExternal = true, IsNullable = false, Name = "string" }); @@ -143,7 +147,7 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance Name = "Date", Declaration = new CodeType { - Name = "Microsoft\\Kiota\\Abstractions\\Types", + Name = $@"{AbstractionsNamespaceName}\Types", IsExternal = true, }, }) @@ -154,7 +158,7 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance Name = "Time", Declaration = new CodeType { - Name = "Microsoft\\Kiota\\Abstractions\\Types", + Name = $@"{AbstractionsNamespaceName}\Types", IsExternal = true, }, }) @@ -184,34 +188,34 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance }; private static readonly AdditionalUsingEvaluator[] defaultUsingEvaluators = { new (static x => x is CodeProperty prop && prop.IsOfKind(CodePropertyKind.RequestAdapter), - "Microsoft\\Kiota\\Abstractions", "RequestAdapter"), + AbstractionsNamespaceName, "RequestAdapter"), new (static x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.RequestGenerator), - "Microsoft\\Kiota\\Abstractions", "HttpMethod", "RequestInformation"), + AbstractionsNamespaceName, "HttpMethod", "RequestInformation"), new (static x => x is CodeClass @class && @class.IsOfKind(CodeClassKind.Model) && @class.Properties.Any(static y => y.IsOfKind(CodePropertyKind.AdditionalData)), - "Microsoft\\Kiota\\Abstractions\\Serialization", "AdditionalDataHolder"), + $@"{AbstractionsNamespaceName}\Serialization", "AdditionalDataHolder"), new (static x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.Serializer), - "Microsoft\\Kiota\\Abstractions\\Serialization", "SerializationWriter"), + $@"{AbstractionsNamespaceName}\Serialization", "SerializationWriter"), new (static x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.Deserializer), - "Microsoft\\Kiota\\Abstractions\\Serialization", "ParseNode"), + $@"{AbstractionsNamespaceName}\Serialization", "ParseNode"), new (static x => x is CodeClass @class && @class.IsOfKind(CodeClassKind.Model), - "Microsoft\\Kiota\\Abstractions\\Serialization", "Parsable"), + $@"{AbstractionsNamespaceName}\Serialization", "Parsable"), new (static x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.ClientConstructor) && method.Parameters.Any(y => y.IsOfKind(CodeParameterKind.BackingStore)), - "Microsoft\\Kiota\\Abstractions\\Store", "BackingStoreFactory", "BackingStoreFactorySingleton"), + $@"{AbstractionsNamespaceName}\Store", "BackingStoreFactory", "BackingStoreFactorySingleton"), new (static x => x is CodeProperty prop && prop.IsOfKind(CodePropertyKind.BackingStore), - "Microsoft\\Kiota\\Abstractions\\Store", "BackingStore", "BackedModel", "BackingStoreFactorySingleton" ), + $@"{AbstractionsNamespaceName}\Store", "BackingStore", "BackedModel", "BackingStoreFactorySingleton" ), new (static x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.RequestExecutor), "Http\\Promise", "Promise"), new (static x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.RequestExecutor), "", "Exception"), - new (static x => x is CodeEnum, "Microsoft\\Kiota\\Abstractions\\", "Enum"), + new (static x => x is CodeEnum, AbstractionsNamespaceName, "Enum"), new(static x => x is CodeProperty {Type.Name: {}} property && property.Type.Name.Equals("DateTime", StringComparison.OrdinalIgnoreCase), "", "\\DateTime"), new(static x => x is CodeProperty {Type.Name: {}} property && property.Type.Name.Equals("DateTimeOffset", StringComparison.OrdinalIgnoreCase), "", "\\DateTime"), - new(static x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.ClientConstructor), "Microsoft\\Kiota\\Abstractions", "ApiClientBuilder"), + new(static x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.ClientConstructor), AbstractionsNamespaceName, "ApiClientBuilder"), new(static x => x is CodeProperty property && property.IsOfKind(CodePropertyKind.QueryParameter) && !string.IsNullOrEmpty(property.SerializationName), "Microsoft\\Kiota\\Abstractions", "QueryParameter"), - new(static x => x is CodeClass codeClass && codeClass.IsOfKind(CodeClassKind.RequestConfiguration), "Microsoft\\Kiota\\Abstractions", "RequestOption"), + new(static x => x is CodeClass codeClass && codeClass.IsOfKind(CodeClassKind.RequestConfiguration), AbstractionsNamespaceName, "RequestOption"), new (static x => x is CodeClass { OriginalComposedType: CodeIntersectionType intersectionType } && intersectionType.Types.Any(static y => !y.IsExternal), - "Microsoft\\Kiota\\Abstractions\\Serialization", "ParseNodeHelper"), + $@"{AbstractionsNamespaceName}\Serialization", "ParseNodeHelper"), new (static x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.RequestExecutor, CodeMethodKind.RequestGenerator) && method.Parameters.Any(static y => y.IsOfKind(CodeParameterKind.RequestBody) && y.Type.Name.Equals(MultipartBodyClassName, StringComparison.OrdinalIgnoreCase)), - @"Microsoft\Kiota\Abstractions", MultipartBodyClassName) + AbstractionsNamespaceName, MultipartBodyClassName) }; private const string MultipartBodyClassName = "MultiPartBody"; diff --git a/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs b/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs index 4a96bd3eec..7aea844e06 100644 --- a/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs +++ b/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs @@ -855,7 +855,7 @@ private void WriteModelFactoryMethodBody(CodeMethod codeElement, CodeClass paren { if (parentClass.DiscriminatorInformation.ShouldWriteDiscriminatorForUnionType || parentClass.DiscriminatorInformation.ShouldWriteDiscriminatorForIntersectionType) writer.WriteLine($"{ResultVarName} = new {parentClass.Name.ToFirstCharacterUpperCase()}();"); - var writeDiscriminatorValueRead = parentClass.DiscriminatorInformation.ShouldWriteParseNodeCheck && !parentClass.DiscriminatorInformation.ShouldWriteDiscriminatorForIntersectionType; + var writeDiscriminatorValueRead = parentClass.DiscriminatorInformation is { ShouldWriteParseNodeCheck: true, ShouldWriteDiscriminatorForIntersectionType: false, HasBasicDiscriminatorInformation: true }; if (writeDiscriminatorValueRead && codeElement.Parameters.OfKind(CodeParameterKind.ParseNode) is CodeParameter parseNodeParameter) @@ -868,7 +868,7 @@ private void WriteModelFactoryMethodBody(CodeMethod codeElement, CodeClass paren if (parentClass.DiscriminatorInformation.ShouldWriteDiscriminatorForInheritedType) WriteFactoryMethodBodyForInheritedModel(parentClass.DiscriminatorInformation.DiscriminatorMappings, writer, codeElement); - else if (parentClass.DiscriminatorInformation.ShouldWriteDiscriminatorForUnionType && parentClass.DiscriminatorInformation.HasBasicDiscriminatorInformation) + else if (parentClass.DiscriminatorInformation is { ShouldWriteDiscriminatorForUnionType: true, HasBasicDiscriminatorInformation: true}) WriteFactoryMethodBodyForUnionModelForDiscriminatedTypes(codeElement, parentClass, writer); else if (parentClass.DiscriminatorInformation.ShouldWriteDiscriminatorForIntersectionType) WriteFactoryMethodBodyForIntersectionModel(codeElement, parentClass, writer); From 490289cb984db3370ea1d3c7d8cca1b380ab979c Mon Sep 17 00:00:00 2001 From: Silas Kenneth Date: Wed, 29 May 2024 15:35:24 +0300 Subject: [PATCH 02/33] Refactor to use constant. --- src/Kiota.Builder/Refiners/PhpRefiner.cs | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/src/Kiota.Builder/Refiners/PhpRefiner.cs b/src/Kiota.Builder/Refiners/PhpRefiner.cs index 19ccb59918..d3b3c7f242 100644 --- a/src/Kiota.Builder/Refiners/PhpRefiner.cs +++ b/src/Kiota.Builder/Refiners/PhpRefiner.cs @@ -75,7 +75,7 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance AddParentClassToErrorClasses( generatedCode, "ApiException", - "Microsoft\\Kiota\\Abstractions" + AbstractionsNamespaceName ); MoveClassesWithNamespaceNamesUnderNamespace(generatedCode); AddConstructorsForDefaultValues(generatedCode, true); @@ -110,7 +110,7 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance } ); cancellationToken.ThrowIfCancellationRequested(); - AddSerializationModulesImport(generatedCode, ["Microsoft\\Kiota\\Abstractions\\ApiClientBuilder"], null, '\\'); + AddSerializationModulesImport(generatedCode, [$@"{AbstractionsNamespaceName}\ApiClientBuilder"], null, '\\'); cancellationToken.ThrowIfCancellationRequested(); AddPropertiesAndMethodTypesImports(generatedCode, true, false, true); CorrectBackingStoreSetterParam(generatedCode); @@ -123,7 +123,7 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance Name = "BaseRequestConfiguration", Declaration = new CodeType { - Name = "Microsoft\\Kiota\\Abstractions", + Name = AbstractionsNamespaceName, IsExternal = true } }); @@ -210,7 +210,7 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance new(static x => x is CodeProperty {Type.Name: {}} property && property.Type.Name.Equals("DateTime", StringComparison.OrdinalIgnoreCase), "", "\\DateTime"), new(static x => x is CodeProperty {Type.Name: {}} property && property.Type.Name.Equals("DateTimeOffset", StringComparison.OrdinalIgnoreCase), "", "\\DateTime"), new(static x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.ClientConstructor), AbstractionsNamespaceName, "ApiClientBuilder"), - new(static x => x is CodeProperty property && property.IsOfKind(CodePropertyKind.QueryParameter) && !string.IsNullOrEmpty(property.SerializationName), "Microsoft\\Kiota\\Abstractions", "QueryParameter"), + new(static x => x is CodeProperty property && property.IsOfKind(CodePropertyKind.QueryParameter) && !string.IsNullOrEmpty(property.SerializationName), AbstractionsNamespaceName, "QueryParameter"), new(static x => x is CodeClass codeClass && codeClass.IsOfKind(CodeClassKind.RequestConfiguration), AbstractionsNamespaceName, "RequestOption"), new (static x => x is CodeClass { OriginalComposedType: CodeIntersectionType intersectionType } && intersectionType.Types.Any(static y => !y.IsExternal), $@"{AbstractionsNamespaceName}\Serialization", "ParseNodeHelper"), @@ -343,7 +343,7 @@ private void AddCollectionValidationUtilImportToModels(CodeElement codeElement) { if (codeElement is CodeClass codeClass && codeClass.Kind == CodeClassKind.Model) { - var typeUtilsUsing = new CodeUsing { Name = "TypeUtils", Declaration = new CodeType { Name = "Microsoft\\Kiota\\Abstractions\\Types", IsExternal = true } }; + var typeUtilsUsing = new CodeUsing { Name = "TypeUtils", Declaration = new CodeType { Name = $@"{AbstractionsNamespaceName}\Types", IsExternal = true } }; if (codeClass.Properties.Any(x => x.Kind == CodePropertyKind.Custom && x.Type is CodeType codeType From 13787c04e30b2d8e973ad3892e902a73b2991182 Mon Sep 17 00:00:00 2001 From: Silas Kenneth Date: Wed, 29 May 2024 15:44:14 +0300 Subject: [PATCH 03/33] Fix formating issue and wrong import. --- src/Kiota.Builder/Refiners/PhpRefiner.cs | 4 ++-- src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/src/Kiota.Builder/Refiners/PhpRefiner.cs b/src/Kiota.Builder/Refiners/PhpRefiner.cs index d3b3c7f242..95985de135 100644 --- a/src/Kiota.Builder/Refiners/PhpRefiner.cs +++ b/src/Kiota.Builder/Refiners/PhpRefiner.cs @@ -32,7 +32,7 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance Name = "BaseRequestBuilder", Declaration = new CodeType { - Name = $@"{AbstractionsNamespaceName}\Serialization", + Name = $@"{AbstractionsNamespaceName}", IsExternal = true } }, AccessModifier.Public); @@ -47,7 +47,7 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance _configuration.UsesBackingStore, static s => s, false, - AbstractionsNamespaceName, + $@"{AbstractionsNamespaceName}\Serialization", "ComposedTypeWrapper" ); ReplaceReservedNames(generatedCode, new PhpReservedNamesProvider(), reservedWord => $"Escaped{reservedWord.ToFirstCharacterUpperCase()}", new HashSet { typeof(CodeEnumOption) }); diff --git a/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs b/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs index 7aea844e06..8b8c870165 100644 --- a/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs +++ b/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs @@ -868,7 +868,7 @@ private void WriteModelFactoryMethodBody(CodeMethod codeElement, CodeClass paren if (parentClass.DiscriminatorInformation.ShouldWriteDiscriminatorForInheritedType) WriteFactoryMethodBodyForInheritedModel(parentClass.DiscriminatorInformation.DiscriminatorMappings, writer, codeElement); - else if (parentClass.DiscriminatorInformation is { ShouldWriteDiscriminatorForUnionType: true, HasBasicDiscriminatorInformation: true}) + else if (parentClass.DiscriminatorInformation is { ShouldWriteDiscriminatorForUnionType: true, HasBasicDiscriminatorInformation: true }) WriteFactoryMethodBodyForUnionModelForDiscriminatedTypes(codeElement, parentClass, writer); else if (parentClass.DiscriminatorInformation.ShouldWriteDiscriminatorForIntersectionType) WriteFactoryMethodBodyForIntersectionModel(codeElement, parentClass, writer); From f3d039d093cbf5a8e284bacabaceff966e8b9025 Mon Sep 17 00:00:00 2001 From: Andrew Omondi Date: Wed, 29 May 2024 17:40:53 +0300 Subject: [PATCH 04/33] WIP: cleanup responses and components. Todo extensions. --- .../Plugins/PluginsGenerationService.cs | 52 ++++++++++++++++++- 1 file changed, 51 insertions(+), 1 deletion(-) diff --git a/src/Kiota.Builder/Plugins/PluginsGenerationService.cs b/src/Kiota.Builder/Plugins/PluginsGenerationService.cs index 7a51fe4b2e..3fa54d249e 100644 --- a/src/Kiota.Builder/Plugins/PluginsGenerationService.cs +++ b/src/Kiota.Builder/Plugins/PluginsGenerationService.cs @@ -51,7 +51,8 @@ public async Task GenerateManifestAsync(CancellationToken cancellationToken = de await using var fileWriter = new StreamWriter(descriptionStream); #pragma warning restore CA2007 // Consider calling ConfigureAwait on the awaited task var descriptionWriter = new OpenApiYamlWriter(fileWriter); - OAIDocument.SerializeAsV3(descriptionWriter); + var trimmedPluginDocument = GetDocumentWithTrimmedComponentsAndResponses(OAIDocument); + trimmedPluginDocument.SerializeAsV3(descriptionWriter); descriptionWriter.Flush(); // write the plugins @@ -93,6 +94,55 @@ public async Task GenerateManifestAsync(CancellationToken cancellationToken = de await writer.FlushAsync(cancellationToken).ConfigureAwait(false); } } + + + private OpenApiDocument GetDocumentWithTrimmedComponentsAndResponses(OpenApiDocument doc) + { + // ensure the info and components are not null + doc.Info ??= new OpenApiInfo(); + doc.Components ??= new OpenApiComponents(); + + if (string.IsNullOrEmpty(doc.Info?.Version)) // filtering fails if there's no version. + return doc; + + //empty out the responses with a single 2XX + foreach (var operation in doc.Paths.SelectMany(static item => item.Value.Operations.Values)) + { + operation.Responses = new OpenApiResponses() + { + { + "2XX",new OpenApiResponse + { + Content = new Dictionary + { + { + "text/plain", new OpenApiMediaType + { + Schema = new OpenApiSchema + { + Type = "string" + } + } + } + } + } + } + }; + } + + // remove unused components + var requestUrls = new Dictionary>(); + var basePath = doc.GetAPIRootUrl(Configuration.OpenAPIFilePath); + foreach (var path in doc.Paths.Where(static path => path.Value.Operations.Count > 0)) + { + var key = string.IsNullOrEmpty(basePath) ? path.Key : $"{basePath}/{path.Key.TrimStart(KiotaBuilder.ForwardSlash)}"; + requestUrls[key] = path.Value.Operations.Keys.Select(static key => key.ToString().ToUpperInvariant()).ToList(); + } + + var predicate = OpenApiFilterService.CreatePredicate(requestUrls: requestUrls, source: doc); + return OpenApiFilterService.CreateFilteredDocument(doc, predicate); + } + private PluginManifestDocument GetV1ManifestDocument(string openApiDocumentPath) { var descriptionForHuman = OAIDocument.Info?.Description.CleanupXMLString() is string d && !string.IsNullOrEmpty(d) ? d : $"Description for {OAIDocument.Info?.Title.CleanupXMLString()}"; From df988e80f53852e901a099fa4bd14aa67bb9f3e0 Mon Sep 17 00:00:00 2001 From: "peter.wurzinger" Date: Wed, 29 May 2024 16:41:42 +0200 Subject: [PATCH 05/33] Fix 4475 by introducing fully qualified names for implementations of ITypeDefinition --- .../Writers/CSharp/CSharpConventionService.cs | 72 ++----------------- .../Writers/CSharp/CodeMethodWriter.cs | 16 +++-- .../CSharp/TypeDefinitionExtensions.cs | 34 +++++++++ 3 files changed, 48 insertions(+), 74 deletions(-) create mode 100644 src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs diff --git a/src/Kiota.Builder/Writers/CSharp/CSharpConventionService.cs b/src/Kiota.Builder/Writers/CSharp/CSharpConventionService.cs index 5a5c1309e8..c975d79c0e 100644 --- a/src/Kiota.Builder/Writers/CSharp/CSharpConventionService.cs +++ b/src/Kiota.Builder/Writers/CSharp/CSharpConventionService.cs @@ -177,7 +177,7 @@ public string GetTypeString(CodeTypeBase code, CodeElement targetElement, bool i throw new InvalidOperationException($"CSharp does not support union types, the union type {code.Name} should have been filtered out by the refiner"); if (code is CodeType currentType) { - var typeName = TranslateTypeAndAvoidUsingNamespaceSegmentNames(currentType, targetElement); + var typeName = TranslateType(currentType); var nullableSuffix = ShouldTypeHaveNullableMarker(code, typeName) && includeNullableInformation ? NullableMarkerAsString : string.Empty; var collectionPrefix = currentType.CollectionKind == CodeTypeCollectionKind.Complex && includeCollectionInformation ? "List<" : string.Empty; var collectionSuffix = currentType.CollectionKind switch @@ -196,76 +196,14 @@ public string GetTypeString(CodeTypeBase code, CodeElement targetElement, bool i throw new InvalidOperationException($"type of type {code?.GetType()} is unknown"); } - private string TranslateTypeAndAvoidUsingNamespaceSegmentNames(CodeType currentType, CodeElement targetElement) - { - var parentElementsHash = targetElement.Parent is CodeClass parentClass ? - parentClass.Methods.Select(static x => x.Name) - .Union(parentClass.Properties.Select(static x => x.Name)) - .Distinct(StringComparer.OrdinalIgnoreCase) - .ToHashSet(StringComparer.OrdinalIgnoreCase) : - new HashSet(0, StringComparer.OrdinalIgnoreCase); - - var typeName = TranslateType(currentType); - var areElementsInSameNamesSpace = DoesTypeExistsInSameNamesSpaceAsTarget(currentType, targetElement); - if (currentType.TypeDefinition != null && - ( - GetNamesInUseByNamespaceSegments(targetElement).Contains(typeName) && !areElementsInSameNamesSpace // match if elements are not in the same namespace and the type name is used in the namespace segments - || parentElementsHash.Contains(typeName) // match if type name is used in the parent elements segments - || !areElementsInSameNamesSpace && DoesTypeExistsInTargetAncestorNamespace(currentType, targetElement) // match if elements are not in the same namespace and the type exists in target ancestor namespace - || !areElementsInSameNamesSpace && DoesTypeExistsInOtherImportedNamespaces(currentType, targetElement) // match if elements is not imported already by another namespace. - ) - ) - return $"{currentType.TypeDefinition.GetImmediateParentOfType().Name}.{typeName}"; - return typeName; - } - - private static bool DoesTypeExistsInSameNamesSpaceAsTarget(CodeType currentType, CodeElement targetElement) - { - return currentType?.TypeDefinition?.GetImmediateParentOfType()?.Name.Equals(targetElement?.GetImmediateParentOfType()?.Name, StringComparison.OrdinalIgnoreCase) ?? false; - } - - private static bool DoesTypeExistsInTargetAncestorNamespace(CodeType currentType, CodeElement targetElement) - { - // Avoid type ambiguity on similarly named classes. Currently, if we have namespaces A and A.B where both namespaces have type T, - // Trying to use type A.B.T in namespace A without using a qualified name will break the build. - // Similarly, if we have type A.B.C.D.T1 that needs to be used within type A.B.C.T2, but there's also a type - // A.B.T1, using T1 in T2 will resolve A.B.T1 even if you have a using statement with A.B.C.D. - var hasChildWithName = false; - if (currentType != null && currentType.TypeDefinition != null && !currentType.IsExternal && targetElement != null) - { - var typeName = currentType.TypeDefinition.Name; - var ns = targetElement.GetImmediateParentOfType(); - var rootNs = ns?.GetRootNamespace(); - while (ns is not null && ns != rootNs && !hasChildWithName) - { - hasChildWithName = ns.GetChildElements(true).OfType().Any(c => c.Name?.Equals(typeName, StringComparison.OrdinalIgnoreCase) == true); - ns = ns.Parent is CodeNamespace n ? n : (ns.GetImmediateParentOfType()); - } - } - return hasChildWithName; - } - - private static bool DoesTypeExistsInOtherImportedNamespaces(CodeType currentType, CodeElement targetElement) - { - if (currentType.TypeDefinition is CodeClass { Parent: CodeNamespace currentTypeNamespace } codeClass) - { - var targetClass = targetElement.GetImmediateParentOfType(); - var importedNamespaces = targetClass.StartBlock.Usings - .Where(codeUsing => !codeUsing.IsExternal // 1. Are defined during generation(not external) - && codeUsing.Declaration?.TypeDefinition != null - && !codeUsing.Name.Equals(currentTypeNamespace.Name, StringComparison.OrdinalIgnoreCase)) // 2. Do not match the namespace of the current type - .Select(static codeUsing => codeUsing.Declaration!.TypeDefinition!.GetImmediateParentOfType()) - .DistinctBy(static declaredNamespace => declaredNamespace.Name); - - return importedNamespaces.Any(importedNamespace => (importedNamespace.FindChildByName(codeClass.Name, false) != null) - || (importedNamespace.FindChildByName(codeClass.Name, false) != null)); - } - return false; - } public override string TranslateType(CodeType type) { ArgumentNullException.ThrowIfNull(type); + + if (type.TypeDefinition is ITypeDefinition typeDefinition) + return typeDefinition.GetFullyQualifiedName(); + return type.Name switch { "integer" => "int", diff --git a/src/Kiota.Builder/Writers/CSharp/CodeMethodWriter.cs b/src/Kiota.Builder/Writers/CSharp/CodeMethodWriter.cs index 66d29a759f..341e5eda25 100644 --- a/src/Kiota.Builder/Writers/CSharp/CodeMethodWriter.cs +++ b/src/Kiota.Builder/Writers/CSharp/CodeMethodWriter.cs @@ -105,7 +105,9 @@ private void WriteRawUrlBuilderBody(CodeClass parentClass, CodeMethod codeElemen { var rawUrlParameter = codeElement.Parameters.OfKind(CodeParameterKind.RawUrl) ?? throw new InvalidOperationException("RawUrlBuilder method should have a RawUrl parameter"); var requestAdapterProperty = parentClass.GetPropertyOfKind(CodePropertyKind.RequestAdapter) ?? throw new InvalidOperationException("RawUrlBuilder method should have a RequestAdapter property"); - writer.WriteLine($"return new {parentClass.Name.ToFirstCharacterUpperCase()}({rawUrlParameter.Name.ToFirstCharacterLowerCase()}, {requestAdapterProperty.Name.ToFirstCharacterUpperCase()});"); + + var fullName = parentClass.GetFullyQualifiedName(); + writer.WriteLine($"return new {fullName}({rawUrlParameter.Name.ToFirstCharacterLowerCase()}, {requestAdapterProperty.Name.ToFirstCharacterUpperCase()});"); } private static readonly CodePropertyTypeComparer CodePropertyTypeForwardComparer = new(); private static readonly CodePropertyTypeComparer CodePropertyTypeBackwardComparer = new(true); @@ -117,13 +119,13 @@ private void WriteFactoryMethodBodyForInheritedModel(CodeMethod codeElement, Cod { writer.WriteLine($"\"{mappedType.Key}\" => new {conventions.GetTypeString(mappedType.Value.AllTypes.First(), codeElement)}(),"); } - writer.WriteLine($"_ => new {parentClass.Name.ToFirstCharacterUpperCase()}(),"); + writer.WriteLine($"_ => new {parentClass.GetFullyQualifiedName()}(),"); writer.CloseBlock("};"); } private const string ResultVarName = "result"; private void WriteFactoryMethodBodyForUnionModel(CodeMethod codeElement, CodeClass parentClass, CodeParameter parseNodeParameter, LanguageWriter writer) { - writer.WriteLine($"var {ResultVarName} = new {parentClass.Name.ToFirstCharacterUpperCase()}();"); + writer.WriteLine($"var {ResultVarName} = new {parentClass.GetFullyQualifiedName()}();"); var includeElse = false; foreach (var property in parentClass.GetPropertiesOfKind(CodePropertyKind.Custom) .OrderBy(static x => x, CodePropertyTypeForwardComparer) @@ -150,7 +152,7 @@ private void WriteFactoryMethodBodyForUnionModel(CodeMethod codeElement, CodeCla } private void WriteFactoryMethodBodyForIntersectionModel(CodeMethod codeElement, CodeClass parentClass, CodeParameter parseNodeParameter, LanguageWriter writer) { - writer.WriteLine($"var {ResultVarName} = new {parentClass.Name.ToFirstCharacterUpperCase()}();"); + writer.WriteLine($"var {ResultVarName} = new {parentClass.GetFullyQualifiedName()}();"); var includeElse = false; foreach (var property in parentClass.GetPropertiesOfKind(CodePropertyKind.Custom) .Where(static x => x.Type is not CodeType propertyType || propertyType.IsCollection || propertyType.TypeDefinition is not CodeClass) @@ -202,7 +204,7 @@ private void WriteFactoryMethodBody(CodeMethod codeElement, CodeClass parentClas else if (parentClass.DiscriminatorInformation.ShouldWriteDiscriminatorForIntersectionType) WriteFactoryMethodBodyForIntersectionModel(codeElement, parentClass, parseNodeParameter, writer); else - writer.WriteLine($"return new {parentClass.Name.ToFirstCharacterUpperCase()}();"); + writer.WriteLine($"return new {parentClass.GetFullyQualifiedName()}();"); } private void WriteRequestBuilderBody(CodeClass parentClass, CodeMethod codeElement, LanguageWriter writer) { @@ -357,7 +359,7 @@ private string GetDeserializationMethodName(CodeTypeBase propType, CodeMethod me return $"GetCollectionOfObjectValues<{propertyType}>({propertyType}.CreateFromDiscriminatorValue){collectionMethod}"; } else if (currentType.TypeDefinition is CodeEnum enumType) - return $"GetEnumValue<{enumType.Name.ToFirstCharacterUpperCase()}>()"; + return $"GetEnumValue<{enumType.GetFullyQualifiedName()}>()"; } return propertyType switch { @@ -662,7 +664,7 @@ private string GetSerializationMethodName(CodeTypeBase propType, CodeMethod meth else return $"WriteCollectionOfObjectValues<{propertyType}>"; else if (currentType.TypeDefinition is CodeEnum enumType) - return $"WriteEnumValue<{enumType.Name.ToFirstCharacterUpperCase()}>"; + return $"WriteEnumValue<{enumType.GetFullyQualifiedName()}>"; } return propertyType switch diff --git a/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs b/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs new file mode 100644 index 0000000000..91e7c58e61 --- /dev/null +++ b/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs @@ -0,0 +1,34 @@ +using System; +using System.Text; +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Extensions; + +namespace Kiota.Builder.Writers.CSharp; + +internal static class TypeDefinitionExtensions +{ + public static string GetFullyQualifiedName(this ITypeDefinition typeDefinition) + { + ArgumentNullException.ThrowIfNull(typeDefinition); + + var fullNameBuilder = new StringBuilder(); + return GetFullyQualifiedName(typeDefinition, fullNameBuilder).ToString(); + } + + private static StringBuilder GetFullyQualifiedName(ITypeDefinition codeClass, StringBuilder fullNameBuilder) + { + fullNameBuilder.Insert(0, codeClass.Name.ToFirstCharacterUpperCase()); + if (codeClass.Parent is CodeClass parentClass) + { + fullNameBuilder.Insert(0, '.'); + return GetFullyQualifiedName(parentClass, fullNameBuilder); + } + if (codeClass.Parent is CodeNamespace ns && !string.IsNullOrEmpty(ns.Name)) + { + fullNameBuilder.Insert(0, '.'); + fullNameBuilder.Insert(0, ns.Name); + } + + return fullNameBuilder; + } +} From f793ece83ce50d8d7597a67de5c4d37dcdbd6fa3 Mon Sep 17 00:00:00 2001 From: "peter.wurzinger" Date: Wed, 29 May 2024 17:05:17 +0200 Subject: [PATCH 06/33] Fix existing tests --- .../Writers/CLI/CliCodeMethodWriterTests.cs | 20 +++++++++---------- .../Writers/CSharp/CodeMethodWriterTests.cs | 6 +++--- .../Writers/CSharp/CodePropertyWriterTests.cs | 8 ++++---- 3 files changed, 17 insertions(+), 17 deletions(-) diff --git a/tests/Kiota.Builder.Tests/Writers/CLI/CliCodeMethodWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/CLI/CliCodeMethodWriterTests.cs index ab5d455c24..1755ff0519 100644 --- a/tests/Kiota.Builder.Tests/Writers/CLI/CliCodeMethodWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/CLI/CliCodeMethodWriterTests.cs @@ -241,7 +241,7 @@ public void WritesIndexerCommands() writer.Write(method); var result = tw.ToString(); - Assert.Contains("var builder = new TestClass", result); + Assert.Contains("var builder = new Test.Name.Sub.TestClass", result); Assert.Contains("var commands = new List();", result); Assert.Contains("commands.Add(builder.BuildTestMethod1());", result); Assert.Contains("commands.AddRange(builder.BuildTestMethod2());", result); @@ -325,7 +325,7 @@ public void WritesMatchingIndexerCommandsIntoExecutableCommand() writer.Write(method); var result = tw.ToString(); - Assert.Contains("var testItemIdx = new TestItemRequestBuilder();", result); + Assert.Contains("var testItemIdx = new Test.Name.Sub.TestItemRequestBuilder();", result); Assert.Contains("var command = testItemIdx.BuildTestMethod1();", result); Assert.Contains("var cmds = testItemIdx.BuildTestMethod2();", result); Assert.DoesNotContain("execCommands.AddRange(cmds.Item1)", result); @@ -418,12 +418,12 @@ public void WritesMatchingIndexerCommandsIntoContainerCommand() writer.Write(method); var result = tw.ToString(); - Assert.Contains("var testItemIndexer = new TestIndexItemRequestBuilder();", result); + Assert.Contains("var testItemIndexer = new Test.Name.Sub.TestIndexItemRequestBuilder();", result); Assert.Contains("var command = testItemIndexer.BuildTestMethod1();", result); Assert.Contains("var cmds = testItemIndexer.BuildTestMethod2();", result); Assert.DoesNotContain("execCommands.AddRange(cmds.Item1);", result); Assert.Contains("nonExecCommands.AddRange(cmds.Item2);", result); - Assert.Contains("var builder = new TestNavItemRequestBuilder", result); + Assert.Contains("var builder = new Test.TestNavItemRequestBuilder", result); Assert.Contains("nonExecCommands.Add(builder.BuildTestMethod11());", result); Assert.Contains("return command;", result); Assert.DoesNotContain("nonExecCommands.Add(builder.BuildTestMethod3());", result); @@ -553,7 +553,7 @@ public void WritesNavCommandThatSkipsReusedNavCommandInstance() var result = tw.ToString(); Assert.Contains("var command = new Command(\"user\");", result); - Assert.Contains("var builder = new TestNavItemRequestBuilder();", result); + Assert.Contains("var builder = new Test.TestNavItemRequestBuilder();", result); Assert.Contains("execCommands.Add(builder.BuildExecutableTestMethod());", result); Assert.Contains("return command;", result); Assert.DoesNotContain("BuildNavTestMethod", result); @@ -596,7 +596,7 @@ public void WritesContainerCommands() var result = tw.ToString(); Assert.Contains("var command = new Command(\"user\");", result); - Assert.Contains("var builder = new TestClass1", result); + Assert.Contains("var builder = new Test.Name.Sub1.Sub2.TestClass1", result); Assert.Contains("nonExecCommands.Add(builder.BuildTestMethod1());", result); Assert.Contains("nonExecCommands.Add(builder.BuildTestMethod2());", result); Assert.Contains("return command;", result); @@ -637,7 +637,7 @@ public void WritesRequestBuilderWithParametersCommands() var result = tw.ToString(); Assert.Contains("var command = new Command(\"user\");", result); - Assert.Contains("var builder = new TestClass1", result); + Assert.Contains("var builder = new Test.Name.Sub1.Sub2.TestClass1", result); Assert.Contains("nonExecCommands.Add(builder.BuildTestMethod1());", result); Assert.Contains("nonExecCommands.Add(builder.BuildTestMethod2());", result); Assert.Contains("return command;", result); @@ -1085,7 +1085,7 @@ public void WritesExecutableCommandForPostRequestWithModelBody() Assert.Contains("command.AddOption(bodyOption);", result); Assert.Contains("var body = invocationContext.ParseResult.GetValueForOption(bodyOption) ?? string.Empty;", result); Assert.Contains("using var stream = new MemoryStream(Encoding.UTF8.GetBytes(body));", result); - Assert.Contains("var model = parseNode.GetObjectValue(Content.CreateFromDiscriminatorValue);", result); + Assert.Contains("var model = parseNode.GetObjectValue(Test.Content.CreateFromDiscriminatorValue);", result); Assert.Contains("if (model is null)", result); Assert.Contains("Console.Error.WriteLine(\"No model data to send.\")", result); Assert.Contains("var requestInfo = CreatePostRequestInformation", result); @@ -1171,7 +1171,7 @@ public void WritesExecutableCommandForPostRequestWithModelBodyAndContentType() Assert.Contains("var body = invocationContext.ParseResult.GetValueForOption(bodyOption) ?? string.Empty;", result); Assert.Contains("var contentType = invocationContext.ParseResult.GetValueForOption(contentTypeOption);", result); Assert.Contains("using var stream = new MemoryStream(Encoding.UTF8.GetBytes(body));", result); - Assert.Contains("var model = parseNode.GetObjectValue(Content.CreateFromDiscriminatorValue);", result); + Assert.Contains("var model = parseNode.GetObjectValue(Test.Content.CreateFromDiscriminatorValue);", result); Assert.Contains("if (model is null)", result); Assert.Contains("Console.Error.WriteLine(\"No model data to send.\")", result); Assert.Contains("var requestInfo = CreatePostRequestInformation(model, contentType", result); @@ -1242,7 +1242,7 @@ public void WritesExecutableCommandForPostRequestWithCollectionModel() Assert.Contains("command.AddOption(bodyOption);", result); Assert.Contains("var body = invocationContext.ParseResult.GetValueForOption(bodyOption) ?? string.Empty;", result); Assert.Contains("using var stream = new MemoryStream(Encoding.UTF8.GetBytes(body));", result); - Assert.Contains("var model = parseNode.GetCollectionOfObjectValues(Content.CreateFromDiscriminatorValue)?.ToList();", result); + Assert.Contains("var model = parseNode.GetCollectionOfObjectValues(Test.Content.CreateFromDiscriminatorValue)?.ToList();", result); Assert.Contains("if (model is null)", result); Assert.Contains("Console.Error.WriteLine(\"No model data to send.\")", result); Assert.Contains("var requestInfo = CreatePostRequestInformation", result); diff --git a/tests/Kiota.Builder.Tests/Writers/CSharp/CodeMethodWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/CSharp/CodeMethodWriterTests.cs index ae87bd46ba..2fe8182ed3 100644 --- a/tests/Kiota.Builder.Tests/Writers/CSharp/CodeMethodWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/CSharp/CodeMethodWriterTests.cs @@ -738,7 +738,7 @@ public void WritesModelFactoryBodyAndDisambiguateAmbiguousImportedTypes() Assert.Contains("var mappingValue = parseNode.GetChildNode(\"@odata.type\")?.GetStringValue()", result); Assert.Contains("return mappingValue switch", result); Assert.Contains("\"namespaceLevelOne.ConflictingModel\" => new namespaceLevelOne.ConflictingModel(),", result); //Assert the disambiguation happens due to the enum imported - Assert.Contains("_ => new ConflictingModelBaseClass()", result); + Assert.Contains("_ => new models.ConflictingModelBaseClass()", result); AssertExtensions.CurlyBracesAreClosed(result); } [Fact] @@ -1532,7 +1532,7 @@ public void WritesConstructorWithEnumValue() writer.Write(method); var result = tw.ToString(); Assert.Contains(parentClass.Name.ToFirstCharacterUpperCase(), result); - Assert.Contains($"{propName.ToFirstCharacterUpperCase()} = {codeEnum.Name.ToFirstCharacterUpperCase()}.{defaultValue.CleanupSymbolName()}", result);//ensure symbol is cleaned up + Assert.Contains($"{propName.ToFirstCharacterUpperCase()} = {modelsNamespace.Name}.{codeEnum.Name.ToFirstCharacterUpperCase()}.{defaultValue.CleanupSymbolName()}", result);//ensure symbol is cleaned up } [Fact] public void WritesConstructorAndIncludesSanitizedEnumValue() @@ -1557,7 +1557,7 @@ public void WritesConstructorAndIncludesSanitizedEnumValue() var result = tw.ToString(); Assert.Contains(parentClass.Name.ToFirstCharacterUpperCase(), result); Assert.Contains("PictureSize.Slash", result);//ensure symbol is cleaned up - Assert.Contains($"{propName.ToFirstCharacterUpperCase()} = {codeEnum.Name.ToFirstCharacterUpperCase()}.{defaultValue.CleanupSymbolName()}", result);//ensure symbol is cleaned up + Assert.Contains($"{propName.ToFirstCharacterUpperCase()} = {modelsNamespace.Name}.{codeEnum.Name.ToFirstCharacterUpperCase()}.{defaultValue.CleanupSymbolName()}", result);//ensure symbol is cleaned up } [Fact] public void WritesConstructorAndDisambiguatesEnumType() diff --git a/tests/Kiota.Builder.Tests/Writers/CSharp/CodePropertyWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/CSharp/CodePropertyWriterTests.cs index a12cd43520..96524a9e7f 100644 --- a/tests/Kiota.Builder.Tests/Writers/CSharp/CodePropertyWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/CSharp/CodePropertyWriterTests.cs @@ -74,7 +74,7 @@ public void WritesRequestBuilder() writer.Write(property); var result = tw.ToString(); Assert.Contains("get =>", result); - Assert.Contains($"new {TypeName}", result); + Assert.Contains($"new {rootNamespace.Name}.{TypeName}", result); Assert.Contains("RequestAdapter", result); Assert.Contains("PathParameters", result); } @@ -103,7 +103,7 @@ public void MapsCustomPropertiesToBackingStore() property.Kind = CodePropertyKind.Custom; writer.Write(property); var result = tw.ToString(); - Assert.Contains("get { return BackingStore?.Get(\"propertyName\"); }", result); + Assert.Contains("get { return BackingStore?.Get<" + rootNamespace.Name + ".SomeCustomClass>(\"propertyName\"); }", result); Assert.Contains("set { BackingStore?.Set(\"propertyName\", value);", result); } [Fact] @@ -113,7 +113,7 @@ public void MapsAdditionalDataPropertiesToBackingStore() property.Kind = CodePropertyKind.AdditionalData; writer.Write(property); var result = tw.ToString(); - Assert.Contains("get { return BackingStore.Get(\"propertyName\") ?? new Dictionary(); }", result); + Assert.Contains("get { return BackingStore.Get<" + rootNamespace.Name + ".SomeCustomClass>(\"propertyName\") ?? new Dictionary(); }", result); Assert.Contains("set { BackingStore.Set(\"propertyName\", value);", result); } [Fact] @@ -207,7 +207,7 @@ public void DisambiguateAmbiguousImportedTypes() // Assert: properties types are disambiguated. Assert.Contains("namespaceLevelOne.SomeCustomClass", result); - Assert.Contains("defaultNamespace.SomeCustomClass", result); + Assert.Contains($"{rootNamespace.Name}.SomeCustomClass", result); } [Fact] public void WritesDeprecationInformation() From 0436897ee81020384a5bc925f67f8ccda4afe415 Mon Sep 17 00:00:00 2001 From: mspearey Date: Thu, 30 May 2024 15:22:24 +0100 Subject: [PATCH 07/33] fix missing models when no multipart encoding exists Model declarations not created when multipart/form-data exists with no encoding in mime content Added check for multipart/form-data with additional check if no other mime content --- .../Extensions/OpenApiOperationExtensions.cs | 9 + src/Kiota.Builder/KiotaBuilder.cs | 40 ++- .../Kiota.Builder.Tests/KiotaBuilderTests.cs | 281 +++++++++++++++++- 3 files changed, 315 insertions(+), 15 deletions(-) diff --git a/src/Kiota.Builder/Extensions/OpenApiOperationExtensions.cs b/src/Kiota.Builder/Extensions/OpenApiOperationExtensions.cs index 6d14c22441..aca62940e2 100644 --- a/src/Kiota.Builder/Extensions/OpenApiOperationExtensions.cs +++ b/src/Kiota.Builder/Extensions/OpenApiOperationExtensions.cs @@ -47,6 +47,15 @@ internal static bool IsMultipartFormDataSchema(this IDictionary source, StructuredMimeTypesCollection structuredMimeTypes) + { + ArgumentNullException.ThrowIfNull(source); + ArgumentNullException.ThrowIfNull(structuredMimeTypes); + if (structuredMimeTypes.Count == 0) return false; + if (!source.ContainsKey(multipartMimeTypes.First())) return false; + if (source.Count == 1) return true; + return structuredMimeTypes.First() == multipartMimeTypes.First(); + } internal static IEnumerable GetValidSchemas(this IDictionary source, StructuredMimeTypesCollection structuredMimeTypes) { ArgumentNullException.ThrowIfNull(source); diff --git a/src/Kiota.Builder/KiotaBuilder.cs b/src/Kiota.Builder/KiotaBuilder.cs index ef05625247..2ef10354c5 100644 --- a/src/Kiota.Builder/KiotaBuilder.cs +++ b/src/Kiota.Builder/KiotaBuilder.cs @@ -1473,26 +1473,37 @@ private void AddRequestBuilderMethodParameters(OpenApiUrlTreeNode currentNode, O if (operation.GetRequestSchema(config.StructuredMimeTypes) is OpenApiSchema requestBodySchema) { CodeTypeBase requestBodyType; - if (operation.RequestBody.Content.IsMultipartFormDataSchema(config.StructuredMimeTypes)) + if (operation.RequestBody.Content.IsMultipartFormDataSchema(config.StructuredMimeTypes) + && operation.RequestBody.Content.IsMultipartTopMimeType(config.StructuredMimeTypes)) { - requestBodyType = new CodeType - { - Name = "MultipartBody", - IsExternal = true, - }; var mediaType = operation.RequestBody.Content.First(x => x.Value.Schema == requestBodySchema).Value; - foreach (var encodingEntry in mediaType.Encoding - .Where(x => !string.IsNullOrEmpty(x.Value.ContentType) && - config.StructuredMimeTypes.Contains(x.Value.ContentType))) + if (mediaType.Encoding.Any()) + { + requestBodyType = new CodeType { Name = "MultipartBody", IsExternal = true, }; + foreach (var encodingEntry in mediaType.Encoding + .Where(x => !string.IsNullOrEmpty(x.Value.ContentType) && + config.StructuredMimeTypes.Contains(x.Value.ContentType))) + { + if (CreateModelDeclarations(currentNode, requestBodySchema.Properties[encodingEntry.Key], + operation, method, $"{operationType}RequestBody", + isRequestBody: true) is CodeType propertyType && + propertyType.TypeDefinition is not null) + multipartPropertiesModels.TryAdd(propertyType.TypeDefinition, true); + } + } + else { - if (CreateModelDeclarations(currentNode, requestBodySchema.Properties[encodingEntry.Key], operation, method, $"{operationType}RequestBody", isRequestBody: true) is CodeType propertyType && - propertyType.TypeDefinition is not null) - multipartPropertiesModels.TryAdd(propertyType.TypeDefinition, true); + requestBodyType = CreateModelDeclarations(currentNode, requestBodySchema, operation, method, + $"{operationType}RequestBody", isRequestBody: true) ?? + throw new InvalidSchemaException(); } } else - requestBodyType = CreateModelDeclarations(currentNode, requestBodySchema, operation, method, $"{operationType}RequestBody", isRequestBody: true) ?? - throw new InvalidSchemaException(); + { + requestBodyType = CreateModelDeclarations(currentNode, requestBodySchema, operation, method, + $"{operationType}RequestBody", isRequestBody: true) ?? + throw new InvalidSchemaException(); + } method.AddParameter(new CodeParameter { Name = "body", @@ -1550,6 +1561,7 @@ private void AddRequestBuilderMethodParameters(OpenApiUrlTreeNode currentNode, O PossibleValues = contentTypes.ToList() }); } + method.AddParameter(new CodeParameter { Name = "requestConfiguration", diff --git a/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs b/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs index 19c2d8a249..6d008c1f2a 100644 --- a/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs +++ b/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs @@ -7180,7 +7180,7 @@ public async Task CleanupSymbolNameDoesNotCauseNameConflictsInQueryParameters() Assert.Equal("int64", select.Type.Name); } [Fact] - public async Task SupportsMultiPartFormAsRequestBody() + public async Task SupportsMultiPartFormAsRequestBodyWithDefaultMimeTypes() { var tempFilePath = Path.Combine(Path.GetTempPath(), Path.GetTempFileName()); await using var fs = await GetDocumentStream(@"openapi: 3.0.1 @@ -7246,6 +7246,285 @@ public async Task SupportsMultiPartFormAsRequestBody() Assert.NotNull(addressClass); } [Fact] + public async Task SupportsMultiPartFormAsRequestBodyWithoutEncodingWithDefaultMimeTypes() + { + var tempFilePath = Path.Combine(Path.GetTempPath(), Path.GetTempFileName()); + await using var fs = await GetDocumentStream(@"openapi: 3.0.1 +info: + title: Example + description: Example + version: 1.0.1 +servers: + - url: https://example.org +paths: + /directoryObject: + post: + requestBody: + content: + multipart/form-data: + schema: + type: object + properties: + id: + type: string + format: uuid + address: + $ref: '#/components/schemas/address' + profileImage: + type: string + format: binary + responses: + '204': + content: + application/json: + schema: + type: string +components: + schemas: + address: + type: object + properties: + street: + type: string + city: + type: string"); + var mockLogger = new Mock>(); + var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath, IncludeAdditionalData = false}, _httpClient); + var document = await builder.CreateOpenApiDocumentAsync(fs); + var node = builder.CreateUriSpace(document); + var codeModel = builder.CreateSourceModel(node); + Assert.NotNull(codeModel); + var rbClass = codeModel.FindChildByName("directoryObjectRequestBuilder"); + Assert.NotNull(rbClass); + var postMethod = rbClass.FindChildByName("Post", false); + Assert.NotNull(postMethod); + var bodyParameter = postMethod.Parameters.FirstOrDefault(static x => x.IsOfKind(CodeParameterKind.RequestBody)); + Assert.NotNull(bodyParameter); + Assert.Equal("directoryObjectPostRequestBody", bodyParameter.Type.Name, StringComparer.OrdinalIgnoreCase); + var addressClass = codeModel.FindChildByName("Address"); + Assert.NotNull(addressClass); + } + [Fact] + public async Task SupportsMultipleContentTypesAsRequestBodyWithDefaultMimeTypes() + { + var tempFilePath = Path.Combine(Path.GetTempPath(), Path.GetTempFileName()); + await using var fs = await GetDocumentStream(@"openapi: 3.0.1 +info: + title: Example + description: Example + version: 1.0.1 +servers: + - url: https://example.org +paths: + /directoryObject: + post: + requestBody: + content: + multipart/form-data: + schema: + type: object + properties: + id: + type: string + format: uuid + address: + $ref: '#/components/schemas/address' + profileImage: + type: string + format: binary + application/json: + schema: + type: object + properties: + id: + type: string + format: uuid + address: + $ref: '#/components/schemas/address' + profileImage: + type: string + format: binary + responses: + '204': + content: + application/json: + schema: + type: string +components: + schemas: + address: + type: object + properties: + street: + type: string + city: + type: string"); + var mockLogger = new Mock>(); + var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath, IncludeAdditionalData = false}, _httpClient); + var document = await builder.CreateOpenApiDocumentAsync(fs); + var node = builder.CreateUriSpace(document); + var codeModel = builder.CreateSourceModel(node); + Assert.NotNull(codeModel); + var rbClass = codeModel.FindChildByName("directoryObjectRequestBuilder"); + Assert.NotNull(rbClass); + var postMethod = rbClass.FindChildByName("Post", false); + Assert.NotNull(postMethod); + var bodyParameter = postMethod.Parameters.FirstOrDefault(static x => x.IsOfKind(CodeParameterKind.RequestBody)); + Assert.NotNull(bodyParameter); + Assert.Equal("directoryObjectPostRequestBody", bodyParameter.Type.Name, StringComparer.OrdinalIgnoreCase); + var addressClass = codeModel.FindChildByName("Address"); + Assert.NotNull(addressClass); + } + [Fact] + public async Task SupportsMultipleContentTypesAsRequestBodyWithMultipartPriorityNoEncoding() + { + var tempFilePath = Path.Combine(Path.GetTempPath(), Path.GetTempFileName()); + await using var fs = await GetDocumentStream(@"openapi: 3.0.1 +info: + title: Example + description: Example + version: 1.0.1 +servers: + - url: https://example.org +paths: + /directoryObject: + post: + requestBody: + content: + multipart/form-data: + schema: + type: object + properties: + id: + type: string + format: uuid + address: + $ref: '#/components/schemas/address' + profileImage: + type: string + format: binary + application/json: + schema: + type: object + properties: + id: + type: string + format: uuid + address: + $ref: '#/components/schemas/address' + profileImage: + type: string + format: binary + responses: + '204': + content: + application/json: + schema: + type: string +components: + schemas: + address: + type: object + properties: + street: + type: string + city: + type: string"); + var mockLogger = new Mock>(); + var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath, IncludeAdditionalData = false, StructuredMimeTypes = new StructuredMimeTypesCollection {"multipart/form-data;q=1", "application/json;q=0.1"}}, _httpClient); + var document = await builder.CreateOpenApiDocumentAsync(fs); + var node = builder.CreateUriSpace(document); + var codeModel = builder.CreateSourceModel(node); + Assert.NotNull(codeModel); + var rbClass = codeModel.FindChildByName("directoryObjectRequestBuilder"); + Assert.NotNull(rbClass); + var postMethod = rbClass.FindChildByName("Post", false); + Assert.NotNull(postMethod); + var bodyParameter = postMethod.Parameters.FirstOrDefault(static x => x.IsOfKind(CodeParameterKind.RequestBody)); + Assert.NotNull(bodyParameter); + Assert.Equal("directoryObjectPostRequestBody", bodyParameter.Type.Name, StringComparer.OrdinalIgnoreCase); + var addressClass = codeModel.FindChildByName("Address"); + Assert.NotNull(addressClass); + } + [Fact] + public async Task SupportsMultipleContentTypesAsRequestBodyWithMultipartPriorityAndEncoding() + { + var tempFilePath = Path.Combine(Path.GetTempPath(), Path.GetTempFileName()); + await using var fs = await GetDocumentStream(@"openapi: 3.0.1 +info: + title: Example + description: Example + version: 1.0.1 +servers: + - url: https://example.org +paths: + /directoryObject: + post: + requestBody: + content: + multipart/form-data: + schema: + type: object + properties: + id: + type: string + format: uuid + address: + $ref: '#/components/schemas/address' + profileImage: + type: string + format: binary + encoding: + id: + contentType: text/plain + address: + contentType: application/json + profileImage: + contentType: image/png + application/json: + schema: + type: object + properties: + id: + type: string + format: uuid + address: + $ref: '#/components/schemas/address' + profileImage: + type: string + format: binary + responses: + '204': + content: + application/json: + schema: + type: string +components: + schemas: + address: + type: object + properties: + street: + type: string + city: + type: string"); + var mockLogger = new Mock>(); + var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath, IncludeAdditionalData = false, StructuredMimeTypes = new StructuredMimeTypesCollection {"multipart/form-data;q=1", "application/json;q=0.1"}}, _httpClient); + var document = await builder.CreateOpenApiDocumentAsync(fs); + var node = builder.CreateUriSpace(document); + var codeModel = builder.CreateSourceModel(node); + Assert.NotNull(codeModel); + var rbClass = codeModel.FindChildByName("directoryObjectRequestBuilder"); + Assert.NotNull(rbClass); + var postMethod = rbClass.FindChildByName("Post", false); + Assert.NotNull(postMethod); + var bodyParameter = postMethod.Parameters.FirstOrDefault(static x => x.IsOfKind(CodeParameterKind.RequestBody)); + Assert.NotNull(bodyParameter); + Assert.Equal("MultipartBody", bodyParameter.Type.Name, StringComparer.OrdinalIgnoreCase); + var addressClass = codeModel.FindChildByName("Address"); + Assert.NotNull(addressClass); + } + [Fact] public async Task ComplexInheritanceStructures() { var tempFilePath = Path.Combine(Path.GetTempPath(), Path.GetTempFileName()); From 9b8ccec1f24ab2dc3cbc3ee03354896cf8ff1392 Mon Sep 17 00:00:00 2001 From: Mathew Spearey Date: Thu, 30 May 2024 15:32:02 +0100 Subject: [PATCH 08/33] Update CHANGELOG.md --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 8a91a69048..34e10fa314 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -45,6 +45,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Fixes a bug where request executors would be missing Untyped parameters in dotnet [#4692](https://github.com/microsoft/kiota/issues/4692) - Fixes a bug where indexers in include/exclude patters were not normalized if the indexer was the last segment without a slash at the end [#4715](https://github.com/microsoft/kiota/issues/4715) - Fixes a bug where CLI generation doesnot handle parameters of type string array. [#4707](https://github.com/microsoft/kiota/issues/4707) +- Fixed a bug where models would not be created when a multipart content schema existed with no encoding [#4734](https://github.com/microsoft/kiota/issues/4734) ## [1.14.0] - 2024-05-02 From 8e482eac84d72c00240dd0b3addcecfaf925349e Mon Sep 17 00:00:00 2001 From: Andrew Omondi Date: Fri, 31 May 2024 09:48:57 +0300 Subject: [PATCH 09/33] Fix format --- src/Kiota.Builder/KiotaBuilder.cs | 2 +- tests/Kiota.Builder.Tests/KiotaBuilderTests.cs | 8 ++++---- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/src/Kiota.Builder/KiotaBuilder.cs b/src/Kiota.Builder/KiotaBuilder.cs index 2ef10354c5..81613e912f 100644 --- a/src/Kiota.Builder/KiotaBuilder.cs +++ b/src/Kiota.Builder/KiotaBuilder.cs @@ -1561,7 +1561,7 @@ private void AddRequestBuilderMethodParameters(OpenApiUrlTreeNode currentNode, O PossibleValues = contentTypes.ToList() }); } - + method.AddParameter(new CodeParameter { Name = "requestConfiguration", diff --git a/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs b/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs index 6d008c1f2a..e70aff7d30 100644 --- a/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs +++ b/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs @@ -7289,7 +7289,7 @@ public async Task SupportsMultiPartFormAsRequestBodyWithoutEncodingWithDefaultMi city: type: string"); var mockLogger = new Mock>(); - var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath, IncludeAdditionalData = false}, _httpClient); + var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath, IncludeAdditionalData = false }, _httpClient); var document = await builder.CreateOpenApiDocumentAsync(fs); var node = builder.CreateUriSpace(document); var codeModel = builder.CreateSourceModel(node); @@ -7360,7 +7360,7 @@ public async Task SupportsMultipleContentTypesAsRequestBodyWithDefaultMimeTypes( city: type: string"); var mockLogger = new Mock>(); - var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath, IncludeAdditionalData = false}, _httpClient); + var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath, IncludeAdditionalData = false }, _httpClient); var document = await builder.CreateOpenApiDocumentAsync(fs); var node = builder.CreateUriSpace(document); var codeModel = builder.CreateSourceModel(node); @@ -7431,7 +7431,7 @@ public async Task SupportsMultipleContentTypesAsRequestBodyWithMultipartPriority city: type: string"); var mockLogger = new Mock>(); - var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath, IncludeAdditionalData = false, StructuredMimeTypes = new StructuredMimeTypesCollection {"multipart/form-data;q=1", "application/json;q=0.1"}}, _httpClient); + var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath, IncludeAdditionalData = false, StructuredMimeTypes = new StructuredMimeTypesCollection { "multipart/form-data;q=1", "application/json;q=0.1" } }, _httpClient); var document = await builder.CreateOpenApiDocumentAsync(fs); var node = builder.CreateUriSpace(document); var codeModel = builder.CreateSourceModel(node); @@ -7509,7 +7509,7 @@ public async Task SupportsMultipleContentTypesAsRequestBodyWithMultipartPriority city: type: string"); var mockLogger = new Mock>(); - var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath, IncludeAdditionalData = false, StructuredMimeTypes = new StructuredMimeTypesCollection {"multipart/form-data;q=1", "application/json;q=0.1"}}, _httpClient); + var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath, IncludeAdditionalData = false, StructuredMimeTypes = new StructuredMimeTypesCollection { "multipart/form-data;q=1", "application/json;q=0.1" } }, _httpClient); var document = await builder.CreateOpenApiDocumentAsync(fs); var node = builder.CreateUriSpace(document); var codeModel = builder.CreateSourceModel(node); From ee3fb51a40c4655fdb61005974c6176e153ee8b2 Mon Sep 17 00:00:00 2001 From: Andrew Omondi Date: Fri, 31 May 2024 11:53:38 +0300 Subject: [PATCH 10/33] Trims unused components from input document --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 34e10fa314..b89ea841e4 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -46,6 +46,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Fixes a bug where indexers in include/exclude patters were not normalized if the indexer was the last segment without a slash at the end [#4715](https://github.com/microsoft/kiota/issues/4715) - Fixes a bug where CLI generation doesnot handle parameters of type string array. [#4707](https://github.com/microsoft/kiota/issues/4707) - Fixed a bug where models would not be created when a multipart content schema existed with no encoding [#4734](https://github.com/microsoft/kiota/issues/4734) +- Trims unused components from output openApi document when generating plugins [#4672](https://github.com/microsoft/kiota/issues/4672) ## [1.14.0] - 2024-05-02 From 964c24fda276cde27fcef11a68efd67f33a3bbf7 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Fri, 31 May 2024 09:01:05 +0000 Subject: [PATCH 11/33] Bump Microsoft.Kiota.Abstractions Bumps the kiota-dependencies group in /it/csharp with 1 update: [Microsoft.Kiota.Abstractions](https://github.com/microsoft/kiota-abstractions-dotnet). Updates `Microsoft.Kiota.Abstractions` from 1.9.3 to 1.9.4 - [Release notes](https://github.com/microsoft/kiota-abstractions-dotnet/releases) - [Changelog](https://github.com/microsoft/kiota-abstractions-dotnet/blob/main/CHANGELOG.md) - [Commits](https://github.com/microsoft/kiota-abstractions-dotnet/compare/v1.9.3...v1.9.4) --- updated-dependencies: - dependency-name: Microsoft.Kiota.Abstractions dependency-type: direct:production update-type: version-update:semver-patch dependency-group: kiota-dependencies ... Signed-off-by: dependabot[bot] --- it/csharp/dotnet.csproj | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/it/csharp/dotnet.csproj b/it/csharp/dotnet.csproj index 92cd67ba5e..b3d5cd54db 100644 --- a/it/csharp/dotnet.csproj +++ b/it/csharp/dotnet.csproj @@ -10,7 +10,7 @@ - + From 9de4bda2a5d2cbeadb60c6a502c91ed33220d380 Mon Sep 17 00:00:00 2001 From: Peter Wurzinger Date: Fri, 31 May 2024 22:00:10 +0200 Subject: [PATCH 12/33] Rename GetFullyQualifiedName to GetFullName and refine some edge cases --- .../Writers/CSharp/CSharpConventionService.cs | 2 +- .../Writers/CSharp/CodeMethodWriter.cs | 14 ++++---- .../CSharp/TypeDefinitionExtensions.cs | 32 ++++++++++++------- 3 files changed, 29 insertions(+), 19 deletions(-) diff --git a/src/Kiota.Builder/Writers/CSharp/CSharpConventionService.cs b/src/Kiota.Builder/Writers/CSharp/CSharpConventionService.cs index c975d79c0e..9522fa678e 100644 --- a/src/Kiota.Builder/Writers/CSharp/CSharpConventionService.cs +++ b/src/Kiota.Builder/Writers/CSharp/CSharpConventionService.cs @@ -202,7 +202,7 @@ public override string TranslateType(CodeType type) ArgumentNullException.ThrowIfNull(type); if (type.TypeDefinition is ITypeDefinition typeDefinition) - return typeDefinition.GetFullyQualifiedName(); + return typeDefinition.GetFullName(); return type.Name switch { diff --git a/src/Kiota.Builder/Writers/CSharp/CodeMethodWriter.cs b/src/Kiota.Builder/Writers/CSharp/CodeMethodWriter.cs index 341e5eda25..796e6a91ac 100644 --- a/src/Kiota.Builder/Writers/CSharp/CodeMethodWriter.cs +++ b/src/Kiota.Builder/Writers/CSharp/CodeMethodWriter.cs @@ -106,7 +106,7 @@ private void WriteRawUrlBuilderBody(CodeClass parentClass, CodeMethod codeElemen var rawUrlParameter = codeElement.Parameters.OfKind(CodeParameterKind.RawUrl) ?? throw new InvalidOperationException("RawUrlBuilder method should have a RawUrl parameter"); var requestAdapterProperty = parentClass.GetPropertyOfKind(CodePropertyKind.RequestAdapter) ?? throw new InvalidOperationException("RawUrlBuilder method should have a RequestAdapter property"); - var fullName = parentClass.GetFullyQualifiedName(); + var fullName = parentClass.GetFullName(); writer.WriteLine($"return new {fullName}({rawUrlParameter.Name.ToFirstCharacterLowerCase()}, {requestAdapterProperty.Name.ToFirstCharacterUpperCase()});"); } private static readonly CodePropertyTypeComparer CodePropertyTypeForwardComparer = new(); @@ -119,13 +119,13 @@ private void WriteFactoryMethodBodyForInheritedModel(CodeMethod codeElement, Cod { writer.WriteLine($"\"{mappedType.Key}\" => new {conventions.GetTypeString(mappedType.Value.AllTypes.First(), codeElement)}(),"); } - writer.WriteLine($"_ => new {parentClass.GetFullyQualifiedName()}(),"); + writer.WriteLine($"_ => new {parentClass.GetFullName()}(),"); writer.CloseBlock("};"); } private const string ResultVarName = "result"; private void WriteFactoryMethodBodyForUnionModel(CodeMethod codeElement, CodeClass parentClass, CodeParameter parseNodeParameter, LanguageWriter writer) { - writer.WriteLine($"var {ResultVarName} = new {parentClass.GetFullyQualifiedName()}();"); + writer.WriteLine($"var {ResultVarName} = new {parentClass.GetFullName()}();"); var includeElse = false; foreach (var property in parentClass.GetPropertiesOfKind(CodePropertyKind.Custom) .OrderBy(static x => x, CodePropertyTypeForwardComparer) @@ -152,7 +152,7 @@ private void WriteFactoryMethodBodyForUnionModel(CodeMethod codeElement, CodeCla } private void WriteFactoryMethodBodyForIntersectionModel(CodeMethod codeElement, CodeClass parentClass, CodeParameter parseNodeParameter, LanguageWriter writer) { - writer.WriteLine($"var {ResultVarName} = new {parentClass.GetFullyQualifiedName()}();"); + writer.WriteLine($"var {ResultVarName} = new {parentClass.GetFullName()}();"); var includeElse = false; foreach (var property in parentClass.GetPropertiesOfKind(CodePropertyKind.Custom) .Where(static x => x.Type is not CodeType propertyType || propertyType.IsCollection || propertyType.TypeDefinition is not CodeClass) @@ -204,7 +204,7 @@ private void WriteFactoryMethodBody(CodeMethod codeElement, CodeClass parentClas else if (parentClass.DiscriminatorInformation.ShouldWriteDiscriminatorForIntersectionType) WriteFactoryMethodBodyForIntersectionModel(codeElement, parentClass, parseNodeParameter, writer); else - writer.WriteLine($"return new {parentClass.GetFullyQualifiedName()}();"); + writer.WriteLine($"return new {parentClass.GetFullName()}();"); } private void WriteRequestBuilderBody(CodeClass parentClass, CodeMethod codeElement, LanguageWriter writer) { @@ -359,7 +359,7 @@ private string GetDeserializationMethodName(CodeTypeBase propType, CodeMethod me return $"GetCollectionOfObjectValues<{propertyType}>({propertyType}.CreateFromDiscriminatorValue){collectionMethod}"; } else if (currentType.TypeDefinition is CodeEnum enumType) - return $"GetEnumValue<{enumType.GetFullyQualifiedName()}>()"; + return $"GetEnumValue<{enumType.GetFullName()}>()"; } return propertyType switch { @@ -664,7 +664,7 @@ private string GetSerializationMethodName(CodeTypeBase propType, CodeMethod meth else return $"WriteCollectionOfObjectValues<{propertyType}>"; else if (currentType.TypeDefinition is CodeEnum enumType) - return $"WriteEnumValue<{enumType.GetFullyQualifiedName()}>"; + return $"WriteEnumValue<{enumType.GetFullName()}>"; } return propertyType switch diff --git a/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs b/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs index 91e7c58e61..6fa8eaa98c 100644 --- a/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs +++ b/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs @@ -7,28 +7,38 @@ namespace Kiota.Builder.Writers.CSharp; internal static class TypeDefinitionExtensions { - public static string GetFullyQualifiedName(this ITypeDefinition typeDefinition) + public static string GetFullName(this ITypeDefinition typeDefinition) { ArgumentNullException.ThrowIfNull(typeDefinition); var fullNameBuilder = new StringBuilder(); - return GetFullyQualifiedName(typeDefinition, fullNameBuilder).ToString(); + return AppendTypeName(typeDefinition, fullNameBuilder).ToString(); } - private static StringBuilder GetFullyQualifiedName(ITypeDefinition codeClass, StringBuilder fullNameBuilder) + private static StringBuilder AppendTypeName(ITypeDefinition typeDefinition, StringBuilder fullNameBuilder) { - fullNameBuilder.Insert(0, codeClass.Name.ToFirstCharacterUpperCase()); - if (codeClass.Parent is CodeClass parentClass) + if (string.IsNullOrEmpty(typeDefinition.Name)) + throw new ArgumentException("Cannot append a full name for a type without a name.", nameof(typeDefinition)); + + fullNameBuilder.Insert(0, typeDefinition.Name.ToFirstCharacterUpperCase()); + if (typeDefinition.Parent is null) + return fullNameBuilder; + + if (typeDefinition.Parent is ITypeDefinition parentTypeDefinition) { fullNameBuilder.Insert(0, '.'); - return GetFullyQualifiedName(parentClass, fullNameBuilder); + return AppendTypeName(parentTypeDefinition, fullNameBuilder); } - if (codeClass.Parent is CodeNamespace ns && !string.IsNullOrEmpty(ns.Name)) + else if (typeDefinition.Parent is CodeNamespace codeNamespace) { - fullNameBuilder.Insert(0, '.'); - fullNameBuilder.Insert(0, ns.Name); - } + if (!string.IsNullOrEmpty(codeNamespace.Name)) + fullNameBuilder.Insert(0, $"{codeNamespace.Name}."); - return fullNameBuilder; + return fullNameBuilder; + } + else + { + throw new InvalidOperationException($"Type {typeDefinition.Name} contains an invalid parent of type {typeDefinition.Parent.GetType().FullName}."); + } } } From 01a4353dc80e2ebdc0e2f7bd2b0ab23a312a6f44 Mon Sep 17 00:00:00 2001 From: Peter Wurzinger Date: Fri, 31 May 2024 22:00:28 +0200 Subject: [PATCH 13/33] Tests for TypeDefinitionExtensions --- .../CSharp/TypeDefinitionExtensionsTests.cs | 140 ++++++++++++++++++ 1 file changed, 140 insertions(+) create mode 100644 tests/Kiota.Builder.Tests/Writers/CSharp/TypeDefinitionExtensionsTests.cs diff --git a/tests/Kiota.Builder.Tests/Writers/CSharp/TypeDefinitionExtensionsTests.cs b/tests/Kiota.Builder.Tests/Writers/CSharp/TypeDefinitionExtensionsTests.cs new file mode 100644 index 0000000000..8425b73e89 --- /dev/null +++ b/tests/Kiota.Builder.Tests/Writers/CSharp/TypeDefinitionExtensionsTests.cs @@ -0,0 +1,140 @@ +using System; +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Writers.CSharp; + +using Xunit; + +namespace Kiota.Builder.Tests.Writers.CSharp; + +public sealed class TypeDefinitionExtensionsTests +{ + [Fact] + public void ReturnsFullNameForTypeWithoutNamespace() + { + var rootNamespace = CodeNamespace.InitRootNamespace(); + var myClass = new CodeClass + { + Name = "myClass" + }; + rootNamespace.AddClass(myClass); + + var fullName = TypeDefinitionExtensions.GetFullName(myClass); + + Assert.Equal("MyClass", fullName); + } + + [Fact] + public void ReturnsFullNameForTypeInNamespace() + { + var rootNamespace = CodeNamespace.InitRootNamespace(); + + var myNamespace = rootNamespace.AddNamespace("MyNamespace"); + var myClass = new CodeClass + { + Name = "myClass", + }; + myNamespace.AddClass(myClass); + + var fullName = TypeDefinitionExtensions.GetFullName(myClass); + + Assert.Equal("MyNamespace.MyClass", fullName); + } + + [Fact] + public void ReturnsFullNameForNestedTypes() + { + var rootNamespace = CodeNamespace.InitRootNamespace(); + + var myNamespace = rootNamespace.AddNamespace("MyNamespace"); + + var myParentClass = new CodeClass + { + Name = "myParentClass" + }; + myNamespace.AddClass(myParentClass); + + var myNestedClass = new CodeClass + { + Name = "myNestedClass", + }; + myParentClass.AddInnerClass(myNestedClass); + + var parentClassFullName = TypeDefinitionExtensions.GetFullName(myParentClass); + var nestedClassFullName = TypeDefinitionExtensions.GetFullName(myNestedClass); + + Assert.Equal("MyNamespace.MyParentClass", parentClassFullName); + Assert.Equal("MyNamespace.MyParentClass.MyNestedClass", nestedClassFullName); + } + + [Fact] + public void ThrowsIfTypeIsNull() + { + Assert.Throws("typeDefinition", () => TypeDefinitionExtensions.GetFullName(null!)); + } + + [Theory] + [InlineData("")] + [InlineData(null)] + public void ThrowsIfTypeDoesNotHaveAName(string typeName) + { + var myClass = new CodeClass + { + Name = typeName + }; + + Assert.Throws("typeDefinition", () => TypeDefinitionExtensions.GetFullName(myClass)); + } + + [Fact] + public void ThrowsIfTypesParentIsInvalid() + { + var myClass = new CodeClass + { + Name = "myClass", + Parent = new CodeConstant() + }; + + Assert.Throws(() => TypeDefinitionExtensions.GetFullName(myClass)); + } + + [Fact] + public void CapitalizesTypeNamesInTypeHierarchyButNotTheNamespace() + { + var rootNamespace = CodeNamespace.InitRootNamespace(); + var myNamespace = rootNamespace.AddNamespace("myNamespace"); + + var myParentClass = new CodeClass + { + Name = "myParentClass" + }; + myNamespace.AddClass(myParentClass); + + var myNestedClass = new CodeClass + { + Name = "myNestedClass", + }; + myParentClass.AddInnerClass(myNestedClass); + + var nestedClassFullName = TypeDefinitionExtensions.GetFullName(myNestedClass); + + Assert.Equal("myNamespace.MyParentClass.MyNestedClass", nestedClassFullName); + } + + [Fact] + public void DoesNotAppendNamespaceSegmentIfNamespaceNameIsEmpty() + { + var rootNamespace = CodeNamespace.InitRootNamespace(); + var myNamespace = rootNamespace.AddNamespace("ThisWillBeEmpty"); + myNamespace.Name = ""; + + var myClass = new CodeClass + { + Name = "myClass" + }; + myNamespace.AddClass(myClass); + + var fullName = TypeDefinitionExtensions.GetFullName(myClass); + + Assert.Equal("MyClass", fullName); + } +} From 1353080fe7ed346e7bf1bc84760808b47db9a95c Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 3 Jun 2024 08:08:27 +0000 Subject: [PATCH 14/33] Bump adm-zip from 0.5.12 to 0.5.13 in /vscode/microsoft-kiota Bumps [adm-zip](https://github.com/cthackers/adm-zip) from 0.5.12 to 0.5.13. - [Release notes](https://github.com/cthackers/adm-zip/releases) - [Changelog](https://github.com/cthackers/adm-zip/blob/master/history.md) - [Commits](https://github.com/cthackers/adm-zip/compare/v0.5.12...v0.5.13) --- updated-dependencies: - dependency-name: adm-zip dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] --- vscode/microsoft-kiota/package-lock.json | 10 +++++----- vscode/microsoft-kiota/package.json | 2 +- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/vscode/microsoft-kiota/package-lock.json b/vscode/microsoft-kiota/package-lock.json index d06238e64b..e344c8b446 100644 --- a/vscode/microsoft-kiota/package-lock.json +++ b/vscode/microsoft-kiota/package-lock.json @@ -10,7 +10,7 @@ "dependencies": { "@vscode/extension-telemetry": "^0.9.6", "@vscode/l10n": "^0.0.18", - "adm-zip": "^0.5.12", + "adm-zip": "^0.5.13", "is-online": "^10.0.0", "original-fs": "^1.2.0", "vscode-jsonrpc": "^8.2.1" @@ -990,11 +990,11 @@ } }, "node_modules/adm-zip": { - "version": "0.5.12", - "resolved": "https://registry.npmjs.org/adm-zip/-/adm-zip-0.5.12.tgz", - "integrity": "sha512-6TVU49mK6KZb4qG6xWaaM4C7sA/sgUMLy/JYMOzkcp3BvVLpW0fXDFQiIzAuxFCt/2+xD7fNIiPFAoLZPhVNLQ==", + "version": "0.5.13", + "resolved": "https://registry.npmjs.org/adm-zip/-/adm-zip-0.5.13.tgz", + "integrity": "sha512-4U51tTl9J8UVEcuKGr6zRzY95tWoAa9l+ureGBNmsfleszjZblm5NyEEL/ZQxkhi86co5mZhSvL2T7gkZ6feYQ==", "engines": { - "node": ">=6.0" + "node": ">=12.0" } }, "node_modules/agent-base": { diff --git a/vscode/microsoft-kiota/package.json b/vscode/microsoft-kiota/package.json index 36bb1adc7f..c75fd6b1e1 100644 --- a/vscode/microsoft-kiota/package.json +++ b/vscode/microsoft-kiota/package.json @@ -441,7 +441,7 @@ "dependencies": { "@vscode/extension-telemetry": "^0.9.6", "@vscode/l10n": "^0.0.18", - "adm-zip": "^0.5.12", + "adm-zip": "^0.5.13", "is-online": "^10.0.0", "original-fs": "^1.2.0", "vscode-jsonrpc": "^8.2.1" From 143a6dbfde5c512416766eba04fedee38cfbb9a4 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 3 Jun 2024 08:08:35 +0000 Subject: [PATCH 15/33] Bump @types/node from 20.12.13 to 20.14.0 in /vscode/microsoft-kiota Bumps [@types/node](https://github.com/DefinitelyTyped/DefinitelyTyped/tree/HEAD/types/node) from 20.12.13 to 20.14.0. - [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases) - [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/node) --- updated-dependencies: - dependency-name: "@types/node" dependency-type: direct:development update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] --- vscode/microsoft-kiota/package-lock.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/vscode/microsoft-kiota/package-lock.json b/vscode/microsoft-kiota/package-lock.json index d06238e64b..a21b73dc0e 100644 --- a/vscode/microsoft-kiota/package-lock.json +++ b/vscode/microsoft-kiota/package-lock.json @@ -483,9 +483,9 @@ "dev": true }, "node_modules/@types/node": { - "version": "20.12.13", - "resolved": "https://registry.npmjs.org/@types/node/-/node-20.12.13.tgz", - "integrity": "sha512-gBGeanV41c1L171rR7wjbMiEpEI/l5XFQdLLfhr/REwpgDy/4U8y89+i8kRiLzDyZdOkXh+cRaTetUnCYutoXA==", + "version": "20.14.0", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.14.0.tgz", + "integrity": "sha512-5cHBxFGJx6L4s56Bubp4fglrEpmyJypsqI6RgzMfBHWUJQGWAAi8cWcgetEbZXHYXo9C2Fa4EEds/uSyS4cxmA==", "dev": true, "dependencies": { "undici-types": "~5.26.4" From 8d2443d09714f7a7ee0ef568b5c2e4438a55f234 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 3 Jun 2024 08:15:14 +0000 Subject: [PATCH 16/33] Bump certifi from 2024.2.2 to 2024.6.2 in /it/python Bumps [certifi](https://github.com/certifi/python-certifi) from 2024.2.2 to 2024.6.2. - [Commits](https://github.com/certifi/python-certifi/compare/2024.02.02...2024.06.02) --- updated-dependencies: - dependency-name: certifi dependency-type: direct:development update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] --- it/python/requirements-dev.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/it/python/requirements-dev.txt b/it/python/requirements-dev.txt index babd89e635..c204c02dae 100644 --- a/it/python/requirements-dev.txt +++ b/it/python/requirements-dev.txt @@ -2,7 +2,7 @@ astroid==3.2.2 ; python_full_version >= '3.7.2' -certifi==2024.2.2 ; python_version >= '3.6' +certifi==2024.6.2 ; python_version >= '3.6' charset-normalizer==3.3.2 ; python_full_version >= '3.7.0' From 94d27f8adcf3c70e0196427a7383e580f0b64259 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 3 Jun 2024 08:15:20 +0000 Subject: [PATCH 17/33] Bump typing-extensions from 4.12.0 to 4.12.1 in /it/python Bumps [typing-extensions](https://github.com/python/typing_extensions) from 4.12.0 to 4.12.1. - [Release notes](https://github.com/python/typing_extensions/releases) - [Changelog](https://github.com/python/typing_extensions/blob/main/CHANGELOG.md) - [Commits](https://github.com/python/typing_extensions/compare/4.12.0...4.12.1) --- updated-dependencies: - dependency-name: typing-extensions dependency-type: direct:development update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] --- it/python/requirements-dev.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/it/python/requirements-dev.txt b/it/python/requirements-dev.txt index babd89e635..c4bc6bdf1b 100644 --- a/it/python/requirements-dev.txt +++ b/it/python/requirements-dev.txt @@ -56,7 +56,7 @@ tomli-w==1.0.0 ; python_version >= '3.7' tomlkit==0.12.5 ; python_version >= '3.7' -typing-extensions==4.12.0 ; python_version >= '3.7' +typing-extensions==4.12.1 ; python_version >= '3.7' urllib3==2.2.1 ; python_version >= '3.7' From 16efc7f36876346bbbbaf10539f5d3c1aca11041 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 3 Jun 2024 08:15:26 +0000 Subject: [PATCH 18/33] Bump zipp from 3.19.0 to 3.19.1 in /it/python Bumps [zipp](https://github.com/jaraco/zipp) from 3.19.0 to 3.19.1. - [Release notes](https://github.com/jaraco/zipp/releases) - [Changelog](https://github.com/jaraco/zipp/blob/main/NEWS.rst) - [Commits](https://github.com/jaraco/zipp/compare/v3.19.0...v3.19.1) --- updated-dependencies: - dependency-name: zipp dependency-type: direct:development update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] --- it/python/requirements-dev.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/it/python/requirements-dev.txt b/it/python/requirements-dev.txt index babd89e635..40310a2824 100644 --- a/it/python/requirements-dev.txt +++ b/it/python/requirements-dev.txt @@ -64,7 +64,7 @@ wrapt==1.15.0 ; python_version < '3.11' yapf==0.40.2 -zipp==3.19.0 ; python_version >= '3.7' +zipp==3.19.1 ; python_version >= '3.7' aiohttp==3.9.5 ; python_version >= '3.6' From 1f2010325077ced61a01b6953f36c6d7ad907c05 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 3 Jun 2024 08:27:26 +0000 Subject: [PATCH 19/33] Bump @types/node from 20.12.13 to 20.14.0 in /it/typescript Bumps [@types/node](https://github.com/DefinitelyTyped/DefinitelyTyped/tree/HEAD/types/node) from 20.12.13 to 20.14.0. - [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases) - [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/node) --- updated-dependencies: - dependency-name: "@types/node" dependency-type: direct:development update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] --- it/typescript/package-lock.json | 8 ++++---- it/typescript/package.json | 2 +- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/it/typescript/package-lock.json b/it/typescript/package-lock.json index 6b55934cd6..df9709f093 100644 --- a/it/typescript/package-lock.json +++ b/it/typescript/package-lock.json @@ -22,7 +22,7 @@ }, "devDependencies": { "@es-exec/esbuild-plugin-start": "^0.0.5", - "@types/node": "^20.12.13", + "@types/node": "^20.14.0", "@typescript-eslint/eslint-plugin": "^7.7.1", "@typescript-eslint/parser": "^7.8.0", "esbuild": "^0.21.4", @@ -827,9 +827,9 @@ "dev": true }, "node_modules/@types/node": { - "version": "20.12.13", - "resolved": "https://registry.npmjs.org/@types/node/-/node-20.12.13.tgz", - "integrity": "sha512-gBGeanV41c1L171rR7wjbMiEpEI/l5XFQdLLfhr/REwpgDy/4U8y89+i8kRiLzDyZdOkXh+cRaTetUnCYutoXA==", + "version": "20.14.0", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.14.0.tgz", + "integrity": "sha512-5cHBxFGJx6L4s56Bubp4fglrEpmyJypsqI6RgzMfBHWUJQGWAAi8cWcgetEbZXHYXo9C2Fa4EEds/uSyS4cxmA==", "dev": true, "dependencies": { "undici-types": "~5.26.4" diff --git a/it/typescript/package.json b/it/typescript/package.json index 35730498cd..3c2654de26 100644 --- a/it/typescript/package.json +++ b/it/typescript/package.json @@ -19,7 +19,7 @@ "prettier": "./.prettierrc.json", "devDependencies": { "@es-exec/esbuild-plugin-start": "^0.0.5", - "@types/node": "^20.12.13", + "@types/node": "^20.14.0", "@typescript-eslint/eslint-plugin": "^7.7.1", "@typescript-eslint/parser": "^7.8.0", "esbuild": "^0.21.4", From 6f79fd945c5583e80cf6ed8d660e37ca1be572dc Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 3 Jun 2024 08:27:37 +0000 Subject: [PATCH 20/33] Bump prettier from 3.2.5 to 3.3.0 in /it/typescript Bumps [prettier](https://github.com/prettier/prettier) from 3.2.5 to 3.3.0. - [Release notes](https://github.com/prettier/prettier/releases) - [Changelog](https://github.com/prettier/prettier/blob/main/CHANGELOG.md) - [Commits](https://github.com/prettier/prettier/compare/3.2.5...3.3.0) --- updated-dependencies: - dependency-name: prettier dependency-type: direct:development update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] --- it/typescript/package-lock.json | 8 ++++---- it/typescript/package.json | 2 +- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/it/typescript/package-lock.json b/it/typescript/package-lock.json index 6b55934cd6..111d9dcbc0 100644 --- a/it/typescript/package-lock.json +++ b/it/typescript/package-lock.json @@ -29,7 +29,7 @@ "eslint": "^8.57.0", "eslint-config-prettier": "^9.1.0", "minimist": "^1.2.8", - "prettier": "^3.2.5", + "prettier": "^3.3.0", "typescript": "^4.9.5" } }, @@ -2878,9 +2878,9 @@ } }, "node_modules/prettier": { - "version": "3.2.5", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.2.5.tgz", - "integrity": "sha512-3/GWa9aOC0YeD7LUfvOG2NiDyhOWRvt1k+rcKhOuYnMY24iiCphgneUfJDyFXd6rZCAnuLBv6UeAULtrhT/F4A==", + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.3.0.tgz", + "integrity": "sha512-J9odKxERhCQ10OC2yb93583f6UnYutOeiV5i0zEDS7UGTdUt0u+y8erxl3lBKvwo/JHyyoEdXjwp4dke9oyZ/g==", "dev": true, "bin": { "prettier": "bin/prettier.cjs" diff --git a/it/typescript/package.json b/it/typescript/package.json index 35730498cd..00ff248b57 100644 --- a/it/typescript/package.json +++ b/it/typescript/package.json @@ -26,7 +26,7 @@ "eslint": "^8.57.0", "eslint-config-prettier": "^9.1.0", "minimist": "^1.2.8", - "prettier": "^3.2.5", + "prettier": "^3.3.0", "typescript": "^4.9.5" }, "dependencies": { From 339e1c26f963dd96b0eef498bab60b00331ac85d Mon Sep 17 00:00:00 2001 From: Andrew Omondi Date: Mon, 3 Jun 2024 11:41:25 +0300 Subject: [PATCH 21/33] Fixes warnings --- CHANGELOG.md | 1 + src/Kiota.Builder/KiotaBuilder.cs | 3 ++- tests/Kiota.Builder.Tests/KiotaBuilderTests.cs | 7 +++++++ 3 files changed, 10 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 34e10fa314..a469a745b7 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -46,6 +46,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Fixes a bug where indexers in include/exclude patters were not normalized if the indexer was the last segment without a slash at the end [#4715](https://github.com/microsoft/kiota/issues/4715) - Fixes a bug where CLI generation doesnot handle parameters of type string array. [#4707](https://github.com/microsoft/kiota/issues/4707) - Fixed a bug where models would not be created when a multipart content schema existed with no encoding [#4734](https://github.com/microsoft/kiota/issues/4734) +- Fixes a bug where warnings about discriminator not being inherited were generated [#4761](https://github.com/microsoft/kiota/issues/4761) ## [1.14.0] - 2024-05-02 diff --git a/src/Kiota.Builder/KiotaBuilder.cs b/src/Kiota.Builder/KiotaBuilder.cs index 81613e912f..226e831e78 100644 --- a/src/Kiota.Builder/KiotaBuilder.cs +++ b/src/Kiota.Builder/KiotaBuilder.cs @@ -2164,7 +2164,8 @@ internal static void AddDiscriminatorMethod(CodeClass newClass, string discrimin } if (baseClass is not null && (result.TypeDefinition is not CodeClass codeClass || codeClass.StartBlock.Inherits is null)) { - logger.LogWarning("Discriminator {ComponentKey} is not inherited from {ClassName}.", componentKey, baseClass.Name); + if (!baseClass.Equals(result.TypeDefinition))// don't log warning if the discriminator points to the base type itself as this is implicitly the default case. + logger.LogWarning("Discriminator {ComponentKey} is not inherited from {ClassName}.", componentKey, baseClass.Name); return null; } return result; diff --git a/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs b/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs index e70aff7d30..fc0df2d7d5 100644 --- a/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs +++ b/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs @@ -3426,6 +3426,13 @@ public async Task AddsDiscriminatorMappingsAllOfImplicit() var entityClass = codeModel.FindChildByName("entity"); var directoryObjectClass = codeModel.FindChildByName("directoryObject"); var userClass = codeModel.FindChildByName("user"); + mockLogger.Verify(logger => logger.Log( + It.Is(logLevel => logLevel == LogLevel.Warning), + It.Is(eventId => eventId.Id == 0), + It.Is((@object, @type) => @object.ToString().Contains(" is not inherited from ", StringComparison.OrdinalIgnoreCase) && @type.Name == "FormattedLogValues"), + It.IsAny(), + It.IsAny>()), + Times.Never); Assert.NotNull(entityClass); Assert.NotNull(directoryObjectClass); Assert.NotNull(userClass); From 69943e574c8a312bc544b0b6941e1cd9ccaff3b9 Mon Sep 17 00:00:00 2001 From: "peter.wurzinger" Date: Mon, 3 Jun 2024 12:07:36 +0200 Subject: [PATCH 22/33] Change encoding to UTF8wBOM --- .../Writers/CSharp/TypeDefinitionExtensionsTests.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/Kiota.Builder.Tests/Writers/CSharp/TypeDefinitionExtensionsTests.cs b/tests/Kiota.Builder.Tests/Writers/CSharp/TypeDefinitionExtensionsTests.cs index 8425b73e89..41c397b34f 100644 --- a/tests/Kiota.Builder.Tests/Writers/CSharp/TypeDefinitionExtensionsTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/CSharp/TypeDefinitionExtensionsTests.cs @@ -1,4 +1,4 @@ -using System; +using System; using Kiota.Builder.CodeDOM; using Kiota.Builder.Writers.CSharp; From 81b4def8b43a4baac9cf8b9d5dac504aa0fb6ab5 Mon Sep 17 00:00:00 2001 From: Andrew Omondi Date: Mon, 3 Jun 2024 16:45:36 +0300 Subject: [PATCH 23/33] Fix formatting --- .../Plugins/PluginsGenerationService.cs | 6 +- .../Plugins/PluginsGenerationServiceTests.cs | 111 ++++++++++++++++++ 2 files changed, 114 insertions(+), 3 deletions(-) diff --git a/src/Kiota.Builder/Plugins/PluginsGenerationService.cs b/src/Kiota.Builder/Plugins/PluginsGenerationService.cs index 3fa54d249e..7b5daca469 100644 --- a/src/Kiota.Builder/Plugins/PluginsGenerationService.cs +++ b/src/Kiota.Builder/Plugins/PluginsGenerationService.cs @@ -103,9 +103,9 @@ private OpenApiDocument GetDocumentWithTrimmedComponentsAndResponses(OpenApiDocu doc.Components ??= new OpenApiComponents(); if (string.IsNullOrEmpty(doc.Info?.Version)) // filtering fails if there's no version. - return doc; + doc.Info!.Version = "1.0"; - //empty out the responses with a single 2XX + //empty out all the responses with a single empty 2XX foreach (var operation in doc.Paths.SelectMany(static item => item.Value.Operations.Values)) { operation.Responses = new OpenApiResponses() @@ -130,7 +130,7 @@ private OpenApiDocument GetDocumentWithTrimmedComponentsAndResponses(OpenApiDocu }; } - // remove unused components + // remove unused components using the OpenApi.Net var requestUrls = new Dictionary>(); var basePath = doc.GetAPIRootUrl(Configuration.OpenAPIFilePath); foreach (var path in doc.Paths.Where(static path => path.Value.Operations.Count > 0)) diff --git a/tests/Kiota.Builder.Tests/Plugins/PluginsGenerationServiceTests.cs b/tests/Kiota.Builder.Tests/Plugins/PluginsGenerationServiceTests.cs index 04e52db82c..d3b4e3cfbb 100644 --- a/tests/Kiota.Builder.Tests/Plugins/PluginsGenerationServiceTests.cs +++ b/tests/Kiota.Builder.Tests/Plugins/PluginsGenerationServiceTests.cs @@ -7,6 +7,8 @@ using Kiota.Builder.Configuration; using Kiota.Builder.Plugins; using Microsoft.Extensions.Logging; +using Microsoft.OpenApi.Models; +using Microsoft.OpenApi.Readers; using Microsoft.OpenApi.Services; using Microsoft.Plugins.Manifest; using Moq; @@ -109,4 +111,113 @@ public async Task GeneratesManifest() private const string ManifestFileName = "client-microsoft.json"; private const string OpenAIPluginFileName = "openai-plugins.json"; private const string OpenApiFileName = "client-openapi.yml"; + + [Fact] + public async Task GeneratesManifestAndCleansUpInputDescription() + { + var simpleDescriptionContent = @"openapi: 3.0.0 +info: + title: test + version: 1.0 +x-test-root-extension: test +servers: + - url: http://localhost/ + description: There's no place like home +paths: + /test: + get: + description: description for test path + responses: + '200': + description: test + '400': + description: client error response + /test/{id}: + get: + description: description for test path with id + operationId: test.WithId + parameters: + - name: id + in: path + required: true + description: The id of the test + schema: + type: integer + format: int32 + responses: + '200': + description: test + '500': + description: api error response +components: + schemas: + microsoft.graph.entity: + title: entity + required: + - '@odata.type' + type: object + properties: + id: + type: string + '@odata.type': + type: string"; + var workingDirectory = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName()); + var simpleDescriptionPath = Path.Combine(workingDirectory) + "description.yaml"; + await File.WriteAllTextAsync(simpleDescriptionPath, simpleDescriptionContent); + var mockLogger = new Mock>(); + var openAPIDocumentDS = new OpenApiDocumentDownloadService(_httpClient, mockLogger.Object); + var outputDirectory = Path.Combine(workingDirectory, "output"); + var generationConfiguration = new GenerationConfiguration + { + OutputPath = outputDirectory, + OpenAPIFilePath = "openapiPath", + PluginTypes = [PluginType.Microsoft], + ClientClassName = "client", + ApiRootUrl = "http://localhost/", //Kiota builder would set this for us + }; + var (openAPIDocumentStream, _) = await openAPIDocumentDS.LoadStreamAsync(simpleDescriptionPath, generationConfiguration, null, false); + var openApiDocument = await openAPIDocumentDS.GetDocumentFromStreamAsync(openAPIDocumentStream, generationConfiguration); + KiotaBuilder.CleanupOperationIdForPlugins(openApiDocument); + var urlTreeNode = OpenApiUrlTreeNode.Create(openApiDocument, Constants.DefaultOpenApiLabel); + + var pluginsGenerationService = new PluginsGenerationService(openApiDocument, urlTreeNode, generationConfiguration, workingDirectory); + await pluginsGenerationService.GenerateManifestAsync(); + + Assert.True(File.Exists(Path.Combine(outputDirectory, ManifestFileName))); + Assert.True(File.Exists(Path.Combine(outputDirectory, OpenApiFileName))); + + // Validate the v2 plugin + var manifestContent = await File.ReadAllTextAsync(Path.Combine(outputDirectory, ManifestFileName)); + using var jsonDocument = JsonDocument.Parse(manifestContent); + var resultingManifest = PluginManifestDocument.Load(jsonDocument.RootElement); + Assert.NotNull(resultingManifest.Document); + Assert.Equal(OpenApiFileName, resultingManifest.Document.Runtimes.OfType().First().Spec.Url); + Assert.Equal(2, resultingManifest.Document.Functions.Count);// all functions are generated despite missing operationIds + Assert.Empty(resultingManifest.Problems);// no problems are expected with names + + var openApiReader = new OpenApiStreamReader(); + + // Validate the original file. + var originalOpenApiFile = File.OpenRead(simpleDescriptionPath); + var originalDocument = openApiReader.Read(originalOpenApiFile, out var originalDiagnostic); + Assert.Empty(originalDiagnostic.Errors); + + Assert.Single(originalDocument.Components.Schemas);// one schema originally + Assert.Single(originalDocument.Extensions); // single unsupported extension at root + Assert.Equal(2, originalDocument.Paths.Count); // document has only two paths + Assert.Equal(2, originalDocument.Paths["/test"].Operations[OperationType.Get].Responses.Count); // 2 responses originally + Assert.Equal(2, originalDocument.Paths["/test/{id}"].Operations[OperationType.Get].Responses.Count); // 2 responses originally + + // Validate the output open api file + var resultOpenApiFile = File.OpenRead(Path.Combine(outputDirectory, OpenApiFileName)); + var resultDocument = openApiReader.Read(resultOpenApiFile, out var diagnostic); + Assert.Empty(diagnostic.Errors); + + // Assertions / validations + Assert.Empty(resultDocument.Components.Schemas);// no schema is referenced. so ensure they are all removed + Assert.Empty(resultDocument.Extensions); // no extension at root (unsupported extension is removed) + Assert.Equal(2, resultDocument.Paths.Count); // document has only two paths + Assert.Single(resultDocument.Paths["/test"].Operations[OperationType.Get].Responses); // other responses are removed from the document + Assert.Single(resultDocument.Paths["/test/{id}"].Operations[OperationType.Get].Responses); // 2 responses originally + } } From a2ae9110b7652a0d69e7ae74cf4f9cb61f4c684b Mon Sep 17 00:00:00 2001 From: Peter Wurzinger Date: Mon, 3 Jun 2024 16:55:32 +0200 Subject: [PATCH 24/33] Accept suggestion of using a switch statement to distinguish nested code elements Co-authored-by: Andrew Omondi --- .../CSharp/TypeDefinitionExtensions.cs | 32 +++++++++---------- 1 file changed, 16 insertions(+), 16 deletions(-) diff --git a/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs b/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs index 6fa8eaa98c..b96d8270db 100644 --- a/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs +++ b/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs @@ -21,24 +21,24 @@ private static StringBuilder AppendTypeName(ITypeDefinition typeDefinition, Stri throw new ArgumentException("Cannot append a full name for a type without a name.", nameof(typeDefinition)); fullNameBuilder.Insert(0, typeDefinition.Name.ToFirstCharacterUpperCase()); - if (typeDefinition.Parent is null) - return fullNameBuilder; - - if (typeDefinition.Parent is ITypeDefinition parentTypeDefinition) - { - fullNameBuilder.Insert(0, '.'); - return AppendTypeName(parentTypeDefinition, fullNameBuilder); - } - else if (typeDefinition.Parent is CodeNamespace codeNamespace) + switch (typeDefinition.Parent) { - if (!string.IsNullOrEmpty(codeNamespace.Name)) - fullNameBuilder.Insert(0, $"{codeNamespace.Name}."); + case null: + return fullNameBuilder; + case ITypeDefinition parentTypeDefinition: + { + fullNameBuilder.Insert(0, '.'); + return AppendTypeName(parentTypeDefinition, fullNameBuilder); + } + case CodeNamespace codeNamespace: + { + if (!string.IsNullOrEmpty(codeNamespace.Name)) + fullNameBuilder.Insert(0, $"{codeNamespace.Name}."); - return fullNameBuilder; - } - else - { - throw new InvalidOperationException($"Type {typeDefinition.Name} contains an invalid parent of type {typeDefinition.Parent.GetType().FullName}."); + return fullNameBuilder; + } + default: + throw new InvalidOperationException($"Type {typeDefinition.Name} contains an invalid parent of type {typeDefinition.Parent.GetType().FullName}."); } } } From 6e602bbbe786a8d2a5eef4b942f8e5e615ec700b Mon Sep 17 00:00:00 2001 From: "peter.wurzinger" Date: Mon, 3 Jun 2024 17:02:15 +0200 Subject: [PATCH 25/33] Add changelog entry --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 34e10fa314..197c5d5e4b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -46,6 +46,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Fixes a bug where indexers in include/exclude patters were not normalized if the indexer was the last segment without a slash at the end [#4715](https://github.com/microsoft/kiota/issues/4715) - Fixes a bug where CLI generation doesnot handle parameters of type string array. [#4707](https://github.com/microsoft/kiota/issues/4707) - Fixed a bug where models would not be created when a multipart content schema existed with no encoding [#4734](https://github.com/microsoft/kiota/issues/4734) +- Types generated by Kiota are now referenced with their full name to avoid namespace ambiguities [#4475](https://github.com/microsoft/kiota/issues/4475) ## [1.14.0] - 2024-05-02 From 4ea2012075d97eaeeaf48aef8d57985e908a22da Mon Sep 17 00:00:00 2001 From: "peter.wurzinger" Date: Tue, 4 Jun 2024 08:50:09 +0200 Subject: [PATCH 26/33] Format whitespaces according to dotnet format --- .../Writers/CSharp/TypeDefinitionExtensions.cs | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs b/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs index b96d8270db..c6ef2fac85 100644 --- a/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs +++ b/src/Kiota.Builder/Writers/CSharp/TypeDefinitionExtensions.cs @@ -26,17 +26,17 @@ private static StringBuilder AppendTypeName(ITypeDefinition typeDefinition, Stri case null: return fullNameBuilder; case ITypeDefinition parentTypeDefinition: - { - fullNameBuilder.Insert(0, '.'); - return AppendTypeName(parentTypeDefinition, fullNameBuilder); - } + { + fullNameBuilder.Insert(0, '.'); + return AppendTypeName(parentTypeDefinition, fullNameBuilder); + } case CodeNamespace codeNamespace: - { - if (!string.IsNullOrEmpty(codeNamespace.Name)) - fullNameBuilder.Insert(0, $"{codeNamespace.Name}."); + { + if (!string.IsNullOrEmpty(codeNamespace.Name)) + fullNameBuilder.Insert(0, $"{codeNamespace.Name}."); - return fullNameBuilder; - } + return fullNameBuilder; + } default: throw new InvalidOperationException($"Type {typeDefinition.Name} contains an invalid parent of type {typeDefinition.Parent.GetType().FullName}."); } From c1956fc9de559abdf84f6b60ecda83bce1a24458 Mon Sep 17 00:00:00 2001 From: Silas Kenneth Date: Tue, 4 Jun 2024 10:56:16 +0300 Subject: [PATCH 27/33] Add CHANGELOG entry. --- CHANGELOG.md | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index a469a745b7..ad45f52ce8 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -44,9 +44,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Fixes a bug where CLI Generation does not handle path parameters of type "string" and format "date", "date-time", "time", etc. [#4615](https://github.com/microsoft/kiota/issues/4615) - Fixes a bug where request executors would be missing Untyped parameters in dotnet [#4692](https://github.com/microsoft/kiota/issues/4692) - Fixes a bug where indexers in include/exclude patters were not normalized if the indexer was the last segment without a slash at the end [#4715](https://github.com/microsoft/kiota/issues/4715) -- Fixes a bug where CLI generation doesnot handle parameters of type string array. [#4707](https://github.com/microsoft/kiota/issues/4707) +- Fixes a bug where CLI generation doesn't handle parameters of type string array. [#4707](https://github.com/microsoft/kiota/issues/4707) - Fixed a bug where models would not be created when a multipart content schema existed with no encoding [#4734](https://github.com/microsoft/kiota/issues/4734) - Fixes a bug where warnings about discriminator not being inherited were generated [#4761](https://github.com/microsoft/kiota/issues/4761) +- Fix scalar member composed type serialization in PHP [#2827](https://github.com/issues/2827) ## [1.14.0] - 2024-05-02 From cfb91577b42727922e8ef007dce9d26f8d0b126c Mon Sep 17 00:00:00 2001 From: Faith Kangai Date: Tue, 4 Jun 2024 10:58:03 +0300 Subject: [PATCH 28/33] chore: Only run if PR is not from a forked context (#4768) --- .github/workflows/project-auto-add.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/project-auto-add.yml b/.github/workflows/project-auto-add.yml index 00534c925a..a6ebb10d30 100644 --- a/.github/workflows/project-auto-add.yml +++ b/.github/workflows/project-auto-add.yml @@ -13,7 +13,7 @@ on: jobs: track_issue: - if: github.actor != 'dependabot[bot]' + if: github.actor != 'dependabot[bot]' && github.event.pull_request.head.repo.fork == false runs-on: ubuntu-latest steps: - name: Generate token From 7a98490c58a4abfeb075610b689a065aa9e902d8 Mon Sep 17 00:00:00 2001 From: Silas Kenneth Date: Tue, 4 Jun 2024 11:06:47 +0300 Subject: [PATCH 29/33] Fix issue link. --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index ad45f52ce8..41b56cd3fc 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -47,7 +47,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Fixes a bug where CLI generation doesn't handle parameters of type string array. [#4707](https://github.com/microsoft/kiota/issues/4707) - Fixed a bug where models would not be created when a multipart content schema existed with no encoding [#4734](https://github.com/microsoft/kiota/issues/4734) - Fixes a bug where warnings about discriminator not being inherited were generated [#4761](https://github.com/microsoft/kiota/issues/4761) -- Fix scalar member composed type serialization in PHP [#2827](https://github.com/issues/2827) +- Fix scalar member composed type serialization in PHP [#2827](https://github.com/microsoft/kiota/issues/2827) ## [1.14.0] - 2024-05-02 From 95cbdcee66698395cf334adce91f77d1ef0d046e Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 4 Jun 2024 08:13:34 +0000 Subject: [PATCH 30/33] Bump Microsoft.Kiota.Abstractions Bumps the kiota-dependencies group in /it/csharp with 1 update: [Microsoft.Kiota.Abstractions](https://github.com/microsoft/kiota-abstractions-dotnet). Updates `Microsoft.Kiota.Abstractions` from 1.9.4 to 1.9.5 - [Release notes](https://github.com/microsoft/kiota-abstractions-dotnet/releases) - [Changelog](https://github.com/microsoft/kiota-abstractions-dotnet/blob/main/CHANGELOG.md) - [Commits](https://github.com/microsoft/kiota-abstractions-dotnet/compare/v1.9.4...v1.9.5) --- updated-dependencies: - dependency-name: Microsoft.Kiota.Abstractions dependency-type: direct:production update-type: version-update:semver-patch dependency-group: kiota-dependencies ... Signed-off-by: dependabot[bot] --- it/csharp/dotnet.csproj | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/it/csharp/dotnet.csproj b/it/csharp/dotnet.csproj index b3d5cd54db..dc7a740ebb 100644 --- a/it/csharp/dotnet.csproj +++ b/it/csharp/dotnet.csproj @@ -10,7 +10,7 @@ - + From a65439cd855568256f58cae410680ff5bfd6f73b Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 4 Jun 2024 08:26:14 +0000 Subject: [PATCH 31/33] Bump @types/node from 20.14.0 to 20.14.1 in /vscode/microsoft-kiota Bumps [@types/node](https://github.com/DefinitelyTyped/DefinitelyTyped/tree/HEAD/types/node) from 20.14.0 to 20.14.1. - [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases) - [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/node) --- updated-dependencies: - dependency-name: "@types/node" dependency-type: direct:development update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] --- vscode/microsoft-kiota/package-lock.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/vscode/microsoft-kiota/package-lock.json b/vscode/microsoft-kiota/package-lock.json index 72a7782fb7..eddb0dcf9e 100644 --- a/vscode/microsoft-kiota/package-lock.json +++ b/vscode/microsoft-kiota/package-lock.json @@ -483,9 +483,9 @@ "dev": true }, "node_modules/@types/node": { - "version": "20.14.0", - "resolved": "https://registry.npmjs.org/@types/node/-/node-20.14.0.tgz", - "integrity": "sha512-5cHBxFGJx6L4s56Bubp4fglrEpmyJypsqI6RgzMfBHWUJQGWAAi8cWcgetEbZXHYXo9C2Fa4EEds/uSyS4cxmA==", + "version": "20.14.1", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.14.1.tgz", + "integrity": "sha512-T2MzSGEu+ysB/FkWfqmhV3PLyQlowdptmmgD20C6QxsS8Fmv5SjpZ1ayXaEC0S21/h5UJ9iA6W/5vSNU5l00OA==", "dev": true, "dependencies": { "undici-types": "~5.26.4" From c70119fd2e7eb5c29e7dfb46a1e5c509c1107c9c Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 4 Jun 2024 08:33:08 +0000 Subject: [PATCH 32/33] Bump @types/node from 20.14.0 to 20.14.1 in /it/typescript Bumps [@types/node](https://github.com/DefinitelyTyped/DefinitelyTyped/tree/HEAD/types/node) from 20.14.0 to 20.14.1. - [Release notes](https://github.com/DefinitelyTyped/DefinitelyTyped/releases) - [Commits](https://github.com/DefinitelyTyped/DefinitelyTyped/commits/HEAD/types/node) --- updated-dependencies: - dependency-name: "@types/node" dependency-type: direct:development update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] --- it/typescript/package-lock.json | 8 ++++---- it/typescript/package.json | 2 +- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/it/typescript/package-lock.json b/it/typescript/package-lock.json index 7223244129..5c9fff831b 100644 --- a/it/typescript/package-lock.json +++ b/it/typescript/package-lock.json @@ -22,7 +22,7 @@ }, "devDependencies": { "@es-exec/esbuild-plugin-start": "^0.0.5", - "@types/node": "^20.14.0", + "@types/node": "^20.14.1", "@typescript-eslint/eslint-plugin": "^7.7.1", "@typescript-eslint/parser": "^7.8.0", "esbuild": "^0.21.4", @@ -827,9 +827,9 @@ "dev": true }, "node_modules/@types/node": { - "version": "20.14.0", - "resolved": "https://registry.npmjs.org/@types/node/-/node-20.14.0.tgz", - "integrity": "sha512-5cHBxFGJx6L4s56Bubp4fglrEpmyJypsqI6RgzMfBHWUJQGWAAi8cWcgetEbZXHYXo9C2Fa4EEds/uSyS4cxmA==", + "version": "20.14.1", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.14.1.tgz", + "integrity": "sha512-T2MzSGEu+ysB/FkWfqmhV3PLyQlowdptmmgD20C6QxsS8Fmv5SjpZ1ayXaEC0S21/h5UJ9iA6W/5vSNU5l00OA==", "dev": true, "dependencies": { "undici-types": "~5.26.4" diff --git a/it/typescript/package.json b/it/typescript/package.json index fa24125607..abc57622aa 100644 --- a/it/typescript/package.json +++ b/it/typescript/package.json @@ -19,7 +19,7 @@ "prettier": "./.prettierrc.json", "devDependencies": { "@es-exec/esbuild-plugin-start": "^0.0.5", - "@types/node": "^20.14.0", + "@types/node": "^20.14.1", "@typescript-eslint/eslint-plugin": "^7.7.1", "@typescript-eslint/parser": "^7.8.0", "esbuild": "^0.21.4", From 517d5f5dfd508af0e13f73ab89e6ef92d5abe176 Mon Sep 17 00:00:00 2001 From: Andrew Omondi Date: Tue, 4 Jun 2024 15:48:19 +0300 Subject: [PATCH 33/33] Fixes missing imports in java generation --- CHANGELOG.md | 1 + src/Kiota.Builder/Refiners/JavaRefiner.cs | 2 +- .../Refiners/JavaLanguageRefinerTests.cs | 10 ++++++---- 3 files changed, 8 insertions(+), 5 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index c5a5c43cf3..98ce954783 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -50,6 +50,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Fixes a bug where warnings about discriminator not being inherited were generated [#4761](https://github.com/microsoft/kiota/issues/4761) - Fix scalar member composed type serialization in PHP [#2827](https://github.com/microsoft/kiota/issues/2827) - Trims unused components from output openApi document when generating plugins [#4672](https://github.com/microsoft/kiota/issues/4672) +- Fixes missing imports for UntypedNode when backingstore is enabled in Java. ## [1.14.0] - 2024-05-02 diff --git a/src/Kiota.Builder/Refiners/JavaRefiner.cs b/src/Kiota.Builder/Refiners/JavaRefiner.cs index 1890dd04c7..db5593f75f 100644 --- a/src/Kiota.Builder/Refiners/JavaRefiner.cs +++ b/src/Kiota.Builder/Refiners/JavaRefiner.cs @@ -264,7 +264,7 @@ private static void AddEnumSetImport(CodeElement currentElement) AbstractionsNamespaceName, "QueryParameters"), new (static x => x is CodeClass @class && @class.OriginalComposedType is CodeIntersectionType intersectionType && intersectionType.Types.Any(static y => !y.IsExternal), SerializationNamespaceName, "ParseNodeHelper"), - new (static x => x is CodeProperty prop && prop.IsOfKind(CodePropertyKind.Custom) && prop.Type.Name.Equals(KiotaBuilder.UntypedNodeName, StringComparison.OrdinalIgnoreCase), + new (static x => (x is CodeMethod @method && @method.IsOfKind(CodeMethodKind.Getter, CodeMethodKind.Setter) && @method.AccessedProperty != null && (@method.AccessedProperty.IsOfKind(CodePropertyKind.Custom) && @method.AccessedProperty.Type.Name.Equals(KiotaBuilder.UntypedNodeName, StringComparison.OrdinalIgnoreCase) )) , SerializationNamespaceName, KiotaBuilder.UntypedNodeName), new (static x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.RequestExecutor, CodeMethodKind.RequestGenerator) && method.Parameters.Any(static y => y.IsOfKind(CodeParameterKind.RequestBody) && y.Type.Name.Equals(MultipartBodyClassName, StringComparison.OrdinalIgnoreCase)), AbstractionsNamespaceName, MultipartBodyClassName) diff --git a/tests/Kiota.Builder.Tests/Refiners/JavaLanguageRefinerTests.cs b/tests/Kiota.Builder.Tests/Refiners/JavaLanguageRefinerTests.cs index 8c36e68216..df2087c009 100644 --- a/tests/Kiota.Builder.Tests/Refiners/JavaLanguageRefinerTests.cs +++ b/tests/Kiota.Builder.Tests/Refiners/JavaLanguageRefinerTests.cs @@ -703,8 +703,10 @@ public async Task SplitsLongRefiners() Assert.Equal("String", model.Methods.First(static x => x.IsOverload).Parameters.First().Type.Name); } - [Fact] - public async Task AddsUsingForUntypedNode() + [Theory] + [InlineData(true)] + [InlineData(false)] + public async Task AddsUsingForUntypedNode(bool usesBackingStore) { var model = root.AddClass(new CodeClass { @@ -720,11 +722,11 @@ public async Task AddsUsingForUntypedNode() IsExternal = true }, }).First(); - await ILanguageRefiner.Refine(new GenerationConfiguration { Language = GenerationLanguage.Java }, root); + await ILanguageRefiner.Refine(new GenerationConfiguration { Language = GenerationLanguage.Java, UsesBackingStore = usesBackingStore }, root); Assert.Equal(KiotaBuilder.UntypedNodeName, property.Type.Name); Assert.NotEmpty(model.StartBlock.Usings); var nodeUsing = model.StartBlock.Usings.Where(static declaredUsing => declaredUsing.Name.Equals(KiotaBuilder.UntypedNodeName, StringComparison.OrdinalIgnoreCase)).ToArray(); - Assert.Single(nodeUsing); + Assert.Equal(2, nodeUsing.Length); // one for the getter and another for setter. Writer will unionise Assert.Equal("com.microsoft.kiota.serialization", nodeUsing[0].Declaration.Name); } #endregion