From 032e5d0c82ca898f1b9ad03067f8d070917cfb74 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Wed, 8 Mar 2023 18:45:45 +0300 Subject: [PATCH 01/36] Add python relative import manager --- .../Python/PythonRelativeImportManager.cs | 55 +++++++++++++++++++ 1 file changed, 55 insertions(+) create mode 100644 src/Kiota.Builder/Writers/Python/PythonRelativeImportManager.cs diff --git a/src/Kiota.Builder/Writers/Python/PythonRelativeImportManager.cs b/src/Kiota.Builder/Writers/Python/PythonRelativeImportManager.cs new file mode 100644 index 0000000000..27cafa4f83 --- /dev/null +++ b/src/Kiota.Builder/Writers/Python/PythonRelativeImportManager.cs @@ -0,0 +1,55 @@ +using System; +using System.Collections.Generic; +using System.Linq; + +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Extensions; + +namespace Kiota.Builder.Writers.Python; +public class PythonRelativeImportManager : RelativeImportManager +{ + public PythonRelativeImportManager(string namespacePrefix, char namespaceSeparator) : base(namespacePrefix, namespaceSeparator) + { + } + /// + /// Returns the relative import path for the given using and import context namespace. + /// + /// The using to import into the current namespace context + /// The current namespace + /// The import symbol, it's alias if any and the relative import path + public override (string, string, string) GetRelativeImportPathForUsing(CodeUsing codeUsing, CodeNamespace currentNamespace) + { + if (codeUsing?.IsExternal ?? true) + return (string.Empty, string.Empty, string.Empty);//it's an external import, add nothing + var (importSymbol, typeDef) = codeUsing.Declaration?.TypeDefinition is CodeElement td ? td switch + { + CodeFunction f => (f.Name.ToFirstCharacterLowerCase(), td), + _ => (td.Name.ToSnakeCase(), td), + } : (codeUsing.Name, null); + + if (typeDef == null) + return (importSymbol, codeUsing.Alias, "."); // it's relative to the folder, with no declaration (default failsafe) + var importPath = GetImportRelativePathFromNamespaces(currentNamespace, + typeDef.GetImmediateParentOfType()); + return (importSymbol, codeUsing.Alias, importPath); + } + protected new string GetImportRelativePathFromNamespaces(CodeNamespace currentNamespace, CodeNamespace importNamespace) + { + var result = currentNamespace.GetDifferential(importNamespace, prefix, separator); + return result.State switch + { + NamespaceDifferentialTrackerState.Same => ".", + NamespaceDifferentialTrackerState.Downwards => $".{GetRemainingImportPath(result.DownwardsSegments)}", + NamespaceDifferentialTrackerState.Upwards => GetUpwardsMoves(result.UpwardsMovesCount), + NamespaceDifferentialTrackerState.UpwardsAndThenDownwards => $"{GetUpwardsMoves(result.UpwardsMovesCount)}{GetRemainingImportPath(result.DownwardsSegments)}", + _ => throw new NotImplementedException(), + }; + } + protected static new string GetUpwardsMoves(int UpwardsMovesCount) => string.Join("", Enumerable.Repeat(".", UpwardsMovesCount)) + (UpwardsMovesCount > 0 ? "." : string.Empty); + protected static new string GetRemainingImportPath(IEnumerable remainingSegments) + { + if (remainingSegments.Any()) + return remainingSegments.Select(x => x.ToFirstCharacterLowerCase()).Aggregate((x, y) => $"{x}.{y}"); + return string.Empty; + } +} From 7423f0d8cee16b834b76ee906d6ff2047bb9f1b2 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 06:09:16 +0300 Subject: [PATCH 02/36] Add a codeusingwriter for python --- .../Writers/Python/CodeUsingWriter.cs | 135 ++++++++++++++++++ 1 file changed, 135 insertions(+) create mode 100644 src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs diff --git a/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs b/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs new file mode 100644 index 0000000000..b29212923f --- /dev/null +++ b/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs @@ -0,0 +1,135 @@ +using System.Linq; + +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Extensions; + + +namespace Kiota.Builder.Writers.Python; +public class CodeUsingWriter +{ + private readonly PythonRelativeImportManager _relativeImportManager; + public CodeUsingWriter(string clientNamespaceName) + { + _relativeImportManager = new PythonRelativeImportManager(clientNamespaceName, '.'); + } + /// + /// Writes external imports for a given code element. + /// + /// The element to write external usings + /// An instance of the language writer + /// void + public void WriteExternalImports(ClassDeclaration codeElement, LanguageWriter writer) + { + var externalImportSymbolsAndPaths = codeElement.Usings + .Where(static x => x.IsExternal) + .Select(x => (x.Name, string.Empty, x.Declaration?.Name)) + .GroupBy(x => x.Item3) + .OrderBy(x => x.Key); + if (externalImportSymbolsAndPaths.Any()) + { + foreach (var codeUsing in externalImportSymbolsAndPaths) + if (!string.IsNullOrWhiteSpace(codeUsing.Key)) + { + if (codeUsing.Key == "-") + writer.WriteLine($"import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct().OrderBy(x => x).Aggregate((x, y) => x + ", " + y)}"); + else + writer.WriteLine($"from {codeUsing.Key.ToSnakeCase()} import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct().OrderBy(x => x).Aggregate((x, y) => x + ", " + y)}"); + } + writer.WriteLine(); + } + } + /// + /// Writes error mapping imports for a given code class. + /// + /// The CodeClass from which to write error mapping usings + /// An instance of the language writer + /// void + public void WriteInternalErrorMappingImports(CodeClass parentClass, LanguageWriter writer) + { + var parentNameSpace = parentClass.GetImmediateParentOfType(); + var internalErrorMappingImportSymbolsAndPaths = parentClass.Usings + .Where(x => !x.IsExternal) + .Where(x => x.Declaration?.TypeDefinition is CodeClass codeClass && codeClass.IsErrorDefinition) + .Select(x => _relativeImportManager.GetRelativeImportPathForUsing(x, parentNameSpace)) + .GroupBy(x => x.Item3) + .Where(x => !string.IsNullOrEmpty(x.Key)) + .OrderBy(x => x.Key); + WriteCodeUsings(internalErrorMappingImportSymbolsAndPaths, writer); + } + /// + /// Writes all internal imports for a given code class. + /// + /// The CodeClass from which to write internal usings + /// An instance of the language writer + /// void + public void WriteInternalImports(CodeClass parentClass, LanguageWriter writer) + { + var parentNameSpace = parentClass.GetImmediateParentOfType(); + var internalImportSymbolsAndPaths = parentClass.Usings + .Where(x => !x.IsExternal) + .Select(x => _relativeImportManager.GetRelativeImportPathForUsing(x, parentNameSpace)) + .GroupBy(x => x.Item3) + .Where(x => !string.IsNullOrEmpty(x.Key)) + .OrderBy(x => x.Key); + WriteCodeUsings(internalImportSymbolsAndPaths, writer); + } + /// + /// Writes conditional internal imports for a given code element for type checking environments. + /// + /// The element to write internal usings from + /// An instance of the language writer + /// The code namespace of the code element + /// void + public void WriteConditionalInternalImports(ClassDeclaration codeElement, LanguageWriter writer, CodeNamespace parentNameSpace) + { + var internalImportSymbolsAndPaths = codeElement.Usings + .Where(x => !x.IsExternal) + .Select(x => _relativeImportManager.GetRelativeImportPathForUsing(x, parentNameSpace)) + .GroupBy(x => x.Item3) + .Where(x => !string.IsNullOrEmpty(x.Key)) + .OrderBy(x => x.Key); + if (internalImportSymbolsAndPaths.Any()) + { + writer.WriteLine("if TYPE_CHECKING:"); + writer.IncreaseIndent(); + foreach (var codeUsing in internalImportSymbolsAndPaths) + writer.WriteLine($"from {codeUsing.Key.ToSnakeCase()} import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct().OrderBy(x => x).Aggregate((x, y) => x + ", " + y)}"); + writer.DecreaseIndent(); + writer.WriteLine(); + + } + } + /// + /// Writes local imports for a given type. + /// + /// The parent CodeClass of the given type + /// The name of the given type. Must be a valid using declaration name in the parentClass + /// An instance of the language writer + /// void + public void WriteDeferredImport(CodeClass parentClass, string typeName, LanguageWriter writer) + { + var parentNamespace = parentClass.GetImmediateParentOfType(); + var internalImportSymbolsAndPaths = parentClass.Usings + .Where(x => !x.IsExternal) + .Where(x => string.Equals(x.Declaration?.Name, typeName)) + .Select(x => _relativeImportManager.GetRelativeImportPathForUsing(x, parentNamespace)) + .GroupBy(x => x.Item3) + .Where(x => !string.IsNullOrEmpty(x.Key)) + .OrderBy(x => x.Key); + WriteCodeUsings(internalImportSymbolsAndPaths, writer); + } + + private static void WriteCodeUsings(IOrderedEnumerable> importSymbolsAndPaths, LanguageWriter writer) + { + if (importSymbolsAndPaths.Any()) + { + foreach (var codeUsing in importSymbolsAndPaths) + writer.WriteLine($"from {codeUsing.Key.ToSnakeCase()} import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct().OrderBy(x => x).Aggregate((x, y) => x + ", " + y)}"); + writer.WriteLine(); + } + } + private static string GetAliasedSymbol(string symbol, string alias) + { + return string.IsNullOrEmpty(alias) ? symbol : $"{symbol} as {alias}"; + } +} From 5fe8d47d029016882fe7bc11b1295b8a7eef67b6 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 06:33:32 +0300 Subject: [PATCH 03/36] Add option to include parent namespace when adding discriminator mapping usings --- .../Refiners/CommonLanguageRefiner.cs | 18 ++++++++++++++++-- 1 file changed, 16 insertions(+), 2 deletions(-) diff --git a/src/Kiota.Builder/Refiners/CommonLanguageRefiner.cs b/src/Kiota.Builder/Refiners/CommonLanguageRefiner.cs index c6b033c277..18bd95e58d 100644 --- a/src/Kiota.Builder/Refiners/CommonLanguageRefiner.cs +++ b/src/Kiota.Builder/Refiners/CommonLanguageRefiner.cs @@ -953,7 +953,7 @@ protected static void AddParentClassToErrorClasses(CodeElement currentElement, s } CrawlTree(currentElement, x => AddParentClassToErrorClasses(x, parentClassName, parentClassNamespace, addNamespaceToInheritDeclaration)); } - protected static void AddDiscriminatorMappingsUsingsToParentClasses(CodeElement currentElement, string parseNodeInterfaceName, bool addFactoryMethodImport = false, bool addUsings = true) + protected static void AddDiscriminatorMappingsUsingsToParentClasses(CodeElement currentElement, string parseNodeInterfaceName, bool addFactoryMethodImport = false, bool addUsings = true, bool includeParentNamespace = false) { if (currentElement is CodeMethod currentMethod && currentMethod.Parent is CodeClass parentClass && @@ -963,7 +963,21 @@ currentMethod.Parent is CodeClass parentClass && (parentClass.DiscriminatorInformation?.HasBasicDiscriminatorInformation ?? false) && parentClass.GetImmediateParentOfType() is CodeNamespace parentClassNamespace) { - if (addUsings) + if (addUsings && includeParentNamespace) + declaration.AddUsings(parentClass.DiscriminatorInformation.DiscriminatorMappings + .Select(static x => x.Value) + .OfType() + .Where(static x => x.TypeDefinition != null) + .Select(x => new CodeUsing + { + Name = x.TypeDefinition!.GetImmediateParentOfType().Name, + Declaration = new CodeType + { + Name = x.TypeDefinition.Name, + TypeDefinition = x.TypeDefinition, + }, + }).ToArray()); + else if (addUsings && !includeParentNamespace) declaration.AddUsings(parentClass.DiscriminatorInformation.DiscriminatorMappings .Select(static x => x.Value) .OfType() From a8ac9252243330ede2ff45953970016c0fba5608 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 06:36:09 +0300 Subject: [PATCH 04/36] Call method to add discriminator mapping usings to parent classes --- src/Kiota.Builder/Refiners/PythonRefiner.cs | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/src/Kiota.Builder/Refiners/PythonRefiner.cs b/src/Kiota.Builder/Refiners/PythonRefiner.cs index e7759544db..2161d335e7 100644 --- a/src/Kiota.Builder/Refiners/PythonRefiner.cs +++ b/src/Kiota.Builder/Refiners/PythonRefiner.cs @@ -90,6 +90,12 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance AddQueryParameterMapperMethod( generatedCode ); + AddDiscriminatorMappingsUsingsToParentClasses( + generatedCode, + "ParseNode", + addUsings: true, + includeParentNamespace: true + ); RemoveHandlerFromRequestBuilder(generatedCode); }, cancellationToken); } From 4b705b703e13dd541effab7fbeaf0a43c0616d20 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 06:37:41 +0300 Subject: [PATCH 05/36] Add type checking import --- src/Kiota.Builder/Refiners/PythonRefiner.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Kiota.Builder/Refiners/PythonRefiner.cs b/src/Kiota.Builder/Refiners/PythonRefiner.cs index 2161d335e7..e7b4aada74 100644 --- a/src/Kiota.Builder/Refiners/PythonRefiner.cs +++ b/src/Kiota.Builder/Refiners/PythonRefiner.cs @@ -103,7 +103,7 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance private const string AbstractionsPackageName = "kiota_abstractions"; private static readonly AdditionalUsingEvaluator[] defaultUsingEvaluators = { new (static x => x is CodeClass, "__future__", "annotations"), - new (static x => x is CodeClass, "typing", "Any, Callable, Dict, List, Optional, Union"), + new (static x => x is CodeClass, "typing", "Any, Callable, Dict, List, Optional, TYPE_CHECKING, Union"), new (static x => x is CodeClass, $"{AbstractionsPackageName}.utils", "lazy_import"), new (static x => x is CodeProperty prop && prop.IsOfKind(CodePropertyKind.RequestAdapter), $"{AbstractionsPackageName}.request_adapter", "RequestAdapter"), From c69c96193813250349bf7d3b9b611dec1c560e95 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 06:38:21 +0300 Subject: [PATCH 06/36] Remove lazy import using --- src/Kiota.Builder/Refiners/PythonRefiner.cs | 1 - 1 file changed, 1 deletion(-) diff --git a/src/Kiota.Builder/Refiners/PythonRefiner.cs b/src/Kiota.Builder/Refiners/PythonRefiner.cs index e7b4aada74..fea1ba72de 100644 --- a/src/Kiota.Builder/Refiners/PythonRefiner.cs +++ b/src/Kiota.Builder/Refiners/PythonRefiner.cs @@ -104,7 +104,6 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance private static readonly AdditionalUsingEvaluator[] defaultUsingEvaluators = { new (static x => x is CodeClass, "__future__", "annotations"), new (static x => x is CodeClass, "typing", "Any, Callable, Dict, List, Optional, TYPE_CHECKING, Union"), - new (static x => x is CodeClass, $"{AbstractionsPackageName}.utils", "lazy_import"), new (static x => x is CodeProperty prop && prop.IsOfKind(CodePropertyKind.RequestAdapter), $"{AbstractionsPackageName}.request_adapter", "RequestAdapter"), new (static x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.RequestGenerator), From bc7fa3671b7169b6fcc9973862314b3d2b43639f Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 06:50:55 +0300 Subject: [PATCH 07/36] Update class declaration writer --- .../Python/CodeClassDeclarationWriter.cs | 92 +++++-------------- 1 file changed, 25 insertions(+), 67 deletions(-) diff --git a/src/Kiota.Builder/Writers/Python/CodeClassDeclarationWriter.cs b/src/Kiota.Builder/Writers/Python/CodeClassDeclarationWriter.cs index ee434f11b4..a5d4eb5976 100644 --- a/src/Kiota.Builder/Writers/Python/CodeClassDeclarationWriter.cs +++ b/src/Kiota.Builder/Writers/Python/CodeClassDeclarationWriter.cs @@ -1,22 +1,36 @@ using System; using System.Linq; - using Kiota.Builder.CodeDOM; using Kiota.Builder.Extensions; namespace Kiota.Builder.Writers.Python; public class CodeClassDeclarationWriter : BaseElementWriter { - - public CodeClassDeclarationWriter(PythonConventionService conventionService) : base(conventionService) + private readonly CodeUsingWriter _codeUsingWriter; + public CodeClassDeclarationWriter(PythonConventionService conventionService, string clientNamespaceName) : base(conventionService) { + _codeUsingWriter = new(clientNamespaceName); } public override void WriteCodeElement(ClassDeclaration codeElement, LanguageWriter writer) { ArgumentNullException.ThrowIfNull(codeElement); - ArgumentNullException.ThrowIfNull(writer); - WriteExternalImports(codeElement, writer); // external imports before internal imports - WriteInternalImports(codeElement, writer); + if (writer == null) throw new ArgumentNullException(nameof(writer)); + var parentNamespace = codeElement.GetImmediateParentOfType(); + _codeUsingWriter.WriteExternalImports(codeElement, writer); // external imports before internal imports + _codeUsingWriter.WriteConditionalInternalImports(codeElement, writer, parentNamespace); + + if (codeElement.Parent is CodeClass parentClass) + { + if (codeElement.Inherits != null) + _codeUsingWriter.WriteDeferredImport(parentClass, codeElement.Inherits.Name, writer); + if (codeElement.Implements.Any()) + { + foreach (var implement in codeElement.Implements) + _codeUsingWriter.WriteDeferredImport(parentClass, implement.Name, writer); + } + + } + var abcClass = !codeElement.Implements.Any() ? string.Empty : $"{codeElement.Implements.Select(static x => x.Name.ToFirstCharacterUpperCase()).Aggregate((x, y) => x + ", " + y)}"; var derivation = codeElement.Inherits is CodeType inheritType && @@ -24,72 +38,16 @@ public override void WriteCodeElement(ClassDeclaration codeElement, LanguageWrit !string.IsNullOrEmpty(inheritSymbol) ? inheritSymbol : abcClass; + + + if (codeElement.Parent?.Parent is CodeClass) { writer.WriteLine("@dataclass"); } writer.WriteLine($"class {codeElement.Name.ToFirstCharacterUpperCase()}({derivation}):"); writer.IncreaseIndent(); - if (codeElement.Parent is CodeClass parentClass) - conventions.WriteShortDescription(parentClass.Documentation.Description, writer); - } - - private static void WriteExternalImports(ClassDeclaration codeElement, LanguageWriter writer) - { - var externalImportSymbolsAndPaths = codeElement.Usings - .Where(static x => x.IsExternal) - .Select(x => (x.Name, string.Empty, x.Declaration?.Name)) - .GroupBy(x => x.Item3) - .OrderBy(x => x.Key); - if (externalImportSymbolsAndPaths.Any()) - { - foreach (var codeUsing in externalImportSymbolsAndPaths) - if (!string.IsNullOrWhiteSpace(codeUsing.Key)) - { - if (codeUsing.Key == "-") - writer.WriteLine($"import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct().OrderBy(x => x).Aggregate((x, y) => x + ", " + y)}"); - else - writer.WriteLine($"from {codeUsing.Key.ToSnakeCase()} import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct().OrderBy(x => x).Aggregate((x, y) => x + ", " + y)}"); - } - writer.WriteLine(); - } - } - - private static void WriteInternalImports(ClassDeclaration codeElement, LanguageWriter writer) - { - var internalImportSymbolsAndPaths = codeElement.Usings - .Where(x => !x.IsExternal) - .Select(static x => GetImportPathForUsing(x)) - .GroupBy(x => x.Item3) - .Where(x => !string.IsNullOrEmpty(x.Key)) - .OrderBy(x => x.Key); - if (internalImportSymbolsAndPaths.Any()) - { - foreach (var codeUsing in internalImportSymbolsAndPaths) - foreach (var symbol in codeUsing.Select(static x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct().OrderBy(static x => x)) - writer.WriteLine($"{symbol} = lazy_import('{codeUsing.Key.ToSnakeCase().Replace("._", ".")}.{symbol}')"); - writer.WriteLine(); - } - } - private static string GetAliasedSymbol(string symbol, string alias) - { - return string.IsNullOrEmpty(alias) ? symbol : $"{symbol} as {alias}"; - } - - /// - /// Returns the import path for the given using and import context namespace. - /// - /// The using to import into the current namespace context - /// The import symbol, it's alias if any and the import path - private static (string, string, string) GetImportPathForUsing(CodeUsing codeUsing) - { - var typeDef = codeUsing.Declaration?.TypeDefinition; - if (typeDef == null) - return (codeUsing.Name, codeUsing.Alias, ""); // it's relative to the folder, with no declaration or type definition (default failsafe) - - var importSymbol = typeDef.Name.ToSnakeCase(); - - var importPath = typeDef.GetImmediateParentOfType().Name; - return (importSymbol, codeUsing.Alias, importPath); + if (codeElement.Parent is CodeClass parent) + conventions.WriteShortDescription(parent.Documentation.Description, writer); } } From 4a8eb3779824520db870e5de68ef697ed10f1c32 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 06:57:19 +0300 Subject: [PATCH 08/36] Update property writer to add local imports for request builders with no parameters --- src/Kiota.Builder/Writers/Python/CodePropertyWriter.cs | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/src/Kiota.Builder/Writers/Python/CodePropertyWriter.cs b/src/Kiota.Builder/Writers/Python/CodePropertyWriter.cs index 36c089e87e..f121223e65 100644 --- a/src/Kiota.Builder/Writers/Python/CodePropertyWriter.cs +++ b/src/Kiota.Builder/Writers/Python/CodePropertyWriter.cs @@ -5,7 +5,11 @@ namespace Kiota.Builder.Writers.Python; public class CodePropertyWriter : BaseElementWriter { - public CodePropertyWriter(PythonConventionService conventionService) : base(conventionService) { } + private readonly CodeUsingWriter _codeUsingWriter; + public CodePropertyWriter(PythonConventionService conventionService, string clientNamespaceName) : base(conventionService) + { + _codeUsingWriter = new(clientNamespaceName); + } public override void WriteCodeElement(CodeProperty codeElement, LanguageWriter writer) { var returnType = conventions.GetTypeString(codeElement.Type, codeElement, true, writer); @@ -21,6 +25,7 @@ public override void WriteCodeElement(CodeProperty codeElement, LanguageWriter w writer.WriteLine($"def {codeElement.Name.ToSnakeCase()}(self) -> {returnType}:"); writer.IncreaseIndent(); conventions.WriteShortDescription(codeElement.Documentation.Description, writer); + _codeUsingWriter.WriteDeferredImport(parentClass, codeElement.Type.Name, writer); conventions.AddRequestBuilderBody(parentClass, returnType, writer); writer.CloseBlock(string.Empty); break; From d02f6b6738d6477c75b84e13f8e335099e643542 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 07:02:55 +0300 Subject: [PATCH 09/36] Add logic to write discriminator method body --- .../Writers/Python/CodeMethodWriter.cs | 30 ++++++++++++++++++- 1 file changed, 29 insertions(+), 1 deletion(-) diff --git a/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs b/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs index 2ad44548a6..7ccaae4665 100644 --- a/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs +++ b/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs @@ -8,8 +8,10 @@ namespace Kiota.Builder.Writers.Python; public class CodeMethodWriter : BaseElementWriter { - public CodeMethodWriter(PythonConventionService conventionService, bool usesBackingStore) : base(conventionService) + private readonly CodeUsingWriter _codeUsingWriter; + public CodeMethodWriter(PythonConventionService conventionService, string clientNamespaceName, bool usesBackingStore) : base(conventionService) { + _codeUsingWriter = new(clientNamespaceName); _usesBackingStore = usesBackingStore; } private readonly bool _usesBackingStore; @@ -73,6 +75,9 @@ public override void WriteCodeElement(CodeMethod codeElement, LanguageWriter wri case CodeMethodKind.QueryParametersMapper: WriteQueryParametersMapper(codeElement, parentClass, writer); break; + case CodeMethodKind.Factory: + WriteFactoryMethodBody(codeElement, parentClass, writer); + break; case CodeMethodKind.RawUrlConstructor: throw new InvalidOperationException("RawUrlConstructor is not supported in python"); case CodeMethodKind.RequestBuilderBackwardCompatibility: @@ -83,6 +88,29 @@ public override void WriteCodeElement(CodeMethod codeElement, LanguageWriter wri } writer.CloseBlock(string.Empty); } + private const string DiscriminatorMappingVarName = "mapping_value"; + private const string NodeVarName = "mapping_value_node"; + private void WriteFactoryMethodBody(CodeMethod codeElement, CodeClass parentClass, LanguageWriter writer) + { + var parseNodeParameter = codeElement.Parameters.OfKind(CodeParameterKind.ParseNode) ?? throw new InvalidOperationException("Factory method should have a ParseNode parameter"); + var writeDiscriminatorValueRead = parentClass.DiscriminatorInformation.ShouldWriteParseNodeCheck && !parentClass.DiscriminatorInformation.ShouldWriteDiscriminatorForIntersectionType; + if (writeDiscriminatorValueRead) + { + writer.WriteLine($"{NodeVarName} = {parseNodeParameter.Name.ToSnakeCase()}.get_child_node(\"{parentClass.DiscriminatorInformation.DiscriminatorPropertyName}\")"); + writer.StartBlock($"if {NodeVarName}:"); + writer.WriteLine($"{DiscriminatorMappingVarName} = {NodeVarName}.get_str_value()"); + foreach (var mappedType in parentClass.DiscriminatorInformation.DiscriminatorMappings.OrderBy(static x => x.Key)) + { + writer.StartBlock($"if {DiscriminatorMappingVarName} == \"{mappedType.Key}\":"); + var mappedTypeName = mappedType.Value.AllTypes.First().Name; + _codeUsingWriter.WriteDeferredImport(parentClass, mappedTypeName, writer); + writer.WriteLine($"return {mappedTypeName.ToSnakeCase()}.{mappedTypeName.ToFirstCharacterUpperCase()}()"); + writer.DecreaseIndent(); + } + writer.DecreaseIndent(); + } + writer.WriteLine($"return {parentClass.Name.ToFirstCharacterUpperCase()}()"); + } private void WriteIndexerBody(CodeMethod codeElement, CodeClass parentClass, string returnType, LanguageWriter writer) { if (parentClass.GetPropertyOfKind(CodePropertyKind.PathParameters) is CodeProperty pathParametersProperty && From fb1868acc7db8e8053721c5726fd7ad11bad1213 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 07:04:49 +0300 Subject: [PATCH 10/36] Write local imports in indexers and request builders with parameters --- src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs b/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs index 7ccaae4665..c83cf3b088 100644 --- a/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs +++ b/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs @@ -113,6 +113,7 @@ private void WriteFactoryMethodBody(CodeMethod codeElement, CodeClass parentClas } private void WriteIndexerBody(CodeMethod codeElement, CodeClass parentClass, string returnType, LanguageWriter writer) { + _codeUsingWriter.WriteDeferredImport(parentClass, codeElement.ReturnType.Name, writer); if (parentClass.GetPropertyOfKind(CodePropertyKind.PathParameters) is CodeProperty pathParametersProperty && codeElement.OriginalIndexer != null) conventions.AddParametersAssignment(writer, pathParametersProperty.Type, $"self.{pathParametersProperty.Name}", @@ -121,6 +122,7 @@ private void WriteIndexerBody(CodeMethod codeElement, CodeClass parentClass, str } private void WriteRequestBuilderWithParametersBody(CodeMethod codeElement, CodeClass parentClass, string returnType, LanguageWriter writer) { + _codeUsingWriter.WriteDeferredImport(parentClass, codeElement.ReturnType.Name, writer); var codePathParameters = codeElement.Parameters .Where(x => x.IsOfKind(CodeParameterKind.Path)); conventions.AddRequestBuilderBody(parentClass, returnType, writer, pathParameters: codePathParameters); From 6134aa4b874ac75522bf0ee3adb4121164fdfdf0 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 07:21:52 +0300 Subject: [PATCH 11/36] Add deferred imports for request executors and deserializers --- src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs b/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs index c83cf3b088..7624d4abb9 100644 --- a/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs +++ b/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs @@ -311,6 +311,7 @@ private static void WriteDefaultMethodBody(CodeMethod codeElement, LanguageWrite } private void WriteDeserializerBody(CodeMethod codeElement, CodeClass parentClass, LanguageWriter writer, bool inherits) { + _codeUsingWriter.WriteInternalImports(parentClass, writer); writer.WriteLine("fields = {"); writer.IncreaseIndent(); foreach (var otherProp in parentClass.GetPropertiesOfKind(CodePropertyKind.Custom).Where(static x => !x.ExistsInBaseType)) @@ -352,6 +353,7 @@ private void WriteRequestExecutorBody(CodeMethod codeElement, RequestParams requ var errorMappingVarName = "None"; if (codeElement.ErrorMappings.Any()) { + _codeUsingWriter.WriteInternalErrorMappingImports(parentClass, writer); errorMappingVarName = "error_mapping"; writer.WriteLine($"{errorMappingVarName}: Dict[str, ParsableFactory] = {{"); writer.IncreaseIndent(); @@ -365,6 +367,7 @@ private void WriteRequestExecutorBody(CodeMethod codeElement, RequestParams requ writer.IncreaseIndent(); writer.WriteLine("raise Exception(\"Http core is null\") "); writer.DecreaseIndent(); + _codeUsingWriter.WriteDeferredImport(parentClass, codeElement.ReturnType.Name, writer); writer.WriteLine($"return await self.request_adapter.{genericTypeForSendMethod}(request_info,{newFactoryParameter} {errorMappingVarName})"); } private string GetReturnTypeWithoutCollectionSymbol(CodeMethod codeElement, string fullTypeName) From bcde39fe5408e7d1bc7fdcd5615ec46fcea5df24 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 07:26:27 +0300 Subject: [PATCH 12/36] Use snake case to match variable name passed to request builder --- src/Kiota.Builder/Writers/Python/PythonConventionService.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Kiota.Builder/Writers/Python/PythonConventionService.cs b/src/Kiota.Builder/Writers/Python/PythonConventionService.cs index 6058ee5594..e2dc24bebd 100644 --- a/src/Kiota.Builder/Writers/Python/PythonConventionService.cs +++ b/src/Kiota.Builder/Writers/Python/PythonConventionService.cs @@ -25,7 +25,7 @@ internal void AddRequestBuilderBody(CodeClass parentClass, string returnType, La var urlTemplateParams = string.IsNullOrEmpty(urlTemplateVarName) && parentClass.GetPropertyOfKind(CodePropertyKind.PathParameters) is CodeProperty pathParametersProperty ? $"self.{pathParametersProperty.Name.ToSnakeCase()}" : urlTemplateVarName; - var pathParametersSuffix = !(pathParameters?.Any() ?? false) ? string.Empty : $", {string.Join(", ", pathParameters.Select(x => $"{x.Name}"))}"; + var pathParametersSuffix = !(pathParameters?.Any() ?? false) ? string.Empty : $", {string.Join(", ", pathParameters.Select(x => $"{x.Name.ToSnakeCase()}"))}"; if (parentClass.GetPropertyOfKind(CodePropertyKind.RequestAdapter) is CodeProperty requestAdapterProp) writer.WriteLine($"return {returnType}(self.{requestAdapterProp.Name.ToSnakeCase()}, {urlTemplateParams}{pathParametersSuffix})"); } From 504668b3f9dbc116a3fa830aee3d96da086fe1c7 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 07:29:10 +0300 Subject: [PATCH 13/36] Update python writer with missing parameters --- src/Kiota.Builder/Writers/Python/PythonWriter.cs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/Kiota.Builder/Writers/Python/PythonWriter.cs b/src/Kiota.Builder/Writers/Python/PythonWriter.cs index b0ebfc2987..52d69e61de 100644 --- a/src/Kiota.Builder/Writers/Python/PythonWriter.cs +++ b/src/Kiota.Builder/Writers/Python/PythonWriter.cs @@ -7,11 +7,11 @@ public PythonWriter(string rootPath, string clientNamespaceName, bool usesBackin { PathSegmenter = new PythonPathSegmenter(rootPath, clientNamespaceName); var conventionService = new PythonConventionService(); - AddOrReplaceCodeElementWriter(new CodeClassDeclarationWriter(conventionService)); + AddOrReplaceCodeElementWriter(new CodeClassDeclarationWriter(conventionService, clientNamespaceName)); AddOrReplaceCodeElementWriter(new CodeBlockEndWriter()); AddOrReplaceCodeElementWriter(new CodeEnumWriter(conventionService)); - AddOrReplaceCodeElementWriter(new CodeMethodWriter(conventionService, usesBackingStore)); - AddOrReplaceCodeElementWriter(new CodePropertyWriter(conventionService)); + AddOrReplaceCodeElementWriter(new CodeMethodWriter(conventionService, clientNamespaceName, usesBackingStore)); + AddOrReplaceCodeElementWriter(new CodePropertyWriter(conventionService, clientNamespaceName)); AddOrReplaceCodeElementWriter(new CodeTypeWriter(conventionService)); AddOrReplaceCodeElementWriter(new CodeNameSpaceWriter(conventionService)); } From 68788b512efcd4bd5341fc6f9456ecf23ec36de3 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 09:24:48 +0300 Subject: [PATCH 14/36] Add incluparentnamespace parameter when crawling tree --- src/Kiota.Builder/Refiners/CommonLanguageRefiner.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Kiota.Builder/Refiners/CommonLanguageRefiner.cs b/src/Kiota.Builder/Refiners/CommonLanguageRefiner.cs index 18bd95e58d..301a01fd2b 100644 --- a/src/Kiota.Builder/Refiners/CommonLanguageRefiner.cs +++ b/src/Kiota.Builder/Refiners/CommonLanguageRefiner.cs @@ -1012,7 +1012,7 @@ type.TypeDefinition is CodeClass modelClass && }); } } - CrawlTree(currentElement, x => AddDiscriminatorMappingsUsingsToParentClasses(x, parseNodeInterfaceName, addFactoryMethodImport, addUsings)); + CrawlTree(currentElement, x => AddDiscriminatorMappingsUsingsToParentClasses(x, parseNodeInterfaceName, addFactoryMethodImport, addUsings, includeParentNamespace)); } protected static void ReplaceLocalMethodsByGlobalFunctions(CodeElement currentElement, Func nameUpdateCallback, Func? usingsCallback, params CodeMethodKind[] kindsToReplace) { From a7f9e86df6c075134c90e704c49f3ef5cadbc1aa Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 19:33:15 +0300 Subject: [PATCH 15/36] Add code class declaration writer tests --- .../Python/CodeClassDeclarationWriterTests.cs | 35 +++++++++++++++---- 1 file changed, 28 insertions(+), 7 deletions(-) diff --git a/tests/Kiota.Builder.Tests/Writers/Python/CodeClassDeclarationWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Python/CodeClassDeclarationWriterTests.cs index b949009287..4d85582f39 100644 --- a/tests/Kiota.Builder.Tests/Writers/Python/CodeClassDeclarationWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/Python/CodeClassDeclarationWriterTests.cs @@ -13,6 +13,8 @@ public class CodeClassDeclarationWriterTests : IDisposable { private const string DefaultPath = "./"; private const string DefaultName = "name"; + + private const string ClientNamespaceName = "graph"; private readonly CodeNamespace root; private readonly CodeNamespace ns; private readonly StringWriter tw; @@ -23,7 +25,7 @@ public class CodeClassDeclarationWriterTests : IDisposable public CodeClassDeclarationWriterTests() { writer = LanguageWriter.GetLanguageWriter(GenerationLanguage.Python, DefaultPath, DefaultName); - codeElementWriter = new CodeClassDeclarationWriter(new PythonConventionService()); + codeElementWriter = new CodeClassDeclarationWriter(new PythonConventionService(), ClientNamespaceName); tw = new StringWriter(); writer.SetTextWriter(tw); root = CodeNamespace.InitRootNamespace(); @@ -42,7 +44,7 @@ public void Dispose() [Fact] public void Defensive() { - var codeClassDeclarationWriter = new CodeClassDeclarationWriter(new PythonConventionService()); + var codeClassDeclarationWriter = new CodeClassDeclarationWriter(new PythonConventionService(), ClientNamespaceName); Assert.Throws(() => codeClassDeclarationWriter.WriteCodeElement(null, writer)); var declaration = parentClass.StartBlock; Assert.Throws(() => codeClassDeclarationWriter.WriteCodeElement(declaration, null)); @@ -74,12 +76,29 @@ public void WritesImplementation() public void WritesInheritance() { var declaration = parentClass.StartBlock; + var interfaceDef = new CodeInterface + { + Name = "someInterface", + }; + ns.AddInterface(interfaceDef); + var nUsing = new CodeUsing + { + Name = "graph", + Declaration = new() + { + Name = "someInterface", + TypeDefinition = interfaceDef, + } + }; + declaration.AddUsings(nUsing); declaration.Inherits = new() { Name = "someInterface" }; codeElementWriter.WriteCodeElement(declaration, writer); var result = tw.ToString(); + Assert.Contains("if TYPE_CHECKING:", result); + Assert.Contains("from . import some_interface", result); Assert.Contains("(some_interface.SomeInterface):", result); } [Fact] @@ -129,7 +148,7 @@ public void WritesExternalImportsWithoutPath() Assert.Contains("import Objects", result); } [Fact] - public void WritesInternalImportsSubNamespace() + public void WritesConditionalInternalImportsSubNamespace() { var declaration = parentClass.StartBlock; var subNS = ns.AddNamespace($"{ns.Name}.messages"); @@ -150,11 +169,12 @@ public void WritesInternalImportsSubNamespace() declaration.AddUsings(nUsing); codeElementWriter.WriteCodeElement(declaration, writer); var result = tw.ToString(); - Assert.Contains("message = lazy_import('graphtests.models.messages.message')", result); + Assert.Contains("if TYPE_CHECKING:", result); + Assert.Contains("from .messages import message", result); } [Fact] - public void WritesInternalImportsSameNamespace() + public void WritesConditionalInternalImportsSameNamespace() { var declaration = parentClass.StartBlock; var messageClassDef = new CodeClass @@ -174,7 +194,8 @@ public void WritesInternalImportsSameNamespace() declaration.AddUsings(nUsing); codeElementWriter.WriteCodeElement(declaration, writer); var result = tw.ToString(); - Assert.Contains("message = lazy_import('graphtests.models.message')", result); + Assert.Contains("if TYPE_CHECKING:", result); + Assert.Contains("from . import message", result); } [Fact] public void WritesInternalImportsNoTypeDef() @@ -191,6 +212,6 @@ public void WritesInternalImportsNoTypeDef() declaration.AddUsings(nUsing); codeElementWriter.WriteCodeElement(declaration, writer); var result = tw.ToString(); - Assert.DoesNotContain("message = lazy_import('graphtests.models.message')", result); + Assert.DoesNotContain("from . import message", result); } } From 2860b65a96c84713fb707bdf6f8b309b5e063065 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 19:56:56 +0300 Subject: [PATCH 16/36] Add property writer tests --- .../Writers/Python/CodePropertyWriterTests.cs | 23 +++++++++++++++++-- 1 file changed, 21 insertions(+), 2 deletions(-) diff --git a/tests/Kiota.Builder.Tests/Writers/Python/CodePropertyWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Python/CodePropertyWriterTests.cs index 640a84c57d..97f697ccfb 100644 --- a/tests/Kiota.Builder.Tests/Writers/Python/CodePropertyWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/Python/CodePropertyWriterTests.cs @@ -15,6 +15,7 @@ public class CodePropertyWriterTests : IDisposable private readonly LanguageWriter writer; private readonly CodeProperty property; private readonly CodeClass parentClass; + private readonly CodeNamespace ns; private const string PropertyName = "propertyName"; private const string TypeName = "Somecustomtype"; public CodePropertyWriterTests() @@ -23,11 +24,12 @@ public CodePropertyWriterTests() tw = new StringWriter(); writer.SetTextWriter(tw); var root = CodeNamespace.InitRootNamespace(); + ns = root.AddNamespace("graphtests.models"); parentClass = new CodeClass { Name = "parentClass" }; - root.AddClass(parentClass); + ns.AddClass(parentClass); property = new CodeProperty { Name = PropertyName, @@ -36,6 +38,22 @@ public CodePropertyWriterTests() Name = TypeName } }; + var subNS = ns.AddNamespace($"{ns.Name}.somecustomtype"); + var somecustomtypeClassDef = new CodeClass + { + Name = "Somecustomtype", + }; + subNS.AddClass(somecustomtypeClassDef); + var nUsing = new CodeUsing + { + Name = somecustomtypeClassDef.Name, + Declaration = new() + { + Name = somecustomtypeClassDef.Name, + TypeDefinition = somecustomtypeClassDef, + } + }; + parentClass.StartBlock.AddUsings(nUsing); parentClass.AddProperty(property, new() { Name = "pathParameters", @@ -67,8 +85,9 @@ public void WritesRequestBuilder() writer.Write(property); var result = tw.ToString(); Assert.Contains("@property", result); - Assert.Contains("def property_name(", result); + Assert.Contains("def property_name(self) -> somecustomtype.Somecustomtype:", result); Assert.Contains("This is a request builder", result); + Assert.Contains("from .somecustomtype import somecustomtype", result); Assert.Contains($"return {TypeName.ToLower()}.{TypeName}(", result); Assert.Contains("self.request_adapter", result); Assert.Contains("self.path_parameters", result); From 0fe2f5f8b2729375fc5312a739a45311ed19af9c Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 9 Mar 2023 20:21:48 +0300 Subject: [PATCH 17/36] Add codemethodwriter tests --- .../Writers/Python/CodeMethodWriterTests.cs | 177 +++++++++++++++++- 1 file changed, 176 insertions(+), 1 deletion(-) diff --git a/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs index 416fbf99c7..6e69c3ae73 100644 --- a/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs @@ -17,7 +17,9 @@ public class CodeMethodWriterTests : IDisposable private readonly LanguageWriter writer; private readonly CodeMethod method; private readonly CodeClass parentClass; + private readonly CodeClass childClass; private readonly CodeNamespace root; + private const string ClientNamespaceName = "graph"; private const string MethodName = "method_name"; private const string ReturnTypeName = "Somecustomtype"; private const string MethodDescription = "some description"; @@ -34,6 +36,11 @@ public CodeMethodWriterTests() Name = "parentClass" }; root.AddClass(parentClass); + childClass = new CodeClass + { + Name = "childClass" + }; + root.AddClass(childClass); method = new CodeMethod { Name = MethodName, @@ -489,7 +496,7 @@ public void WritesMethodSyncDescription() [Fact] public void Defensive() { - var codeMethodWriter = new CodeMethodWriter(new PythonConventionService(), false); + var codeMethodWriter = new CodeMethodWriter(new PythonConventionService(), ClientNamespaceName, false); Assert.Throws(() => codeMethodWriter.WriteCodeElement(null, writer)); Assert.Throws(() => codeMethodWriter.WriteCodeElement(method, null)); var originalParent = method.Parent; @@ -538,6 +545,21 @@ public void DoesNotAddAsyncInformationOnSyncMethods() public void WritesFactoryMethods() { method.Kind = CodeMethodKind.Factory; + method.AddParameter(new CodeParameter + { + Name = "parseNode", + Kind = CodeParameterKind.ParseNode, + Type = new CodeType + { + Name = "ParseNode", + TypeDefinition = new CodeClass + { + Name = "ParseNode", + }, + IsExternal = true, + }, + Optional = false, + }); writer.Write(method); var result = tw.ToString(); Assert.Contains("@staticmethod", result); @@ -545,6 +567,159 @@ public void WritesFactoryMethods() } [Fact] + public void WritesModelFactoryBody() + { + parentClass.Kind = CodeClassKind.Model; + childClass.Kind = CodeClassKind.Model; + childClass.StartBlock.Inherits = new CodeType + { + Name = "parentClass", + TypeDefinition = parentClass, + }; + method.Kind = CodeMethodKind.Factory; + method.ReturnType = new CodeType + { + Name = "parentClass", + TypeDefinition = parentClass, + }; + method.IsStatic = true; + parentClass.DiscriminatorInformation.AddDiscriminatorMapping("ns.childclass", new CodeType + { + Name = "childClass", + TypeDefinition = childClass, + }); + parentClass.DiscriminatorInformation.DiscriminatorPropertyName = "@odata.type"; + method.AddParameter(new CodeParameter + { + Name = "parseNode", + Kind = CodeParameterKind.ParseNode, + Type = new CodeType + { + Name = "ParseNode", + TypeDefinition = new CodeClass + { + Name = "ParseNode", + }, + IsExternal = true, + }, + Optional = false, + }); + writer.Write(method); + var result = tw.ToString(); + Assert.Contains("mapping_value_node = parse_node.get_child_node(\"@odata.type\")", result); + Assert.Contains("if mapping_value_node:", result); + Assert.Contains("mapping_value = mapping_value_node.get_str_value()", result); + Assert.Contains("if mapping_value == \"ns.childclass\"", result); + Assert.Contains("return child_class.ChildClass()", result); + Assert.Contains("return ParentClass()", result); + } + [Fact] + public void DoesntWriteFactoryConditionalsOnMissingParameter() + { + parentClass.Kind = CodeClassKind.Model; + childClass.Kind = CodeClassKind.Model; + childClass.StartBlock.Inherits = new CodeType + { + Name = "parentClass", + TypeDefinition = parentClass, + }; + method.Kind = CodeMethodKind.Factory; + method.ReturnType = new CodeType + { + Name = "parentClass", + TypeDefinition = parentClass, + }; + method.IsStatic = true; + parentClass.DiscriminatorInformation.AddDiscriminatorMapping("ns.childclass", new CodeType + { + Name = "childClass", + TypeDefinition = childClass, + }); + parentClass.DiscriminatorInformation.DiscriminatorPropertyName = "@odata.type"; + Assert.Throws(() => writer.Write(method)); + } + [Fact] + public void DoesntWriteFactoryConditionalsOnEmptyPropertyName() + { + parentClass.Kind = CodeClassKind.Model; + childClass.Kind = CodeClassKind.Model; + childClass.StartBlock.Inherits = new CodeType + { + Name = "parentClass", + TypeDefinition = parentClass, + }; + method.Kind = CodeMethodKind.Factory; + method.ReturnType = new CodeType + { + Name = "parentClass", + TypeDefinition = parentClass, + }; + method.IsStatic = true; + parentClass.DiscriminatorInformation.AddDiscriminatorMapping("ns.childclass", new CodeType + { + Name = "childClass", + TypeDefinition = childClass, + }); + parentClass.DiscriminatorInformation.DiscriminatorPropertyName = string.Empty; + method.AddParameter(new CodeParameter + { + Name = "parseNode", + Kind = CodeParameterKind.ParseNode, + Type = new CodeType + { + Name = "ParseNode", + TypeDefinition = new CodeClass + { + Name = "ParseNode", + }, + IsExternal = true, + }, + Optional = false, + }); + writer.Write(method); + var result = tw.ToString(); + Assert.DoesNotContain("mapping_value_node = parse_node.get_child_node(\"@odata.type\")", result); + Assert.DoesNotContain("if mapping_value_node:", result); + Assert.DoesNotContain("mapping_value = mapping_value_node.get_str_value", result); + Assert.DoesNotContain("if mapping_value == \"ns.childclass\"", result); + Assert.Contains("return ParentClass()", result); + } + [Fact] + public void DoesntWriteFactorySwitchOnEmptyMappings() + { + parentClass.Kind = CodeClassKind.Model; + method.Kind = CodeMethodKind.Factory; + method.ReturnType = new CodeType + { + Name = "parentClass", + TypeDefinition = parentClass, + }; + method.IsStatic = true; + parentClass.DiscriminatorInformation.DiscriminatorPropertyName = "@odata.type"; + method.AddParameter(new CodeParameter + { + Name = "parseNode", + Kind = CodeParameterKind.ParseNode, + Type = new CodeType + { + Name = "ParseNode", + TypeDefinition = new CodeClass + { + Name = "ParseNode", + }, + IsExternal = true, + }, + Optional = false, + }); + writer.Write(method); + var result = tw.ToString(); + Assert.DoesNotContain("mapping_value_node = parse_node.get_child_node(\"@odata.type\")", result); + Assert.DoesNotContain("if mapping_value_node:", result); + Assert.DoesNotContain("mapping_value = mapping_value_node.get_str_value", result); + Assert.DoesNotContain("if mapping_value == \"ns.childclass\"", result); + Assert.Contains("return ParentClass()", result); + } + [Fact] public void WritesPublicMethodByDefault() { writer.Write(method); From b858fa4913e8367fb2fd81fe14cd31cbaa72242f Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Fri, 10 Mar 2023 18:16:51 +0300 Subject: [PATCH 18/36] Fix formatting issues --- src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs | 2 +- src/Kiota.Builder/Writers/Python/PythonRelativeImportManager.cs | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs b/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs index b29212923f..5416c1cbd0 100644 --- a/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs +++ b/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs @@ -1,4 +1,4 @@ -using System.Linq; +using System.Linq; using Kiota.Builder.CodeDOM; using Kiota.Builder.Extensions; diff --git a/src/Kiota.Builder/Writers/Python/PythonRelativeImportManager.cs b/src/Kiota.Builder/Writers/Python/PythonRelativeImportManager.cs index 27cafa4f83..2cda2dfa3b 100644 --- a/src/Kiota.Builder/Writers/Python/PythonRelativeImportManager.cs +++ b/src/Kiota.Builder/Writers/Python/PythonRelativeImportManager.cs @@ -1,4 +1,4 @@ -using System; +using System; using System.Collections.Generic; using System.Linq; From ef293544d9955978956e2a1c6688e246450a5763 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Fri, 10 Mar 2023 20:31:02 +0300 Subject: [PATCH 19/36] Update method writer tests --- .../Writers/Python/CodeMethodWriterTests.cs | 73 ++++++++++++++++++- 1 file changed, 71 insertions(+), 2 deletions(-) diff --git a/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs index 6e69c3ae73..d688e2ad1a 100644 --- a/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs @@ -41,6 +41,21 @@ public CodeMethodWriterTests() Name = "childClass" }; root.AddClass(childClass); + var returnTypeClassDef = new CodeClass + { + Name = ReturnTypeName, + }; + root.AddClass(returnTypeClassDef); + var nUsing = new CodeUsing + { + Name = returnTypeClassDef.Name, + Declaration = new() + { + Name = returnTypeClassDef.Name, + TypeDefinition = returnTypeClassDef, + } + }; + parentClass.StartBlock.AddUsings(nUsing); method = new CodeMethod { Name = MethodName, @@ -203,6 +218,20 @@ private void AddInheritanceClass() Name = "someParentClass" }; } + + private void AddCodeUsings() + { + var nUsing = new CodeUsing + { + Name = childClass.Name, + Declaration = new() + { + Name = childClass.Name, + TypeDefinition = childClass, + } + }; + parentClass.StartBlock.AddUsings(nUsing); + } private void AddRequestBodyParameters(bool useComplexTypeForBody = false) { var stringType = new CodeType @@ -287,26 +316,59 @@ public void WritesRequestExecutorBody() var error4XX = root.AddClass(new CodeClass { Name = "Error4XX", + IsErrorDefinition = true + }).First(); var error5XX = root.AddClass(new CodeClass { Name = "Error5XX", + IsErrorDefinition = true }).First(); var error401 = root.AddClass(new CodeClass { Name = "Error401", + IsErrorDefinition = true }).First(); + parentClass.StartBlock.AddUsings(new() + { + Name = error401.Name, + Declaration = new() + { + Name = error401.Name, + TypeDefinition = error401, + } + }, + new() + { + Name = error5XX.Name, + Declaration = new() + { + Name = error5XX.Name, + TypeDefinition = error5XX, + } + }, + new() + { + Name = error4XX.Name, + Declaration = new() + { + Name = error4XX.Name, + TypeDefinition = error4XX, + } + }); method.AddErrorMapping("4XX", new CodeType { Name = "Error4XX", TypeDefinition = error4XX }); method.AddErrorMapping("5XX", new CodeType { Name = "Error5XX", TypeDefinition = error5XX }); - method.AddErrorMapping("403", new CodeType { Name = "Error403", TypeDefinition = error401 }); + method.AddErrorMapping("401", new CodeType { Name = "Error401", TypeDefinition = error401 }); AddRequestBodyParameters(); writer.Write(method); var result = tw.ToString(); Assert.Contains("request_info", result); + Assert.Contains("from . import error4_x_x, error401, error5_x_x", result); Assert.Contains("error_mapping: Dict[str, ParsableFactory] =", result); Assert.Contains("\"4XX\": error4_x_x.Error4XX", result); Assert.Contains("\"5XX\": error5_x_x.Error5XX", result); - Assert.Contains("\"403\": error403.Error403", result); + Assert.Contains("\"401\": error401.Error401", result); + Assert.Contains("from . import somecustomtype", result); Assert.Contains("send_async", result); Assert.Contains("raise Exception", result); } @@ -330,6 +392,7 @@ public void WritesRequestExecutorBodyForCollections() AddRequestBodyParameters(); writer.Write(method); var result = tw.ToString(); + Assert.Contains("from . import somecustomtype", result); Assert.Contains("send_collection_async", result); } [Fact] @@ -388,6 +451,7 @@ public void WritesInheritedDeSerializerBody() AddInheritanceClass(); writer.Write(method); var result = tw.ToString(); + Assert.Contains("from . import somecustomtype", result); Assert.Contains("super_fields = super()", result); Assert.Contains("fields.update(super_fields)", result); Assert.Contains("return fields", result); @@ -400,6 +464,7 @@ public void WritesDeSerializerBody() AddSerializationProperties(); writer.Write(method); var result = tw.ToString(); + Assert.Contains("from . import somecustomtype", result); Assert.Contains("get_str_value", result); Assert.Contains("get_int_value", result); Assert.Contains("get_float_value", result); @@ -589,6 +654,7 @@ public void WritesModelFactoryBody() TypeDefinition = childClass, }); parentClass.DiscriminatorInformation.DiscriminatorPropertyName = "@odata.type"; + AddCodeUsings(); method.AddParameter(new CodeParameter { Name = "parseNode", @@ -610,6 +676,7 @@ public void WritesModelFactoryBody() Assert.Contains("if mapping_value_node:", result); Assert.Contains("mapping_value = mapping_value_node.get_str_value()", result); Assert.Contains("if mapping_value == \"ns.childclass\"", result); + Assert.Contains("from . import child_class", result); Assert.Contains("return child_class.ChildClass()", result); Assert.Contains("return ParentClass()", result); } @@ -753,6 +820,7 @@ public void WritesIndexer() Name = "string", } }; + writer.Write(method); var result = tw.ToString(); Assert.Contains("self.request_adapter", result); @@ -776,6 +844,7 @@ public void WritesPathParameterRequestBuilder() }); writer.Write(method); var result = tw.ToString(); + Assert.Contains("from . import somecustomtype", result); Assert.Contains("self.request_adapter", result); Assert.Contains("self.path_parameters", result); Assert.Contains("path_param", result); From cf7427edf437eca39c5dfb75a7a24bef4ec3d7dc Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Fri, 10 Mar 2023 20:55:09 +0300 Subject: [PATCH 20/36] Add codeusing writer tests --- .../Writers/Python/CodeUsingWriterTests.cs | 70 +++++++++++++++++++ 1 file changed, 70 insertions(+) create mode 100644 tests/Kiota.Builder.Tests/Writers/Python/CodeUsingWriterTests.cs diff --git a/tests/Kiota.Builder.Tests/Writers/Python/CodeUsingWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Python/CodeUsingWriterTests.cs new file mode 100644 index 0000000000..6f51ba6ac9 --- /dev/null +++ b/tests/Kiota.Builder.Tests/Writers/Python/CodeUsingWriterTests.cs @@ -0,0 +1,70 @@ +using System.IO; +using System.Linq; +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Writers; +using Kiota.Builder.Writers.Python; +using Xunit; + +namespace Kiota.Builder.Tests.Writers.Python; + +public class CodeUsingWriterTests +{ + private const string DefaultPath = "./"; + private const string DefaultName = "name"; + private readonly LanguageWriter writer; + private readonly StringWriter tw; + private readonly CodeNamespace root; + public CodeUsingWriterTests() + { + writer = LanguageWriter.GetLanguageWriter(GenerationLanguage.TypeScript, DefaultPath, DefaultName); + tw = new StringWriter(); + writer.SetTextWriter(tw); + root = CodeNamespace.InitRootNamespace(); + } + [Fact] + public void WritesAliasedSymbol() + { + var usingWriter = new CodeUsingWriter("foo"); + var codeClass = root.AddClass(new CodeClass + { + Name = "bar", + }).First(); + var us = new CodeUsing + { + Name = "bar", + Alias = "baz", + Declaration = new CodeType + { + Name = "bar", + TypeDefinition = codeClass, + }, + }; + codeClass.AddUsing(us); + usingWriter.WriteInternalImports(codeClass, writer); + var result = tw.ToString(); + Assert.Contains("from . import bar as baz", result); + } + [Fact] + public void DoesntAliasRegularSymbols() + { + var usingWriter = new CodeUsingWriter("foo"); + var codeClass = root.AddClass(new CodeClass + { + Name = "bar", + + }).First(); + var us = new CodeUsing + { + Name = "bar", + Declaration = new CodeType + { + Name = "bar", + TypeDefinition = codeClass, + }, + }; + codeClass.AddUsing(us); + usingWriter.WriteInternalImports(codeClass, writer); + var result = tw.ToString(); + Assert.Contains("from . import bar", result); + } +} From 0f131201169defb1ca6b2fc7ed4d369f2817496c Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Fri, 10 Mar 2023 20:56:27 +0300 Subject: [PATCH 21/36] Fix formatting --- .../Kiota.Builder.Tests/Writers/Python/CodeUsingWriterTests.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/Kiota.Builder.Tests/Writers/Python/CodeUsingWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Python/CodeUsingWriterTests.cs index 6f51ba6ac9..6e42efbe03 100644 --- a/tests/Kiota.Builder.Tests/Writers/Python/CodeUsingWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/Python/CodeUsingWriterTests.cs @@ -1,4 +1,4 @@ -using System.IO; +using System.IO; using System.Linq; using Kiota.Builder.CodeDOM; using Kiota.Builder.Writers; From 07f5d0c06112005e0e7af3b54c607ea199b6e08e Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Fri, 10 Mar 2023 21:01:05 +0300 Subject: [PATCH 22/36] Add changelog entry --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 1944e238d4..5508b67f5f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -14,6 +14,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [1.0.1] - 2023-03-11 - Fixed a bug where double would not be mapped properly. +- Changed python import mechanism to facilitate code completion. [#2380](https://github.com/microsoft/kiota/issues/2380) +- Fixed a bug where discriminator methods were missing possible types in Python [#2381](https://github.com/microsoft/kiota/issues/2381) ## [1.0.0] - 2023-03-10 From 7a2d47c6e1961053b2c9a151ef42cb15a5c19818 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Mon, 13 Mar 2023 20:30:36 +0300 Subject: [PATCH 23/36] Add type annotation for discriminator fields variable --- src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs b/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs index 7624d4abb9..265a353e73 100644 --- a/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs +++ b/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs @@ -312,7 +312,7 @@ private static void WriteDefaultMethodBody(CodeMethod codeElement, LanguageWrite private void WriteDeserializerBody(CodeMethod codeElement, CodeClass parentClass, LanguageWriter writer, bool inherits) { _codeUsingWriter.WriteInternalImports(parentClass, writer); - writer.WriteLine("fields = {"); + writer.WriteLine("fields: Dict[str, function] = {"); writer.IncreaseIndent(); foreach (var otherProp in parentClass.GetPropertiesOfKind(CodePropertyKind.Custom).Where(static x => !x.ExistsInBaseType)) { From e3ac57a0f6b64df3171f3042bba7ea9a2390f69c Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Mon, 13 Mar 2023 20:36:20 +0300 Subject: [PATCH 24/36] Update test --- .../Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs | 1 + 1 file changed, 1 insertion(+) diff --git a/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs index d688e2ad1a..dc91674605 100644 --- a/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs @@ -465,6 +465,7 @@ public void WritesDeSerializerBody() writer.Write(method); var result = tw.ToString(); Assert.Contains("from . import somecustomtype", result); + Assert.Contains("fields: Dict[str, function] =", result); Assert.Contains("get_str_value", result); Assert.Contains("get_int_value", result); Assert.Contains("get_float_value", result); From d02fd21169d46266fe204447e130a60b8e0a873a Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Mon, 13 Mar 2023 23:06:15 +0300 Subject: [PATCH 25/36] Fix fields type hint --- src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs | 2 +- .../Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs b/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs index 265a353e73..e261121b22 100644 --- a/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs +++ b/src/Kiota.Builder/Writers/Python/CodeMethodWriter.cs @@ -312,7 +312,7 @@ private static void WriteDefaultMethodBody(CodeMethod codeElement, LanguageWrite private void WriteDeserializerBody(CodeMethod codeElement, CodeClass parentClass, LanguageWriter writer, bool inherits) { _codeUsingWriter.WriteInternalImports(parentClass, writer); - writer.WriteLine("fields: Dict[str, function] = {"); + writer.WriteLine("fields: Dict[str, Callable[[Any], None]] = {"); writer.IncreaseIndent(); foreach (var otherProp in parentClass.GetPropertiesOfKind(CodePropertyKind.Custom).Where(static x => !x.ExistsInBaseType)) { diff --git a/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs index dc91674605..a7840af2ae 100644 --- a/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs @@ -465,7 +465,7 @@ public void WritesDeSerializerBody() writer.Write(method); var result = tw.ToString(); Assert.Contains("from . import somecustomtype", result); - Assert.Contains("fields: Dict[str, function] =", result); + Assert.Contains("fields: Dict[str, Callable[[Any], None]] =", result); Assert.Contains("get_str_value", result); Assert.Contains("get_int_value", result); Assert.Contains("get_float_value", result); From 9e2cc792ecfd3b4273265d2cd09db14e938dfe35 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Thu, 16 Mar 2023 18:01:43 +0300 Subject: [PATCH 26/36] Bump test coverage in classdeclarationwriter --- .../Writers/Python/CodeClassDeclarationWriterTests.cs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/tests/Kiota.Builder.Tests/Writers/Python/CodeClassDeclarationWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Python/CodeClassDeclarationWriterTests.cs index 4d85582f39..da4a13f080 100644 --- a/tests/Kiota.Builder.Tests/Writers/Python/CodeClassDeclarationWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/Python/CodeClassDeclarationWriterTests.cs @@ -54,6 +54,7 @@ public void WritesSimpleDeclaration() { codeElementWriter.WriteCodeElement(parentClass.StartBlock, writer); var result = tw.ToString(); + Assert.DoesNotContain("@dataclass", result); Assert.Contains("class ParentClass()", result); } [Fact] @@ -70,6 +71,7 @@ public void WritesImplementation() }); codeElementWriter.WriteCodeElement(declaration, writer); var result = tw.ToString(); + Assert.DoesNotContain("()", result); Assert.Contains("(SecondInterface, SomeInterface):", result); } [Fact] From 7cec2a2a0b74551641c8ddd2bba2990cc1fd6f55 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Tue, 21 Mar 2023 17:39:04 +0300 Subject: [PATCH 27/36] Add python code element order comparer --- src/Kiota.Builder/CodeElementOrderComparer.cs | 6 ++++-- .../CodeElementOrderComparerPython.cs | 19 +++++++++++++++++++ 2 files changed, 23 insertions(+), 2 deletions(-) create mode 100644 src/Kiota.Builder/CodeElementOrderComparerPython.cs diff --git a/src/Kiota.Builder/CodeElementOrderComparer.cs b/src/Kiota.Builder/CodeElementOrderComparer.cs index 619aecc450..d2d64d40d6 100644 --- a/src/Kiota.Builder/CodeElementOrderComparer.cs +++ b/src/Kiota.Builder/CodeElementOrderComparer.cs @@ -36,8 +36,10 @@ protected virtual int GetTypeFactor(CodeElement element) _ => 0, }; } - private static readonly int methodKindWeight = 10; - protected static int GetMethodKindFactor(CodeElement element) + + protected virtual int methodKindWeight { get; } = 10; + + protected virtual int GetMethodKindFactor(CodeElement element) { if (element is CodeMethod method) return method.Kind switch diff --git a/src/Kiota.Builder/CodeElementOrderComparerPython.cs b/src/Kiota.Builder/CodeElementOrderComparerPython.cs new file mode 100644 index 0000000000..00a9bfa0ae --- /dev/null +++ b/src/Kiota.Builder/CodeElementOrderComparerPython.cs @@ -0,0 +1,19 @@ +using Kiota.Builder.CodeDOM; + +namespace Kiota.Builder; +public class CodeElementOrderComparerPython : CodeElementOrderComparer +{ + protected override int methodKindWeight { get; } = 200; + protected override int GetMethodKindFactor(CodeElement element) + { + if (element is CodeMethod method) + return method.Kind switch + { + CodeMethodKind.ClientConstructor => 1, + CodeMethodKind.Constructor => 0, + CodeMethodKind.RawUrlConstructor => 3, + _ => 2, + }; + return 0; + } +} From 17078da77a16689da4eabb81c2cd89ab42e15af0 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Tue, 21 Mar 2023 17:39:25 +0300 Subject: [PATCH 28/36] Add code renderer for python --- src/Kiota.Builder/CodeRenderers/CodeRenderer.cs | 5 +++-- src/Kiota.Builder/CodeRenderers/PythonCodeRenderer.cs | 9 +++++++++ 2 files changed, 12 insertions(+), 2 deletions(-) create mode 100644 src/Kiota.Builder/CodeRenderers/PythonCodeRenderer.cs diff --git a/src/Kiota.Builder/CodeRenderers/CodeRenderer.cs b/src/Kiota.Builder/CodeRenderers/CodeRenderer.cs index a60d5a9ba9..fd20967c94 100644 --- a/src/Kiota.Builder/CodeRenderers/CodeRenderer.cs +++ b/src/Kiota.Builder/CodeRenderers/CodeRenderer.cs @@ -15,11 +15,11 @@ namespace Kiota.Builder.CodeRenderers; /// public class CodeRenderer { - public CodeRenderer(GenerationConfiguration configuration) + public CodeRenderer(GenerationConfiguration configuration, CodeElementOrderComparer? elementComparer = null) { ArgumentNullException.ThrowIfNull(configuration); _configuration = configuration; - _rendererElementComparer = configuration.ShouldRenderMethodsOutsideOfClasses ? new CodeElementOrderComparerWithExternalMethods() : new CodeElementOrderComparer(); + _rendererElementComparer = elementComparer ?? (configuration.ShouldRenderMethodsOutsideOfClasses ? new CodeElementOrderComparerWithExternalMethods() : new CodeElementOrderComparer()); } public async Task RenderCodeNamespaceToSingleFileAsync(LanguageWriter writer, CodeElement codeElement, string outputFile, CancellationToken cancellationToken) { @@ -91,6 +91,7 @@ public static CodeRenderer GetCodeRender(GenerationConfiguration config) => config.Language switch { GenerationLanguage.TypeScript => new TypeScriptCodeRenderer(config), + GenerationLanguage.Python => new PythonCodeRenderer(config), _ => new CodeRenderer(config), }; diff --git a/src/Kiota.Builder/CodeRenderers/PythonCodeRenderer.cs b/src/Kiota.Builder/CodeRenderers/PythonCodeRenderer.cs new file mode 100644 index 0000000000..c207545a3c --- /dev/null +++ b/src/Kiota.Builder/CodeRenderers/PythonCodeRenderer.cs @@ -0,0 +1,9 @@ +using System.Linq; +using Kiota.Builder.CodeDOM; +using Kiota.Builder.Configuration; + +namespace Kiota.Builder.CodeRenderers; +public class PythonCodeRenderer : CodeRenderer +{ + public PythonCodeRenderer(GenerationConfiguration configuration) : base(configuration, new CodeElementOrderComparerPython()) { } +} From b3d0f1b20a70a792e0e8d3e9e5037813f8420599 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Wed, 22 Mar 2023 00:33:54 +0300 Subject: [PATCH 29/36] Write methods before properties --- .../CodeElementOrderComparerPython.cs | 15 +++++++++++++++ 1 file changed, 15 insertions(+) diff --git a/src/Kiota.Builder/CodeElementOrderComparerPython.cs b/src/Kiota.Builder/CodeElementOrderComparerPython.cs index 00a9bfa0ae..a15461e36e 100644 --- a/src/Kiota.Builder/CodeElementOrderComparerPython.cs +++ b/src/Kiota.Builder/CodeElementOrderComparerPython.cs @@ -3,6 +3,21 @@ namespace Kiota.Builder; public class CodeElementOrderComparerPython : CodeElementOrderComparer { + protected override int GetTypeFactor(CodeElement element) + { + return element switch + { + CodeUsing => 1, + ClassDeclaration => 2, + InterfaceDeclaration => 3, + CodeMethod => 4, + CodeIndexer => 5, + CodeProperty => 6, + CodeClass => 7, + BlockEnd => 8, + _ => 0, + }; + } protected override int methodKindWeight { get; } = 200; protected override int GetMethodKindFactor(CodeElement element) { From 71bf77542911848189bd89476afcfc2483b664cb Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Wed, 22 Mar 2023 00:34:21 +0300 Subject: [PATCH 30/36] Add python element comparer tests --- .../CodeDOM/CodeElementComparerPythonTests.cs | 89 +++++++++++++++++++ 1 file changed, 89 insertions(+) create mode 100644 tests/Kiota.Builder.Tests/CodeDOM/CodeElementComparerPythonTests.cs diff --git a/tests/Kiota.Builder.Tests/CodeDOM/CodeElementComparerPythonTests.cs b/tests/Kiota.Builder.Tests/CodeDOM/CodeElementComparerPythonTests.cs new file mode 100644 index 0000000000..04818ffb50 --- /dev/null +++ b/tests/Kiota.Builder.Tests/CodeDOM/CodeElementComparerPythonTests.cs @@ -0,0 +1,89 @@ +using System; +using System.Collections.Generic; + +using Kiota.Builder.CodeDOM; + +using Xunit; + +namespace Kiota.Builder.Tests.CodeDOM; +public class CodeElementComparerPythonTests +{ + [Fact] + public void OrdersWithMethodWithinClass() + { + var root = CodeNamespace.InitRootNamespace(); + var comparer = new CodeElementOrderComparerPython(); + var codeClass = new CodeClass + { + Name = "Class" + }; + root.AddClass(codeClass); + var method = new CodeMethod + { + Name = "Method", + ReturnType = new CodeType + { + Name = "string" + } + }; + codeClass.AddMethod(method); + method.AddParameter(new CodeParameter + { + Name = "param", + Type = new CodeType + { + Name = "string" + } + }); + var dataSet = new List> { + new(null, null, 0), + new(null, new CodeClass(), -1), + new(new CodeClass(), null, 1), + new(new CodeUsing(), new CodeProperty() { + Name = "prop", + Type = new CodeType { + Name = "string" + } + }, -1100), + new(new CodeIndexer() { + ReturnType = new CodeType { + Name = "string" + }, + IndexType = new CodeType { + Name = "string" + } + }, new CodeProperty() { + Name = "prop", + Type = new CodeType { + Name = "string" + } + }, -1100), + new(method, new CodeProperty() { + Name = "prop", + Type = new CodeType { + Name = "string" + } + }, -899), + new(method, codeClass, -699), + new(new CodeMethod() { + Kind = CodeMethodKind.Constructor, + ReturnType = new CodeType + { + Name = "null", + } + }, method, -301), + new(new CodeMethod() { + Kind = CodeMethodKind.ClientConstructor, + ReturnType = new CodeType + { + Name = "null", + } + }, method, -301), + + }; + foreach (var dataEntry in dataSet) + { + Assert.Equal(dataEntry.Item3, comparer.Compare(dataEntry.Item1, dataEntry.Item2)); + } + } +} From 3b4015720bc7903b7e8c953c7f3289f6c175bff3 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Wed, 22 Mar 2023 00:45:35 +0300 Subject: [PATCH 31/36] Fix formatting --- .../CodeDOM/CodeElementComparerPythonTests.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/Kiota.Builder.Tests/CodeDOM/CodeElementComparerPythonTests.cs b/tests/Kiota.Builder.Tests/CodeDOM/CodeElementComparerPythonTests.cs index 04818ffb50..c09496381e 100644 --- a/tests/Kiota.Builder.Tests/CodeDOM/CodeElementComparerPythonTests.cs +++ b/tests/Kiota.Builder.Tests/CodeDOM/CodeElementComparerPythonTests.cs @@ -1,4 +1,4 @@ -using System; +using System; using System.Collections.Generic; using Kiota.Builder.CodeDOM; From 5cf3d83fc87ac4b0e53849b2884c2e8e090f5a97 Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Wed, 22 Mar 2023 00:49:58 +0300 Subject: [PATCH 32/36] Update CHANGELOG --- CHANGELOG.md | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 0c5a38edc0..55b670112a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -14,7 +14,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Changed - Fixed a bug where a CLI client would not set the content types for requests. (Shell) - +- Fixed linting errors by re-ordering methods and properties in Python. +- Changed python import mechanism to facilitate code completion. [#2380](https://github.com/microsoft/kiota/issues/2380) +- Fixed a bug where discriminator methods were missing possible types in Python [#2381](https://github.com/microsoft/kiota/issues/2381) - Fixed a bug where boolean or number enums would be mapped to enums instead of primitive types. [#2367](https://github.com/microsoft/kiota/issues/2367) - Fixed a bug where CSharp inherited constructor name was incorrect. [#2351](https://github.com/microsoft/kiota/issues/2351) - Fixed a bug where java refiner would emit method's parameters types without normalizing the name. @@ -24,11 +26,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Fixed a bug where java writer would emit incorrect type names in case of compound types - Fixed a bug where go refiner would emit incorrect code when inlining error parents + ## [1.0.1] - 2023-03-11 - Fixed a bug where double would not be mapped properly. -- Changed python import mechanism to facilitate code completion. [#2380](https://github.com/microsoft/kiota/issues/2380) -- Fixed a bug where discriminator methods were missing possible types in Python [#2381](https://github.com/microsoft/kiota/issues/2381) ## [1.0.0] - 2023-03-10 From a6c530161cd9486b6080631eec2e3220579b7bcf Mon Sep 17 00:00:00 2001 From: samwelkanda Date: Wed, 22 Mar 2023 01:20:06 +0300 Subject: [PATCH 33/36] Remove suppressions for passing it tests --- it/config.json | 12 ------------ 1 file changed, 12 deletions(-) diff --git a/it/config.json b/it/config.json index 954b81e5ec..7e4916f3f3 100644 --- a/it/config.json +++ b/it/config.json @@ -74,19 +74,11 @@ { "Language": "php", "Rationale": "https://github.com/microsoft/kiota/issues/2351" - }, - { - "Language": "python", - "Rationale": "https://github.com/microsoft/kiota/issues/2351" } ] }, "./tests/Kiota.Builder.IntegrationTests/NoUnderscoresInModel.yaml": { "Suppressions": [ - { - "Language": "python", - "Rationale": "https://github.com/microsoft/kiota/issues/2361" - }, { "Language": "ruby", "Rationale": "https://github.com/microsoft/kiota/issues/2374" @@ -124,10 +116,6 @@ { "Language": "php", "Rationale": "https://github.com/microsoft/kiota/issues/2378" - }, - { - "Language": "python", - "Rationale": "https://github.com/microsoft/kiota/issues/2381" } ] }, From 49e7fc7a775a44dd1a354621cc029f8fd0f3e7d5 Mon Sep 17 00:00:00 2001 From: Vincent Biret Date: Wed, 22 Mar 2023 09:28:16 -0400 Subject: [PATCH 34/36] Apply suggestions from code review --- .../Python/CodeClassDeclarationWriter.cs | 9 ++-- .../Writers/Python/CodeUsingWriter.cs | 52 +++++++++---------- 2 files changed, 29 insertions(+), 32 deletions(-) diff --git a/src/Kiota.Builder/Writers/Python/CodeClassDeclarationWriter.cs b/src/Kiota.Builder/Writers/Python/CodeClassDeclarationWriter.cs index a5d4eb5976..4c0048939e 100644 --- a/src/Kiota.Builder/Writers/Python/CodeClassDeclarationWriter.cs +++ b/src/Kiota.Builder/Writers/Python/CodeClassDeclarationWriter.cs @@ -14,7 +14,7 @@ public CodeClassDeclarationWriter(PythonConventionService conventionService, str public override void WriteCodeElement(ClassDeclaration codeElement, LanguageWriter writer) { ArgumentNullException.ThrowIfNull(codeElement); - if (writer == null) throw new ArgumentNullException(nameof(writer)); + ArgumentNullException.ThrowIfNull(writer); var parentNamespace = codeElement.GetImmediateParentOfType(); _codeUsingWriter.WriteExternalImports(codeElement, writer); // external imports before internal imports _codeUsingWriter.WriteConditionalInternalImports(codeElement, writer, parentNamespace); @@ -23,11 +23,8 @@ public override void WriteCodeElement(ClassDeclaration codeElement, LanguageWrit { if (codeElement.Inherits != null) _codeUsingWriter.WriteDeferredImport(parentClass, codeElement.Inherits.Name, writer); - if (codeElement.Implements.Any()) - { - foreach (var implement in codeElement.Implements) - _codeUsingWriter.WriteDeferredImport(parentClass, implement.Name, writer); - } + foreach (var implement in codeElement.Implements) + _codeUsingWriter.WriteDeferredImport(parentClass, implement.Name, writer); } diff --git a/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs b/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs index 5416c1cbd0..4051f1f04b 100644 --- a/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs +++ b/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs @@ -22,18 +22,18 @@ public void WriteExternalImports(ClassDeclaration codeElement, LanguageWriter wr { var externalImportSymbolsAndPaths = codeElement.Usings .Where(static x => x.IsExternal) - .Select(x => (x.Name, string.Empty, x.Declaration?.Name)) - .GroupBy(x => x.Item3) - .OrderBy(x => x.Key); + .Select(static x => (x.Name, string.Empty, x.Declaration?.Name)) + .GroupBy(static x => x.Item3) + .OrderBy(static x => x.Key); if (externalImportSymbolsAndPaths.Any()) { foreach (var codeUsing in externalImportSymbolsAndPaths) if (!string.IsNullOrWhiteSpace(codeUsing.Key)) { - if (codeUsing.Key == "-") - writer.WriteLine($"import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct().OrderBy(x => x).Aggregate((x, y) => x + ", " + y)}"); + if ("-".Equals(codeUsing.Key, StringComparison.OrdinalIgnoreCase)) + writer.WriteLine($"import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct(StringComparer.OrdinalIgnoreCase).Order(StringComparer.OrdinalIgnoreCase).Aggregate(static (x, y) => x + ", " + y)}"); else - writer.WriteLine($"from {codeUsing.Key.ToSnakeCase()} import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct().OrderBy(x => x).Aggregate((x, y) => x + ", " + y)}"); + writer.WriteLine($"from {codeUsing.Key.ToSnakeCase()} import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct(StringComparer.OrdinalIgnoreCase).Order(StringComparer.OrdinalIgnoreCase).Aggregate(static (x, y) => x + ", " + y)}"); } writer.WriteLine(); } @@ -48,12 +48,12 @@ public void WriteInternalErrorMappingImports(CodeClass parentClass, LanguageWrit { var parentNameSpace = parentClass.GetImmediateParentOfType(); var internalErrorMappingImportSymbolsAndPaths = parentClass.Usings - .Where(x => !x.IsExternal) - .Where(x => x.Declaration?.TypeDefinition is CodeClass codeClass && codeClass.IsErrorDefinition) + .Where(static x => !x.IsExternal) + .Where(static x => x.Declaration?.TypeDefinition is CodeClass codeClass && codeClass.IsErrorDefinition) .Select(x => _relativeImportManager.GetRelativeImportPathForUsing(x, parentNameSpace)) - .GroupBy(x => x.Item3) - .Where(x => !string.IsNullOrEmpty(x.Key)) - .OrderBy(x => x.Key); + .GroupBy(static x => x.Item3) + .Where(static x => !string.IsNullOrEmpty(x.Key)) + .OrderBy(static x => x.Key, StringComparer.OrdinalIgnoreCase); WriteCodeUsings(internalErrorMappingImportSymbolsAndPaths, writer); } /// @@ -66,11 +66,11 @@ public void WriteInternalImports(CodeClass parentClass, LanguageWriter writer) { var parentNameSpace = parentClass.GetImmediateParentOfType(); var internalImportSymbolsAndPaths = parentClass.Usings - .Where(x => !x.IsExternal) + .Where(static x => !x.IsExternal) .Select(x => _relativeImportManager.GetRelativeImportPathForUsing(x, parentNameSpace)) - .GroupBy(x => x.Item3) - .Where(x => !string.IsNullOrEmpty(x.Key)) - .OrderBy(x => x.Key); + .GroupBy(static x => x.Item3) + .Where(static x => !string.IsNullOrEmpty(x.Key)) + .OrderBy(static x => x.Key, StringComparer.OrdinalIgnoreCase); WriteCodeUsings(internalImportSymbolsAndPaths, writer); } /// @@ -83,17 +83,17 @@ public void WriteInternalImports(CodeClass parentClass, LanguageWriter writer) public void WriteConditionalInternalImports(ClassDeclaration codeElement, LanguageWriter writer, CodeNamespace parentNameSpace) { var internalImportSymbolsAndPaths = codeElement.Usings - .Where(x => !x.IsExternal) + .Where(static x => !x.IsExternal) .Select(x => _relativeImportManager.GetRelativeImportPathForUsing(x, parentNameSpace)) - .GroupBy(x => x.Item3) - .Where(x => !string.IsNullOrEmpty(x.Key)) - .OrderBy(x => x.Key); + .GroupBy(static x => x.Item3) + .Where(static x => !string.IsNullOrEmpty(x.Key)) + .OrderBy(static x => x.Key, StringComparer.OrdinalIgnoreCase); if (internalImportSymbolsAndPaths.Any()) { writer.WriteLine("if TYPE_CHECKING:"); writer.IncreaseIndent(); foreach (var codeUsing in internalImportSymbolsAndPaths) - writer.WriteLine($"from {codeUsing.Key.ToSnakeCase()} import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct().OrderBy(x => x).Aggregate((x, y) => x + ", " + y)}"); + writer.WriteLine($"from {codeUsing.Key.ToSnakeCase()} import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct(StringComparer.OrdinalIgnoreCase).Order(StringComparer.OrdinalIgnoreCase).Aggregate(static (x, y) => x + ", " + y)}"); writer.DecreaseIndent(); writer.WriteLine(); @@ -110,12 +110,12 @@ public void WriteDeferredImport(CodeClass parentClass, string typeName, Language { var parentNamespace = parentClass.GetImmediateParentOfType(); var internalImportSymbolsAndPaths = parentClass.Usings - .Where(x => !x.IsExternal) - .Where(x => string.Equals(x.Declaration?.Name, typeName)) + .Where(static x => !x.IsExternal) + .Where(x => typeName.Equals(x.Declaration?.Name, StringComparison.OrdinalIgnoreCase)) .Select(x => _relativeImportManager.GetRelativeImportPathForUsing(x, parentNamespace)) - .GroupBy(x => x.Item3) - .Where(x => !string.IsNullOrEmpty(x.Key)) - .OrderBy(x => x.Key); + .GroupBy(static x => x.Item3) + .Where(static x => !string.IsNullOrEmpty(x.Key)) + .OrderBy(static x => x.Key, StringComparer.OrdinalIgnoreCase); WriteCodeUsings(internalImportSymbolsAndPaths, writer); } @@ -124,7 +124,7 @@ private static void WriteCodeUsings(IOrderedEnumerable GetAliasedSymbol(x.Item1, x.Item2)).Distinct().OrderBy(x => x).Aggregate((x, y) => x + ", " + y)}"); + writer.WriteLine($"from {codeUsing.Key.ToSnakeCase()} import {codeUsing.Select(x => GetAliasedSymbol(x.Item1, x.Item2)).Distinct(StringComparer.OrdinalIgnoreCase).Order(StringComparer.OrdinalIgnoreCase).Aggregate(static (x, y) => x + ", " + y)}"); writer.WriteLine(); } } From de51794ac10bdd27d8149f49ff80479b4369bd98 Mon Sep 17 00:00:00 2001 From: Vincent Biret Date: Wed, 22 Mar 2023 09:31:07 -0400 Subject: [PATCH 35/36] Update src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs --- src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs b/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs index 4051f1f04b..650bb75936 100644 --- a/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs +++ b/src/Kiota.Builder/Writers/Python/CodeUsingWriter.cs @@ -1,5 +1,5 @@ -using System.Linq; - +using System; +using System.Linq; using Kiota.Builder.CodeDOM; using Kiota.Builder.Extensions; From ecf2e0776001351715b5a54059fdae90ff4b2b89 Mon Sep 17 00:00:00 2001 From: Vincent Biret Date: Wed, 22 Mar 2023 09:39:21 -0400 Subject: [PATCH 36/36] - fixes failing unit test Signed-off-by: Vincent Biret --- .../Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs index a7840af2ae..87f193c452 100644 --- a/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/Python/CodeMethodWriterTests.cs @@ -363,7 +363,7 @@ public void WritesRequestExecutorBody() writer.Write(method); var result = tw.ToString(); Assert.Contains("request_info", result); - Assert.Contains("from . import error4_x_x, error401, error5_x_x", result); + Assert.Contains("from . import error401, error4_x_x, error5_x_x", result); Assert.Contains("error_mapping: Dict[str, ParsableFactory] =", result); Assert.Contains("\"4XX\": error4_x_x.Error4XX", result); Assert.Contains("\"5XX\": error5_x_x.Error5XX", result);