Skip to content
/ crave Public

Structure a node project your way with the ability to require models, controllers, or any file dynamically.

License

Notifications You must be signed in to change notification settings

ssmereka/crave

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

91 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Crave gives you the ability to structure your application's files any way you like without the burden of manually requiring each file's location. Take these file structures for example:

Left: A common node application file structure where each folder is required dynamically. This works great, but restricts where your files must be located.

Right: Crave allows you to move files where ever you like and will require them dynamically. This grouping by feature makes related code easier to find and transfer between projects.

Getting Started

Install Crave using npm and save it as a dependency in your package.json.

npm install crave --save

You can require Crave just like every other node.js module.

var crave = require('crave');

Controllers, modules, and other files you wish to require before starting your server should be structured like this:

// This is the Crave type, it tells crave when and
// if to load this file.  You can use any text you want.
// ~> Controller

// Export a function whose parameters are global values
// needed in the controller's logic.
module.exports = function (app, config) {

  // Place your controller logic inside this method.

  // An example route that sends "Hello"
  app.get('/', function (req, res, next) {
    res.send("Hello");
  });
};

Use Crave's directory method to require your files dynamically. They will be required in the order you specify and will recieve the parameters you pass.

var crave = require('crave'),
    express = require('express');

// Create an express application object.
var app = express();

// Create a method to start the server.
var startServerMethod = function(err, filesRequired, returnValues) {
  if(err) return console.log(err);

  // A list of files that were required by crave, or where attempted to be
  // required by crave are listed in the filesRequired parameter.
  console.log(filesRequired);
  
  // A list of returned values, or errors, from each file required are located in the 
  // returnValues parameter.
  console.log(returnValues);

  var server = app.listen(3000, function() {
    console.log("Listening on http://127.0.0.1:3000");
  });
}

// Define the directory where Crave will search for files in.
// Crave will do a recursive search, looking in children folders.
var directoryToLoad = '/path/to/my/app/files';

// Define an ordered list of file types for Crave to load.
// This is the Crave type we just talked about in the
// earlier example:  " // ~> Controller "
var types = [ "controller" ];

// Crave will now load the files in the given directory with the specified types.
// Each file will recieve the application and configuration objects as parameters.
// You can pass any number of parameters into the files being loaded by continuing
// to overload the crave.driectory method.  Once Crave has loaded all the files
// the callback method will be called, aka startServerMethod.
crave.directory(directoryToLoad, types, startServerMethod, app, config);

Errors

There are two types of errors passive and blocking. A blocking error is typically non-recoverable or requires the server to respond in some way. Where a passive error can be ignored, allowing the server to operate as normal or close to normal.

All errors are logged to the console by default and can be toggle off using the error flag in the Crave configuration object.

Blocking Errors

If a blocking error occurs while requiring files crave will return the error instantly to the callback method provided. Your server can then determine the best course of action.

// ...
var startServerMethod = function(err) {
  if(err) return console.log(err);
}
// ...
crave.directory(directoryToLoad, types, startServerMethod, app, config);

Passive Errors

If, for example, a file contains an error and as a result Crave fails to require the file, then a passive error would be generated. Passive errors are passed to the callback method in the 3rd parameter, known as returnValues. Where returnValues is a list of values returned from each required file in the same order as the list of filesRequired.

// ...
var startServerMethod = function(err, filesRequired, returnValues) {
  if(err) return console.log(err);
  
  console.log(returnValues);
}
// ...
crave.directory(directoryToLoad, types, startServerMethod, app, config);

Instead of undefined or the expected return value you will instead see something like the following:

{
  "error": "~/crave/examples/simple/app/error/errors_controller.js contains an error an therefore could not be required by Crave.",
  "stack": "TypeError: undefined is not a function\n    at requireFiles (~/crave/libs/index.js:450:32)\n    at ~/crave/libs/index.js:512:28\n    at createListFromCache (~/crave/libs/index.js:242:3)\n    at ~/crave/libs/index.js:506:11\n    at saveCache (~/crave/libs/index.js:111:12)\n    at updateDirectoryInCache (~/crave/libs/index.js:218:3)\n    at ~/crave/libs/index.js:402:5\n    at ~/crave/libs/index.js:304:15\n    at ~/crave/libs/index.js:316:15\n    at ~/crave/libs/index.js:371:22"
}

Config

You can configure Crave using the setConfig(myConfigObject) method. Pass along an object with any of the properties you wish to override. For example:

var crave = require('crave');
var express = require(express);

var app = express();

crave.setConfig({
  cache: {
    enable: true              // Enable caching of the list of files to require.
  },
  debug: true,                // Display log messages in crave.
  identification: {
    type: 'string',           // Search for a string after an identifier in each file.
    identifier: "(>^_^)>"     // The identifier preceding the string.  Example: (>^_^)> Controller
  }
})

crave.directory("/path/to/directory", [ "controller" ], function(err) { console.log(err || "success"), app);

The available properties are:

Property Type Default Description
cache Object An object containing configuration properties related to file path caching.
cache.enable Boolean false when true, the file path cache is used.
cache.path String /data/cache.json An absolute path to where the file path cache is stored or will be stored.
debug Boolean false When true, Crave will display log messages.
error Boolean true When true, Crave will display error log messages.
identification Object An object containing configuration properties related to how files are recognized and grouped.
identification.type String string Describes the method on each file to find an identifier. Available options are string or filename, where either the file is searched or a filename is searched looking for the identifier specified, respectively.
identification.identifier String ~> A unique string used to indicate the following string indicates the grouping name for a file.
trace Boolean false When true, Crave will display trace log messages.

File Identification

You may have noticed that you can change how Crave identifies a file as a model, controller, or etc. There are two ways to identify a file, either using a string of text in the file or by filename.

String

When using the string option, crave will search all the text in every file looking for the unique identifier you specify. Once an identifier is found the following string will be evaluated as a possible type (e.g. controller).

String identification is the default for Crave and an example can be found in the Getting Started section.

Filename

When using the filename option, crave will search each filename for the unique identifier you specified. Once an identifier is found the following characters will be evaluated as a possible type (e.g. controller).

Crave can be configured to use filenames using the configuration object. Here is an example:

var crave = require('crave');
var express = require(express);

var app = express();

crave.setConfig({
  identification: {
    type: 'filename',
    identifier: "_"
  }
})

crave.directory("/path/to/directory", [ "model", "controller" ], function(err) { console.log(err || "success"), app);

Using the configuration described above the following file structure:

Crave would generate a list of files to require that looks like this:

~/myproject/app/device/device_model.js
~/myproject/app/user/user_model.js
~/myproject/app/device/device_controller.js
~/myproject/app/user/user_controller.js

Cache

Searching for files or inside files can take some time. In a development environment this time is negligible, however in a production environment we should avoid it. So when in production you should enable the cache.

crave.setConfig({
  cache: {
    enable: true
  }
});

Once the cache is enabled and crave.directory() is called, then crave will save the ordered list of files to a file. After that each time crave.directory() is called the same list of files will be required until it is cleared, even if the server is restarted or new files are added.

Clear Cache

Crave can of course delete the cache using the clearCache() method. Once the cache is deleted, a new list will be generated and saved the next time crave.directory() is called. Lets look at an example:

// Keeps track of how many time we have restarted the server.
var restartCounter = 0;

// Create a method to start the server.
var startServerMethod = function(err) {
  if(err) return console.log(err);

  var server = app.listen(3000, function() {
    console.log("Listening on http://127.0.0.1:3000");

    if(restartCounter < 1) {
      console.log("Restarting server for the " + restartCounter + " time."); // 1st

      // Increment the restart counter and restart the server
      restartCounter++;
      app.close();

      // Crave recognizes that the cache is enabled and already exists.  The cached
      // list is used to require all the files.  Even though we requested that "model"
      // also be included, they will not.
      crave.directory("/path/to/directory", [ "controller", "model" ], startServerMethod, app);
    } else if(restartCounter < 2) {
      console.log("Restarting server for the " + restartCounter + " time."); // 2nd

      // Increment the restart counter and restart the server
      restartCounter++;
      app.close();

      // Clear the cache
      crave.clearCache();

      // Crave recognizes that the cache is enabled and does not exist.  This triggers
      // crave to search for all the files to require and rebuild the cache.  This time
      // "model" will be included.
      crave.directory("/path/to/directory", [ "controller", "model" ], startServerMethod, app);
    }
  });
}

// Enable the cache for crave.
crave.setConfig({
  cache: {
    enable: true
  }
});

// Trigger crave to search for all the files to require.  Once found this ordered
// list will be saved to a file.  Once this file is created, it will never change
// until you tell crave to remove it.
crave.directory("/path/to/directory", [ "controller" ], startServerMethod, app);

Cache Path

The cache file is stored by default in the crave module folder at /data/cache.json. You can change this location by specifying an absolute path in the path property of the configuration object.

crave.setConfig({
  cache: {
    enable: true,
    path: "/Absolute/Path/To/Custom/Cache/File.json"
  }
})

Debug

Debugging crave can be done using the debug, trace, and error flags that can be toggled on/off using the config. When enabling these flags additional logging will be enabled allowing you to find issues within Crave easier.

Documentation

Further documentation can be found in the wiki.

About

Structure a node project your way with the ability to require models, controllers, or any file dynamically.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published