Microse (stands for Micro Remote Object Serving Engine) is a light-weight engine that provides applications the ability to serve modules as RPC services, whether in another process or in another machine.
This is the Node.js version of microse implementation. For API reference, please check the API documentation, or the Protocol Reference.
Other implementations:
- microse-py Python implementation
- microse-swoole PHP implementation based on swoole
Important Note:
Since v1.4.0, Codec CLONE
now uses
v8.serialize
and v8.deserialize
for data serialization, and the codec
option is only set on the client side,
the server will respond accordingly.
npm i microse
In order to use microse, one must create a root ModuleProxyApp
instance and
augment it to the global scope, so other files can directly use it as a root
namespace without importing the module.
// src/app.ts
import { ModuleProxyApp } from "microse";
// Expose and merge the app as a namespace under the global scope.
declare global {
namespace app { }
}
// Create the instance.
export const App = global["app"] = new ModuleProxyApp("app", __dirname);
In other files, just define and export a default class, and merge the type to
the app
namespace, so that another file can access it directly without import.
(NOTE: Microse offers first priority of the default
export, if a module
doesn't have a default export, microse will try to load all exports instead.)
// Be aware that the namespace must correspond to the filename.
// src/bootstrap.ts
import { ModuleProxy } from "microse";
declare global {
namespace app {
const bootstrap: ModuleProxy<Bootstrap>
}
}
export default class Bootstrap {
init() {
// ...
}
}
// src/models/User.ts
import { ModuleProxy } from "microse";
declare global {
namespace app {
namespace models {
// a class with constructor parameters must use the signature
// `typeof T`.
const User: ModuleProxy<typeof User>
}
}
}
export default class User {
constructor(private name: string) { }
getName() {
return this.name;
}
setName(name: string) {
this.name = name
}
}
And other files can access to the modules via the namespace:
// src/index.ts
import "./app";
// Accessing the module as a singleton and calling its function directly.
app.bootstrap.init();
// Using `new` syntax on the module to create a new instance.
var user = new app.models.User("Mr. Handsome");
console.log(user.getName()); // Mr. Handsome
If a module doesn't have a default class exported, the module it it self will be used instead, moreover, if the module exports an default object, it will be used as a prototype when creating a new instance of the module, and a deep clone will be created, if an argument is passed, it will be merged into the new object.
// src/config.ts
import { ModuleProxy } from "microse";
declare global {
namespace app {
const config: {
get(key: string): Promise<any>
}
}
}
export const hostname = "127.0.0.1";
export const port = 80;
export async function get(key: string) {
// some async operations...
return value
}
// Use `exports` property to access the module's original exports:
const config = app.config.exports;
print(`${config.hostname}:${config.port}`) // 127.0.0.1:80
// Functions can be called directly:
let value = await app.config.get("someKey");
The above examples access the modules and their functions in the current process, but we can do more, we can serve a module as a remote service, and calls its functions as remote procedures.
For example, if I want to serve a user service in a different process and communicate via RPC channel, I just have to do this:
// src/services/user.ts
import { ModuleProxy } from "microse";
declare global {
namespace app {
namespace services {
const user: ModuleProxy<UserService>
}
}
}
// It is recommended not to define the 'constructor' or use a non-parameter
// constructor.
export default class UserService {
private users: { firstName: string, lastName: string }[] = [
{ firstName: "David", lastName: "Wood" },
// ...
];
// Any method that will potentially be called remotely shall be async.
async getFullName(firstName: string) {
let user = this.users.find(user => {
return user.firstName === firstName;
});
return user ? `${firstName} ${user.lastName}` : void 0;
}
}
// src/server.ts
import { App } from "./app";
(async () => {
let channel = await App.serve("ws://localhost:4000");
await channel.register(app.services.user);
console.log("Server started!");
})();
Just try ts-node --files src/server
(or node dist/server
),
and the service will be started immediately.
And in client-side code, connect to the service before using remote functions:
// client.ts
import { App } from "./app";
(async () => {
let channel = await App.connect("ws://localhost:4000");
// Once registered, all functions of the service module will be remotized.
await channel.register(app.services.user);
// Accessing the instance in local style but actually calling remote.
let fullName = await app.services.user.getFullName("David");
console.log(fullName); // David Wood
})();
NOTE: to ship a service in multiple server nodes, just create and connect to multiple channels, and register the service to each of them, when calling remote functions, microse will automatically calculate routes and redirect traffics to them.
NOTE: RPC calling will serialize all input and output data, those data that cannot be serialized will be lost during transmission.
When in the need of transferring large data, generator functions could be a great help, unlike general functions, which may block network traffic when transmitting large data since they send the data as a whole, generator functions, on the other hand, will transfer the data piece by piece.
// src/services/user.ts
import { ModuleProxy } from "microse";
declare global {
namespace app {
namespace services {
const user: ModuleProxy<UserService>
}
}
}
export default class UserService {
private friends = {
"David": [
{ firstName: "Albert", lastName: "Einstein" },
{ firstName: "Nicola", lastName: "Tesla" },
// ...
],
// ...
}
async *getFriendsOf(name: string) {
let friends = this.friends[name]
if (friends) {
for (let friend of friends) {
yield `${friend.firstName} ${friend.lastName}`;
}
}
return "We are buddies";
}
}
// index.ts
(async () => {
let generator = app.services.user.getFriendsOf("David");
for await (let name of generator) {
console.log(name);
// Albert Einstein
// Nicola tesla
// ...
}
// The following usage gets the same result.
let generator2 = app.services.user.getFriendsOf();
while (true) {
let { value, done } = await generator2.next();
console.log(value);
// NOTE: calling next() will return the returning value of the generator
// as well, so the output would be:
//
// Albert Einstein
// Nicola tesla
// ...
// We are buddies
if (done) {
break;
}
}
})();
Microse provides a way to support life cycle functions, if a service class has
an init()
method, it will be used for asynchronous initiation, and if the
class has a destroy()
method, it will be used for asynchronous destruction.
With these feature, the service class can, for example, connect to a database
when starting the server and release the connection when the server shuts down.
// src/services/user.ts
declare global {
namespace app {
namespace services {
const user: ModuleProxy<UserService>
}
}
}
export default class UserService {
async init() {
// ...
}
async destroy() {
// ...
}
}
Microse also supports running as a standalone client outside the main program
codebase, Instead of importing from the main module, we import the
microse/client
sub-module, which is designed to be run in standalone Node.js
programs or even web apps. The client will not actually load any modules since
there are no such files, instead, it just map the module names so you can use
them as usual.
In the following example, we assume that app.services.user
service is served
by a Python program, and we can use it in our Node.js program as usual.
import { ModuleProxyApp } from "microse/client";
const app = global.app = new ModuleProxyApp("app"); // no path needed
(async () => {
channel = await app.connect("ws://localhost:4000");
await channel.register(app.services.user);
let fullName = await app.services.user.getFullName("David");
console.log(fullName); // David Wood
})();
And declare the service as an interface:
import { ModuleProxy } from "microse/client";
declare global {
namespace app {
namespace services {
const user: ModuleProxy<UserService>;
}
}
}
interface UserService {
getFullName(firstName: number): Promise<string>;
}
In NodeJS (with CommonJS module solution), require
and import
will
immediately load the corresponding module and make a reference in the current
scope. That means, if the module doesn't finish initiation, e.g. circular
import, the application may not work as expected. And if the module file is
modified, the application won't be able to reload that module without
restarting the program.
Microse, on the other hand, based on the namespace and ES6 proxy, it creates a "soft-link" of the module, and only import the module when truly needed (calling a method/function or create a new instance). And since it's soft-linked, when the module file is changed, it has the ability to wipe out the memory cache and reload the module with very few side-effects.
// Watch file changes and hot-reload modules.
App.watch();
Lifecycle function will work well during hot-reloading, if a program file is
changed and the program is about to reload it, the destroy()
function will be
called to release sources before unloading the module, and after the module is
reloaded, the init()
function will be called again to initiate the new service.
Once a remote service enters hot-reloading stage, it will be marked as unavailable temporarily, all remote calls will be automatically avoided from redirecting traffic to that server until the module finishes reloading.
This implementation supports fast-forwarding in the same process, that means, if it detects that the target remote instance is served in the current process, the function will always be called locally and prevent unnecessary network traffic.