From 3803437472908ee15649db5b9d8491e07ec15a6b Mon Sep 17 00:00:00 2001 From: silaskenneth Date: Tue, 24 Oct 2023 18:56:20 +0300 Subject: [PATCH 1/4] [FIX] Include the type which a promise resolves to. This commit also includes: - Removing the try catch in the request executor and instead just allow the exception to be thrown. - Refactor the type resolution logic in the PHPConventionService. --- src/Kiota.Builder/Refiners/PhpRefiner.cs | 2 +- .../Writers/Php/CodeMethodWriter.cs | 18 ++++++------------ .../Php/CodeClassDeclarationWriterTests.cs | 3 +-- .../Writers/Php/CodeMethodWriterTests.cs | 5 +---- 4 files changed, 9 insertions(+), 19 deletions(-) diff --git a/src/Kiota.Builder/Refiners/PhpRefiner.cs b/src/Kiota.Builder/Refiners/PhpRefiner.cs index ff068c8a95..1d12aef87d 100644 --- a/src/Kiota.Builder/Refiners/PhpRefiner.cs +++ b/src/Kiota.Builder/Refiners/PhpRefiner.cs @@ -192,7 +192,7 @@ public override Task Refine(CodeNamespace generatedCode, CancellationToken cance "Microsoft\\Kiota\\Abstractions\\Store", "BackingStoreFactory", "BackingStoreFactorySingleton"), new (x => x is CodeProperty prop && prop.IsOfKind(CodePropertyKind.BackingStore), "Microsoft\\Kiota\\Abstractions\\Store", "BackingStore", "BackedModel", "BackingStoreFactorySingleton" ), - new (x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.RequestExecutor), "Http\\Promise", "Promise", "RejectedPromise"), + new (x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.RequestExecutor), "Http\\Promise", "Promise"), new (x => x is CodeMethod method && method.IsOfKind(CodeMethodKind.RequestExecutor), "", "Exception"), new (x => x is CodeEnum, "Microsoft\\Kiota\\Abstractions\\", "Enum"), new(static x => x is CodeProperty {Type.Name: {}} property && property.Type.Name.Equals("DateTime", StringComparison.OrdinalIgnoreCase), "", "\\DateTime"), diff --git a/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs b/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs index b439ea8e0a..802183a72b 100644 --- a/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs +++ b/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs @@ -259,7 +259,7 @@ private static void AssignPropertyFromParameter(CodeClass parentClass, CodeMetho private void WriteMethodPhpDocs(CodeMethod codeMethod, LanguageWriter writer) { var methodDescription = codeMethod.Documentation.Description; - + var methodThrows = codeMethod.IsOfKind(CodeMethodKind.RequestExecutor); var hasMethodDescription = !string.IsNullOrEmpty(methodDescription.Trim()); if (!hasMethodDescription && !codeMethod.Parameters.Any()) { @@ -276,13 +276,15 @@ private void WriteMethodPhpDocs(CodeMethod codeMethod, LanguageWriter writer) { var nullableSuffix = (codeMethod.ReturnType.IsNullable ? "|null" : ""); returnDocString = (codeMethod.Kind == CodeMethodKind.RequestExecutor) - ? "@return Promise" + ? $"@return Promise<{returnDocString}|null>" : $"@return {returnDocString}{nullableSuffix}"; } else returnDocString = String.Empty; + + var throwsArray = methodThrows ? new[] { "@throws Exception" } : Array.Empty(); conventions.WriteLongDescription(codeMethod.Documentation, writer, - parametersWithOrWithoutDescription.Union(new[] { returnDocString }) + parametersWithOrWithoutDescription.Union(new[] { returnDocString }).Union(throwsArray) ); } @@ -294,6 +296,7 @@ private string GetDocCommentReturnType(CodeMethod codeMethod) CodeMethodKind.Deserializer => "array", CodeMethodKind.Getter when codeMethod.AccessedProperty?.IsOfKind(CodePropertyKind.AdditionalData) ?? false => "array", CodeMethodKind.Getter when codeMethod.AccessedProperty?.Type.IsCollection ?? false => $"array<{conventions.TranslateType(codeMethod.AccessedProperty.Type)}>", + CodeMethodKind.RequestExecutor when codeMethod.ReturnType.IsCollection => $"array<{conventions.TranslateType(codeMethod.ReturnType)}>", _ => conventions.GetTypeString(codeMethod.ReturnType, codeMethod) }; } @@ -735,8 +738,6 @@ private void WriteRequestExecutorBody(CodeMethod codeElement, CodeClass parentCl var returnTypeName = conventions.GetTypeString(codeElement.ReturnType, codeElement, false); writer.WriteLine($"$requestInfo = $this->{generatorMethodName}({joinedParams});"); - writer.WriteLine("try {"); - writer.IncreaseIndent(); var errorMappings = codeElement.ErrorMappings; var hasErrorMappings = false; var errorMappingsVarName = "$errorMappings"; @@ -774,13 +775,6 @@ private void WriteRequestExecutorBody(CodeMethod codeElement, CodeClass parentCl var requestAdapterProperty = parentClass.GetPropertyOfKind(CodePropertyKind.RequestAdapter) ?? throw new InvalidOperationException("Request adapter property not found"); writer.WriteLine( $"return {GetPropertyCall(requestAdapterProperty, string.Empty)}->{methodName}({RequestInfoVarName}{finalReturn}, {(hasErrorMappings ? $"{errorMappingsVarName}" : "null")});"); - - writer.DecreaseIndent(); - writer.WriteLine("} catch(Exception $ex) {"); - writer.IncreaseIndent(); - writer.WriteLine("return new RejectedPromise($ex);"); - writer.DecreaseIndent(); - writer.WriteLine("}"); } private static void WriteApiConstructorBody(CodeClass parentClass, CodeMethod codeMethod, LanguageWriter writer) diff --git a/tests/Kiota.Builder.Tests/Writers/Php/CodeClassDeclarationWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Php/CodeClassDeclarationWriterTests.cs index 64e1cba133..582208c25c 100644 --- a/tests/Kiota.Builder.Tests/Writers/Php/CodeClassDeclarationWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/Php/CodeClassDeclarationWriterTests.cs @@ -134,8 +134,7 @@ public async Task ImportRequiredClassesWhenContainsRequestExecutor() codeElementWriter.WriteCodeElement(dec, writer); var result = tw.ToString(); - Assert.Contains("use Http\\Promise\\Promise;", result); - Assert.Contains("use Http\\Promise\\RejectedPromise;", result); + Assert.Contains(@"use Http\Promise\Promise;", result); Assert.Contains("use Exception;", result); } diff --git a/tests/Kiota.Builder.Tests/Writers/Php/CodeMethodWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Php/CodeMethodWriterTests.cs index c0f8cd3783..458323eb5e 100644 --- a/tests/Kiota.Builder.Tests/Writers/Php/CodeMethodWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/Php/CodeMethodWriterTests.cs @@ -287,9 +287,7 @@ public async void WriteRequestExecutor() Assert.Contains("public function post(): Promise", result); Assert.Contains("$requestInfo = $this->createPostRequestInformation();", result); - Assert.Contains("RejectedPromise", result); Assert.Contains("@link https://learn.microsoft.com/ Learning", result); - Assert.Contains("catch(Exception $ex)", result); Assert.Contains("'401' => [Error401::class, 'createFromDiscriminatorValue']", result); Assert.Contains("return $this->requestAdapter->sendPrimitiveAsync($requestInfo, StreamInterface::class, $errorMappings);", result); } @@ -380,11 +378,10 @@ public async void WritesRequestExecutorForEnumTypes() _codeMethodWriter.WriteCodeElement(codeMethod, languageWriter); var result = stringWriter.ToString(); + Assert.Contains("@throws Exception", result); Assert.Contains("public function post(): Promise", result); Assert.Contains("$requestInfo = $this->createPostRequestInformation();", result); - Assert.Contains("RejectedPromise", result); Assert.Contains("@link https://learn.microsoft.com/ Learning", result); - Assert.Contains("catch(Exception $ex)", result); Assert.Contains("'401' => [Error401::class, 'createFromDiscriminatorValue']", result); Assert.Contains("return $this->requestAdapter->sendPrimitiveAsync($requestInfo, PhoneNumberPrefix::class, $errorMappings);", result); } From f90a550fd0827460648d8cede344c046da792a80 Mon Sep 17 00:00:00 2001 From: silaskenneth Date: Wed, 25 Oct 2023 12:26:19 +0300 Subject: [PATCH 2/4] Add CHANGELOG entry. --- CHANGELOG.md | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index c1d4b55209..12640336e3 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -19,8 +19,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Changed +- Include the type which a promise resolves to in PHPDocs for PHP. [#3542](https://github.com/microsoft/kiota/issues/3542) - Aligns header management in Python with other languages. [#3430](https://github.com/microsoft/kiota/issues/3430) -- Fixed parameters that are in camelcase to snakecase in Python. [#3525](https://github.com/microsoft/kiota/issues/3525 +- Fixed parameters that are in camelcase to snakecase in Python. [#3525](https://github.com/microsoft/kiota/issues/3525) - Fixed missing imports for method parameters that are query parameters. - Replaces the use of "new" by "override" and "virtual" in CSharp models. - Fixed query parameters type mapping for arrays. [#3354](https://github.com/microsoft/kiota/issues/3354) From c0121607548276984c8ae49f3d724751b7fb7c4a Mon Sep 17 00:00:00 2001 From: silaskenneth Date: Mon, 30 Oct 2023 13:12:38 +0300 Subject: [PATCH 3/4] Add var doc comment to executors that return `sendPrimitive`. --- src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs | 13 +++++++++++-- .../Writers/Php/CodeMethodWriterTests.cs | 7 +++++-- 2 files changed, 16 insertions(+), 4 deletions(-) diff --git a/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs b/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs index 802183a72b..9882aeb8d5 100644 --- a/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs +++ b/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs @@ -773,8 +773,17 @@ private void WriteRequestExecutorBody(CodeMethod codeElement, CodeClass parentCl ? $", '{returnTypeName}'" : returnEnumType; var requestAdapterProperty = parentClass.GetPropertyOfKind(CodePropertyKind.RequestAdapter) ?? throw new InvalidOperationException("Request adapter property not found"); - writer.WriteLine( - $"return {GetPropertyCall(requestAdapterProperty, string.Empty)}->{methodName}({RequestInfoVarName}{finalReturn}, {(hasErrorMappings ? $"{errorMappingsVarName}" : "null")});"); + + var methodCall = $"{GetPropertyCall(requestAdapterProperty, string.Empty)}->{methodName}({RequestInfoVarName}{finalReturn}, {(hasErrorMappings ? $"{errorMappingsVarName}" : "null")});"; + if (methodName.Contains("sendPrimitive", StringComparison.OrdinalIgnoreCase)) + { + writer.WriteLines($"/** @var Promise<{GetDocCommentReturnType(codeElement)}|null> $result", + $"$result = {methodCall}", + "return $result;" + ); + } + else writer.WriteLines( + $"return {methodCall}"); } private static void WriteApiConstructorBody(CodeClass parentClass, CodeMethod codeMethod, LanguageWriter writer) diff --git a/tests/Kiota.Builder.Tests/Writers/Php/CodeMethodWriterTests.cs b/tests/Kiota.Builder.Tests/Writers/Php/CodeMethodWriterTests.cs index 458323eb5e..f9d62812c0 100644 --- a/tests/Kiota.Builder.Tests/Writers/Php/CodeMethodWriterTests.cs +++ b/tests/Kiota.Builder.Tests/Writers/Php/CodeMethodWriterTests.cs @@ -289,7 +289,8 @@ public async void WriteRequestExecutor() Assert.Contains("$requestInfo = $this->createPostRequestInformation();", result); Assert.Contains("@link https://learn.microsoft.com/ Learning", result); Assert.Contains("'401' => [Error401::class, 'createFromDiscriminatorValue']", result); - Assert.Contains("return $this->requestAdapter->sendPrimitiveAsync($requestInfo, StreamInterface::class, $errorMappings);", result); + Assert.Contains("$result = $this->requestAdapter->sendPrimitiveAsync($requestInfo, StreamInterface::class, $errorMappings);", result); + Assert.Contains("return $result;", result); } [Fact] @@ -383,7 +384,9 @@ public async void WritesRequestExecutorForEnumTypes() Assert.Contains("$requestInfo = $this->createPostRequestInformation();", result); Assert.Contains("@link https://learn.microsoft.com/ Learning", result); Assert.Contains("'401' => [Error401::class, 'createFromDiscriminatorValue']", result); - Assert.Contains("return $this->requestAdapter->sendPrimitiveAsync($requestInfo, PhoneNumberPrefix::class, $errorMappings);", result); + Assert.Contains("/** @var Promise $result", result); + Assert.Contains("$result = $this->requestAdapter->sendPrimitiveAsync($requestInfo, PhoneNumberPrefix::class, $errorMappings);", result); + Assert.Contains("return $result;", result); } public static IEnumerable SerializerProperties => new List From eda6bcb9db7eb7459caea20094604cb730f53387 Mon Sep 17 00:00:00 2001 From: silaskenneth Date: Mon, 30 Oct 2023 13:53:47 +0300 Subject: [PATCH 4/4] Update doc comments for generated code. --- src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs b/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs index 9882aeb8d5..5108394264 100644 --- a/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs +++ b/src/Kiota.Builder/Writers/Php/CodeMethodWriter.cs @@ -777,7 +777,7 @@ private void WriteRequestExecutorBody(CodeMethod codeElement, CodeClass parentCl var methodCall = $"{GetPropertyCall(requestAdapterProperty, string.Empty)}->{methodName}({RequestInfoVarName}{finalReturn}, {(hasErrorMappings ? $"{errorMappingsVarName}" : "null")});"; if (methodName.Contains("sendPrimitive", StringComparison.OrdinalIgnoreCase)) { - writer.WriteLines($"/** @var Promise<{GetDocCommentReturnType(codeElement)}|null> $result", + writer.WriteLines($"/** @var Promise<{GetDocCommentReturnType(codeElement)}|null> $result */", $"$result = {methodCall}", "return $result;" );