LightNode is a Micro RPC/REST Framework built on OWIN. LightNode is a good alternative to the ASP.NET Web API and Nancy if you make a simple API. It is like Ruby's Grape framework, Slack API's HTTP RPC-style methods. Implementation of the API is lightweight, powerful debugging supports with Glimpse, client code generation by T4 for PCL(HttpClient) and Unity3D.
LightNode 2 has been started. You can install from -Pre
packages.
PM> Install-Package LightNode -Pre
Simple ASP.NET Core Startup
using LightNode;
public class Startup
{
public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
{
app.UseLightNode(typeof(Startup));
}
}
LightNode 2 includes swagger package. You can map for Swagger.
public class Startup
{
public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
{
app.Map("/api", builder =>
{
builder.UseLightNode(typeof(Startup));
});
app.Map("/swagger", builder =>
{
var xmlName = "AspNetCoreSample.xml";
var xmlPath = Path.Combine(Path.GetDirectoryName(Assembly.GetEntryAssembly().Location), xmlName);
builder.UseLightNodeSwagger(new LightNode.Swagger.SwaggerOptions("AspNetCoreSample", "/api")
{
XmlDocumentPath = xmlPath,
IsEmitEnumAsString = true
});
});
}
}
binary from NuGet, LightNode.Server
PM> Install-Package LightNode.Server
Server implementation is very easy, built up Owin and implements LightNodeContract
.
// Owin Startup
public class Startup
{
public void Configuration(IAppBuilder app)
{
app.UseLightNode();
}
}
// implement LightNodeContract, all public methods become API.
// You can access {ClassName}/{MethodName}
// Ex. http://localhost/My/Echo?x=test
public class My : LightNodeContract
{
// return value is response body serialized by ContentTypeFormatter(default is JSON).
public string Echo(string x)
{
return x;
}
// support async! return type allows void, T, Task and Task<T>.
// parameter supports array, nullable and optional parameter.
public Task<int> Sum(int x, int? y, int z = 1000)
{
return Task.Run(() => x + y.Value + z);
}
}
Compile, run, very quick! LightNode calls class as Contract, method as Operation.
Parameter model bindings supports only basic pattern, can't use complex type. allow types are "string, DateTime, DateTimeOffset, Boolean, Decimal, Char, TimeSpan, Int16, Int32, Int64, UInt16, UInt32, UInt64, Single, Double, SByte, Byte, Enum and each Nullable types and array(except byte[]. If you want to use byte[], use Base64 string instead of byte[] or see receive byte[] section)
Return type allows all serializable(ContentFormatter support) type.
LightNode supports filter. The implementation is like middleware pipeline.
public class SampleFilterAttribute : LightNodeFilterAttribute
{
public override async Task Invoke(OperationContext operationContext, Func<Task> next)
{
try
{
// OnBeforeAction
await next(); // next filter or operation handler
// OnAfterAction
}
catch
{
// OnExeception
}
finally
{
// OnFinally
}
}
}
Filter can be attached contract(class), operation(method) and global. Execution pipeline is formed is sorted by Order all specified. Range is -int.MaxValue to int.MaxValue. Default Order of all filters is int.MaxValue.
Difference between Middleware and Filter is who knows operation context. Filter is located after the parameter binding. Therefore, it is possible check attributes(operationContext.IsAttributeDefined
, operationContext.GetAttributes
).
The default status code, can't find operation returns 404, failed operation returns 500, success and has value returns 200, success and no value returns 204. If returns arbitrary status code, throw ReturnStatusCodeException
.
throw new ReturnStatusCodeException(System.Net.HttpStatusCode.Unauthorized);
LightNode fully supports Glimpse! Currently Glimpse does not support Owin but if you host on Microsoft.Owin.Host.SystemWeb
Glimpse works. You can download Glimpse plugin from NuGet.
There are configuration sample.
public void Configuration(Owin.IAppBuilder app)
{
app.EnableGlimpse(); // This is Glimpse.LightNode's helper for enable Glimpse
app.MapWhen(x => !x.Request.Path.Value.StartsWith("/glimpse.axd", StringComparison.OrdinalIgnoreCase), x =>
{
x.UseLightNode(new LightNodeOptions()
{
// for Glimpse Profiling
OperationCoordinatorFactory = new GlimpseProfilingOperationCoordinatorFactory()
});
});
}
Access glimpse.axd and Click Standalone Glimpse Launch Now! Click History window and Inspect. You can see Filter and Execution elapsed on Timeline tab.
Check the LightNode tab, you can monitor everything. Parameters, Result, Exectuion Phase, Response, and LightNodeOptions.
If encounts exception, LightNode tab shows exception on Result.
My recommended glimpse configuration.
<!-- sometimes Glimpse rewrite response for display tab, but API no needs, set RuntimePolicy PersitResults -->
<glimpse defaultRuntimePolicy="PersistResults" endpointBaseUri="~/Glimpse.axd">
<tabs>
<ignoredTypes>
<!-- no needs only Owin -->
<add type="Glimpse.AspNet.Tab.Cache, Glimpse.AspNet" />
<add type="Glimpse.AspNet.Tab.Routes, Glimpse.AspNet" />
<add type="Glimpse.AspNet.Tab.Session, Glimpse.AspNet" />
</ignoredTypes>
</tabs>
<runtimePolicies>
<ignoredTypes>
<!-- If API's client no use cookie, ignore control cookie -->
<add type="Glimpse.Core.Policy.ControlCookiePolicy, Glimpse.Core" />
<!-- for improvement LightNode debugging -->
<add type="Glimpse.Core.Policy.StatusCodePolicy, Glimpse.Core" />
<!-- If not Ajax -->
<add type="Glimpse.Core.Policy.AjaxPolicy, Glimpse.Core" />
<!-- If run on remote -->
<add type="Glimpse.AspNet.Policy.LocalPolicy, Glimpse.AspNet" />
</ignoredTypes>
</runtimePolicies>
</glimpse>
Ignore ControlCookiePolicy is very important. But we can't indistinguishable request. Glimpse handle group by cookie. You can add glimpseid cookie for example
var req = WebRequest.CreateHttp("http://localhost:41932/Member/Random?seed=13");
req.CookieContainer = new CookieContainer();
req.CookieContainer.Add(new Uri("http://localhost:41932"), new Cookie("glimpseid", "UserId:4"));
LightNode supports Swagger for API Explorer(currently Swagger supports is experimental, only shows parameters).
Middleware available in NuGet.
Swagger-UI file is embedded in LightNode.Swagger. You can enable only UseLightNodeSwagger
.
// Currently LightNode.Swagger only supports POST so you needs AcceptVerbs.Post
app.Map("/api", builder =>
{
builder.UseLightNode(new LightNodeOptions(AcceptVerbs.Get | AcceptVerbs.Post, new JilContentFormatter(), new GZipJilContentFormatter())
{
ParameterEnumAllowsFieldNameParse = true, // If you want to use enums human readable display on Swagger, set to true
ErrorHandlingPolicy = ErrorHandlingPolicy.ReturnInternalServerErrorIncludeErrorDetails,
OperationMissingHandlingPolicy = OperationMissingHandlingPolicy.ReturnErrorStatusCodeIncludeErrorDetails
});
});
// Mapping to swagger path
app.Map("/swagger", builder =>
{
// If you want to additional info for Swagger, load xmlDoc file.
// LightNode.Swagger loads methods's summary, remarks, param for info.
var xmlName = "LightNode.Sample.GlimpseUse.xml";
var xmlPath = System.AppDomain.CurrentDomain.BaseDirectory + "\\bin\\" + xmlName; // or HttpContext.Current.Server.MapPath("~/bin/" + xmlName);
builder.UseLightNodeSwagger(new Swagger.SwaggerOptions("LightNodeSample", "/api") // baseApi is LightNode's root
{
XmlDocumentPath = xmlPath,
IsEmitEnumAsString = true
});
});
Okay, for example jump to http://localhost:41932/Swagger/
, you can see all API info and Swagger specification json can download from api-default.json
. If you host multiple LightNode engine, you can select target engine from {engineID}.json
. {engineID}
is from ILightNodeOptions.ServerEngineId
.
If you can't run swagger on hosting IIS, maybe conflicts static file handling. Please remoe StaticFile handler and register OwinHttpHandler for all paths.
<system.webServer>
<handlers>
<remove name="StaticFile" />
<!-- If use with Glimpse, glimpse handler must be first -->
<add name="Glimpse" path="glimpse.axd" verb="GET" type="Glimpse.AspNet.HttpHandler, Glimpse.AspNet" preCondition="integratedMode" />
<add name="OWIN" path="*" verb="*" type="Microsoft.Owin.Host.SystemWeb.OwinHttpHandler" />
</handlers>
</system.webServer>
If you want to customize index.html(or others) for authentication etc. You can use ResolveCustomResource
.
app.Map("/swagger", builder =>
{
builder.UseLightNodeSwagger(new LightNode.Swagger.SwaggerOptions("MySample", "/api")
{
ResolveCustomResource = (filePath, loadedEmbeddedBytes) =>
{
if (filePath == "index.html")
{
using (var resourceStream = typeof(Startup).Assembly.GetManifestResourceStream("MySample.Swagger.index.html"))
using (var ms = new MemoryStream())
{
resourceStream.CopyTo(ms);
return ms.ToArray();
}
}
return loadedEmbeddedBytes;
}
});
});
You can use LightNode with ASP.NET MVC. A simple solution is to change the root path.
public void Configuration(IAppBuilder app)
{
app.Map("/api", x =>
{
x.UseLightNode();
});
}
Default content formatter is JavaScriptContentFormatter
(application/json) and bundling formatters are TextContentFormatter
(text/plain), HtmlContentFormatter
(text/html), RawOctetStreamContentFormatter
(application/octet-straem), XmlContentFormatter
(application/xml), DataContractContentFormatter
(application/xml), DataContractJsonContentFormatter
(application/json).
More useful ContentFormatters(for JsonNet(JSON), Jil(JSON/JSON+GZip), Jil+LZ4, ProtoBuf, MsgPack) are available.
Configuration sample
public class Startup
{
public void Configuration(Owin.IAppBuilder app)
{
// default is Json, If Accept-Encoding=gzip then Json+GZip
app.UseLightNode(new LightNodeOptions(AcceptVerbs.Get | AcceptVerbs.Post,
new JilContentFormatter(), new GZipJilContentFormatter()));
}
}
public class Sample : LightNodeContract
{
// use specified content formatter, select verb per operation
[OperationOption(AcceptVerbs.Get, typeof(HtmlContentFormatterFactory))]
public string Html()
{
return "<html><body>aaa</body></html>";
}
// LightNode's default is GET | POST and default can customize.
// change per operation verb by OperationOption(AcceptVerbs)
// [Get/Post/Put/Delete/Patch]Attribute for its shortcut
[Post]
public int PostOnly()
{
return 0;
}
}
LightNode isn't allow byte[] argument. If you want to receive byte[] that use Base64 string instead of byte[]. But you needs to avoid Base64, you can take raw stream from Environment
.
[Post, IgnoreClientGenerate]
public int PostByte() // zero argument
{
// Take raw stream
var body = this.Environment["owin.RequestBody"] as Stream;
byte[] bodyBytes;
using (var ms = new MemoryStream())
{
body.CopyTo(ms);
bodyBytes = ms.ToArray();
}
return bodyBytes.Length;
}
If you want to receive multipart/form-data
, you can parse by ReadAsMultipartAsync of System.Net.Http.Formatting.dll
.
var body = this.Environment["owin.RequestBody"] as Stream;
var multipart = await new StreamContent(body).ReadAsMultipartAsync();
If you return byte[] array, you maybe should avoid Json(or other) ContentFormatter. You can use RawOctetStreamContentFormatterFactory
.
[IgnoreClientGenerate]
[OperationOption(AcceptVerbs.Post, typeof(RawOctetStreamContentFormatterFactory))]
public byte[] EchoByte()
{
var body = this.Environment["owin.RequestBody"] as Stream;
byte[] bodyBytes;
using (var ms = new MemoryStream())
{
body.CopyTo(ms);
bodyBytes = ms.ToArray();
}
return bodyBytes;
}
If you needs return html, set content formatter to HtmlContentFormatter/HtmlContentFormatterFactory
. If you needs template engine, you can use RazorEngine. Simple helper base contract.
public abstract class RazorContractBase : LightNode.Server.LightNodeContract
{
static readonly IRazorEngineService razor = CreateRazorEngineService();
IRazorEngineService CreateRazorEngineService()
{
var config = new TemplateServiceConfiguration();
config.DisableTempFileLocking = true;
config.CachingProvider = new DefaultCachingProvider(_ => { });
config.TemplateManager = new DelegateTemplateManager(name =>
{
// import from "Views" directory
var viewPath = System.IO.Path.Combine(System.AppDomain.CurrentDomain.BaseDirectory, "Views", name);
return System.IO.File.ReadAllText(viewPath);
});
return RazorEngineService.Create(config);
}
protected string View(string viewName)
{
return View(viewName, new object());
}
protected string View(string viewName, object model)
{
var type = model.GetType();
if (razor.IsTemplateCached(viewName, type))
{
return razor.Run(viewName, type, model);
}
else
{
return razor.RunCompile(viewName, type, model);
}
}
}
And you can make shortcut of OperationOptionAttribute.
public class Html : LightNode.Server.OperationOptionAttribute
{
public Html(AcceptVerbs acceptVerbs = AcceptVerbs.Get | AcceptVerbs.Post)
: base(acceptVerbs, typeof(HtmlContentFormatterFactory))
{
}
}
LightNode is like RPC but REST. Public API follows a simple rule. Address is {ClassName}/{MethodName}
, and it's case insensitive. GET parameter use QueryString. POST parameter use x-www-form-urlencoded. Response type follows configured ContentFormatter. Receiver can select response type use url extension(.xml, .json etc...) or Accept header.
You can use other OWIN Middleware(for example ASP.NET Identity). LightNode can copmose with there. LightNode provides only core framework.
LightNode can enable per assembly. For the versioning strategy with project separating.
app.Map("/v1", x =>
{
x.UseLightNode(new LightNodeOptions(), typeof(v1Contract).Assembly);
});
app.Map("/v2", x =>
{
x.UseLightNode(new LightNodeOptions(), typeof(v2Contract).Assembly);
});
Client code generation
Client side implementation of the REST API is often painful. LightNode solves by T4 code generation.
// Open .tt file and configure four steps.
<#@ assembly name="$(SolutionDir)\Performance\LightNode.Performance\bin\LightNode.Performance.dll" #>
<#
// ------------- T4 Configuration ------------- //
// 1. Set LightNodeContract assemblies(and all dependency) path to above #@ assembly name # directive
// 2. Set Namespace & ClientName & Namespace
var clientName = "LightNodeClient";
var namespaceName = "LightNode.Client";
// 3. Set DefaultContentFormatter Construct String
var defaultContentFormatter = "new LightNode.Formatter.JsonNetContentFormatter()";
// 4. Set Additional using Namespace
var usingNamespaces = new [] {"System.Linq"};
// 5. Set append "Async" suffix to method name(ex: CalcAsync or Calc)
var addAsyncSuffix = true;
// ----------End T4 Configuration ------------- //
// generated code is like RPC Style.
// {ClassName}.{MethodName}({Parameters})
var client = new LightNodeClient("http://localhost");
await client.Me.EchoAsync("test");
var sum = await client.Me.SumAsync(1, 10, 100);
Client is very simple but very easy for use. Currently code generation provides for Portable Class Library(HttpClinet) and Unity3D(with UniRx).
You can download from NuGet.
Note:Client generation currently supports POST only.
LightNode expose logging interface ILightNodeOptions.Logger
and provides optional logger for Systen.Diagnostics.Tracing.EventSource, it's send to ETW(EventTrace for Windows) and you can subscribe easily by Microsoft's Semantic Logging Application Block
Library.
var option = new LightNodeOptions
{
Logger = LightNode.Diagnostics.LightNodeEventSource.Log
};
app.UseLightNode(option);
If throws unhandled exception, LightNode's default no handles exception and pass thru other middleware. This option is useful for debugging with Glimpse or other diagnostics middleware such as Microsoft.Owin.Diagnostics's UseErrorPage. You can also use LightNode's Builtin diagnostics system - LightNodeOptions.ErrorHandlingPolicy ErrorHandlingPolicy.ReturnInternalServerErrorIncludeErrorDetails
. It's show simply error string.
If LightNode can't create OperationContext(for example 404), default returns StatusCode and description string. This can customize OperationMissingHandlingPolicy
, If ThrowException
then throws OperationNotFoundException
.
// Default
var option = new LightNodeOptions()
{
ErrorHandlingPolicy = ErrorHandlingPolicy.ThrowException,
OperationMissingHandlingPolicy = OperationMissingHandlingPolicy.ReturnErrorStatusCodeIncludeErrorDetails,
};
app.UseLightNode(option);
LightNode is fastest framework.
Performance source code is in LightNode/Performance. Enviroment is "Windows 8.1/CPU Core i7-3770K(3.5GHz)/Memory 32GB" and disabled firewall and windows defender. Orange and Green bar is hosted on IIS(System.Web). LightNode(Green bar)'s performance is nearly raw handler. Gray bar is reference, LightNode on Helios - Microsoft.Owin.Host.IIS gots extremely performance.
LightNode is using AppVeyor CI. You can check unit test status.
1.6.6.1, 2.0.4-beta
1.6.5.1, 2.0.3-beta
2.0.0-beta - 2016-07/28
1.6.4 - 2015-11-27
1.6.3 - 2015-10-06
1.6.2(Only LightNode.MsgPack) - 2015-08-18
1.6.1 - 2015-08-11
1.6.0 - 2015-07-16
1.5.0 - 2015-06-25
1.4.0 - 2015-05-05
1.3.2(Only LightNode.Client.PCL.T4) - 2015-05-01
1.3.0/1.3.1 - 2015-04-28
1.2.1 - 2015-04-22
1.2.0 - 2015-04-19
1.1.0 - 2015-02-21
1.0.0 - 2015-02-16
0.4.0 - 2015-01-27
0.3.0 - 2014-05-12
0.2.0 - 2014-01-14
0.1.1 - 2013-12-23