npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

packetjs-di

v2.1.0

Published

Lightweight and powerful dependency injection container for Node.js and JavaScript apps written in TypeScript.

Downloads

92

Readme

Packet.js DI

packet.js di

node install size npm bundle size zip npm bundle size Dependencies npm downloads Build status MIT License Coverage Status Known Vulnerabilities

Packet.js DI is a lightweight Dependency Injection container written in TypeScript and made for JavaScript applications like Node.js, Angular, React, Vue, Vanilla JS, etc.

Packet.js is designed to be simple, powerful and easy to use. It provides a lightweight and efficient way to manage dependencies in your JavaScript/Node applications. With its lazy loading feature, it only instantiates services when they are actually needed, improving performance and reducing memory usage.

Additionally, it supports both object-oriented programming and functional programming, making it versatile and adaptable to different coding styles. Whether you're building a small application or a large-scale project, Packet.js can help you organize and manage your dependencies effectively.

The main features of Packet.js are:

  • Lazy loading: It only instantiates services when they are actually needed, improving performance and reducing memory usage.
  • Object-oriented and Functional programming: It supports both object-oriented and functional programming, making it versatile and adaptable to different coding styles.
  • Dependency injection: It provides a simple and flexible way to manage Dependency Inversion or Inversion of Control (IoC), which helps to improve the design and uncoupling of the application components.
  • Middleware management: It intercepts and modifies requests and responses from the services, before they are sent to the next service in the middleware chain, which can be very useful for correction and error management, adding new features, etc.
  • Caching: It provides a caching service that can be used to improve the performance of your application.
  • Global container: Provides a way to retrieve the global container across the application through the import statement.
  • Performant: It is designed with focus on performance and memory usage.

Table of Contents

Installation

Using npm:

npm i packetjs-di

Using yarn:

yarn add packetjs-di

Importing

// Using Node.js `require()`
const { default: packetJsDi, Container } = require('packetjs-di');

// Using ES6 imports
import packetJsDi, { Container } from 'packetjs-di';

packetJsDi variable is the global instance of the container.

Basic usage

Register a service:

// file container.config.ts (js or ts) 
import { Container } from 'packetjs-di';

const container = new Container();

container.add('Service', () => {
  return new SomeService();
});

export default container;

Get the service in the application:

import container from 'container.config';

const service = container.get('Service'); // Instantiate the service

With typescript:

import container from 'container.config';
import { Service } from "./@types";

const service = container.get<Service>('Service');

Generic types are supported. This allows you to use the get() method with your own types. For more information, see the Typescript section.

Global Container

Getting the global container:

import { Container } from 'packetjs-di';

const globalContainer = Container.getContainer();

Note that getContainer() method is a static method of the Container class.

You can also get the global container using the import statement:

// Import the global container using CommonJS modules
const { default: packetJsDi } = require('packetjs-di');

// Import the global container using ES6 imports
import packetJsDi from 'packetjs-di';

Register a service using the global container from the import statement:

import packetJsDi from 'packetjs-di';

packetJsDi.add('Service', () => {
  return new SomeService();
});

Get the service in the application using the global container from the import statement:

import packetJsDi from 'packetjs-di';

const service = packetJsDi.get('Service'); // Instantiate the service

In both cases, packetJsDi will always have the same global instance of the container. Except in environments rendered in SSR vs CSR, like Next.js, Angular, etc., where the container instance may be different depending on the rendering environment.

Registering services

The add() method allows you to register each service in the container.

container.add(key, callback, options);

The most basic example is the following (Object-oriented programming):

container.add('Service', () => new Service());

And, for example, with a React custom hook (Functional programming):

container.add('useCustomHook', () => useCustomHook()); 

Note that the service statement must be wrapped in a callback function, and in both cases, should return an instance of the class or the function.

See React Considerations for more information about React hooks and singleton option.

By default, the singleton option is enabled. To disable it, use the singleton option:

container.add('Service', () => new Service(), { singleton: false });

This disables the singleton option for the service's get() method, and it is equivalent to using the getFactory() method instead.

Caching is disabled by default. To enable it, use the cache option:

container.add('Service', () => new Service(), { cache: true });

See more information for caching in the Caching/Memoization section

You can also protect the service statement from changes by using the freeze option:

container.add('Service', () => new Service(), { freeze: true });

By default, the registered services are not protected. This means that if you declare the same service twice, the second one will overwrite the first one.

container.add('Service', () => new Service1());
container.add('Service', () => new Service2());

container.get('Service'); // new Service2() will be returned

With the freeze option enabled, the service statement will be protected from changes, and the service cannot be modified. This is useful for services that you want to protect from any changes once they are set up.

More details about the add() method can be found in the Container API

Getting services

get(key, options)

The usual way to get a service is using the get(key) method:

const service = container.get('Service');
  • options: { proxyMiddleware: boolean }. See the Container API for more details.

This instantiates the service only the first time, and returns the same instance for each subsequent call. Singleton option is enabled by default in the registration add() method.

getAll(options)

The getAll() method returns an object with all the services registered in the container:

const services = container.getAll();
  • options: { proxyMiddleware: boolean }. See the Container API for more details.

getAll() does not instantiate any of the services. It only returns a callback function for each service that will make the service instance when called.

Then, with this object, we can apply different techniques to get each service:

Destructuring

const { Service1, Service2 } = container.getAll();

const instance1 = Service1(); // Make the instance of the Service1. 
const instance2 = Service2(); // Make the instance of the Service2.

Method chaining

// Instantiate Service1 and Service2.
const instance1 = container.getAll().Service1();
const instance2 = container.getAll().Service2();

Global variable

const services = container.getAll();

const instance1 = services.Service1(); // Instantiate the Service1.
const instance2 = services.Service2(); // Instantiate the Service2.

getFactory(key, options)

If you need a factory for the registered services, you can use the getFactory method, which always returns a different instance of the service:

const instance1 = container.getFactory('Service');
const instance2 = container.getFactory('Service');
// instance1 !== instance2
  • options: { proxyMiddleware: boolean }. See the Container API for more details.

Instead, with get() method, with the default singleton option enabled:

const instance1 = container.get('Service');
const instance2 = container.get('Service');
// instance1 === instance2

Adding properties

You can add configuration properties, which are useful for service registration or making settings available throughout the application.

To add properties from a file in JSON format, use the following code:

// Add properties from some file in json format
const properties = require('./index.json');
container.addProps(properties);

To retrieve the properties, use the code below:

const properties = container.getProps();

You can also get the properties when registering a new service as follows:

container.add('Service', ({ props }) => {
  return new Service(props.someProperty);
});

To add new properties to existing ones:

container.addProps({
  newProperty: { foo: 'bar' }
});

The addProps() method will overwrite any existing properties that have the same name.

Injecting Dependencies

This is the most common way to manage multiple dependencies. For instance, we'll create a service that depends on the database and dao service.

The callback function will be passed the container and the properties: { container, props }. These provide access to any service declared in the container and retrieve configuration or project properties.

// Configuring some service, which depends on the dao
container.add('Service', ({ container }) => {
  return new Service(container.get('dao'));
});

// Configuring the dao service, which depends on the db
container.add('dao', ({ container }) => {
  return new dao(container.get('db'));
});

// Configuring the database, which depends on the properties
container.add('db', ({ props }) => {
  return new someDatabase(
    process.env.USER,
    process.env.PASS,
    props.someConfigurationProperty
  );
});

Note that with dependency injection container, the registration order for each service is irrelevant. Each registered service is called on demand when the main service is invoked.

Below is an example of how to use the container:

// Instantiate each dependency on demand. Lazy instantiation.
const foo = container.get('Service').getBar(id);
// Or like this
const foo = container.getAll().Service().getBar(id);

In the preceding snippet, the Service has a single dependency. However, you can inject as many as needed:

// Configuring a service which depends on many services
container.add('Service', ({ container }) => {
  const service1 = container.get('Service1');
  const service2 = container.get('Service2');
  const service3 = container.get('Service3');
  return new Service(service1, service2, service3);
});

Middleware

The concept of middleware remains similar to other frameworks like Express, but it is adapted to intercept and modify the requests and responses of each of the container's services, instead of handling HTTP requests.

js-service-middleware.png

Middleware Stack

The middleware stack is a chain of functions that execute in sequence when an operation is performed on a particular service.

Each middleware can modify the request, perform additional operations and/or modify the response.

next Function

The next function is used to pass control to the next middleware in the chain and execute the underlying operation.

next returns the response, which can be synchronous or a promise, and which resolves with the result of the operation or rejects with an error.

Interception and Modification

The middleware can intercept the request before it is sent to the service, modify the request parameters, add additional logic and manipulate the response after it is received.

The middleware only intercept calls to methods or functions of previously instantiated services. But, they do not intercept service instantiation.

The middleware with a higher priority value runs first. Conversely, if you need to create a middleware that runs closer to the service execution, assign it a lower priority value. Note that the priority value must be an integer. Negative values are also allowed.

Registering Middleware

middleware.add(key, middleware, options)

The middleware.add() method allows you to add a middleware to the middleware stack.

container.middleware.add(key, middleware, options);

Arguments:

  • key: Unique key for the new middleware, which should match the service name.
  • middleware: Callback function with the signature (next, context, args) => {}
    • next(): Function that accepts args as a parameter, and is responsible for calling the next middleware in the stack, and return the result of calling the next middleware.
    • context: { serviceName: string, methodName: string, container: Container }
    • args: The parameters array passed to the operation performed by the called method or function.
  • options: { priority: number, name: string }

See the Middleware API for more details.

The most basic example of a middleware, without any modifications, is as follows:

container.middleware.add('Service', (next, context, args) => {
  return next(args);
});

The next() function receives the arguments from the middleware. You can modify these arguments before calling the next middleware if needed.

Internally, if the next() function is called with empty arguments, it will use the default arguments from the service method instead.

See Service Middleware Considerations for more information about using the container inside a middleware.

middleware.addGlobal(middleware, options)

Global middleware are similar to service middleware, but they are applied to all the services in the container.

js-global-middleware.png

The signature of addGlobal() method is similar to add() method, without the key parameter:

container.middleware.addGlobal(middleware, options);

See the Middleware API for more details.

The most basic example of a global middleware, without any modifications, is as follows:

container.middleware.addGlobal((next, context, args) => {
  return next(args);
});

Global middleware, which have higher priority than service middleware, are always executed first.

See Global Middleware Considerations for more information about using the container inside a global middleware.

Middleware example using axios

The following example shows how to add a middleware to intercept the requests for the axios library.

First, we declare the axios instance to the container:

import { Container } from 'packetjs-di';
import axios from 'axios';
import properties from './properties/index.json';
import { Properties } from './@types';

// Create the container
const container = new Container();

// Add configuration properties
container.addProps(properties);

// Add the axios service
container.add('axios', ({ props }: { props: Properties }) => {
  return axios.create({
    baseURL: props.axios.baseURL, // jsonplaceholder.typicode.com
  });
});

Next, we register the middleware:

import { AxiosResponse } from 'axios';
import { Comments, HelperDefinition } from "./@types";

container.middleware.add('axios', async (next, context, args) => {
  // We can access the service name, method name and container
  const { container, methodName } = context;

  // We can override the arguments if needed
  console.log('arguments', args);

  // Call the next middleware and get the response
  const response: AxiosResponse = await next(args);

  // Here we can modify the response
  if (methodName === 'get' && response.status === 200 && response.config.url === '/comments') {
    const { data: comments }: { data: Comments[] } = response;

    // Get the Helper from the container
    const Helper = container.get<HelperDefinition>('Helper');

    // Update comments by reference
    comments.forEach((comment) => {
      comment.random = Helper.getRandom();
      comment.uniqId = Helper.getUniqId();
    });
  }

  return response;
}, {
  priority: 1,
  name: 'axiosService'
});

The context object contains information about the service name, method name and container. You can get any service declared in the container through the Container instance using the get(), getAll() or getFactory() methods.

Here's how to make the request that will be processed by the middleware:

const axios = container.get<Axios>('axios');

axios.get('/comments').then(({ data: comments }) => {
  console.log(comments);
});

Caching/Memoization

The dependency injection container is able to cache the methods or functions of each service, which significantly improves the performance of applications that call functions that are expensive to compute. It is also useful for large applications that make multiple calls to methods or functions that perform the same operation over and over again.

The caching system is implemented on a middleware with the lowest priority, allowing the middleware to execute as close to the service execution as possible.

js-cache-middleware.png

See Pure Functions Considerations for more information about using the caching system with non-pure functions.

Caching all methods

To enable the caching for all methods, you can use the cache option when registering the service:

container.add('Service', () => {
  return new Service();
}, { cache: true });

Caching only for specific methods

With the methods option you can specify a list of methods that should be handled by the cache if the cache is enabled.

container.add('Service', () => {
  return new Service();
}, {
  cache: true,
  methods: ['someMethod1', 'someMethod2'],
});

This caches only the methods someMethod1 and someMethod2 of the service.

Caching all methods except specific ones

With the excludeMode option you can specify a list of methods that should not be cached. This is useful if you have a large number of methods and only want to exclude some of them.

container.add('Service', () => {
  return new Service();
}, {
  cache: true,
  methods: ['someMethod1', 'someMethod2'],
  excludeMode: true,
});

This caches all the methods of the service except the methods someMethod1 and someMethod2.

No caching

The default behavior, without the cache option, is to not cache any method of the service.

container.add('Service', () => {
  return new Service();
});

Typescript

Packet.js is built on top of TypeScript and supports TypeScript projects. This section will describe how to use Packet.js with TypeScript and Generic Types.

get<T>(key, options)

The get<T>() method returns the instance of the registered service. So, to define the type of the service, you can do the following:

import { Service } from './@types';

const service = container.get<Service>('Service');

getFactory<T>(key, options)

Similar to the get<T>() method, the getFactory<T>() method returns an instance for the registered service, and the type of the service is defined using the getFactory<T>() method.

import { Service } from './@types';

const service = container.getFactory<Service>('Service');

getAll<T>(options)

The getAll<T>() method returns an object with the following structure:

{
  Service1: () => { // Logic for the registered service 1 };
  Service2: () => { // Logic for the registered service 2 };
  ...
  ...
}

So, to define the type of each service, you can do the following:

import { Service1, Service2 } from './@types';

interface ServicesMap {
  Service1: () => Service1; // Declare the type for the service 1
  Service2: () => Service2; // Declare the type for the service 2
  ...
  ...
}

Then we can use it like this:

const services = container.getAll<ServicesMap>();

getProps<T>()

The getProps<T>() method returns an object with the properties registered in the container. So, to define the type of the properties, you can do the following:

import { Props } from './@types';

const props = container.getProps<Props>();

Considerations

This section outlines some of the considerations that should be taken into account when using Packet.js.

API Reference

Container API

| Method | Arguments | Description | |-------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------| | constructor(options) | options:: { freeze: boolean, proxyMiddleware: boolean }Properties: freeze: Indicates if the services in the container should be frozen (unmodifiable). By default, it is false.proxyMiddleware: Indicates if the services in the container should be proxied with a middleware. By default, the proxy middleware is enabled for all services. | Constructor for the container. | | add(key, callback, options) | key: Unique key for the new service or function.callback({ container, props }): Allows you to add a new service or function to the container. The callback function will receive the container and the properties as arguments.options: Configuration object for registering the service or function: { cache: boolean, methods: array, excludeMode: boolean, freeze: boolean, singleton: boolean }Properties:cache: if true enable memoization.methods: array of methods to be cached.excludeMode: if true, exclude the methods, from methods array, to be cached.freeze: if true, freeze the service registration.singleton: if false, disable the singleton option for the service. By default, the singleton option is enabled.For more information about caching system see the Caching/Memoization section. | Register a new service or function to the container. | | get(key, options) | key: Unique key of the service or function to get.options: { proxyMiddleware: boolean }Properties: proxyMiddleware: if false, disable the proxy middleware. By default, the proxy middleware is enabled. | Always gets the same instance for a concrete service. (singleton option enabled) | | getAll(options) | options: Same as get(key, options) method. | Gets all services and return them as an object. Each service, of this object, is wrapped in a callback function, and called on demand when call this function. | | getFactory(key, options) | The arguments are the same as get(key, options) method. | Always gets a new instance for a concrete service. | | addProps(props) | props: Configuration properties object in JSON format. | | | getProps() | | Gets the configuration properties object. | | getContainer() | | Gets the global container. (static method) |

Middleware API (container.middleware)

| Method | Arguments | Description | |----------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------------------------------------------------| | add(key, middleware, options) | key: Unique key for the new middleware, which should match the service name.middleware(next, context, args): Callback function to register a new middleware. next(): call the next middleware in the stack, and return the result of calling the next middleware. Accepts the args as a parameter.context: { serviceName: string, methodName: string, container: Container } serviceName: Name of the service called.methodName: Name of the method or function called.container: Container instance.args: Array with the parameters from the method or function called.options: { priority: number, name: string }:Properties:priority: sets the priority of the middleware. A higher priority means that the middleware will be executed first. By default, if no priority is set, the priority of middleware will be in order of registration.name: sets the name of the middleware. The name is not required. Only used for debugging purposes.For more information about middleware see the Registering Middleware section. | Add a new middleware to the middleware stack. | | addGlobal(middleware, options) | middleware(next, context, args): Same as middleware.add() method.options: Same as middleware.add() method. | Add a new global middleware to the middleware stack. |

Examples

To see a list of practical examples of how to use Packet.js in various scenarios, click here.

Support

If you encounter any issues or bugs while using packetjs, or if you have any feature requests, please feel free to open an issue in our GitHub repository. You can report issues, track progress, and engage in discussions with the community via the following link:

Before submitting a new issue, we encourage you to search through the existing issues to see if the problem has already been reported or discussed. This helps reduce duplicates and allows us to address concerns more efficiently.

For any other questions or inquiries, you can also reach out to the maintainers directly through the GitHub discussion board.

We appreciate your feedback and contributions to making packetjs even better!

Credits

Packet.js is inspired by pimple, a small PHP dependency injection container.

License

MIT