Skip to content

kristofferkarlsson93/nodeJs-DependencyLoader

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

65 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Nodejs DependencyLoader

No more passing around dependencies from file to file. This tool takes care of that for you. This is an automated dependency loader. It will inject all required dependencies in to your module without you having to do anything.

Usage

Install it

npm i @krikar/dependencyloader

Start it

snippet 1.0

const DependencyLoader = require('@krikar/dependencyloader');
const MyStartModule = require('./your/module/path');
const rootPath = __dirname;

const dependencyLoader = DependencyLoader(rootPath);
const myStartModule = dependencyLoader.load('myStartModule', MyStartModule );

myStartModule.myFunc();

Make use of it

snippet 2.0

// in file: MyStartModule.js
module.exports = function ({ dep1, dep2, dep3 }) {
    return { myFunc };
    
    function myFunc() {
        dep1.doStuff()
    }
};

For more detailed example see below. Detailed example

How does it work?

API

Functions Input Returns
DependencyLoader Your projects entry path An instance of the DependencyLoader containing the method load
dependencyLoader.load A name for your module and the uninstantiated module The instantiated module
dependencyLoader.feed An array with objects containing the name and module [{ moduleName: 'myModule', module: {} }]

The DependencyLoader requires you to specify a path from which it will start searching for dependencies. The easiest way of doing this is to provide it with the nodejs __dirname.

When the DependencyLoader has been instantiated, access is granted to the load method and the feed method.

The load method takes two parameters. First the name of the module and secondly the module itself. It returns the instantiated module.

The feed method takes an array of objects containing the moduleName and the instantiated module you want to feed the dependency loader with. This method can be used for hooking up your npm-dependencies with the dependencyLoader. For example you could feed the dependencyLoader with Express, and thus get access to express wherever you want by specifying it in your module. WARNING It is often a better practice to make a wrapper to your dependencies.

Modules

A module should ideally implement the Revealing Module Pattern and must take a destructed object as parameter. The destructed keys will get the instantiated dependencies as value. See usage in snippet 2.0

Dependency loading

The dependencyLoader.load will start searching the project tree to find files matching the dependency names in the module provided. If the dependencies them self have dependencies they also will be found and instantiated.

It will find dependencies that matches the following criteria. The search is case insensitive.

  • File name is equal to the module name
  • File name is index.js but parent directory matches the module name

It ignores folders with the following names

  • test
  • tests
  • node_modules

Detailed usage

To better illustrate the use of the dependencyLoader a more detailed example is provided.

The following example can also be found in full in the github Example folder in the repo.

|
| - | Controllers
| - - | Login
| - - - | LoginWithEmailController.js 
|
| - | DatabaseGateways
| - - | UserDataBaseGateway.js
|
| - | Helpers
| - - | TokenCreator
| - - - | index.js
| - - | ResponceCreator.js
|
| - | Providers
| - - | UserProvider.js
|
|- | app.js

Let's dive in to app.js.

const DependencyLoader = require('@krikar/dependencyloader');
const LoginWithEmailController = require('./Controllers/Login/LoginWithEmailController.js');

const dependencyLoader = DependencyLoader(__dirname);
const loginWithEmailController = dependencyLoader.load('loginWithEmailController', LoginWithEmailController);

loginWithEmailController.login('someMail@ignore.mail');

Here you would of course have some server-setup code too. But for this example that is not necessary.

When the DependencyLoader is required and instantiated the load function is executed to load the first module (LoginWithEmailController). As seen in the fileTree that module is located under Controllers/Login/. The module looks like this.

// LoginWithEmailController.js
module.exports = function ({ emailValidator, userProvider, responseCreator, tokenCreator }) {
    return {
        login
    };

    function login (email) {
        if (emailValidator.validates(email)) {
            const user = userProvider.getUserByEmail();
            const token = tokenCreator.createForUser(user);
            responseCreator.sendSuccess({ token });
        } else {
            responseCreator.sendBadCredentials();
        }
    }
};

As seen in the exported function the module requires four dependencies. When this module is loaded through the dependencyLoader the dependencyLoader will find these four dependencies in the project and inject them in to the module. So as long as the dependencies specified can be found, they will be injected.

If the modules dependencies themselves requires dependencies they will also be loaded, as seen in UserProvider, which is dependent on UserDatabaseGateway.

// Userprovider.js
module.exports = function ({ userDatabaseGateway }) {
    return {
        getUserByEmail
    };

    function getUserByEmail(email) {
        return userDatabaseGateway.getByEmail(email);
    }
};

When a dependency has been loaded once it is added to the cache.

Versions

  • 1.0 - First release.
  • 1.1
    • Adding the feed function.
    • Fixing a bug that occurred when dependencies are chopped down on new line.
      • 1.1.1
        • Better errors
        • can now use the dependencyLoader as a dependency
      • 1.1.2
        • Documentation updated
      • 1.1.3
        • Updating dev dependency in test runner that made Github complain.
      • 1.1.4
        • Documentation updated

About

An implementation of an automated dependency loader in javascript

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published