Working on a client code generation feature to help speed up using an api built with this library.
This commit is contained in:
parent
f8704e425b
commit
19ccdb9026
@ -34,6 +34,10 @@ namespace MontoyaTech.Rest.Net.Example
|
||||
new Route(HttpRequestMethod.Get, "/json", Json)
|
||||
);
|
||||
|
||||
Console.WriteLine(CodeGenerator.GenerateCSharpClient(listener.Routes));
|
||||
|
||||
Console.ReadLine();
|
||||
|
||||
listener.RequestPreProcessEvent += (HttpListenerContext context) => {
|
||||
Console.WriteLine("Request start: " + context.Request.RawUrl);
|
||||
return true;
|
||||
@ -56,21 +60,25 @@ namespace MontoyaTech.Rest.Net.Example
|
||||
listener.Block();
|
||||
}
|
||||
|
||||
[RouteGroup("Test")]
|
||||
public static HttpListenerResponse Status(HttpListenerContext context)
|
||||
{
|
||||
return context.Response.WithStatus(HttpStatusCode.OK).WithText("Everything is operational. 👍");
|
||||
}
|
||||
|
||||
[RouteGroup("Test")]
|
||||
public static HttpListenerResponse Add(HttpListenerContext context, double a, double b)
|
||||
{
|
||||
return context.Response.WithStatus(HttpStatusCode.OK).WithText((a + b).ToString());
|
||||
}
|
||||
|
||||
[RouteGroup("Test")]
|
||||
public static HttpListenerResponse Signup(HttpListenerContext context, User user)
|
||||
{
|
||||
return context.Response.WithStatus(HttpStatusCode.OK).WithText("User:" + user.Name);
|
||||
}
|
||||
|
||||
[RouteGroup("Test")]
|
||||
public static HttpListenerResponse Json(HttpListenerContext context)
|
||||
{
|
||||
return context.Response.WithStatus(HttpStatusCode.OK).WithJson(new User("Rest.Net"));
|
||||
|
78
Rest.Net/CodeGenerator.cs
Normal file
78
Rest.Net/CodeGenerator.cs
Normal file
@ -0,0 +1,78 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.ComponentModel.DataAnnotations;
|
||||
using System.Linq;
|
||||
using System.Reflection;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace MontoyaTech.Rest.Net
|
||||
{
|
||||
/// <summary>
|
||||
/// A class that can take a set of routes and generate code
|
||||
/// for a client that can be used to interact with them.
|
||||
/// </summary>
|
||||
public class CodeGenerator
|
||||
{
|
||||
/// <summary>
|
||||
/// Generates a CSharp client for a given set of routes.
|
||||
/// </summary>
|
||||
/// <param name="routes"></param>
|
||||
/// <returns></returns>
|
||||
public static string GenerateCSharpClient(IList<Route> routes)
|
||||
{
|
||||
var writer = new CodeWriter();
|
||||
|
||||
writer.WriteLine("public class Client").WriteLine("{").Indent();
|
||||
writer.WriteSpacer();
|
||||
writer.WriteLine("public string BaseUrl = null;");
|
||||
writer.WriteSpacer();
|
||||
writer.WriteLine("public Client(string baseUrl)").WriteLine("{").Indent();
|
||||
writer.WriteLine("this.BaseUrl = baseUrl;");
|
||||
writer.Outdent().WriteLine("}");
|
||||
|
||||
GenerateCSharpRouteClasses(routes, writer);
|
||||
|
||||
writer.Outdent().WriteLine("}");
|
||||
|
||||
return writer.ToString();
|
||||
}
|
||||
|
||||
private static void GenerateCSharpRouteClasses(IList<Route> routes, CodeWriter writer)
|
||||
{
|
||||
var groups = new Dictionary<string, List<Route>>();
|
||||
|
||||
//Go through all the routes and group them.
|
||||
foreach (var route in routes)
|
||||
{
|
||||
//Get the method that this route is tied to.
|
||||
var methodInfo = route.Target.GetMethodInfo();
|
||||
|
||||
//See if this method defines the group for us.
|
||||
var routeGroup = methodInfo.GetCustomAttribute<RouteGroup>();
|
||||
|
||||
//Group this route.
|
||||
string group = (routeGroup != null ? routeGroup.Name : methodInfo.DeclaringType.Name);
|
||||
|
||||
if (groups.ContainsKey(group))
|
||||
groups[group].Add(route);
|
||||
else
|
||||
groups.Add(group, new List<Route>() { route });
|
||||
}
|
||||
}
|
||||
|
||||
private static void GenerateCSharpRouteClass(string name, List<Route> routes, CodeWriter writer)
|
||||
{
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Generates a Javascript client for a given set of routes.
|
||||
/// </summary>
|
||||
/// <param name="routes"></param>
|
||||
/// <returns></returns>
|
||||
public static string GenerateJavascriptClient(IList<Route> routes)
|
||||
{
|
||||
return null;
|
||||
}
|
||||
}
|
||||
}
|
398
Rest.Net/CodeWriter.cs
Normal file
398
Rest.Net/CodeWriter.cs
Normal file
@ -0,0 +1,398 @@
|
||||
using Newtonsoft.Json.Linq;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace MontoyaTech.Rest.Net
|
||||
{
|
||||
/// <summary>
|
||||
/// The outline of a writer that helps with generating code.
|
||||
/// </summary>
|
||||
internal class CodeWriter
|
||||
{
|
||||
/// <summary>
|
||||
/// The internal string builder.
|
||||
/// </summary>
|
||||
private StringBuilder Builder = new StringBuilder();
|
||||
|
||||
/// <summary>
|
||||
/// The current number of characters written..
|
||||
/// </summary>
|
||||
public int Length
|
||||
{
|
||||
get
|
||||
{
|
||||
return Builder.Length;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// The current number of indents.
|
||||
/// </summary>
|
||||
private int Indents = 0;
|
||||
|
||||
/// <summary>
|
||||
/// Whether or not the writer is pending an indent that needs
|
||||
/// to be handled.
|
||||
/// </summary>
|
||||
private bool PendingIndent = false;
|
||||
|
||||
/// <summary>
|
||||
/// Creates a new default CodeWriter.
|
||||
/// </summary>
|
||||
public CodeWriter() { }
|
||||
|
||||
/// <summary>
|
||||
/// Creates a new default text writer and copies the indent data from the passed
|
||||
/// text writer.
|
||||
/// </summary>
|
||||
/// <param name="writer"></param>
|
||||
public CodeWriter(CodeWriter writer)
|
||||
{
|
||||
this.Indents = writer.Indents;
|
||||
|
||||
//If we have indents, then we must set pending indent.
|
||||
if (this.Indents > 0)
|
||||
this.PendingIndent = true;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes text to the writer.
|
||||
/// </summary>
|
||||
/// <param name="text"></param>
|
||||
public CodeWriter Write(string text)
|
||||
{
|
||||
if (this.PendingIndent)
|
||||
{
|
||||
for (var i = 0; i < Indents; i++)
|
||||
this.Builder.Append(' ');
|
||||
|
||||
this.PendingIndent = false;
|
||||
}
|
||||
|
||||
this.Builder.Append(text);
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes a character to the writer.
|
||||
/// </summary>
|
||||
/// <param name="char"></param>
|
||||
public CodeWriter Write(char @char)
|
||||
{
|
||||
if (this.PendingIndent)
|
||||
{
|
||||
for (var i = 0; i < Indents; i++)
|
||||
this.Builder.Append(' ');
|
||||
|
||||
this.PendingIndent = false;
|
||||
}
|
||||
|
||||
this.Builder.Append(@char);
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes text to the writer and a newline.
|
||||
/// </summary>
|
||||
/// <param name="text"></param>
|
||||
public CodeWriter WriteLine(string text)
|
||||
{
|
||||
if (this.PendingIndent)
|
||||
{
|
||||
for (var i = 0; i < Indents; i++)
|
||||
this.Builder.Append(' ');
|
||||
|
||||
this.PendingIndent = false;
|
||||
}
|
||||
|
||||
this.Builder.Append(text);
|
||||
this.Builder.Append('\n');
|
||||
this.PendingIndent = true;
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes a character and new line to the writer.
|
||||
/// </summary>
|
||||
/// <param name="char"></param>
|
||||
public CodeWriter WriteLine(char @char)
|
||||
{
|
||||
if (this.PendingIndent)
|
||||
{
|
||||
for (var i = 0; i < Indents; i++)
|
||||
this.Builder.Append(' ');
|
||||
|
||||
this.PendingIndent = false;
|
||||
}
|
||||
|
||||
this.Builder.Append(@char);
|
||||
this.Builder.Append('\n');
|
||||
this.PendingIndent = true;
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes text to the writer if the condition is met.
|
||||
/// </summary>
|
||||
/// <param name="condition"></param>
|
||||
/// <param name="text"></param>
|
||||
/// <returns></returns>
|
||||
public CodeWriter WriteAssert(bool condition, string text)
|
||||
{
|
||||
if (condition)
|
||||
return this.Write(text);
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes a character to the writer if the condition is met.
|
||||
/// </summary>
|
||||
/// <param name="condition"></param>
|
||||
/// <param name="char"></param>
|
||||
/// <returns></returns>
|
||||
public CodeWriter WriteAssert(bool condition, char @char)
|
||||
{
|
||||
if (condition)
|
||||
return this.Write(@char);
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes text to the writer and a newline if the condition is met.
|
||||
/// </summary>
|
||||
/// <param name="condition"></param>
|
||||
/// <param name="text"></param>
|
||||
/// <returns></returns>
|
||||
public CodeWriter WriteLineAssert(bool condition, string text)
|
||||
{
|
||||
if (condition)
|
||||
return this.WriteLine(text);
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes a character and new line to the writer if the condition is met.
|
||||
/// </summary>
|
||||
/// <param name="condition"></param>
|
||||
/// <param name="char"></param>
|
||||
/// <returns></returns>
|
||||
public CodeWriter WriteLineAssert(bool condition, char @char)
|
||||
{
|
||||
if (condition)
|
||||
return this.WriteLine(@char);
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes the contents of another text writer into this one.
|
||||
/// </summary>
|
||||
/// <param name="writer"></param>
|
||||
public CodeWriter Write(CodeWriter writer)
|
||||
{
|
||||
this.Builder.Append(writer.Builder.ToString());
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Inserts a new line to the writer.
|
||||
/// </summary>
|
||||
public CodeWriter NewLine()
|
||||
{
|
||||
this.Builder.Append('\n');
|
||||
this.PendingIndent = true;
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Inserts a new line to the writer if the condition is met.
|
||||
/// </summary>
|
||||
public CodeWriter NewLineAssert(bool condition)
|
||||
{
|
||||
if (condition)
|
||||
{
|
||||
this.Builder.Append('\n');
|
||||
this.PendingIndent = true;
|
||||
}
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes a space to the writer unless there is already one.
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
public CodeWriter WriteSpacer()
|
||||
{
|
||||
if (this.Builder.Length >= 1 && this.Builder[this.Builder.Length - 1] != ' ')
|
||||
this.Builder.Append(' ');
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes a separator to the writer unless one wouldn't make sense.
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
public CodeWriter WriteSeparator()
|
||||
{
|
||||
if (this.Builder.Length > 0)
|
||||
{
|
||||
var offset = this.Builder.Length - 1;
|
||||
var prev = this.Builder[offset];
|
||||
while (offset >= 0 && (prev == ' ' || prev == '\t' || prev == '\n' || prev == '\r'))
|
||||
{
|
||||
offset--;
|
||||
prev = this.Builder[offset];
|
||||
}
|
||||
|
||||
if (prev != '(' && prev != '{' && prev != '[' && prev != ',')
|
||||
this.Builder.Append(", ");
|
||||
}
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes a separator to the writer and text if a condition is met unless a separator here isn't needed, then just text is written.
|
||||
/// </summary>
|
||||
/// <param name="condition"></param>
|
||||
/// <param name="text"></param>
|
||||
/// <returns></returns>
|
||||
public CodeWriter WriteSeparatorAssert(bool condition, string text)
|
||||
{
|
||||
if (condition)
|
||||
{
|
||||
if (this.Builder.Length > 0)
|
||||
{
|
||||
var offset = this.Builder.Length - 1;
|
||||
var prev = this.Builder[offset];
|
||||
while (offset >= 0 && (prev == ' ' || prev == '\t' || prev == '\n' || prev == '\r'))
|
||||
{
|
||||
offset--;
|
||||
prev = this.Builder[offset];
|
||||
}
|
||||
|
||||
if (prev != '(' && prev != '{' && prev != '[' && prev != ',')
|
||||
this.Builder.Append(", ");
|
||||
}
|
||||
|
||||
if (text != null)
|
||||
this.Builder.Append(text);
|
||||
}
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes a blank line as a spacer to the writer
|
||||
/// unless there is already one.
|
||||
/// </summary>
|
||||
public CodeWriter WriteBreak()
|
||||
{
|
||||
if (this.Builder.Length >= 2)
|
||||
{
|
||||
var prevChar = this.Builder[this.Builder.Length - 2];
|
||||
|
||||
if (prevChar != '\n')
|
||||
{
|
||||
this.Builder.Append('\n');
|
||||
this.PendingIndent = true;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
this.Builder.Append('\n');
|
||||
this.PendingIndent = true;
|
||||
}
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Indents the writer one time.
|
||||
/// </summary>
|
||||
public CodeWriter Indent()
|
||||
{
|
||||
this.Indents += 4;
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Indents the writer one time if the condition is met.
|
||||
/// </summary>
|
||||
public CodeWriter IndentAssert(bool condition)
|
||||
{
|
||||
if (condition)
|
||||
this.Indents += 4;
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Removes one indent from the writer.
|
||||
/// </summary>
|
||||
public CodeWriter Outdent()
|
||||
{
|
||||
if (this.Indents > 0)
|
||||
this.Indents -= 4;
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Removes one indent from the writer if the condition is met.
|
||||
/// </summary>
|
||||
public CodeWriter OutdentAssert(bool condition)
|
||||
{
|
||||
if (condition && this.Indents > 0)
|
||||
this.Indents -= 4;
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Resets all indents from the writer.
|
||||
/// </summary>
|
||||
public CodeWriter ResetIndents()
|
||||
{
|
||||
this.Indents = 0;
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Removes the last character from the writer.
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
public CodeWriter Remove()
|
||||
{
|
||||
if (this.Builder.Length > 0)
|
||||
this.Builder.Remove(this.Builder.Length - 1, 1);
|
||||
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets all the written data from the writer.
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
public override string ToString()
|
||||
{
|
||||
return this.Builder.ToString();
|
||||
}
|
||||
}
|
||||
}
|
@ -25,7 +25,7 @@ namespace MontoyaTech.Rest.Net
|
||||
/// <summary>
|
||||
/// The target function to invoke if this route is invoked.
|
||||
/// </summary>
|
||||
private Func<HttpListenerContext, HttpListenerResponse> Target;
|
||||
internal Func<HttpListenerContext, HttpListenerResponse> Target;
|
||||
|
||||
/// <summary>
|
||||
/// Whether or not to close the response after the route is invoked.
|
||||
|
37
Rest.Net/RouteGroup.cs
Normal file
37
Rest.Net/RouteGroup.cs
Normal file
@ -0,0 +1,37 @@
|
||||
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 controls what group a route belongs to.
|
||||
/// </summary>
|
||||
[AttributeUsage(AttributeTargets.Method, AllowMultiple = false, Inherited = true)]
|
||||
public class RouteGroup : Attribute
|
||||
{
|
||||
/// <summary>
|
||||
/// The name of the group this Route belongs to.
|
||||
/// </summary>
|
||||
public string Name = null;
|
||||
|
||||
/// <summary>
|
||||
/// Creates a new default route group.
|
||||
/// </summary>
|
||||
public RouteGroup() { }
|
||||
|
||||
/// <summary>
|
||||
/// Creates a new route group with the name of the group.
|
||||
/// </summary>
|
||||
/// <param name="name"></param>
|
||||
public RouteGroup(string name)
|
||||
{
|
||||
if (string.IsNullOrWhiteSpace(name))
|
||||
throw new ArgumentException("Must not be null or empty", nameof(name));
|
||||
|
||||
this.Name = name;
|
||||
}
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user