Working on C# client code generation.

This commit is contained in:
MattMo 2023-02-04 08:26:09 -08:00
parent 19ccdb9026
commit 6bb01464e7
7 changed files with 298 additions and 9 deletions

View File

@ -0,0 +1,47 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace MontoyaTech.Rest.Net.Example
{
public class Client
{
public string BaseUrl = null;
public TestFunctions Test;
public class TestFunctions
{
public Client Client;
public TestFunctions(Client client)
{
this.Client = client;
}
public void Status()
{
}
public void Add()
{
}
public void Signup()
{
}
public void Json()
{
}
}
public Client(string baseUrl)
{
this.BaseUrl = baseUrl;
this.Test = new TestFunctions(this);
}
}
}

View File

@ -34,7 +34,9 @@ namespace MontoyaTech.Rest.Net.Example
new Route(HttpRequestMethod.Get, "/json", Json)
);
Console.WriteLine(CodeGenerator.GenerateCSharpClient(listener.Routes));
string code = CodeGenerator.GenerateCSharpClient(listener.Routes);
Console.WriteLine(code);
Console.ReadLine();
@ -61,24 +63,35 @@ namespace MontoyaTech.Rest.Net.Example
}
[RouteGroup("Test")]
[RouteResponse(typeof(string))]
public static HttpListenerResponse Status(HttpListenerContext context)
{
return context.Response.WithStatus(HttpStatusCode.OK).WithText("Everything is operational. 👍");
}
[RouteGroup("Test")]
[RouteResponse(typeof(string))]
public static HttpListenerResponse Add(HttpListenerContext context, double a, double b)
{
return context.Response.WithStatus(HttpStatusCode.OK).WithText((a + b).ToString());
}
[RouteGroup("Test")]
[RouteResponse(typeof(string))]
public static HttpListenerResponse Signup(HttpListenerContext context, User user)
{
return context.Response.WithStatus(HttpStatusCode.OK).WithText("User:" + user.Name);
}
[RouteGroup("Test")]
[RouteRequest(typeof(User))]
public static HttpListenerResponse SignupRequest(HttpListenerContext context)
{
return context.Response.WithStatus(HttpStatusCode.OK);
}
[RouteGroup("Test")]
[RouteResponse(typeof(User))]
public static HttpListenerResponse Json(HttpListenerContext context)
{
return context.Response.WithStatus(HttpStatusCode.OK).WithJson(new User("Rest.Net"));

View File

@ -24,21 +24,25 @@ namespace MontoyaTech.Rest.Net
var writer = new CodeWriter();
writer.WriteLine("public class Client").WriteLine("{").Indent();
writer.WriteSpacer();
writer.WriteLine("public string BaseUrl = null;");
writer.WriteSpacer();
GenerateCSharpRouteClasses(routes, writer, out List<string> routeClasses);
writer.WriteBreak();
writer.WriteLine("public Client(string baseUrl)").WriteLine("{").Indent();
writer.WriteLine("this.BaseUrl = baseUrl;");
writer.Outdent().WriteLine("}");
GenerateCSharpRouteClasses(routes, writer);
foreach (var @class in routeClasses)
writer.WriteLine($"this.{@class} = new {@class}Functions(this);");
writer.Outdent().WriteLine("}");
writer.Outdent().WriteLine("}");
return writer.ToString();
}
private static void GenerateCSharpRouteClasses(IList<Route> routes, CodeWriter writer)
private static void GenerateCSharpRouteClasses(IList<Route> routes, CodeWriter writer, out List<string> routeClasses)
{
var groups = new Dictionary<string, List<Route>>();
@ -46,11 +50,15 @@ namespace MontoyaTech.Rest.Net
foreach (var route in routes)
{
//Get the method that this route is tied to.
var methodInfo = route.Target.GetMethodInfo();
var methodInfo = route.GetTarget().GetMethodInfo();
//See if this method defines the group for us.
var routeGroup = methodInfo.GetCustomAttribute<RouteGroup>();
//If there wasn't a route group on the method, see if there is one on the declaring type.
if (routeGroup == null)
routeGroup = methodInfo.DeclaringType.GetCustomAttribute<RouteGroup>();
//Group this route.
string group = (routeGroup != null ? routeGroup.Name : methodInfo.DeclaringType.Name);
@ -59,10 +67,51 @@ namespace MontoyaTech.Rest.Net
else
groups.Add(group, new List<Route>() { route });
}
//Generate classes for these groups
foreach (var group in groups)
GenerateCSharpRouteClass(group.Key, group.Value, writer);
//Set the route classes
routeClasses = new List<string>();
routeClasses.AddRange(groups.Select(group => group.Key));
}
private static void GenerateCSharpRouteClass(string name, List<Route> routes, CodeWriter writer)
{
writer.WriteBreak();
writer.WriteLine($"public {name}Functions {name};");
writer.WriteBreak();
writer.WriteLine($"public class {name}Functions").WriteLine("{").Indent();
writer.WriteLine("public Client Client;");
writer.WriteBreak();
writer.WriteLine($"public {name}Functions(Client client)").WriteLine("{").Indent();
writer.WriteLine("this.Client = client;");
writer.Outdent().WriteLine("}");
foreach (var route in routes)
GenerateCSharpRouteFunction(route, writer);
writer.Outdent().WriteLine("}");
}
private static void GenerateCSharpRouteFunction(Route route, CodeWriter writer)
{
writer.WriteBreak();
var methodInfo = route.GetTarget().GetMethodInfo();
writer.WriteLine($"public void {methodInfo.Name}()").WriteLine("{").Indent();
writer.Outdent().WriteLine("}");
}
/// <summary>

View File

@ -25,7 +25,7 @@ namespace MontoyaTech.Rest.Net
/// <summary>
/// The target function to invoke if this route is invoked.
/// </summary>
internal Func<HttpListenerContext, HttpListenerResponse> Target;
private Func<HttpListenerContext, HttpListenerResponse> Target;
/// <summary>
/// Whether or not to close the response after the route is invoked.
@ -47,6 +47,13 @@ namespace MontoyaTech.Rest.Net
/// <param name="closeResponse"></param>
public Route(string method, string syntax, Func<HttpListenerContext, HttpListenerResponse> target, bool closeResponse = true)
{
if (target == null)
throw new ArgumentNullException(nameof(target));
else if (string.IsNullOrWhiteSpace(method))
throw new ArgumentException("Cannot be null or empty", nameof(method));
else if (string.IsNullOrWhiteSpace(syntax))
throw new ArgumentException("Cannot be null or empty", nameof(syntax));
this.Method = method;
this.Syntax = syntax;
this.Target = target;
@ -117,6 +124,15 @@ namespace MontoyaTech.Rest.Net
{
this.Target.Invoke(context);
}
/// <summary>
/// Returns the target delegate for this route.
/// </summary>
/// <returns></returns>
public virtual Delegate GetTarget()
{
return this.Target;
}
}
public class Route<T1> : Route
@ -125,6 +141,13 @@ namespace MontoyaTech.Rest.Net
public Route(string method, string syntax, Func<HttpListenerContext, T1, HttpListenerResponse> target, bool closeResponse = true)
{
if (target == null)
throw new ArgumentNullException(nameof(target));
else if (string.IsNullOrWhiteSpace(method))
throw new ArgumentException("Cannot be null or empty", nameof(method));
else if (string.IsNullOrWhiteSpace(syntax))
throw new ArgumentException("Cannot be null or empty", nameof(syntax));
this.Method = method;
this.Syntax = syntax;
this.Target = target;
@ -138,6 +161,11 @@ namespace MontoyaTech.Rest.Net
{
this.Target.DynamicInvoke(context, RouteArgumentConverter.Convert<T1>(arguments[0]));
}
public override Delegate GetTarget()
{
return this.Target;
}
}
public class Route<T1, T2> : Route
@ -146,6 +174,13 @@ namespace MontoyaTech.Rest.Net
public Route(string method, string syntax, Func<HttpListenerContext, T1, T2, HttpListenerResponse> target, bool closeResponse = true)
{
if (target == null)
throw new ArgumentNullException(nameof(target));
else if (string.IsNullOrWhiteSpace(method))
throw new ArgumentException("Cannot be null or empty", nameof(method));
else if (string.IsNullOrWhiteSpace(syntax))
throw new ArgumentException("Cannot be null or empty", nameof(syntax));
this.Method = method;
this.Syntax = syntax;
this.Target = target;
@ -163,6 +198,11 @@ namespace MontoyaTech.Rest.Net
RouteArgumentConverter.Convert<T2>(arguments[1])
);
}
public override Delegate GetTarget()
{
return this.Target;
}
}
public class Route<T1, T2, T3> : Route
@ -171,6 +211,13 @@ namespace MontoyaTech.Rest.Net
public Route(string method, string syntax, Func<HttpListenerContext, T1, T2, T3, HttpListenerResponse> target, bool closeResponse = true)
{
if (target == null)
throw new ArgumentNullException(nameof(target));
else if (string.IsNullOrWhiteSpace(method))
throw new ArgumentException("Cannot be null or empty", nameof(method));
else if (string.IsNullOrWhiteSpace(syntax))
throw new ArgumentException("Cannot be null or empty", nameof(syntax));
this.Method = method;
this.Syntax = syntax;
this.Target = target;
@ -189,6 +236,11 @@ namespace MontoyaTech.Rest.Net
RouteArgumentConverter.Convert<T3>(arguments[2])
);
}
public override Delegate GetTarget()
{
return this.Target;
}
}
public class Route<T1, T2, T3, T4> : Route
@ -197,6 +249,13 @@ namespace MontoyaTech.Rest.Net
public Route(string method, string syntax, Func<HttpListenerContext, T1, T2, T3, T4, HttpListenerResponse> target, bool closeResponse = true)
{
if (target == null)
throw new ArgumentNullException(nameof(target));
else if (string.IsNullOrWhiteSpace(method))
throw new ArgumentException("Cannot be null or empty", nameof(method));
else if (string.IsNullOrWhiteSpace(syntax))
throw new ArgumentException("Cannot be null or empty", nameof(syntax));
this.Method = method;
this.Syntax = syntax;
this.Target = target;
@ -216,6 +275,11 @@ namespace MontoyaTech.Rest.Net
RouteArgumentConverter.Convert<T4>(arguments[3])
);
}
public override Delegate GetTarget()
{
return this.Target;
}
}
public class Route<T1, T2, T3, T4, T5> : Route
@ -224,6 +288,13 @@ namespace MontoyaTech.Rest.Net
public Route(string method, string syntax, Func<HttpListenerContext, T1, T2, T3, T4, T5, HttpListenerResponse> target, bool closeResponse = true)
{
if (target == null)
throw new ArgumentNullException(nameof(target));
else if (string.IsNullOrWhiteSpace(method))
throw new ArgumentException("Cannot be null or empty", nameof(method));
else if (string.IsNullOrWhiteSpace(syntax))
throw new ArgumentException("Cannot be null or empty", nameof(syntax));
this.Method = method;
this.Syntax = syntax;
this.Target = target;
@ -244,6 +315,11 @@ namespace MontoyaTech.Rest.Net
RouteArgumentConverter.Convert<T5>(arguments[4])
);
}
public override Delegate GetTarget()
{
return this.Target;
}
}
public class Route<T1, T2, T3, T4, T5, T6> : Route
@ -252,6 +328,13 @@ namespace MontoyaTech.Rest.Net
public Route(string method, string syntax, Func<HttpListenerContext, T1, T2, T3, T4, T5, T6, HttpListenerResponse> target, bool closeResponse = true)
{
if (target == null)
throw new ArgumentNullException(nameof(target));
else if (string.IsNullOrWhiteSpace(method))
throw new ArgumentException("Cannot be null or empty", nameof(method));
else if (string.IsNullOrWhiteSpace(syntax))
throw new ArgumentException("Cannot be null or empty", nameof(syntax));
this.Method = method;
this.Syntax = syntax;
this.Target = target;
@ -273,6 +356,11 @@ namespace MontoyaTech.Rest.Net
RouteArgumentConverter.Convert<T6>(arguments[5])
);
}
public override Delegate GetTarget()
{
return this.Target;
}
}
public class Route<T1, T2, T3, T4, T5, T6, T7> : Route
@ -281,6 +369,13 @@ namespace MontoyaTech.Rest.Net
public Route(string method, string syntax, Func<HttpListenerContext, T1, T2, T3, T4, T5, T6, T7, HttpListenerResponse> target, bool closeResponse = true)
{
if (target == null)
throw new ArgumentNullException(nameof(target));
else if (string.IsNullOrWhiteSpace(method))
throw new ArgumentException("Cannot be null or empty", nameof(method));
else if (string.IsNullOrWhiteSpace(syntax))
throw new ArgumentException("Cannot be null or empty", nameof(syntax));
this.Method = method;
this.Syntax = syntax;
this.Target = target;
@ -303,6 +398,11 @@ namespace MontoyaTech.Rest.Net
RouteArgumentConverter.Convert<T7>(arguments[6])
);
}
public override Delegate GetTarget()
{
return this.Target;
}
}
public class Route<T1, T2, T3, T4, T5, T6, T7, T8> : Route
@ -311,6 +411,13 @@ namespace MontoyaTech.Rest.Net
public Route(string method, string syntax, Func<HttpListenerContext, T1, T2, T3, T4, T5, T6, T7, T8, HttpListenerResponse> target, bool closeResponse = true)
{
if (target == null)
throw new ArgumentNullException(nameof(target));
else if (string.IsNullOrWhiteSpace(method))
throw new ArgumentException("Cannot be null or empty", nameof(method));
else if (string.IsNullOrWhiteSpace(syntax))
throw new ArgumentException("Cannot be null or empty", nameof(syntax));
this.Method = method;
this.Syntax = syntax;
this.Target = target;
@ -334,5 +441,10 @@ namespace MontoyaTech.Rest.Net
RouteArgumentConverter.Convert<T8>(arguments[7])
);
}
public override Delegate GetTarget()
{
return this.Target;
}
}
}

View File

@ -9,7 +9,7 @@ namespace MontoyaTech.Rest.Net
/// <summary>
/// The outline of an attribute that controls what group a route belongs to.
/// </summary>
[AttributeUsage(AttributeTargets.Method, AllowMultiple = false, Inherited = true)]
[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, AllowMultiple = false, Inherited = true)]
public class RouteGroup : Attribute
{
/// <summary>

34
Rest.Net/RouteRequest.cs Normal file
View File

@ -0,0 +1,34 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace MontoyaTech.Rest.Net
{
/// <summary>
/// The outline of an attribute that defines a routes request type.
/// </summary>
[AttributeUsage(AttributeTargets.Method, AllowMultiple = false, Inherited = true)]
public class RouteRequest : Attribute
{
/// <summary>
/// The type of the request.
/// </summary>
public Type RequestType = null;
/// <summary>
/// Creates a default route request.
/// </summary>
public RouteRequest() { }
/// <summary>
/// Creates a route request with the request type.
/// </summary>
/// <param name="requestType"></param>
public RouteRequest(Type requestType)
{
this.RequestType = requestType;
}
}
}

34
Rest.Net/RouteResponse.cs Normal file
View File

@ -0,0 +1,34 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace MontoyaTech.Rest.Net
{
/// <summary>
/// The outline of an attribute that defines a routes response type.
/// </summary>
[AttributeUsage(AttributeTargets.Method, AllowMultiple = false, Inherited = true)]
public class RouteResponse : Attribute
{
/// <summary>
/// The type of the request.
/// </summary>
public Type ResponseType = null;
/// <summary>
/// Creates a default route response.
/// </summary>
public RouteResponse() { }
/// <summary>
/// Creates a route response with the response type.
/// </summary>
/// <param name="responseType"></param>
public RouteResponse(Type responseType)
{
this.ResponseType = responseType;
}
}
}