Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add shell language support #738

Merged
merged 88 commits into from
Feb 16, 2022
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
Show all changes
88 commits
Select commit Hold shift + click to select a range
ad5cf7d
Add shell writer based on C# writer
calebkiage Oct 18, 2021
7ec5869
Fix method writer reference
calebkiage Oct 18, 2021
5ee05a0
Add shell writer based on C# writer
calebkiage Oct 18, 2021
e0061ea
Fix method writer reference
calebkiage Oct 18, 2021
f3bf3f6
Fix generation error
calebkiage Oct 19, 2021
b36cf92
Add code class kind for shell language
calebkiage Oct 21, 2021
5d370c7
Remove unused code
calebkiage Oct 21, 2021
e7ce85f
Rename AddCodeElementWriter function to reflect updated behavior whic…
calebkiage Oct 21, 2021
0802de4
- adds path and query parameters for non fluent languages
baywet Oct 21, 2021
0c668ec
Update shell refiner to add command builders
calebkiage Oct 25, 2021
d23f2a7
Merge branch 'feat/cli_generator' of https://github.com/microsoft/kio…
calebkiage Oct 25, 2021
87d12e6
Remove parameters from build command
calebkiage Oct 25, 2021
cdd23bb
Update command handler
calebkiage Oct 25, 2021
4a78a55
Apply suggested Linq optimization
calebkiage Oct 26, 2021
c43d8fa
Add code to set up command tree
calebkiage Oct 26, 2021
14995f9
Update import declarations in generated code
calebkiage Oct 27, 2021
1c613e5
Merge remote-tracking branch 'origin/main' into feat/cli_generator
calebkiage Oct 27, 2021
7162adb
Generate code to output command results
calebkiage Nov 1, 2021
2bb0dad
Read request body input as json string
calebkiage Nov 2, 2021
d65d247
Add file IO support for stream data upload & download
calebkiage Nov 3, 2021
fe9b1e8
Merge branch main into feat/cli_generator
calebkiage Nov 10, 2021
84b2fc5
Fix NullReferenceException when there are no parameters
calebkiage Dec 2, 2021
9b418e7
Apply suggested code changes
calebkiage Dec 2, 2021
e567be5
Handle edge cases that broke the generated code build
calebkiage Dec 3, 2021
4597b62
Merge branch 'main' into feat/cli_generator
calebkiage Dec 6, 2021
8772b4a
Order methods for deterministic output
calebkiage Dec 6, 2021
0ba0e09
Add missing import
calebkiage Dec 6, 2021
b23dc85
Throw exception for attempts to use CommandBuilders in CSharp
calebkiage Dec 6, 2021
1d0f3c3
Fix bug with command builder writer
calebkiage Dec 7, 2021
a3ea49b
Fix stack overflow exception in builder methods
calebkiage Dec 7, 2021
7d6745f
- fixes a bug where path and query parameters of type collection woul…
baywet Dec 8, 2021
db176be
Update src/Kiota.Builder/Writers/Shell/ShellCodeMethodWriter.cs
calebkiage Dec 9, 2021
8e7236f
Add required flag to options
calebkiage Dec 9, 2021
0c81ef4
Merge branch 'feat/cli_generator' of https://github.com/microsoft/kio…
calebkiage Dec 9, 2021
af989d1
Use generator method to add query parameters to request
calebkiage Dec 9, 2021
d593305
Merge remote-tracking branch 'origin/main' into feat/cli_generator
calebkiage Jan 19, 2022
a0b4c3a
Update shell writer to accomodate breaking changes in System.Commandl…
calebkiage Jan 19, 2022
9ecf819
Merge pull request #1038 from microsoft/chore/update_system_commandline
baywet Jan 19, 2022
7f19244
Update method writer to use output formatter
calebkiage Jan 25, 2022
0610a1f
Fix duplicated output in generated code for streams
calebkiage Jan 26, 2022
6ee2632
Merge branch 'main' into feat/cli_generator
calebkiage Jan 26, 2022
839c8c0
Fix issue with some commands being skipped.
calebkiage Jan 27, 2022
fad1be8
Merge branch 'feat/cli_generator' into feat/output_formatter
calebkiage Jan 27, 2022
f835f83
Print server response if an error occurs in the request
calebkiage Jan 27, 2022
1a23e25
Refactor for generation performance
calebkiage Jan 28, 2022
e2f825f
Move from using response handler and use SendPrimitiveAsync<Stream>
calebkiage Feb 4, 2022
cfba6ef
Add CLI commons package
calebkiage Feb 9, 2022
9a2fdf7
Add missing types
calebkiage Feb 9, 2022
fa2bec3
Update dependabot config
calebkiage Feb 10, 2022
20169c1
Merge remote-tracking branch 'origin/main' into chore/commons_package
calebkiage Feb 10, 2022
eee1036
Merge remote-tracking branch 'origin/main' into feat/output_formatter
calebkiage Feb 10, 2022
e285e77
Merge branch 'feat/output_formatter' into chore/commons_package
calebkiage Feb 10, 2022
00471e9
Add shell language to main readme
calebkiage Feb 10, 2022
5bb480e
Merge pull request #1180 from microsoft/chore/commons_package
calebkiage Feb 10, 2022
9bc455d
Merge remote-tracking branch 'origin/main' into feat/cli_generator
calebkiage Feb 10, 2022
0188007
Merge branch 'feat/cli_generator' into feat/output_formatter
calebkiage Feb 10, 2022
8e6f55f
Merge pull request #1071 from microsoft/feat/output_formatter
calebkiage Feb 10, 2022
ef8c275
Fix possible null pointer exception
calebkiage Feb 10, 2022
d562acc
Fix code smells
calebkiage Feb 10, 2022
d2e6be9
Simplify WriteCommandBuilderBody function
calebkiage Feb 10, 2022
bf6614c
Correct core types in shell refiner
calebkiage Feb 11, 2022
1f61f7e
Remove raw url constructor overload
calebkiage Feb 11, 2022
9aed749
Simplify CreateCommandType method
calebkiage Feb 11, 2022
8cc0438
Remove unused request methods
calebkiage Feb 11, 2022
61d6311
Check for whitespace as well as null on string values
calebkiage Feb 11, 2022
a6f9d7d
Refactor duplicated code
calebkiage Feb 11, 2022
3eaa361
Get serialization contect type from generator method
calebkiage Feb 11, 2022
5cfb3a1
Add IsOfKind extension method for CodeParameter
calebkiage Feb 11, 2022
3bd2e32
Merge branch 'main' into feat/cli_generator
calebkiage Feb 11, 2022
e06f262
Fix build error
calebkiage Feb 11, 2022
0ac97f3
Remove unnecessary extension method
calebkiage Feb 11, 2022
1fe18c4
Fix unintended revert in CodeMethodWriter after conflict resolution
calebkiage Feb 11, 2022
c9352e7
Refactor code
calebkiage Feb 14, 2022
81670b1
Add shell code method writer tests
calebkiage Feb 14, 2022
b166e23
Replace if...else with switch expression
calebkiage Feb 14, 2022
791578d
Use string.Equals function
calebkiage Feb 14, 2022
bf6d4b9
Add additional unit tests
calebkiage Feb 15, 2022
e27e15a
Add unit tests
calebkiage Feb 16, 2022
11d6e82
Refactor code
calebkiage Feb 16, 2022
65e245c
Update default value to handle string option types
calebkiage Feb 16, 2022
f4aaa20
Add assertion for optional parameters
calebkiage Feb 16, 2022
bfff3a1
Rename root build command
calebkiage Feb 16, 2022
ebb18a1
Merge remote-tracking branch 'origin/main' into feat/cli_generator
calebkiage Feb 16, 2022
f5597b8
Fix failing tests
calebkiage Feb 16, 2022
97671bd
Simplify code complexity
calebkiage Feb 16, 2022
5c0347c
Use OfKind extension method
calebkiage Feb 16, 2022
a7cf475
Add error handling support to shell generation
calebkiage Feb 16, 2022
d5d3c02
Fix failing tests
calebkiage Feb 16, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion src/Kiota.Builder/CodeDOM/CodeMethod.cs
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,8 @@ public enum CodeMethodKind
ClientConstructor,
RequestBuilderBackwardCompatibility,
RequestBuilderWithParameters,
RawUrlConstructor
RawUrlConstructor,
CommandBuilder
}
public enum HttpMethod {
Get,
Expand Down
6 changes: 3 additions & 3 deletions src/Kiota.Builder/Extensions/StringExtensions.cs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
using System;
using System;
using System.Linq;
using System.Security.Cryptography;
using System.Text;
Expand Down Expand Up @@ -26,7 +26,7 @@ public static string ReplaceValueIdentifier(this string original) =>
public static string TrimQuotes(this string original) =>
original?.Trim('\'', '"');

public static string ToSnakeCase(this string name)
public static string ToSnakeCase(this string name, char separator = '_')
{
if(string.IsNullOrEmpty(name)) return name;
var chunks = name.Split('-', StringSplitOptions.RemoveEmptyEntries);
Expand All @@ -40,7 +40,7 @@ public static string ToSnakeCase(this string name)
foreach (var item in identifier[1..])
{
if(char.IsUpper(item)) {
sb.Append('_');
sb.Append(separator);
sb.Append(char.ToLowerInvariant(item));
} else {
sb.Append(item);
Expand Down
27 changes: 13 additions & 14 deletions src/Kiota.Builder/Refiners/ShellRefiner.cs
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ public override void Refine(CodeNamespace generatedCode)
RemoveModelClasses(generatedCode);
RemoveEnums(generatedCode);
RemoveConstructors(generatedCode);
TurnRequestBuildersIntoCommandBuilders(generatedCode);
CreateCommandBuilders(generatedCode);
AddAsyncSuffix(generatedCode);
AddInnerClasses(generatedCode, false);
CapitalizeNamespacesFirstLetters(generatedCode);
Expand Down Expand Up @@ -92,13 +92,10 @@ private static void RemoveConstructors(CodeElement currentElement)
CrawlTree(currentElement, RemoveConstructors);
}

private static void TurnRequestBuildersIntoCommandBuilders(CodeElement currentElement)
private static void CreateCommandBuilders(CodeElement currentElement)
{
if (currentElement is CodeClass currentClass && currentClass.IsOfKind(CodeClassKind.RequestBuilder))
{

(currentClass.StartBlock as CodeClass.Declaration).IsStatic = true;

// Replace Nav Properties with BuildXXXCommand methods
var navProperties = currentClass.GetChildElements().Where(e => e is CodeProperty prop && prop.IsOfKind(CodePropertyKind.RequestBuilder)).Cast<CodeProperty>();
foreach (var navProp in navProperties)
Expand All @@ -107,21 +104,23 @@ private static void TurnRequestBuildersIntoCommandBuilders(CodeElement currentEl
currentClass.AddMethod(method);
currentClass.RemoveChildElement(navProp);
}
// Change signtature of RequestExecutors
// Clone executors & convert to build command
var requestMethods = currentClass.GetChildElements().Where(e => e is CodeMethod method && method.IsOfKind(CodeMethodKind.RequestExecutor)).Cast<CodeMethod>();
calebkiage marked this conversation as resolved.
Show resolved Hide resolved
foreach (var requestMethod in requestMethods)
calebkiage marked this conversation as resolved.
Show resolved Hide resolved
{
requestMethod.IsAsync = false;
requestMethod.IsStatic = true;
requestMethod.Name = $"Build{requestMethod.Name}Command";
requestMethod.ReturnType = CreateCommandType(requestMethod);
CodeMethod clone = requestMethod.Clone() as CodeMethod;
clone.IsAsync = false;
clone.Name = $"Build{clone.Name}Command";
clone.ReturnType = CreateCommandType(requestMethod);
clone.MethodKind = CodeMethodKind.CommandBuilder;
currentClass.AddMethod(clone);
}

var buildMethod = new CodeMethod
calebkiage marked this conversation as resolved.
Show resolved Hide resolved
{
Name = "Build",
IsStatic = true,
IsAsync = false
IsAsync = false,
MethodKind = CodeMethodKind.CommandBuilder
};
buildMethod.AddParameter(new CodeParameter { Name = "httpCore", Type = new CodeType { Name = "IHttpCore", IsExternal = true } });
calebkiage marked this conversation as resolved.
Show resolved Hide resolved
// Add calls to BuildMethods here..
Expand All @@ -133,7 +132,7 @@ private static void TurnRequestBuildersIntoCommandBuilders(CodeElement currentEl
currentClass.AddMethod(buildMethod);

}
CrawlTree(currentElement, TurnRequestBuildersIntoCommandBuilders);
CrawlTree(currentElement, CreateCommandBuilders);
}

private static CodeType CreateCommandType(CodeElement parent)
Expand All @@ -151,7 +150,7 @@ private static CodeMethod CreateBuildCommandMethod(CodeProperty navProperty, Cod
codeMethod.IsAsync = false;
baywet marked this conversation as resolved.
Show resolved Hide resolved
codeMethod.IsStatic = true;
codeMethod.Name = $"Build{navProperty.Name.ToFirstCharacterUpperCase()}Command";
codeMethod.MethodKind = CodeMethodKind.RequestBuilderWithParameters;
codeMethod.MethodKind = CodeMethodKind.CommandBuilder;
codeMethod.ReturnType = CreateCommandType(codeMethod);
return codeMethod;
}
Expand Down
29 changes: 19 additions & 10 deletions src/Kiota.Builder/Writers/CSharp/CodeMethodWriter.cs
Original file line number Diff line number Diff line change
Expand Up @@ -22,22 +22,31 @@ public override void WriteCodeElement(CodeMethod codeElement, LanguageWriter wri
WriteMethodDocumentation(codeElement, writer);
WriteMethodPrototype(codeElement, writer, returnType, inherits, isVoid);
writer.IncreaseIndent();
var requestBodyParam = codeElement.Parameters.OfKind(CodeParameterKind.RequestBody);
var queryStringParam = codeElement.Parameters.OfKind(CodeParameterKind.QueryParameter);
var headersParam = codeElement.Parameters.OfKind(CodeParameterKind.Headers);
var optionsParam = codeElement.Parameters.OfKind(CodeParameterKind.Options);
var requestParams = new RequestParams(requestBodyParam, queryStringParam, headersParam, optionsParam);
foreach(var parameter in codeElement.Parameters.Where(x => !x.Optional).OrderBy(x => x.Name)) {
var parameterName = parameter.Name.ToFirstCharacterLowerCase();
if(nameof(String).Equals(parameter.Type.Name, StringComparison.OrdinalIgnoreCase))
writer.WriteLine($"if(string.IsNullOrEmpty({parameterName})) throw new ArgumentNullException(nameof({parameterName}));");
else
writer.WriteLine($"_ = {parameterName} ?? throw new ArgumentNullException(nameof({parameterName}));");
}
switch(codeElement.MethodKind) {
HandleMethodKind(codeElement, writer, inherits, parentClass, isVoid);
writer.DecreaseIndent();
calebkiage marked this conversation as resolved.
Show resolved Hide resolved
writer.WriteLine("}");
calebkiage marked this conversation as resolved.
Show resolved Hide resolved
}

protected virtual void HandleMethodKind(CodeMethod codeElement, LanguageWriter writer, bool inherits, CodeClass parentClass, bool isVoid)
{
var returnType = conventions.GetTypeString(codeElement.ReturnType, codeElement);
var requestBodyParam = codeElement.Parameters.OfKind(CodeParameterKind.RequestBody);
var queryStringParam = codeElement.Parameters.OfKind(CodeParameterKind.QueryParameter);
var headersParam = codeElement.Parameters.OfKind(CodeParameterKind.Headers);
var optionsParam = codeElement.Parameters.OfKind(CodeParameterKind.Options);
var requestParams = new RequestParams(requestBodyParam, queryStringParam, headersParam, optionsParam);
switch (codeElement.MethodKind)
{
case CodeMethodKind.Serializer:
WriteSerializerBody(inherits, codeElement, parentClass, writer);
break;
break;
case CodeMethodKind.RequestGenerator:
WriteRequestGeneratorBody(codeElement, requestParams, parentClass, writer);
break;
Expand All @@ -63,12 +72,12 @@ public override void WriteCodeElement(CodeMethod codeElement, LanguageWriter wri
throw new InvalidOperationException("getters and setters are automatically added on fields in dotnet");
case CodeMethodKind.RequestBuilderBackwardCompatibility:
throw new InvalidOperationException("RequestBuilderBackwardCompatibility is not supported as the request builders are implemented by properties.");
case CodeMethodKind.CommandBuilder:
calebkiage marked this conversation as resolved.
Show resolved Hide resolved
break;
default:
writer.WriteLine("return null;");
break;
break;
}
writer.DecreaseIndent();
writer.WriteLine("}");
}
private void WriteRequestBuilderBody(CodeClass parentClass, CodeMethod codeElement, LanguageWriter writer)
{
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -34,8 +34,7 @@ public override void WriteCodeElement(CodeClass.Declaration codeElement, Languag
var derivation = derivedTypes.Any() ? ": " + derivedTypes.Select(x => x.ToFirstCharacterUpperCase()).Aggregate((x, y) => $"{x}, {y}") + " " : string.Empty;
calebkiage marked this conversation as resolved.
Show resolved Hide resolved
if (codeElement.Parent is CodeClass parentClass)
conventions.WriteShortDescription(parentClass.Description, writer);
var staticModifier = codeElement.IsStatic ? "static " : string.Empty;
writer.WriteLine($"public {staticModifier}class {codeElement.Name.ToFirstCharacterUpperCase()} {derivation}{{");
writer.WriteLine($"public class {codeElement.Name.ToFirstCharacterUpperCase()} {derivation}{{");
writer.IncreaseIndent();
}
}
Expand Down
84 changes: 64 additions & 20 deletions src/Kiota.Builder/Writers/Shell/ShellCodeMethodWriter.cs
Original file line number Diff line number Diff line change
Expand Up @@ -13,28 +13,72 @@ public ShellCodeMethodWriter(CSharpConventionService conventionService) : base(c
{
}

protected override void WriteRequestExecutorBody(CodeMethod codeElement, RequestParams requestParams, bool isVoid, string returnType, LanguageWriter writer)
protected override void HandleMethodKind(CodeMethod codeElement, LanguageWriter writer, bool inherits, CodeClass parentClass, bool isVoid)
{
if (codeElement.HttpMethod == null) throw new InvalidOperationException("http method cannot be null");
base.HandleMethodKind(codeElement, writer, inherits, parentClass, isVoid);
if (codeElement.MethodKind == CodeMethodKind.CommandBuilder)
{
var returnType = conventions.GetTypeString(codeElement.ReturnType, codeElement);
var requestBodyParam = codeElement.Parameters.OfKind(CodeParameterKind.RequestBody);
var queryStringParam = codeElement.Parameters.OfKind(CodeParameterKind.QueryParameter);
var headersParam = codeElement.Parameters.OfKind(CodeParameterKind.Headers);
var optionsParam = codeElement.Parameters.OfKind(CodeParameterKind.Options);
var requestParams = new RequestParams(requestBodyParam, queryStringParam, headersParam, optionsParam);
WriteCommandBuilderBody(codeElement, requestParams, isVoid, returnType, writer);
}
}

protected void WriteCommandBuilderBody(CodeMethod codeElement, RequestParams requestParams, bool isVoid, string returnType, LanguageWriter writer)
{
if (codeElement.HttpMethod == null)
{
// Build method
// Puts together the BuildXXCommand objects. Needs a nav property name e.g. users
// Command("users") -> Command("get")
} else
{
var isStream = conventions.StreamTypeName.Equals(returnType, StringComparison.OrdinalIgnoreCase);
var generatorMethodName = (codeElement.Parent as CodeClass)
.Methods
.FirstOrDefault(x => x.IsOfKind(CodeMethodKind.RequestGenerator) && x.HttpMethod == codeElement.HttpMethod)
?.Name;
var parametersList = new CodeParameter[] { requestParams.requestBody, requestParams.queryString, requestParams.headers, requestParams.options }
.Select(x => x?.Name).Where(x => x != null).Aggregate((x, y) => $"{x}, {y}");
writer.WriteLine($"var command = new Command(\"{codeElement.HttpMethod.ToString().ToLower()}\") {{");
writer.IncreaseIndent();
writer.WriteLine($"Handler = CommandHandler.Create<>(async () => {{");
writer.IncreaseIndent();
writer.WriteLine($"var requestInfo = {generatorMethodName}({parametersList});");
writer.WriteLine($"{(isVoid ? string.Empty : "return ")}await HttpCore.{GetSendRequestMethodName(isVoid, isStream, codeElement.ReturnType.IsCollection, returnType)}(requestInfo, responseHandler);");
writer.DecreaseIndent();
writer.WriteLine("})");
writer.DecreaseIndent();
writer.WriteLine("};");
writer.WriteLine("// Create options for all the parameters"); // investigate exploding query params

foreach (var option in codeElement.Parameters)
{
if (option.ParameterKind == CodeParameterKind.ResponseHandler)
{
continue;
}
var optionBuilder = new StringBuilder("new Option(");
optionBuilder.Append($"\"{option.Name}\"");
if (option.DefaultValue != null)
{
optionBuilder.Append($", getDefaultValue: ()=> {option.DefaultValue}");
}

if (!String.IsNullOrEmpty(option.Description))
{
optionBuilder.Append($", description: \"{option.Description}\"");
}

var isStream = conventions.StreamTypeName.Equals(returnType, StringComparison.OrdinalIgnoreCase);
var generatorMethodName = (codeElement.Parent as CodeClass)
.Methods
.FirstOrDefault(x => x.IsOfKind(CodeMethodKind.RequestGenerator) && x.HttpMethod == codeElement.HttpMethod)
?.Name;
var parametersList = new CodeParameter[] { requestParams.requestBody, requestParams.queryString, requestParams.headers, requestParams.options }
.Select(x => x?.Name).Where(x => x != null).Aggregate((x, y) => $"{x}, {y}");
writer.WriteLine($"var command = new Command(\"{codeElement.HttpMethod.ToString().ToLower()}\") {{");
writer.IncreaseIndent();
writer.WriteLine($"Handler = CommandHandler.Create<>(async () => {{");
writer.IncreaseIndent();
writer.WriteLine($"var requestInfo = {generatorMethodName}({parametersList});");
writer.WriteLine($"{(isVoid ? string.Empty : "return ")}await HttpCore.{GetSendRequestMethodName(isVoid, isStream, codeElement.ReturnType.IsCollection, returnType)}(requestInfo, responseHandler);");
writer.DecreaseIndent();
writer.WriteLine("})");
writer.DecreaseIndent();
writer.WriteLine("};");
writer.WriteLine("// Create options for all the parameters");
optionBuilder.Append(')');
writer.WriteLine($"command.AddOption({optionBuilder});");
writer.WriteLine($"// {option.Type.Name}"); //GetTypeString
}
}
}
}
}