The node.js documentation generation utility by angular.js and other projects.
Try out the Dgeni example project at https://github.com/petebacondarwin/dgeni-example
You'll need node.js and a bunch of npm modules installed to use Dgeni. Get node.js from here: http://nodejs.org/.
In the project you want to document you install Dgeni by running:
npm install dgeni --save
This will install Dgeni and any modules that Dgeni depends upon.
Dgeni on its own doesn't do much. You much configure it with Packages that contain Services and Processors. It is the Processors that actually convert your source files to documentation files.
To run the processors we create a new instance of Dgeni
, providing to it an array of Packages
to load. Then simply call the generate()
method on this instance. The generate()
method runs the
processors asynchronously and returns a Promise to the generated documents.
var Dgeni = require('dgeni');
var packages = [require('./myPackage')];
var dgeni = new Dgeni(packages);
dgeni.generate().then(function(docs) {
console.log(docs.length, 'docs generated');
});
Dgeni is normally used from a build tool such as Gulp or Grunt but it does also come with a command line tool.
If you install Dgeni globally then you can run it from anywhere:
npm install -g dgeni
dgeni some/package.js
If Dgeni is only installed locally then you either have to specify the path explicitly:
npm install dgeni
node_modules/.bin/dgeni some/package.js
or you can run the tool in an npm script:
{
...
scripts: {
docs: 'dgeni some/package.js'
}
...
}
The usage is:
dgeni path/to/mainPackage [path/to/other/packages ...] [--log level]
You must provide the path to one or more Dgeni Packages to load. You can, optionally, set the logging level.
Services, Processors, configuration values and templates are be bundled into a Package
. Packages
can depend upon other Packages. In this way you can build up your custom configuration on
top of an existing configuration.
Dgeni provides a Package
type that you instantiate. This instance has methods to register Services and
Processors, and to configure the properties of Processors:
var Package = require('dgeni').Package;
var myPackage = new Package('myPackage', ['packageDepencency1', 'packageDependency2']);
myPackage.processor(require('./processors/processor1'));
myPackage.processor(require('./processors/processor2'));
myPackage.factory(require('./services/service1'));
myPackage.factory(require('./services/service2'));
myPackage.config(function(processor1, service2) {
service2.someProperty = 'some value';
processor1.specialService = service2;
});
Dgeni makes significant use of Dependency Injection (DI) to instantiate objects. Objects that
will be instantiated by the DI system must be provided by a factory function, which is registered
in a Package, either as a Processor, by myPackage.processor(factoryFn)
, or as a Service,
by myPackage.factory(factoryFn)
.
The parameters to a factory function are dependencies on other services that the DI system must find or instantiate and provide to the factory function.
car.js:
module.exports = function car(function(engine, wheels) {
return {
drive: function() {
engine.start();
wheels.turn();
}
};
})
Here we have defined a car
service, which depends upon two other services, engine
and wheels
defined elsewhere. Note that this car
service doesn't care how and where these dependencies are
defined. It relies on the DI system to provide them when needed.
The car
service returned by the factory function is an object containing one method, drive()
,
which in turn calls methods on engine
and wheels
.
You then register the Service with a Package:
myPackage.js:
var Package = require('dgeni').Package;
module.exports = new Package('myPackage')
.factory(require('./car'));
This car Service is then available to any other Service, Processor or configuration block:
var Package = require('dgeni').Package;
module.exports = new Package('greenTaxiPackage', ['myPackage'])
.factory(function taxi(car) {
return {
orderTaxi: function(place) { car.driveTo(place); }
};
})
.config(function(car) {
car.fuel = 'hybrid';
});
Processors are Services that contain a $process(docs)
method. The processors are run
one after another in a pipeline. Each Processor takes the collection documents from the previous
Processor and manipulates it, maybe inserting new documents or adding meta data to documents that are
there already.
Processors can expose properties that tell Dgeni where in the pipeline they should be run and how to validate the configuration of the Processor.
$enabled
- if set tofalse
then this Processor will not be included in the pipeline$runAfter
- an array of strings, where each string is the name of a Processor that must appear earlier in the pipeline than this Processor$runBefore
- an array of strings, where each string is the name of a Processor that must appear later in the pipeline than this one$validate
- a http://validatejs.org/ constraint object that Dgeni uses to validate the properties of this Processor.
You define Processors just like you would a Service:
myDocProcessor.js:
module.exports = function myDocProcessor(dependency1, dependency2) {
return {
$process: function (docs) {
//... do stuff with the docs ...
},
$runAfter: ['otherProcessor1'],
$runBefore: ['otherProcessor2', 'otherProcessor3'],
$validate: {
myProperty: { presence: true }
},
myProperty: 'some config value'
};
};
You then register the Processor with a Package: myPackage.js:
var Package = require('dgeni').Package;
module.exports = new Package('myPackage')
.processor(require('./myDocProcessor'));
The $process(docs)
method can be synchronous or asynchronous:
- If synchronous then it should return
undefined
or a new array of documents. If it returns a new array of docs then this array will replace the previousdocs
array. - If asynchronous then it must return a Promise, which should resolve to
undefined
or a new collection of documents. By returning a Promise, the processor tells Dgeni that it is asynchronous and Dgeni will wait for the promise to resolve before calling the next processor.
Here is an example of an asynchronous Processor
var qfs = require('q-io/fs');
module.exports = function readFileProcessor() {
return {
filePath: 'some/file.js',
$process(docs) {
return qfs.readFile(this.filePath).then(function(response) {
docs.push(response.data);
});
}
};
The dgeni-packages repository contains a many Processors - from basic essentials to complex, angular.js specific. These processors are grouped into Packages:
-
base
- contains the basic file reading and writing Processors as well as an abstract rendering Processor. -
jsdoc
- depends uponbase
and adds Processors and Services to support parsing and extracting jsdoc style tags from comments in code. -
nunjucks
- provides a nunjucks based rendering engine. -
ngdoc
- depends uponjsdoc
andnunjucks
and adds additional processing for the AngularJS extensions to jsdoc. -
examples
- depends uponjsdoc
and provides Processors for extracting examples from jsdoc comments and converting them to files that can be run. -
dgeni
- support for documenting dgeni Packages.
You can define processors that don't do anything but act as markers for stages of the
processing. You can use these markers in $runBefore
and $runAfter
properties to ensure
that your Processor is run at the right time.
The Packages is dgeni-packages define some of these marker processors. Here is a list of these in the order that Dgeni will add them to the processing pipeline:
- reading-files (defined in base)
- files-read (defined in base)
- parsing-tags (defined in jsdoc)
- tags-parsed (defined in jsdoc)
- extracting-tags (defined in jsdoc)
- tags-extracted (defined in jsdoc)
- processing-docs (defined in base)
- docs-processed (defined in base)
- adding-extra-docs (defined in base)
- extra-docs-added (defined in base)
- computing-ids (defined in base)
- ids-computed (defined in base)
- computing-paths (defined in base)
- paths-computed (defined in base)
- rendering-docs (defined in base)
- docs-rendered (defined in base)
- writing-files (defined in base)
- files-written (defined in base)
You can configure the Services and Processors defined in a Package or its dependencies by registering Configuration Blocks with the Package. These are functions that can be injected with Services and Processors by the DI system, giving you the opportunity to set properties on them.
You register a Configuration Block by calling config(configFn)
on a Package.
myPackage.config(function(readFilesProcessor) {
readFilesProcessor.sourceFiles = ['src/**/*.js'];
});