Overview
Target Frameworks
Platforms
Prerequisites
Installation
Usage
API
Extensibility
Performance
Building and Testing
Projects Using this Library
Related Concepts
Contributing
About
Jering.Javascript.NodeJS enables you to invoke javascript in NodeJS, from C#. With this ability, you can use javascript libraries and scripts from your C# projects.
You can use this library as a replacement for the recently obsoleted Microsoft.AspNetCore.NodeServices.
InvokeFromFileAsync<T>
replacesINodeService
'sInvokeAsync<T>
andInvokeExportAsync<T>
.
This library is flexible; you can use a dependency injection (DI) based API or a static API, also, you can invoke both in-memory and on-disk javascript.
Static API example:
string javascriptModule = @"
module.exports = (callback, x, y) => { // Module must export a function that takes a callback as its first parameter
var result = x + y; // Your javascript logic
callback(null /* If an error occurred, provide an error object or message */, result); // Call the callback when you're done.
}";
// Invoke javascript
int result = await StaticNodeJSService.InvokeFromStringAsync<int>(javascriptModule, args: new object[] { 3, 5 });
// result == 8
Assert.Equal(8, result);
DI based API example:
string javascriptModule = @"
module.exports = (callback, x, y) => { // Module must export a function that takes a callback as its first parameter
var result = x + y; // Your javascript logic
callback(null /* If an error occurred, provide an error object or message */, result); // Call the callback when you're done.
}";
// Create an INodeJSService
var services = new ServiceCollection();
services.AddNodeJS();
ServiceProvider serviceProvider = services.BuildServiceProvider();
INodeJSService nodeJSService = serviceProvider.GetRequiredService<INodeJSService>();
// Invoke javascript
int result = await nodeJSService.InvokeFromStringAsync<int>(javascriptModule, args: new object[] { 3, 5 });
// result == 8
Assert.Equal(8, result);
- .NET Standard 2.0
- .NET Framework 4.6.1
Works on Windows, macOS, and Linux systems.
You'll need to install NodeJS and add the NodeJS executable's directory to the Path
environment variable (automatically done by the official installer).
Using Package Manager:
PM> Install-Package Jering.Javascript.NodeJS
Using .Net CLI:
> dotnet add package Jering.Javascript.NodeJS
This library provides a DI based API to facilitate extensibility and testability. You can use any DI framework that has adapters for Microsoft.Extensions.DependencyInjection. Here, we'll use vanilla Microsoft.Extensions.DependencyInjection:
var services = new ServiceCollection();
services.AddNodeJS();
ServiceProvider serviceProvider = services.BuildServiceProvider();
INodeJSService nodeJSService = serviceProvider.GetRequiredService<INodeJSService>();
The default implementation of INodeJSService
is HttpNodeJSService
, which manages a NodeJS process that it sends javascript invocations to via HTTP.
INodeJSService
is a singleton service and INodeJSService
's members are thread safe.
Where possible, inject INodeJSService
into your types or share an INodeJSService
.
This avoids the overhead of killing and creating NodeJS processes repeatedly.
When you're done, you can dispose of an INodeJSService
by calling
nodeJSService.Dispose();
or
serviceProvider.Dispose(); // Calls Dispose on objects it has instantiated that are disposable
Disposing of an INodeJSService
kills its associated NodeJS process.
Note that even if Dispose
isn't called, the NodeJS process is killed when the application shuts down - if the application shuts down gracefully.
If the application doesn't shutdown gracefully, the NodeJS process will kill itself when it detects that its parent has been killed.
Essentially, manually disposing of INodeJSService
s isn't mandatory.
This library provides a static API as an alternative. The StaticNodeJSService
type wraps an INodeJSService
, exposing most of its public members.
Whether you use the static API or the DI based API depends on your development needs. If you're already using DI, if you want to mock
out javascript invocations in your tests or if you want to overwrite services, use the DI based API. Otherwise,
use the static API. Example usage:
string result = await StaticNodeJSService
.InvokeFromStringAsync<Result>("module.exports = (callback, message) => callback(null, message);", args: new[] { "success" });
Assert.Equal("success", result);
To invoke javascript, you'll need a NodeJS module that exports either a function or an object containing functions. Exported functions can be of two forms:
These functions take a callback as their first argument, and call the callback when they're done.
The callback takes two optional arguments:
- The first argument is an error or an error message. It must be of type
Error
orstring
. - The second argument is the result. It must be a JSON-serializable type, a
string
, or astream.Readable
.
This is known as an error-first callback. Such callbacks are commonly used for error handling in NodeJS asynchronous code (check out NodeJS Event Loop for more information on asynchrony in NodeJS).
This is a module that exports a valid function:
module.exports = (callback, arg1, arg2, arg3) => {
... // Do something with args
callback(null, result);
}
This is a module that exports an object containing valid functions:
module.exports = {
doSomething: (callback, arg1) => {
... // Do something with arg
callback(null, result);
},
doSomethingElse: (callback) => {
... // Do something else
callback(null, result);
}
}
Async functions are syntactic sugar for functions with callback parameters (check out Callbacks, Promises and Async/Await for a summary on how callbacks, promises and async/await are related).
This is a module that exports a valid function:
module.exports = async (arg1, arg2) => {
... // Do something with args
return result;
}
And this is a module that exports an object containing valid functions:
module.exports = {
doSomething: async (arg1, arg2, arg3, arg4) => {
... // Do something with args
// async functions can explicitly return promises
return new Promise((resolve, reject) => {
resolve(result);
});
},
doSomethingElse: async (arg1) => {
... // Do something with arg
return result;
}
}
If an error is thrown in an async function, the error message is sent back to the calling .Net process, where an InvocationException
is thrown:
module.exports = async () => {
throw new Error('error message');
}
If you have a javascript file named exampleModule.js
(located in NodeJSProcessOptions.ProjectPath
):
module.exports = (callback, message) => callback(null, { resultMessage: message });
And a .Net class Result
:
public class Result
{
public string Message { get; set; }
}
You can invoke the javascript using InvokeFromFileAsync<T>
:
Result result = await nodeJSService.InvokeFromFileAsync<Result>("exampleModule.js", args: new[] { "success" });
Assert.Equal("success", result.Message);
If you change exampleModule.js
to export an object containing functions:
module.exports = {
appendExclamationMark: (callback, message) => callback(null, { resultMessage: message + '!' }),
appendFullStop: (callback, message) => callback(null, { resultMessage: message + '.' })
}
You can invoke a specific function by providing an export's name:
Result result = await nodeJSService.InvokeFromFileAsync<Result>("exampleModule.js", "appendExclamationMark", args: new[] { "success" });
Assert.Equal("success!", result.Message);
When using InvokeFromFileAsync
, NodeJS always caches the module using the .js
file's absolute path as cache identifier. This is great for
performance, since the file will not be reread or recompiled on subsequent invocations.
You can invoke javascript in string form using InvokeFromStringAsync<T>
:
string module = "module.exports = (callback, message) => callback(null, { resultMessage: message });";
// Invoke javascript
Result result = await nodeJSService.InvokeFromStringAsync<Result>(module, args: new[] { "success" });
Assert.Equal("success", result.Message);
In the above example, the module string is sent to NodeJS and recompiled on every invocation. If you're going to invoke a module repeatedly, to avoid resending and recompiling, you'll want to have NodeJS cache the module. To do this, you must specify a custom cache identifier, since unlike a file, a string has no "absolute file path" for NodeJS to use as cache identifier. Once NodeJS has cached the module, invoke directly from the NodeJS cache:
string cacheIdentifier = "exampleModule";
// Try to invoke from the NodeJS cache
(bool success, Result result) = await nodeJSService.TryInvokeFromCacheAsync<Result>(cacheIdentifier, args: new[] { "success" });
// If the module hasn't been cached, cache it. If the NodeJS process dies and restarts, the cache will be invalidated, so always check whether success is false.
if(!success)
{
// This is a trivialized example. In practice, to avoid holding large module strings in memory, you might retrieve the module
// string from an on-disk or remote source, like a file.
string moduleString = "module.exports = (callback, message) => callback(null, { resultMessage: message });";
// Send the module string to NodeJS where it's compiled, invoked and cached.
result = await nodeJSService.InvokeFromStringAsync<Result>(moduleString, cacheIdentifier, args: new[] { "success" });
}
Assert.Equal("success", result.ResultMessage);
We recommend using the following InvokeFromStringAsync<T>
overload to perform the above example's operations.
The above example is really there to explain what this overload does.
If you've enabled concurrency, you must use this overload:
string module = "module.exports = (callback, message) => callback(null, { resultMessage: message });";
string cacheIdentifier = "exampleModule";
// This is a trivialized example. In practice, to avoid holding large module strings in memory, you might retrieve the module
// string from an on-disk or remote source, like a file.
Func<string> moduleFactory = () => module;
// Initially, sends only cacheIdentifier to NodeJS, in an attempt to invoke from the NodeJS cache. If the module hasn't been cached, creates the module string using moduleFactory and
// sends it to NodeJS where it's compiled, invoked and cached.
Result result = await nodeJSService.InvokeFromStringAsync<Result>(moduleFactory, cacheIdentifier, args: new[] { "success" });
Assert.Equal("success", result.Message);
Like when invoking javascript form a file, if the module exports an object containing functions, you can invoke a specific function by specifying its name.
You can invoke javascript in stream form using InvokeFromStreamAsync<T>
:
// Write the module to a MemoryStream for demonstration purposes.
streamWriter.Write("module.exports = (callback, message) => callback(null, {resultMessage: message});");
streamWriter.Flush();
memoryStream.Position = 0;
Result result = await nodeJSService.InvokeFromStreamAsync<Result>(memoryStream, args: new[] { "success" });
Assert.Equal("success", result.Message);
InvokeFromStreamAsync
behaves in a similar manner to InvokeFromStringAsync
, refer to Invoking Javascript in String Form for details on caching and more.
This method provides a way to avoid allocating a string if the source of the module is a stream. Avoiding string
allocations can improve performance.
This library uses the ASP.NET Core options pattern. While developed for ASP.NET Core, this pattern can be used by other types of applications. The NodeJS process and the service that manages the process are both configurable, for example:
var services = new ServiceCollection();
services.AddNodeJS();
// Options for the NodeJSProcess, here we enable debugging
services.Configure<NodeJSProcessOptions>(options => options.NodeAndV8Options = "--inspect-brk");
// Options for the service that manages the process, here we make its timeout infinite
services.Configure<OutOfProcessNodeJSServiceOptions>(options => options.TimeoutMS = -1);
ServiceProvider serviceProvider = services.BuildServiceProvider();
INodeJSService nodeJSService = serviceProvider.GetRequiredService<INodeJSService>();
The static API exposes a method for configuring options:
StaticNodeJSService.Configure<OutOfProcessNodeJSServiceOptions>(options => options.TimeoutMS = -1);
Configurations made using StaticNodeJSService.Configure<T>
only apply to javascript invocations made using the static API.
Ideally, such configurations should be done before the first javascript invocation.
Any existing NodeJS process is killed and a new one is created in the first javascript invocation after every StaticNodeJSService.Configure<T>
call.
Re-creating the NodeJS process is resource intensive. Also, if you're using the static API from multiple threads and
the NodeJS process is performing invocations for other threads, you might get unexpected results.
The next two sections list all available options.
Option | Type | Description | Default |
---|---|---|---|
ProjectPath | string |
The base path for resolving paths of NodeJS modules on disk. If this value is null , whitespace or an empty string and the application is an ASP.NET Core application, project path is IHostingEnvironment.ContentRootPath |
The current directory (value returned by Directory.GetCurrentDirectory() ) |
NodeAndV8Options | string |
NodeJS and V8 options in the form "[NodeJS options] [V8 options]". The full list of NodeJS options can be found here: https://nodejs.org/api/cli.html#cli_options. | null |
Port | int |
The port that the server running on NodeJS will listen on. If set to 0, the OS will choose the port. | 0 |
EnvironmentVariables | IDictionary<string, string> |
The environment variables for the NodeJS process. The full list of NodeJS environment variables can be found here: https://nodejs.org/api/cli.html#cli_environment_variables. If this value doesn't contain an element with key "NODE_ENV" and the application is an ASP.NET Core application, an element with key "NODE_ENV" is added with value "development" if IHostingEnvironment.EnvironmentName is EnvironmentName.Development or "production" otherwise. |
An Empty IDictionary<string, string> |
Option | Type | Description | Default |
---|---|---|---|
TimeoutMS | int |
The maximum duration to wait for the NodeJS process to connect and to wait for responses to invocations. If this value is negative, the maximum duration is infinite. | 60000 |
NumRetries | int |
The number of times an invocation is retried. If set to a negative value, invocations are retried indefinitely. If the module source of an invocation is an unseekable stream, the invocation isn't retried. If you require retries for such streams, copy their contents to a MemoryStream . |
1 |
Concurrency | Concurrency |
The concurrency mode for invocations. By default, this value is Concurrency.None and invocations are executed synchronously by a single NodeJS process; mode pros: lower memory overhead and supports all modules, cons: less performant.If this value is Concurrency.MultiProcess , ConcurrencyDegree NodeJS processes are created and invocations are distributed among them using round-robin load balancing; mode pros: more performant, cons: higher memory overhead and doesn't work with modules that have persistent state. |
Concurrency.None |
ConcurrencyDegree | int |
The concurrency degree. If Concurrency is Concurrency.MultiProcess , this value is the number of NodeJS processes. If this value is less than or equal to 0, concurrency degree is the number of logical processors the current machine has. This value does nothing if Concurrency is Concurrency.None . |
0 |
EnableFileWatching | bool |
The value specifying whether file watching is enabled. If file watching is enabled, when a file in WatchPath with name matching a pattern in WatchFileNamePatterns changes, NodeJS is restarted. |
false |
WatchPath | string |
The path of the directory to watch for file changes. If this value is null , the path NodeJSProcessOptions.ProjectPath is watched. This value does nothing if EnableFileWatching is false . |
null |
WatchSubdirectories | bool |
The value specifying whether to watch subdirectories of WatchPath . This value does nothing if EnableFileWatching is false . |
true |
WatchFileNamePatterns | IEnumerable<string> |
The file name patterns to watch. In a pattern, "*" represents 0 or more of any character and "?" represents 0 or 1 of any character. For example, "TestFile1.js" matches the pattern "*File?.js". This value does nothing if EnableFileWatching is false . |
["*.js", "*.jsx", "*.ts", "*.tsx", "*.json", "*.html"] |
WatchGracefulShutdown | bool |
The value specifying whether NodeJS processes shutdown gracefully when a file changes. If this value is true, NodeJS processes shutdown gracefully. Otherwise they're killed immediately. This value does nothing if EnableFileWatching is false .What's a graceful shutdown? When a file changes, a new NodeJS process is created and subsequent invocations are sent to it. The old NodeJS process might still be handling earlier invocations. If graceful shutdown is enabled, the old NodeJS process is killed after its invocations complete. If graceful shutdown is disabled, the old NodeJS process is killed immediately and invocations are retried in the new NodeJS process if retries remain (see NumRetries ).Should I use graceful shutdown? Shutting down gracefully is safer: chances of an invocation exhausting retries and failing is lower, also, you won't face issues from an invocation terminating midway. However, graceful shutdown does incur a tiny performance cost and invocations complete using the outdated version of your script. Weigh these factors for your script and use-case to decide whether to use graceful shutdown. |
true |
These are the steps for debugging javascript invoked using INodeJSService:
- Create an INodeJSService using the example options in the previous section (
NodeJSProcessOptions.NodeAndV8Options
=--inspect-brk
andOutOfProcessNodeJSServiceOptions.TimeoutMS
=-1
). - Add
debugger
statements to your javascript module. - Call a javascript invoking method.
- Navigate to
chrome://inspect/
in Chrome. - Click "Open dedicated DevTools for Node".
- Click continue to advance to your
debugger
statements.
To enable concurrency, set OutOfProcessNodeJSServiceOptions.Concurrency
to Concurrency.MultiProcess
:
services.Configure<OutOfProcessNodeJSServiceOptions>(options => {
options.Concurrency = Concurrency.MultiProcess; // Concurrency.None by default
options.ConcurrencyDegree = 8; // Number of processes. Defaults to the number of logical processors on your machine.
);
(see Configuring INodeJSService for more information on configuring)
All invocations will be distributed among multiple NodeJS processes using round-robin load balancing.
Enabling concurrency significantly speeds up CPU-bound workloads. For example, consider the following benchmarks:
Method | Mean | Error | StdDev | Gen 0 | Gen 1 | Gen 2 | Allocated |
---|---|---|---|---|---|---|---|
INodeJSService_Concurrency_MultiProcess | 400.3 ms | 0.62 ms | 0.58 ms | - | - | - | 134.95 KB |
INodeJSService_Concurrency_None | 2,500.2 ms | 0.51 ms | 0.48 ms | - | - | - | 135.13 KB |
INodeServices_Concurrency | 2,500.2 ms | 0.49 ms | 0.46 ms | - | - | - | 246.98 KB |
BenchmarkDotNet=v0.12.0, OS=Windows 10.0.18362
Intel Core i7-7700 CPU 3.60GHz (Kaby Lake), 1 CPU, 8 logical and 4 physical cores
.NET Core SDK=3.0.100
[Host] : .NET Core 3.0.0 (CoreCLR 4.700.19.46205, CoreFX 4.700.19.46214), X64 RyuJIT
DefaultJob : .NET Core 3.0.0 (CoreCLR 4.700.19.46205, CoreFX 4.700.19.46214), X64 RyuJIT
These benchmarks invoke javascript asynchronously, as most applications would (view complete source here):
const int numTasks = 25;
var results = new Task<string>[numTasks];
for (int i = 0; i < numTasks; i++)
{
results[i] = _nodeJSService.InvokeFromFileAsync<string>(DUMMY_CONCURRENCY_MODULE);
}
return await Task.WhenAll(results);
Where the DUMMY_CONCURRENCY_MODULE
file contains:
// Minimal processor blocking logic
module.exports = (callback) => {
// Block processor
var end = new Date().getTime() + 100; // 100ms block
while (new Date().getTime() < end) { /* do nothing */ }
callback(null);
};
For INodeJSService
with Concurrency.MultiProcessing
, multiple NodeJS processes perform invocations concurrently, so the benchmark takes ~400ms ((25 tasks x 100ms) / number-of-logical-processors + overhead-from-unrelated-processes).
In the other two benchmarks, a single NodeJS process performs invocations synchronously, so those benchmarks take ~2500ms (25 tasks x 100ms).
-
You can't use concurrency if you persist data between invocations. For example, with concurrency enabled:
const string javascriptModule = @" var lastResult; module.exports = (callback, x) => { var result = x + (lastResult ? lastResult : 0); // Use persisted value here lastResult = result; // Persist callback(null, result); }"; // result == 3 int result = await StaticNodeJSService.InvokeFromStringAsync<int>(javascriptModule, "customIdentifier", args: new object[] { 3 }); // expected 8, but result == 5 since different processes perform the invocations result = await StaticNodeJSService.InvokeFromStringAsync<int>(javascriptModule, "customIdentifier", args: new object[] { 5 });
This should not be a problem in most cases.
-
Higher memory overhead. This isn't typically an issue - a standard workstation can host dozens of NodeJS processes, and in cloud scenarios you'll typically have memory proportional to the number of logical processors.
-
Concurrency may not speed up workloads with lots of asynchronous operations. For example if your workload spends lots of time waiting on a databases, more NodeJS processes will not speed things up significantly.
-
With concurrency enabled, you can't use the following pattern to invoke from NodeJS's cache:
string cacheIdentifier = "exampleModule"; // If you have an even number of NodeJS processes, success will always be false since the resulting caching attempt is // sent to the next NodeJS process. (bool success, Result result) = await nodeJSService.TryInvokeFromCacheAsync<Result>(cacheIdentifier, args: new[] { "success" }); // False, so we attempt to cache if(!success) { string moduleString = "module.exports = (callback, message) => callback(null, { resultMessage: message });"; // Because of round-robin load balancing, this caching attempt is sent to the next NodeJS process. result = await nodeJSService.InvokeFromStringAsync<Result>(moduleString, cacheIdentifier, args: new[] { "success" }); } Assert.Equal("success", result.ResultMessage);
Instead, call an overload that atomically handles caching and invoking:
string module = "module.exports = (callback, message) => callback(null, { resultMessage: message });"; string cacheIdentifier = "exampleModule"; // This is a trivialized example. In practice, to avoid holding large module strings in memory, you might retrieve the module // string from an on-disk or remote source, like a file. Func<string> moduleFactory = () => module; // Initially, sends only cacheIdentifier to NodeJS, in an attempt to invoke from the NodeJS cache. If the module hasn't been cached, creates the module string using moduleFactory and // sends it to NodeJS where it's compiled, invoked and cached. Result result = await nodeJSService.InvokeFromStringAsync<Result>(moduleFactory, cacheIdentifier, args: new[] { "success" }); Assert.Equal("success", result.Message);
Task<T> InvokeFromFileAsync<T>(string modulePath, string exportName = null, object[] args = null, CancellationToken cancellationToken = default(CancellationToken));
Invokes a function exported by a NodeJS module on disk.
-
T
- Description: The type of object this method will return. It can be a JSON-serializable type,
string
, orStream
.
- Description: The type of object this method will return. It can be a JSON-serializable type,
-
modulePath
- Type:
string
- Description: The path to the NodeJS module (i.e., JavaScript file) relative to
NodeJSProcessOptions.ProjectPath
.
- Type:
-
exportName
- Type:
string
- Description: The function in the module's exports to be invoked. If unspecified, the module's exports object is assumed to be a function, and is invoked.
- Type:
-
args
- Type:
object[]
- Description: The sequence of JSON-serializable and/or
string
arguments to be passed to the function to invoke.
- Type:
-
cancellationToken
- Type:
CancellationToken
- Description: The cancellation token for the asynchronous operation.
- Type:
The task representing the asynchronous operation.
ConnectionException
- Thrown if unable to connect to NodeJS.
InvocationException
- Thrown if a NodeJS error occurs.
- Thrown if the invocation request times out.
ObjectDisposedException
- Thrown if this has been disposed or if it attempts to use one of its dependencies that has been disposed.
OperationCanceledException
- Thrown if
cancellationToken
is cancelled.
- Thrown if
If we have a file named exampleModule.js
(located in NodeJSProcessOptions.ProjectPath
), with contents:
module.exports = (callback, message) => callback(null, { resultMessage: message });
And we have the class Result
:
public class Result
{
public string Message { get; set; }
}
The following assertion will pass:
Result result = await nodeJSService.InvokeFromFileAsync<Result>("exampleModule.js", args: new[] { "success" });
Assert.Equal("success", result.Message);
Task<T> InvokeFromStringAsync<T>(string moduleString, string newCacheIdentifier = null, string exportName = null, object[] args = null, CancellationToken cancellationToken = default(CancellationToken));
Invokes a function exported by a NodeJS module in string form.
-
T
- Description: The type of object this method will return. It can be a JSON-serializable type,
string
, orStream
.
- Description: The type of object this method will return. It can be a JSON-serializable type,
-
moduleString
- Type:
string
- Description: The module in
string
form.
- Type:
-
newCacheIdentifier
- Type:
string
- Description: The modules's cache identifier in the NodeJS module cache. If unspecified, the module will not be cached.
- Type:
-
exportName
- Type:
string
- Description: The function in the module's exports to be invoked. If unspecified, the module's exports object is assumed to be a function, and is invoked.
- Type:
-
args
- Type:
object[]
- Description: The sequence of JSON-serializable and/or
string
arguments to be passed to the function to invoke.
- Type:
-
cancellationToken
- Type:
CancellationToken
- Description: The cancellation token for the asynchronous operation.
- Type:
The task representing the asynchronous operation.
ConnectionException
- Thrown if unable to connect to NodeJS.
InvocationException
- Thrown if a NodeJS error occurs.
- Thrown if the invocation request times out.
ObjectDisposedException
- Thrown if this has been disposed or if it attempts to use one of its dependencies that has been disposed.
OperationCanceledException
- Thrown if
cancellationToken
is cancelled.
- Thrown if
Using the class Result
:
public class Result
{
public string Message { get; set; }
}
The following assertion will pass:
Result result = await nodeJSService.InvokeFromStringAsync<Result>("module.exports = (callback, message) => callback(null, { resultMessage: message });",
args: new[] { "success" });
Assert.Equal("success", result.Message);
Task<T> InvokeFromStreamAsync<T>(Stream moduleStream, string newCacheIdentifier = null, string exportName = null, object[] args = null, CancellationToken cancellationToken = default(CancellationToken));
Invokes a function exported by a NodeJS module in Stream form.
-
T
- Description: The type of object this method will return. It can be a JSON-serializable type,
string
, orStream
.
- Description: The type of object this method will return. It can be a JSON-serializable type,
-
moduleStream
- Type:
Stream
- Description: The module in
Stream
form.
- Type:
-
newCacheIdentifier
- Type:
string
- Description: The modules's cache identifier in the NodeJS module cache. If unspecified, the module will not be cached.
- Type:
-
exportName
- Type:
string
- Description: The function in the module's exports to be invoked. If unspecified, the module's exports object is assumed to be a function, and is invoked.
- Type:
-
args
- Type:
object[]
- Description: The sequence of JSON-serializable and/or
string
arguments to be passed to the function to invoke.
- Type:
-
cancellationToken
- Type:
CancellationToken
- Description: The cancellation token for the asynchronous operation.
- Type:
The task representing the asynchronous operation.
ConnectionException
- Thrown if unable to connect to NodeJS.
InvocationException
- Thrown if a NodeJS error occurs.
- Thrown if the invocation request times out.
ObjectDisposedException
- Thrown if this has been disposed or if it attempts to use one of its dependencies that has been disposed.
OperationCanceledException
- Thrown if
cancellationToken
is cancelled.
- Thrown if
Using the class Result
:
public class Result
{
public string Message { get; set; }
}
The following assertion will pass:
using (var memoryStream = new MemoryStream())
using (var streamWriter = new StreamWriter(memoryStream))
{
// Write the module to a MemoryStream for demonstration purposes.
streamWriter.Write("module.exports = (callback, message) => callback(null, {resultMessage: message});");
streamWriter.Flush();
memoryStream.Position = 0;
Result result = await nodeJSService.InvokeFromStreamAsync<Result>(memoryStream, args: new[] { "success" });
Assert.Equal("success", result.Message);
}
Task<(bool, T)> TryInvokeFromCacheAsync<T>(string moduleCacheIdentifier, string exportName = null, object[] args = null, CancellationToken cancellationToken = default(CancellationToken));
Attempts to invoke a function exported by a NodeJS module cached by NodeJS.
-
T
- Description: The type of object this method will return. It can be a JSON-serializable type,
string
, orStream
.
- Description: The type of object this method will return. It can be a JSON-serializable type,
-
moduleCacheIdentifier
- Type:
string
- Description: The cache identifier of the module.
- Type:
-
exportName
- Type:
string
- Description: The function in the module's exports to be invoked. If unspecified, the module's exports object is assumed to be a function, and is invoked.
- Type:
-
args
- Type:
object[]
- Description: The sequence of JSON-serializable and/or
string
arguments to be passed to the function to invoke.
- Type:
-
cancellationToken
- Type:
CancellationToken
- Description: The cancellation token for the asynchronous operation.
- Type:
The task representing the asynchronous operation. On completion, the task returns a (bool, T)
with the bool set to true on
success and false otherwise.
ConnectionException
- Thrown if unable to connect to NodeJS.
InvocationException
- Thrown if a NodeJS error occurs.
- Thrown if the invocation request times out.
ObjectDisposedException
- Thrown if this has been disposed or if it attempts to use one of its dependencies that has been disposed.
OperationCanceledException
- Thrown if
cancellationToken
is cancelled.
- Thrown if
Using the class Result
:
public class Result
{
public string Message { get; set; }
}
The following assertion will pass:
// Cache the module
string cacheIdentifier = "exampleModule";
await nodeJSService.InvokeFromStringAsync<Result>("module.exports = (callback, message) => callback(null, { resultMessage: message });",
cacheIdentifier,
args: new[] { "success" });
// Invoke from cache
(bool success, Result result) = await nodeJSService.TryInvokeFromCacheAsync<Result>(cacheIdentifier, args: new[] { "success" });
Assert.True(success);
Assert.Equal("success", result.Message);
This library's behaviour can be customized by implementing public interfaces and overwriting their default DI services. For example, if we have objects that
can't be serialized using the default JSON serialization logic, we can implement IJsonService
:
// Create a custom implementation of IJsonService
public class MyJsonService : IJsonService
{
public T Deserialize<T>(JsonReader jsonReader)
{
... // Custom deserializetion logic
}
public void Serialize(JsonWriter jsonWriter, object value)
{
... // Custom serialization logic
}
}
And overwrite its default DI service:
var services = new ServiceCollection();
services.AddNodeJS();
// Overwrite the default DI service
services.AddSingleton<IJsonService, MyJsonService>();
ServiceProvider serviceProvider = services.BuildServiceProvider();
INodeJSService nodeJSService = serviceProvider.GetRequiredService<INodeJSService>();
This is the list of implementable interfaces:
Interface | Description |
---|---|
IJsonService |
An abstraction for JSON serialization/deserialization. |
IHttpClientService |
An abstraction for HttpClient . |
INodeJSProcessFactory |
An abstraction for NodeJS process creation. |
IHttpContentFactory |
An abstraction for HttpContent creation. |
INodeJSService |
An abstraction for invoking code in NodeJS. |
IEmbeddedResourcesService |
An abstraction for reading of embedded resources. |
This library is heavily inspired by Microsoft.AspNetCore.NodeServices. While the main additions to this library are ways to invoke in-memory javascript, this library also provides better performance.
Inter-process communication latency benchmarks:
Method | Mean | Error | StdDev | Gen 0 | Gen 1 | Gen 2 | Allocated |
---|---|---|---|---|---|---|---|
INodeJSService_Latency_InvokeFromFile | 104.0 us | 0.64 us | 0.56 us | 1.2207 | - | - | 5.69 KB |
INodeJSService_Latency_InvokeFromFile_GracefulShutdownEnabled | 104.2 us | 0.65 us | 0.57 us | 1.2207 | - | - | 5.91 KB |
INodeJSService_Latency_InvokeFromCache | 100.7 us | 0.47 us | 0.44 us | 1.2207 | - | - | 5.76 KB |
INodeServices_Latency | 114.8 us | 1.12 us | 0.99 us | 2.4414 | - | - | 10.25 KB |
NodeJS v12.13.0
BenchmarkDotNet=v0.12.0, OS=Windows 10.0.18362
Intel Core i7-7700 CPU 3.60GHz (Kaby Lake), 1 CPU, 8 logical and 4 physical cores
.NET Core SDK=3.0.100
[Host] : .NET Core 3.0.0 (CoreCLR 4.700.19.46205, CoreFX 4.700.19.46214), X64 RyuJIT
DefaultJob : .NET Core 3.0.0 (CoreCLR 4.700.19.46205, CoreFX 4.700.19.46214), X64 RyuJIT
View source here.
Asynchronous invocations benchmarks:
Method | Mean | Error | StdDev | Gen 0 | Gen 1 | Gen 2 | Allocated |
---|---|---|---|---|---|---|---|
INodeJSService_Concurrency_MultiProcess | 400.2 ms | 0.44 ms | 0.42 ms | - | - | - | 134.76 KB |
INodeJSService_Concurrency_None | 2,500.4 ms | 0.45 ms | 0.42 ms | - | - | - | 134.76 KB |
INodeServices_Concurrency | 2,500.2 ms | 0.49 ms | 0.46 ms | - | - | - | 245.78 KB |
BenchmarkDotNet=v0.12.0, OS=Windows 10.0.18362
Intel Core i7-7700 CPU 3.60GHz (Kaby Lake), 1 CPU, 8 logical and 4 physical cores
.NET Core SDK=3.0.100
[Host] : .NET Core 3.0.0 (CoreCLR 4.700.19.46205, CoreFX 4.700.19.46214), X64 RyuJIT
DefaultJob : .NET Core 3.0.0 (CoreCLR 4.700.19.46205, CoreFX 4.700.19.46214), X64 RyuJIT
View source here.
Real world benchmarks. These use the syntax highlighter, Prism, to highlight C#:
Method | Mean | Error | StdDev | Gen 0 | Gen 1 | Gen 2 | Allocated |
---|---|---|---|---|---|---|---|
INodeJSService_RealWorkload | 1.269 ms | 0.0150 ms | 0.0140 ms | 54.6875 | 11.7188 | - | 224.55 KB |
INodeServices_RealWorkload | 2.236 ms | 0.0148 ms | 0.0131 ms | 70.3125 | - | - | 283.93 KB |
NodeJS v12.13.0
BenchmarkDotNet=v0.12.0, OS=Windows 10.0.18362
Intel Core i7-7700 CPU 3.60GHz (Kaby Lake), 1 CPU, 8 logical and 4 physical cores
.NET Core SDK=3.0.100
[Host] : .NET Core 3.0.0 (CoreCLR 4.700.19.46205, CoreFX 4.700.19.46214), X64 RyuJIT
DefaultJob : .NET Core 3.0.0 (CoreCLR 4.700.19.46205, CoreFX 4.700.19.46214), X64 RyuJIT
View source here.
How long it takes for NodeJS to restart and begin processing invocations:
Method | Mean | Error | StdDev | Gen 0 | Gen 1 | Gen 2 | Allocated |
---|---|---|---|---|---|---|---|
HttpNodeJSService_FileWatching_GracefulShutdownEnabled_MoveToNewProcess | 48.64 ms | 0.943 ms | 0.882 ms | - | - | - | 276.99 KB |
HttpNodeJSService_FileWatching_GracefulShutdownDisabled_MoveToNewProcess | 49.75 ms | 0.987 ms | 1.416 ms | - | - | - | 276.62 KB |
NodeJS v12.13.0
BenchmarkDotNet=v0.12.0, OS=Windows 10.0.18362
Intel Core i7-7700 CPU 3.60GHz (Kaby Lake), 1 CPU, 8 logical and 4 physical cores
.NET Core SDK=3.0.100
[Host] : .NET Core 3.0.0 (CoreCLR 4.700.19.46205, CoreFX 4.700.19.46214), X64 RyuJIT
DefaultJob : .NET Core 3.0.0 (CoreCLR 4.700.19.46205, CoreFX 4.700.19.46214), X64 RyuJIT
View source here.
You can build and test this project in Visual Studio 2017/2019.
Jering.Web.SyntaxHighlighters.HighlightJS - Use the Syntax Highlighter, HighlightJS, from C#.
Jering.Web.SyntaxHighlighters.Prism - Use the Syntax Highlighter, Prism, from C#.
NodeReact.NET - Library to render React library components on the server-side with C# as well as on the client.
NodeJS is a javascript runtime. Essentially, it provides some built-in libraries and executes javascript. Similarities can be drawn to the Core Common Language Runtime (CoreCLR), which provides a set of base libraries and executes .NET Intermediate Language (typically generated by compiling C# or some other .NET language).
Under the hood, NodeJS uses V8 to execute javascript. While this library could have been built to invoke javascript directly in V8, invoking javascript in NodeJS affords both access to NodeJS's built-in modules and the ability to use most of the modules hosted by npm.
NodeJS modules are a kind of javascript module. The concept of javascript modules can seem far more complicated than it really is, not least because of the existence of competing specifications (CommonJS, AMD, ES6, ...), and the existence of multiple implementations of each specification (SystemJS, RequireJS, Dojo, NodeJS, ...). In reality, javascript modules such as NodeJS modules are really simple. In the following sections, we will go through the what, how and why of NodeJS modules.
The following is a valid NodeJS module. Lets imagine that it exists in a file, flavours.js
:
// Note that the module variable isn't declared (no "var module = {}")
module.exports = ['chocolate', 'strawberry', 'vanilla'];
The following is another valid NodeJS module, we will use it as an entry script (to be supplied to node
on the command line). Lets imagine that it exists in a file, printer.js
,
in the same directory as flavours.js
:
var flavours = require('./flavours.js');
flavours.forEach((flavour) => console.log(flavour));
If we run node printer.js
on the command line, the flavours get printed:
PS C:\NodeJS_Modules_Example> node printer.js
chocolate
strawberry
vanilla
In general, a NodeJS module is simply a block of javascript with module.exports
and/or require
statements. These statements are explained in the next section.
NodeJS's logic for managing modules is contained in its require
function. In the example above, require('./flavours.js')
executes the following steps:
- Resolves the absolute path of
flavours.js
toC:/NodeJS_Modules_Example/flavours.js
. - Checks whether the NodeJS module cache (a simple javascript object) has a property with name
C:/NodeJS_Modules_Example/flavours.js
, and finds that it does not (the module has not been cached). - Reads the contents of
C:/NodeJS_Modules_Example/flavours.js
into memory. - Wraps the contents of
C:/NodeJS_Modules_Example/flavour.js
in a function by appending and prepending strings. The resulting function looks like the following:// Note how the require function and a module object are supplied by the wrapper. function (exports, require, module, __filename, __dirname){ module.exports = ['chocolate', 'strawberry', 'vanilla']; }
- Creates the
module
object and passes it to the generated function. - Adds the
module
object (now containing an array as itsexports
property) to the NodeJS module cache using the property nameC:/NodeJS_Modules_Example/flavours.js
. - Returns
module.exports
.
If the flavours module is required again, the cached module
object is retrieved in step 2, and its exports object is returned. This means that module exports are not immutable, for example,
if we replace the contents of printer.js
with the following:
var flavours = require('./flavours.js');
flavours.forEach((flavour) => console.log(flavour));
// Clear the array
flavours.length = 0;
// Add three new flavours
flavours.push('apple');
flavours.push('green tea');
flavours.push('sea salt');
// Require the module again, turns out that require returns a reference to the same array
flavours = require('./flavours.js');
flavours.forEach((flavour) => console.log(flavour));
Running node printer.js
on the command line prints the following flavours:
PS C:\Users\Jeremy\Desktop\JSTest> node entry.js
chocolate
strawberry
vanilla
apple
green tea
sea salt
To answer this question, lets consider the impetus for the creation of javascript modules in general. Web pages used to include scripts like so:
<html>
...
<script type="text/javascript" src="path/to/coolLibrary.js"></script>
<script type="text/javascript" src="path/to/myScript.js"></script>
...
</html>
Browsers would load the scripts like so:
// Contents of coolLibrary.js
var coolLibraryPrivateObject = ...;
function CoolLibraryPublicFunction(){
... // Do something with coolLibraryPrivateObject, and return some value
}
// Contents of myScript.js
var myVar = CoolLibraryPublicFunction();
... // Do something with myVar
Note how everything in the example above is in the same scope. coolLibraryPrivateObject
can be accessed in myscript.js
. How
can we hide the private object? We can encapsulate cool library in a function:
var module = {};
// This is an immediately invoked function expression, shorthand for assigning the function to a variable then calling it - https://developer.mozilla.org/en-US/docs/Glossary/IIFE
(function(module){
// Contents of coolLibrary.js
var coolLibraryPrivateObject = ...;
function CoolLibraryPublicFunction(){
... // Do something with coolLibraryPrivateObject, and return some value
}
module.exports = CoolLibraryPublicFunction;
})(module)
// Contents of myScript.js
var myVar = module.exports(); // We assigned CoolLibraryPublicFunction to module.exports
... // Do something with myVar
We've successfully hidden coolLibraryPrivateObject
from the global scope using a module-esque pattern. Apart from hiding private objects, this pattern also prevents global namespace pollution.
NodeJS modules serve a similar purpose. By wrapping modules in functions, NodeJS creates a closure for each module so internal details can be kept private.
Contributions are welcome!
Follow @JeringTech for updates and more.