tsnode-proxify

Proxy-based AOP library for node.js. It allows you to extend/provide QoS handler and enable QoS to your methods via typescript decorators(metadata-programming syntax) without invasiveness, which increase modularity for your application

Usage no npm install needed!

<script type="module">
  import tsnodeProxify from 'https://cdn.skypack.dev/tsnode-proxify';
</script>

README

Introduction

tsnode-proxify is a proxy-based method hook and AOP library for node.js with typescript. It allows you to extend/provide customized QoS handler and apply these QoS features via typescript decorators(metadata-programming-like syntax) without invasiveness to existing code logic, which increase modularity for your application.

tsnode-proxify depends on typescript decorator feature which is marked as experimental feature in typescript. In the meanwhile, tsnode-proxify is still a beta version project for now.

Background

Before we jump into the tsnode-proxify, let's take one step back to take a look what problems could be resolved by proxy pattern and AOP, why it is so important in JEE.

Proxy pattern

The proxy is a well-used design pattern. We can see it in either high level software architecture design, e.g: api-gateway and service mesh in microservices, or a narrow-down specific programming module. Generally speaking, the proxy pattern provides the capablity to implement a contract interface, but adds special functionality on-the-fly behind the sense.

AOP

Aspect Oriented Programming(AOP) addresses the problem of cross-cutting concerns. It is a complement to Object-Oriented-Programming(OOP). Not like OOP, which provide class as an key unit of modularity, AOP focus on aspect as it's unit of modularity, the aspect extract and modularization the code which cut across multiple classes/components for the same concern solution in a system. Such concerns are often termed cross-cutting concerns in AOP literature. In a real-life system, the typical cross-cutting concerns includes logging, security, transaction management, caching, validation, etc. Providing AOP capability is important to increase modularity, and normally be a fundamental feature in a middleware platform.

AOP in Java

AOP is really popular in java. In a JEE runtime, the transaction, security are noramlly provided as AOP aspects under the hood. In SOA, the SCA runtime also heavily depends on the AOP to provide QoS, IT-Specific features around the business logic. In Spring, the AOP is actually delivered as a base component in fundamental layer and open to upper stack. Indeed, per my experiences on JEE server development, AOP provides an excellent solution for the problems in enterprise application field to increase modularity and make the system more loose-coupled, especially in the middleware product development.

Implementing an AOP framework to advise method execution, the proxy pattern is perfect fit here. The aspect module code can be abstracted, prepared and injected at before and after points of the method and also be able to recieve the execution context, arguments and output(or fault) to the target operation like it was there. That is reason we usually see the proxy pattern in an AOP framework. In a pure java world, some typical approaches to achieve this:

  • Weave(static-way)
    • Compile-time weaving(e.g: using AspectJ compiler, which can provide complete AOP implementation)
    • Load-time weaving(e.g: AspectJ LTW, which depends on the JVM option -javaagent, provide complete AOP implementation)
  • JDK Proxy(dynamical-way, for java interface and method execution join point only)
  • CGlib Proxy(dynamical-way, adopt to both class and interface, method execution join point only, but need the proxied method to be public)

In a pure java world, using dynamical proxy way for AOP implementation, it usually be used in conjunction with a IoC container, which can take over the responsibility of proxy instance construction and injection. Hide these implementation details and provide a clean/easy-to-use programming model to developer.

Motivation

As mentioned above, the AOP framework can bring us so many benefits to improve the software modularity. Inspired by JEE experiences, I believe it would be helpful if we have similar framework in node.js. That is reason I come across to tsnode-proxify project. The goal of tsnode-proxify is NOT to provide a complete AOP implementation, it primarily focus on method execution join point for the moment, will consider some IoC features to enable the property injection of proxied object in next step.

Concepts

Before we dig into the tsnode-proxify, we need to clarify some concepts.

  1. Interaction style for a method call in node.js

    • Sync: The method completion should be the same time point as method invocation being returned.
    e.g: 
    greet(name: string): string {
      return 'Hello, ' + name;
    }
    
    • Async: The method completion will be done in a future certain time point after the moment of method invocation being returned.
    e.g: 
    greet(name: string, cb: Function): void{
      setTimeout(function() {
        let reval = 'Hello, ' + name;
        cb(null, reval);
      }, 10); 
    }
    
  2. Completion hints: invocation completion hint is the concept relevant to how tsnode-proxify runtime understand after of an execution join point. It should be mark as the moment of method logic execution done.

    • target method getting returned (of course, this is for sync style method only)

    • the callback method getting called (adopt to both sync and async style method)

    • the returned promise's status change from pending to resolved or rejected (for async method with promise as return value only)

  3. tsnode-proxify can support below interaction style and completion hints combinations with before and after advise join points

    • sync-return

    • sync-callback

    • async-callback

    • async-promise

tsnode-proxify enable the aspect modularity to be implemented as an Interceptor class(declared by @Interceptor decorator) for a specific QoS intention, which can be dynamically injected into the join-point if a desired @QoS declaration being claimed on the target method.

@Interceptor({
  "interactionStyle": InteractionStyleType.SYNC
})
  class NoopInterceptor extends AbstractInterceptor{
  constructor(config: any) {
    super(config);	
  }
  
  // called before get into before and after advise
  public canProcess(context: InvocationContext, callback: canProcessCallbackFn): void {
    callback(null, true);	
  }
  
  // being called at before advise
  public handleRequest(context: InvocationContext, done: doneFn): void {
    done();	
  }
  
  // being called at after advise with output
  public handleResponse(context: InvocationContext, done: doneFn): void {
    done();	
  }
  
  // being called at after advise with fault 
  public handleFault(context: InvocationContext, done: doneFn): void {
    done();	
  }
  
  public getName(): string {
    return 'NoopInterceptor';	
  }
}

Notable, If the target method is sync interaction style(especially, expected a synchronous return value), all of interceptors applied to that method should also be sync interaction style.

Quick Start

  • Prerequsites
    • node.js
    • typescript toolkits(including tsc and ts-node command)
    • Q based promise, will support native promise soon.

If you don't want to following this step-by-step guide to complete the helloworld sample, you can download the sample from link for a quick start

  • Create your package folder e,g: named hello-tsnode-proxify

    mkdir hello-tsnode-proxify
    cd hello-tsnode-proxify
    mkdir src
    git init
    echo "# My first sample for tsnode-proxify" >> README.md
    git add . && git commit -m "initial commit"
    
  • Create package.json

    npm init -y 
    echo "node_modules" >> .gitignore
    echo "dist" >> .gitignore
    echo "package-lock.json" >> .gitignore
    npm install --save-dev typescript
    npm install --save tsnode-proxify
    
  • Create tsconfig.json

    {
      "compilerOptions": {
        "target": "es5",
        "module": "commonjs",
        "declaration": true,
        "outDir": "./dist",
        "experimentalDecorators": true
      },
      "include": ["src"],
      "exclude": ["node_modules"]
    }
    
  • Add scripts to package.json

    "build": "tsc",
    "start": "node dist/helloworld.js",
    "main": "dist/helloworld.js"
    
  • Create a logger interceptor in file src/logger.ts as below:

    To create a interceptor which encapsulate an cross-cutting concern solution, you just need to implement a class, which has:

    • @Interceptor decorator: declare a class to be registried as interceptor in tsnode-proxify runtime
    • Extend the AbastractInterceptor base class, and implement the methods.
    import {
      Interceptor,
      InteractionStyleType,
      AbstractInterceptor,
      InvocationContext,
      doneFn,
      canProcessCallbackFn,
    } from 'tsnode-proxify';
    
    @Interceptor({
      interactionStyle: InteractionStyleType.SYNC,
    })
    export class Logger extends AbstractInterceptor {
      private LOG_PREFIX: string = '[logger] ';
    
      constructor(config: any) {
        super(config);
      }
    
      private getTargetFullName(context: InvocationContext): string {
        let targetFullName = context.getClassName() + '.' + context.getOperationName();
    
        return targetFullName;
      }
    
      public init(context: InvocationContext, done: doneFn): void {
        console.log(this.LOG_PREFIX + '<init> ');
        done();
      }
    
      public handleRequest(context: InvocationContext, done: doneFn): void {
        console.log(
          this.LOG_PREFIX +
            '<request> ' +
            this.getTargetFullName(context) +
            '; [input]: "' +
            context.input +
            '"; [timestamp]: ' +
            new Date().getTime(),
        );
        done();
      }
    
      public handleResponse(context: InvocationContext, done: doneFn): void {
        console.log(
          this.LOG_PREFIX +
            '<response> ' +
            this.getTargetFullName(context) +
            '; [output]: "' +
            context.output +
            '"; [timestamp]: ' +
            new Date().getTime(),
        );
        done();
      }
    
      public handleFault(context: InvocationContext, done: doneFn): void {
        console.log(
          this.LOG_PREFIX +
            '<fault> ' +
            this.getTargetFullName(context) +
            '; [fault]: ' +
            context.fault +
            '; [timestamp]: ' +
            new Date().getTime(),
        );
        done();
      }
    
      public canProcess(context: InvocationContext, callback: canProcessCallbackFn): void {
        callback(null, true);
      }
    
      public getName(): string {
        return 'Logger';
      }
    }
    
  • Create a component in file src/helloworld.ts as below:

    As you see beow, to "declare" a class to be a component managed, we just need to apply a couple of decorators to it, there is no other differences than a normal class definition.

    • @Component decorator: declare a class to be managed as a component in tsnode-proxify
    • @QoS decorator: declare a method to be proxify and provide before and after advises
    import {
      Component,
      QoS,
      InteractionStyle,
      Completion,
      Callback,
      Fault,
      Output,
      InteractionStyleType,
    } from 'tsnode-proxify';
    import { Logger } from './logger';
    
    @Component()
    class Hello {
      constructor() {}
    
      @InteractionStyle(InteractionStyleType.SYNC)
      @QoS({ interceptorType: Logger })
      greet(name: string): string {
        console.log('[greet]    ==> I am saying hello to', name);
        return 'Hello, ' + name;
      }
    }
    
    // =====================
    //    main
    // ====================
    
    let hello: Hello = new Hello();
    console.log('[result]: "' + hello.greet('World') + '"');
    
  • Run the helloworld sample

    npm run build
    npm run start
    
  • The output result with aspect logger feature:

    > hello-tsnode-proxify@1.0.0 start /home/lizh/tmp/hello-tsnode-proxify
    > node dist/helloworld.js
    
    [logger] <init> 
    [logger] <request> Hello.greet; [input]: "World"; [timestamp]: 1541054935419
    [greet]    ==> I am saying hello to World
    [logger] <response> Hello.greet; [output]: "Hello, World"; [timestamp]: 1541054935420
    [result]: "Hello, World"
    

Notable, to keep the helloworld sample as simple as possible, I don't introduce some other decorators in that sample. If you want to try with promise or callback completion hints invocation, you can refer to stock sample. For more advanced usages(e.g: async/await, multiple interceptors w/ different interaction styles, interceptor slot context, etc), please refer to unit test cases for more details.

Run Unit Tests

You can run the integration tests to get a full picture of what features has been supported by tsnode-proxify so far.

npm run test

> mocha --compilers ts:ts-node/register,tsx:ts-node/register ./test/**/*test.ts

    ...

Integration Tests
    ✓ @QoS on static sync-return method with sync-interceptor
    ✓ @QoS on sync-return method with sync-interceptor
    ✓ @QoS on sync-callback method with sync-interceptor
    ✓ @QoS on async-promise method with sync-interceptor (102ms)
    ✓ @QoS on async-promise method with async-interceptor (102ms)
    ✓ @QoS on async-callback method with sync-interceptor (101ms)
    ✓ @QoS on async-callback method with async-interceptor (252ms)
    ✓ @QoS on sync-callback method which being invoked recursively with a pass-through callback function handler
    ✓ @QoS on a method with recursive invocations, QoSed method is triggered by "this" reference
    ✓ @QoS on sync-return bind()ed method
    ✓ Validation: async style interceptor can NOT be applied to sync style target method
    ✓ Validation: conflict/dumplicated interceptor names
    ✓ @QoS on async/await method with sync interceptor
    ✓ @QoS on async/await method with async interceptor

Join us

If you are interested in this project, please feel free to let me know, any bug-report/comments/suggestions/contribution on this project is appreciated. :-)