diff --git a/CHANGELOG.md b/CHANGELOG.md index 68607e75e9..1a4befd77e 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -15,6 +15,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Fixed a bug in generation when a referenced schema in an allOf was a primitive [#5701](https://github.com/microsoft/kiota/issues/5701). - Fixed a bug where inherited error models would be missing interface declarations. [#5888](https://github.com/microsoft/kiota/issues/5888) +- Fixed a bug where oneOf/anyOf schemas with single references to inheritance or intersections would be missing properties. [#5921](https://github.com/microsoft/kiota/issues/5921) ## [1.21.0] - 2024-12-05 diff --git a/src/Kiota.Builder/Extensions/IListExtensions.cs b/src/Kiota.Builder/Extensions/IListExtensions.cs new file mode 100644 index 0000000000..ec2aed7711 --- /dev/null +++ b/src/Kiota.Builder/Extensions/IListExtensions.cs @@ -0,0 +1,29 @@ +using System.Collections.Generic; + +namespace Kiota.Builder.Extensions; + +public static class IListExtensions +{ + /// + /// Returns the only element of this list when it has count of exactly 1 + /// + /// The contained item type. + /// The items. + /// The only element or null. + internal static T? OnlyOneOrDefault(this IList items) => + items.Count == 1 ? items[0] : default; + + /// + /// Adds the provided to this list. + /// + /// The contained item type. + /// The items. + /// The values to add. + internal static void AddRange(this IList items, IEnumerable values) + { + foreach (var item in values) + { + items.Add(item); + } + } +} diff --git a/src/Kiota.Builder/Extensions/OpenApiSchemaExtensions.cs b/src/Kiota.Builder/Extensions/OpenApiSchemaExtensions.cs index cec1a3cf2b..ec9d9d2f6a 100644 --- a/src/Kiota.Builder/Extensions/OpenApiSchemaExtensions.cs +++ b/src/Kiota.Builder/Extensions/OpenApiSchemaExtensions.cs @@ -98,13 +98,7 @@ public static bool IsInherited(this OpenApiSchema? schema) if (schema is null || !schema.IsInclusiveUnion(0)) return null; var result = new OpenApiSchema(schema); result.AnyOf.Clear(); - foreach (var subSchema in schema.AnyOf) - { - foreach (var property in subSchema.Properties) - { - result.Properties.TryAdd(property.Key, property.Value); - } - } + result.TryAddProperties(schema.AnyOf.SelectMany(static x => x.Properties)); return result; } @@ -113,14 +107,42 @@ public static bool IsInherited(this OpenApiSchema? schema) if (schema is null || !schema.IsExclusiveUnion(0)) return null; var result = new OpenApiSchema(schema); result.OneOf.Clear(); - foreach (var subSchema in schema.OneOf) + result.TryAddProperties(schema.OneOf.SelectMany(static x => x.Properties)); + return result; + } + + internal static OpenApiSchema? MergeSingleInclusiveUnionInheritanceOrIntersectionSchemaEntries(this OpenApiSchema? schema) + { + if (schema is not null + && schema.IsInclusiveUnion(0) + && schema.AnyOf.OnlyOneOrDefault() is OpenApiSchema subSchema + && (subSchema.IsInherited() || subSchema.IsIntersection())) { - foreach (var property in subSchema.Properties) - { - result.Properties.TryAdd(property.Key, property.Value); - } + var result = new OpenApiSchema(schema); + result.AnyOf.Clear(); + result.TryAddProperties(subSchema.Properties); + result.AllOf.AddRange(subSchema.AllOf); + return result; } - return result; + + return null; + } + + internal static OpenApiSchema? MergeSingleExclusiveUnionInheritanceOrIntersectionSchemaEntries(this OpenApiSchema? schema) + { + if (schema is not null + && schema.IsExclusiveUnion(0) + && schema.OneOf.OnlyOneOrDefault() is OpenApiSchema subSchema + && (subSchema.IsInherited() || subSchema.IsIntersection())) + { + var result = new OpenApiSchema(schema); + result.OneOf.Clear(); + result.TryAddProperties(subSchema.Properties); + result.AllOf.AddRange(subSchema.AllOf); + return result; + } + + return null; } internal static OpenApiSchema? MergeIntersectionSchemaEntries(this OpenApiSchema? schema, HashSet? schemasToExclude = default, bool overrideIntersection = false, Func? filter = default) @@ -144,11 +166,17 @@ public static bool IsInherited(this OpenApiSchema? schema) else if (discriminator.Mapping?.Any() ?? false) result.Discriminator.Mapping = discriminator.Mapping.ToDictionary(static x => x.Key, static x => x.Value); - foreach (var propertyToMerge in entriesToMerge.SelectMany(static x => x.Properties)) + result.TryAddProperties(entriesToMerge.SelectMany(static x => x.Properties)); + + return result; + } + + internal static void TryAddProperties(this OpenApiSchema schema, IEnumerable> properties) + { + foreach (var property in properties) { - result.Properties.TryAdd(propertyToMerge.Key, propertyToMerge.Value); + schema.Properties.TryAdd(property.Key, property.Value); } - return result; } public static bool IsIntersection(this OpenApiSchema? schema) diff --git a/src/Kiota.Builder/KiotaBuilder.cs b/src/Kiota.Builder/KiotaBuilder.cs index b4466a5bf0..450e392103 100644 --- a/src/Kiota.Builder/KiotaBuilder.cs +++ b/src/Kiota.Builder/KiotaBuilder.cs @@ -958,7 +958,7 @@ x.Parent is CodeIndexer || var parentNS = x.Parent?.Parent?.Parent as CodeNamespace; CodeClass[] exceptions = x.Parent?.Parent is CodeClass parentClass ? [parentClass] : []; x.TypeDefinition = parentNS?.FindChildrenByName(x.Name) - .Except(exceptions)// the property method should not reference itself as a return type. + .Except(exceptions)// the property method should not reference itself as a return type. .MinBy(shortestNamespaceOrder); // searching down first because most request builder properties on a request builder are just sub paths on the API if (x.TypeDefinition == null) @@ -1833,6 +1833,18 @@ private CodeTypeBase CreateModelDeclarations(OpenApiUrlTreeNode currentNode, Ope return CreateComposedModelDeclaration(currentNode, schema, operation, suffix, codeNamespace, isRequestBody, typeNameForInlineSchema); } + // type: object with single oneOf referring to inheritance or intersection + if (schema.IsObjectType() && schema.MergeSingleExclusiveUnionInheritanceOrIntersectionSchemaEntries() is OpenApiSchema mergedExclusiveUnionSchema) + { + return CreateModelDeclarations(currentNode, mergedExclusiveUnionSchema, operation, parentElement, suffixForInlineSchema, response, typeNameForInlineSchema, isRequestBody); + } + + // type: object with single anyOf referring to inheritance or intersection + if (schema.IsObjectType() && schema.MergeSingleInclusiveUnionInheritanceOrIntersectionSchemaEntries() is OpenApiSchema mergedInclusiveUnionSchema) + { + return CreateModelDeclarations(currentNode, mergedInclusiveUnionSchema, operation, parentElement, suffixForInlineSchema, response, typeNameForInlineSchema, isRequestBody); + } + if (schema.IsObjectType() || schema.HasAnyProperty() || schema.IsEnum() || !string.IsNullOrEmpty(schema.AdditionalProperties?.Type)) { // no inheritance or union type, often empty definitions with only additional properties are used as property bags. @@ -1840,7 +1852,7 @@ private CodeTypeBase CreateModelDeclarations(OpenApiUrlTreeNode currentNode, Ope } if (schema.IsArray() && - !schema.Items.IsArray()) // Only handle collections of primitives and complex types. Otherwise, multi-dimensional arrays would be recursively unwrapped undesirably to lead to incorrect serialization types. + !schema.Items.IsArray()) // Only handle collections of primitives and complex types. Otherwise, multi-dimensional arrays would be recursively unwrapped undesirably to lead to incorrect serialization types. { // collections at root return CreateCollectionModelDeclaration(currentNode, schema, operation, codeNamespace, typeNameForInlineSchema, isRequestBody); @@ -2230,7 +2242,7 @@ internal static void AddDiscriminatorMethod(CodeClass newClass, string discrimin logger.LogWarning("Discriminator {ComponentKey} not found in the OpenAPI document.", componentKey); return null; } - // Call CreateModelDeclarations with isViaDiscriminator=true. This is for a special case where we always generate a base class when types are referenced via a oneOf discriminator. + // Call CreateModelDeclarations with isViaDiscriminator=true. This is for a special case where we always generate a base class when types are referenced via a oneOf discriminator. if (CreateModelDeclarations(currentNode, discriminatorSchema, currentOperation, GetShortestNamespace(currentNamespace, discriminatorSchema), string.Empty, null, string.Empty, false, true) is not CodeType result) { logger.LogWarning("Discriminator {ComponentKey} is not a valid model and points to a union type.", componentKey); @@ -2238,7 +2250,7 @@ internal static void AddDiscriminatorMethod(CodeClass newClass, string discrimin } if (baseClass is not null && (result.TypeDefinition is not CodeClass codeClass || codeClass.StartBlock.Inherits is null)) { - 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. + 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; } diff --git a/tests/Kiota.Builder.Tests/Extensions/OpenApiSchemaExtensionsTests.cs b/tests/Kiota.Builder.Tests/Extensions/OpenApiSchemaExtensionsTests.cs index 6ca611be28..db1ff23416 100644 --- a/tests/Kiota.Builder.Tests/Extensions/OpenApiSchemaExtensionsTests.cs +++ b/tests/Kiota.Builder.Tests/Extensions/OpenApiSchemaExtensionsTests.cs @@ -673,6 +673,351 @@ public void MergesIntersectionRecursively() Assert.Equal("description", result.Description); Assert.True(result.Deprecated); } + + public class MergeSingleInclusiveUnionInheritanceOrIntersectionSchemaEntries + { + [Fact] + public void DoesMergeWithInheritance() + { + var schema = new OpenApiSchema() + { + Type = "object", + AnyOf = + [ + new() + { + Properties = new Dictionary() + { + ["one"] = new OpenApiSchema(), + }, + AllOf = + [ + new() + { + Reference = new() + { + Id = "BaseClass" + }, + }, + new() + { + Type = "object", + Properties = new Dictionary() + { + ["firstName"] = new OpenApiSchema(), + ["lastName"] = new OpenApiSchema() + } + }, + ] + }, + ], + }; + + var result = schema.MergeSingleInclusiveUnionInheritanceOrIntersectionSchemaEntries(); + Assert.True(schema.AnyOf[0].IsInherited()); + Assert.NotNull(result); + Assert.True(result.IsInherited()); + Assert.Contains("one", result.Properties.Keys); + Assert.Empty(result.AnyOf); + Assert.Equal(2, result.AllOf.Count); + } + [Fact] + public void DoesMergeWithIntersection() + { + var schema = new OpenApiSchema() + { + Type = "object", + AnyOf = + [ + new() + { + Properties = new Dictionary() + { + ["one"] = new OpenApiSchema(), + }, + AllOf = + [ + new() + { + Type = "object", + Properties = new Dictionary() + { + ["first"] = new OpenApiSchema(), + } + }, + new() + { + Type = "object", + Properties = new Dictionary() + { + ["second"] = new OpenApiSchema(), + } + }, + new() + { + Type = "object", + Properties = new Dictionary() + { + ["third"] = new OpenApiSchema(), + } + }, + ] + }, + ], + }; + + var result = schema.MergeSingleInclusiveUnionInheritanceOrIntersectionSchemaEntries(); + Assert.NotNull(result); + Assert.True(schema.AnyOf[0].IsIntersection()); + Assert.True(result.IsIntersection()); + Assert.Contains("one", result.Properties.Keys); + Assert.Empty(result.AnyOf); + Assert.Equal(3, result.AllOf.Count); + } + [Fact] + public void DoesNotMergeWithMoreThanOneInclusiveEntry() + { + var schema = new OpenApiSchema() + { + Type = "object", + AnyOf = + [ + new() + { + Properties = new Dictionary() + { + ["one"] = new OpenApiSchema(), + }, + AllOf = + [ + new() + { + Reference = new() + { + Id = "BaseClass" + }, + }, + new() + { + Type = "object", + Properties = new Dictionary() + { + ["firstName"] = new OpenApiSchema(), + ["lastName"] = new OpenApiSchema() + } + }, + ] + }, + new() { Type = "object" }, + ], + }; + + var result = schema.MergeSingleInclusiveUnionInheritanceOrIntersectionSchemaEntries(); + Assert.Null(result); + } + [Fact] + public void DoesNotMergeWithoutInheritanceOrIntersection() + { + var schema = new OpenApiSchema() + { + Type = "object", + AnyOf = + [ + new() + { + AllOf = + [ + new() + { + Type = "object", + Properties = new Dictionary() + { + ["firstName"] = new OpenApiSchema(), + ["lastName"] = new OpenApiSchema() + } + }, + ] + }, + ], + }; + + var result = schema.MergeSingleInclusiveUnionInheritanceOrIntersectionSchemaEntries(); + Assert.Null(result); + } + } + + public class MergeSingleExclusiveUnionInheritanceOrIntersectionSchemaEntries + { + [Fact] + public void DoesMergeWithInheritance() + { + var schema = new OpenApiSchema() + { + Type = "object", + OneOf = + [ + new() + { + Properties = new Dictionary() + { + ["one"] = new OpenApiSchema(), + }, + AllOf = + [ + new() + { + Reference = new() + { + Id = "BaseClass" + }, + }, + new() + { + Type = "object", + Properties = new Dictionary() + { + ["firstName"] = new OpenApiSchema(), + ["lastName"] = new OpenApiSchema() + } + }, + ] + }, + ], + }; + + var result = schema.MergeSingleExclusiveUnionInheritanceOrIntersectionSchemaEntries(); + Assert.True(schema.OneOf[0].IsInherited()); + Assert.NotNull(result); + Assert.True(result.IsInherited()); + Assert.Contains("one", result.Properties.Keys); + Assert.Empty(result.OneOf); + Assert.Equal(2, result.AllOf.Count); + } + [Fact] + public void DoesMergeWithIntersection() + { + var schema = new OpenApiSchema() + { + Type = "object", + OneOf = + [ + new() + { + Properties = new Dictionary() + { + ["one"] = new OpenApiSchema(), + }, + AllOf = + [ + new() + { + Type = "object", + Properties = new Dictionary() + { + ["first"] = new OpenApiSchema(), + } + }, + new() + { + Type = "object", + Properties = new Dictionary() + { + ["second"] = new OpenApiSchema(), + } + }, + new() + { + Type = "object", + Properties = new Dictionary() + { + ["third"] = new OpenApiSchema(), + } + }, + ] + }, + ], + }; + + var result = schema.MergeSingleExclusiveUnionInheritanceOrIntersectionSchemaEntries(); + Assert.NotNull(result); + Assert.True(schema.OneOf[0].IsIntersection()); + Assert.True(result.IsIntersection()); + Assert.Contains("one", result.Properties.Keys); + Assert.Empty(result.OneOf); + Assert.Equal(3, result.AllOf.Count); + } + [Fact] + public void DoesNotMergeWithMoreThanOneExclusiveEntry() + { + var schema = new OpenApiSchema() + { + Type = "object", + OneOf = + [ + new() + { + Properties = new Dictionary() + { + ["one"] = new OpenApiSchema(), + }, + AllOf = + [ + new() + { + Reference = new() + { + Id = "BaseClass" + }, + }, + new() + { + Type = "object", + Properties = new Dictionary() + { + ["firstName"] = new OpenApiSchema(), + ["lastName"] = new OpenApiSchema() + } + }, + ] + }, + new() { Type = "object" }, + ], + }; + + var result = schema.MergeSingleExclusiveUnionInheritanceOrIntersectionSchemaEntries(); + Assert.Null(result); + } + [Fact] + public void DoesNotMergeWithoutInheritanceOrIntersection() + { + var schema = new OpenApiSchema() + { + Type = "object", + OneOf = + [ + new() + { + AllOf = + [ + new() + { + Type = "object", + Properties = new Dictionary() + { + ["firstName"] = new OpenApiSchema(), + ["lastName"] = new OpenApiSchema() + } + }, + ] + }, + ], + }; + + var result = schema.MergeSingleExclusiveUnionInheritanceOrIntersectionSchemaEntries(); + Assert.Null(result); + } + } + [Fact] public void IsArrayFalseOnEmptyItems() { diff --git a/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs b/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs index a62089926f..7acf60fbb3 100644 --- a/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs +++ b/tests/Kiota.Builder.Tests/KiotaBuilderTests.cs @@ -8619,11 +8619,266 @@ public async Task ExclusiveUnionSingleEntriesMergingAsync() Assert.NotNull(oneProperty); var withoutObjectClass = codeModel.FindChildByName("Component2"); - Assert.NotNull(withObjectClass); + Assert.NotNull(withoutObjectClass); var twoProperty = withoutObjectClass.FindChildByName("two", false); Assert.NotNull(twoProperty); } + [Fact] + public async Task ExclusiveUnionInheritanceEntriesMergingAsync() + { + var tempFilePath = Path.Combine(Path.GetTempPath(), Path.GetTempFileName()); + await using var fs = await GetDocumentStreamAsync( +""" +openapi: 3.0.0 +info: + title: "Generator not generating oneOf if the containing schema has type: object" + version: "1.0.0" +servers: + - url: https://mytodos.doesnotexist/ +paths: + /uses-components: + post: + description: Return something + responses: + "200": + description: OK + content: + application/json: + schema: + $ref: "#/components/schemas/UsesComponents" + requestBody: + content: + application/json: + schema: + $ref: "#/components/schemas/UsesComponents" +components: + schemas: + ExampleWithSingleOneOfWithTypeObject: + description: "ExampleWithSingleOneOfWithTypeObject" + type: object + oneOf: + - $ref: "#/components/schemas/Component1" + discriminator: + propertyName: objectType + + ExampleWithSingleOneOfWithoutTypeObject: + description: "ExampleWithSingleOneOfWithoutTypeObject" + oneOf: + - $ref: "#/components/schemas/Component2" + discriminator: + propertyName: objectType + + UsesComponents: + type: object + properties: + component_with_single_oneof_with_type_object: + $ref: "#/components/schemas/ExampleWithSingleOneOfWithTypeObject" + component_with_single_oneof_without_type_object: + $ref: "#/components/schemas/ExampleWithSingleOneOfWithoutTypeObject" + + ComponentCommon: + description: "ComponentCommon" + type: object + required: + - objectType + properties: + objectType: + type: string + common: + type: string + + Component1: + description: "Component1" + type: object + allOf: + - $ref: "#/components/schemas/ComponentCommon" + - type: object + description: "Component1Inner" + properties: + one: + type: string + properties: + anotherOne: + type: string + + Component2: + description: "Component2" + type: object + allOf: + - $ref: "#/components/schemas/ComponentCommon" + - type: object + description: "Component2Inner" + properties: + two: + type: string + properties: + anotherTwo: + type: string +"""); + var mockLogger = new Mock>(); + var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath }, _httpClient); + var document = await builder.CreateOpenApiDocumentAsync(fs); + var node = builder.CreateUriSpace(document); + var codeModel = builder.CreateSourceModel(node); + + // Verify that all three classes referenced by the discriminator inherit from baseDirectoryObject + var withObjectClass = codeModel.FindChildByName("ExampleWithSingleOneOfWithTypeObject"); + Assert.NotNull(withObjectClass); + // ExampleWithSingleOneOfWithTypeObject inherits from ComponentCommon + Assert.Equal("ComponentCommon", withObjectClass.BaseClass?.Name); + var withObjectClassOneProperty = withObjectClass.FindChildByName("one", false); + Assert.NotNull(withObjectClassOneProperty); + var withObjectClassAnotherOneProperty = withObjectClass.FindChildByName("anotherOne", false); + Assert.NotNull(withObjectClassAnotherOneProperty); + + var withoutObjectClass = codeModel.FindChildByName("Component2"); + Assert.NotNull(withoutObjectClass); + // Component2 inherits from ComponentCommon + Assert.Equal("ComponentCommon", withoutObjectClass.BaseClass?.Name); + var withoutObjectClassTwoProperty = withoutObjectClass.FindChildByName("two", false); + Assert.NotNull(withoutObjectClassTwoProperty); + var withoutObjectClassClassAnotherTwoProperty = withoutObjectClass.FindChildByName("anotherTwo", false); + Assert.NotNull(withoutObjectClassClassAnotherTwoProperty); + } + + [Fact] + public async Task ExclusiveUnionIntersectionEntriesMergingAsync() + { + var tempFilePath = Path.Combine(Path.GetTempPath(), Path.GetTempFileName()); + await using var fs = await GetDocumentStreamAsync( +""" +openapi: 3.0.0 +info: + title: "Generator not generating oneOf if the containing schema has type: object" + version: "1.0.0" +servers: + - url: https://mytodos.doesnotexist/ +paths: + /uses-components: + post: + description: Return something + responses: + "200": + description: OK + content: + application/json: + schema: + $ref: "#/components/schemas/UsesComponents" + requestBody: + content: + application/json: + schema: + $ref: "#/components/schemas/UsesComponents" +components: + schemas: + ExampleWithSingleOneOfWithTypeObject: + description: "ExampleWithSingleOneOfWithTypeObject" + type: object + oneOf: + - $ref: "#/components/schemas/Component1" + discriminator: + propertyName: objectType + + ExampleWithSingleOneOfWithoutTypeObject: + description: "ExampleWithSingleOneOfWithoutTypeObject" + oneOf: + - $ref: "#/components/schemas/Component2" + discriminator: + propertyName: objectType + + UsesComponents: + type: object + properties: + component_with_single_oneof_with_type_object: + $ref: "#/components/schemas/ExampleWithSingleOneOfWithTypeObject" + component_with_single_oneof_without_type_object: + $ref: "#/components/schemas/ExampleWithSingleOneOfWithoutTypeObject" + + ComponentCommon: + description: "ComponentCommon" + type: object + required: + - objectType + properties: + objectType: + type: string + common: + type: string + + ComponentCommon2: + description: "ComponentCommon2" + type: object + properties: + common2: + type: string + + Component1: + description: "Component1" + type: object + allOf: + - $ref: "#/components/schemas/ComponentCommon" + - $ref: "#/components/schemas/ComponentCommon2" + - type: object + description: "Component1Self" + properties: + one: + type: string + properties: + anotherOne: + type: string + + Component2: + description: "Component2" + type: object + required: + - objectType + allOf: + - $ref: "#/components/schemas/ComponentCommon" + - $ref: "#/components/schemas/ComponentCommon2" + - type: object + description: "Component2Self" + properties: + two: + type: string + properties: + anotherTwo: + type: string +"""); + var mockLogger = new Mock>(); + var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath }, _httpClient); + var document = await builder.CreateOpenApiDocumentAsync(fs); + var node = builder.CreateUriSpace(document); + var codeModel = builder.CreateSourceModel(node); + + // Verify both scenarios have all the properties available from all schemas + var withObjectClass = codeModel.FindChildByName("ExampleWithSingleOneOfWithTypeObject"); + Assert.NotNull(withObjectClass); + var withObjectClassOneProperty = withObjectClass.FindChildByName("one", false); + Assert.NotNull(withObjectClassOneProperty); + var withObjectClassCommonProperty = withObjectClass.FindChildByName("common", false); + Assert.NotNull(withObjectClassCommonProperty); + var withObjectClassCommon2Property = withObjectClass.FindChildByName("common2", false); + Assert.NotNull(withObjectClassCommon2Property); + var withObjectClassObjectTypeProperty = withObjectClass.FindChildByName("objectType", false); + Assert.NotNull(withObjectClassObjectTypeProperty); + var withObjectClassAnotherOneProperty = withObjectClass.FindChildByName("anotherOne", false); + Assert.NotNull(withObjectClassAnotherOneProperty); + + var withoutObjectClass = codeModel.FindChildByName("Component2"); + Assert.NotNull(withoutObjectClass); + var withoutObjectClassTwoProperty = withoutObjectClass.FindChildByName("two", false); + Assert.NotNull(withoutObjectClassTwoProperty); + var withoutObjectClassCommonProperty = withoutObjectClass.FindChildByName("common", false); + Assert.NotNull(withoutObjectClassCommonProperty); + var withoutObjectClassCommon2Property = withoutObjectClass.FindChildByName("common2", false); + Assert.NotNull(withoutObjectClassCommon2Property); + var withoutObjectClassObjectTypeProperty = withoutObjectClass.FindChildByName("objectType", false); + Assert.NotNull(withoutObjectClassObjectTypeProperty); + var withoutObjectClassClassAnotherTwoProperty = withoutObjectClass.FindChildByName("anotherTwo", false); + Assert.NotNull(withoutObjectClassClassAnotherTwoProperty); + } + [Fact] public async Task InclusiveUnionSingleEntriesMergingAsync() { @@ -8712,6 +8967,261 @@ public async Task InclusiveUnionSingleEntriesMergingAsync() Assert.NotNull(twoProperty); } + [Fact] + public async Task InclusiveUnionInheritanceEntriesMergingAsync() + { + var tempFilePath = Path.Combine(Path.GetTempPath(), Path.GetTempFileName()); + await using var fs = await GetDocumentStreamAsync( +""" +openapi: 3.0.0 +info: + title: "Generator not generating oneOf if the containing schema has type: object" + version: "1.0.0" +servers: + - url: https://mytodos.doesnotexist/ +paths: + /uses-components: + post: + description: Return something + responses: + "200": + description: OK + content: + application/json: + schema: + $ref: "#/components/schemas/UsesComponents" + requestBody: + content: + application/json: + schema: + $ref: "#/components/schemas/UsesComponents" +components: + schemas: + ExampleWithSingleOneOfWithTypeObject: + description: "ExampleWithSingleOneOfWithTypeObject" + type: object + anyOf: + - $ref: "#/components/schemas/Component1" + discriminator: + propertyName: objectType + + ExampleWithSingleOneOfWithoutTypeObject: + description: "ExampleWithSingleOneOfWithoutTypeObject" + anyOf: + - $ref: "#/components/schemas/Component2" + discriminator: + propertyName: objectType + + UsesComponents: + type: object + properties: + component_with_single_oneof_with_type_object: + $ref: "#/components/schemas/ExampleWithSingleOneOfWithTypeObject" + component_with_single_oneof_without_type_object: + $ref: "#/components/schemas/ExampleWithSingleOneOfWithoutTypeObject" + + ComponentCommon: + description: "ComponentCommon" + type: object + required: + - objectType + properties: + objectType: + type: string + common: + type: string + + Component1: + description: "Component1" + type: object + allOf: + - $ref: "#/components/schemas/ComponentCommon" + - type: object + description: "Component1Inner" + properties: + one: + type: string + properties: + anotherOne: + type: string + + Component2: + description: "Component2" + type: object + allOf: + - $ref: "#/components/schemas/ComponentCommon" + - type: object + description: "Component2Inner" + properties: + two: + type: string + properties: + anotherTwo: + type: string +"""); + var mockLogger = new Mock>(); + var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath }, _httpClient); + var document = await builder.CreateOpenApiDocumentAsync(fs); + var node = builder.CreateUriSpace(document); + var codeModel = builder.CreateSourceModel(node); + + // Verify that all three classes referenced by the discriminator inherit from baseDirectoryObject + var withObjectClass = codeModel.FindChildByName("ExampleWithSingleOneOfWithTypeObject"); + Assert.NotNull(withObjectClass); + // ExampleWithSingleOneOfWithTypeObject inherits from ComponentCommon + Assert.Equal("ComponentCommon", withObjectClass.BaseClass?.Name); + var withObjectClassOneProperty = withObjectClass.FindChildByName("one", false); + Assert.NotNull(withObjectClassOneProperty); + var withObjectClassAnotherOneProperty = withObjectClass.FindChildByName("anotherOne", false); + Assert.NotNull(withObjectClassAnotherOneProperty); + + var withoutObjectClass = codeModel.FindChildByName("Component2"); + Assert.NotNull(withoutObjectClass); + // Component2 inherits from ComponentCommon + Assert.Equal("ComponentCommon", withoutObjectClass.BaseClass?.Name); + var withoutObjectClassTwoProperty = withoutObjectClass.FindChildByName("two", false); + Assert.NotNull(withoutObjectClassTwoProperty); + var withoutObjectClassClassAnotherTwoProperty = withoutObjectClass.FindChildByName("anotherTwo", false); + Assert.NotNull(withoutObjectClassClassAnotherTwoProperty); + } + + [Fact] + public async Task InclusiveUnionIntersectionEntriesMergingAsync() + { + var tempFilePath = Path.Combine(Path.GetTempPath(), Path.GetTempFileName()); + await using var fs = await GetDocumentStreamAsync( +""" +openapi: 3.0.0 +info: + title: "Generator not generating oneOf if the containing schema has type: object" + version: "1.0.0" +servers: + - url: https://mytodos.doesnotexist/ +paths: + /uses-components: + post: + description: Return something + responses: + "200": + description: OK + content: + application/json: + schema: + $ref: "#/components/schemas/UsesComponents" + requestBody: + content: + application/json: + schema: + $ref: "#/components/schemas/UsesComponents" +components: + schemas: + ExampleWithSingleOneOfWithTypeObject: + description: "ExampleWithSingleOneOfWithTypeObject" + type: object + anyOf: + - $ref: "#/components/schemas/Component1" + discriminator: + propertyName: objectType + + ExampleWithSingleOneOfWithoutTypeObject: + description: "ExampleWithSingleOneOfWithoutTypeObject" + anyOf: + - $ref: "#/components/schemas/Component2" + discriminator: + propertyName: objectType + + UsesComponents: + type: object + properties: + component_with_single_oneof_with_type_object: + $ref: "#/components/schemas/ExampleWithSingleOneOfWithTypeObject" + component_with_single_oneof_without_type_object: + $ref: "#/components/schemas/ExampleWithSingleOneOfWithoutTypeObject" + + ComponentCommon: + description: "ComponentCommon" + type: object + required: + - objectType + properties: + objectType: + type: string + common: + type: string + + ComponentCommon2: + description: "ComponentCommon2" + type: object + properties: + common2: + type: string + + Component1: + description: "Component1" + type: object + allOf: + - $ref: "#/components/schemas/ComponentCommon" + - $ref: "#/components/schemas/ComponentCommon2" + - type: object + description: "Component1Self" + properties: + one: + type: string + properties: + anotherOne: + type: string + + Component2: + description: "Component2" + type: object + required: + - objectType + allOf: + - $ref: "#/components/schemas/ComponentCommon" + - $ref: "#/components/schemas/ComponentCommon2" + - type: object + description: "Component2Self" + properties: + two: + type: string + properties: + anotherTwo: + type: string +"""); + var mockLogger = new Mock>(); + var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath }, _httpClient); + var document = await builder.CreateOpenApiDocumentAsync(fs); + var node = builder.CreateUriSpace(document); + var codeModel = builder.CreateSourceModel(node); + + // Verify both scenarios have all the properties available from all schemas + var withObjectClass = codeModel.FindChildByName("ExampleWithSingleOneOfWithTypeObject"); + Assert.NotNull(withObjectClass); + var withObjectClassOneProperty = withObjectClass.FindChildByName("one", false); + Assert.NotNull(withObjectClassOneProperty); + var withObjectClassCommonProperty = withObjectClass.FindChildByName("common", false); + Assert.NotNull(withObjectClassCommonProperty); + var withObjectClassCommon2Property = withObjectClass.FindChildByName("common2", false); + Assert.NotNull(withObjectClassCommon2Property); + var withObjectClassObjectTypeProperty = withObjectClass.FindChildByName("objectType", false); + Assert.NotNull(withObjectClassObjectTypeProperty); + var withObjectClassAnotherOneProperty = withObjectClass.FindChildByName("anotherOne", false); + Assert.NotNull(withObjectClassAnotherOneProperty); + + var withoutObjectClass = codeModel.FindChildByName("Component2"); + Assert.NotNull(withoutObjectClass); + var withoutObjectClassTwoProperty = withoutObjectClass.FindChildByName("two", false); + Assert.NotNull(withoutObjectClassTwoProperty); + var withoutObjectClassCommonProperty = withoutObjectClass.FindChildByName("common", false); + Assert.NotNull(withoutObjectClassCommonProperty); + var withoutObjectClassCommon2Property = withoutObjectClass.FindChildByName("common2", false); + Assert.NotNull(withoutObjectClassCommon2Property); + var withoutObjectClassObjectTypeProperty = withoutObjectClass.FindChildByName("objectType", false); + Assert.NotNull(withoutObjectClassObjectTypeProperty); + var withoutObjectClassClassAnotherTwoProperty = withoutObjectClass.FindChildByName("anotherTwo", false); + Assert.NotNull(withoutObjectClassClassAnotherTwoProperty); + } + [Fact] public async Task NestedIntersectionTypeAllOfAsync() { @@ -9022,14 +9532,14 @@ public async Task InheritanceWithAllOfWith3Parts1Schema2InlineAsync(bool reverse '#microsoft.graph.group': '#/components/schemas/microsoft.graph.group' microsoft.graph.group: allOf:" - + (reverseOrder ? "" : @" + + (reverseOrder ? "" : @" - '$ref': '#/components/schemas/microsoft.graph.directoryObject'") + @" - properties: groupprop1: type: 'string' - properties: groupprop2: - type: 'string'" + (!reverseOrder ? "" : @" + type: 'string'" + (!reverseOrder ? "" : @" - '$ref': '#/components/schemas/microsoft.graph.directoryObject'")); var mockLogger = new Mock>(); var builder = new KiotaBuilder(mockLogger.Object, new GenerationConfiguration { ClientClassName = "Graph", OpenAPIFilePath = tempFilePath }, _httpClient);