Skip to content

Micro Remote Object Serving Engine, Node.js implementation

License

Notifications You must be signed in to change notification settings

microse-rpc/microse-node

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

53 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

🌹Microse

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:

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.

Install

npm i microse

Peel The Onion

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.

Example

// 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

Non-class Module

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");

Remote Service

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.

Example

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.

Generator Support

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;
        }
    }
})();

Lifecycle Support

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() {
        // ...
    }
}

Standalone Client

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>;
}

Auto-loading and Hot-reloading

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();

Hot-reloading Supports Lifecycle Functions

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.

Hot-reloading for Remote 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.

Inner-process Fast-forwarding

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.

About

Micro Remote Object Serving Engine, Node.js implementation

Resources

License

Stars

Watchers

Forks

Packages

No packages published