api-builder: add target package

This commit is contained in:
Peter Bacon Darwin 2015-11-06 21:40:30 +00:00
parent 9a6fa4b4c5
commit b68ef96472
9 changed files with 263 additions and 31 deletions

View File

@ -11,6 +11,7 @@ var Q = require("q");
var delPromise = Q.denodeify(del); var delPromise = Q.denodeify(del);
var Minimatch = require("minimatch").Minimatch; var Minimatch = require("minimatch").Minimatch;
var Dgeni = require('dgeni'); var Dgeni = require('dgeni');
var Package = require('dgeni').Package;
var fsExtra = require('fs-extra'); var fsExtra = require('fs-extra');
var fs = fsExtra; var fs = fsExtra;
var exec = require('child_process').exec; var exec = require('child_process').exec;
@ -102,11 +103,14 @@ gulp.task('build-devguide-docs', ['_shred-devguide-examples'], function() {
return buildShredMaps(true); return buildShredMaps(true);
}); });
gulp.task('build-api-docs', ['_shred-api-examples'], function() { gulp.task('build-api-docs', ['build-js-api-docs', 'build-ts-api-docs']);
if (!fs.existsSync(ANGULAR_PROJECT_PATH)) {
throw new Error('build-api-docs task requires the angular2 repo to be at ' + path.resolve(ANGULAR_PROJECT_PATH)); gulp.task('build-ts-api-docs', ['_shred-api-examples'], function() {
} return buildApiDocs('ts');
return buildApiDocs(); });
gulp.task('build-js-api-docs', ['_shred-api-examples'], function() {
return buildApiDocs('js');
}); });
gulp.task('_shred-devguide-examples', ['_shred-clean-devguide'], function() { gulp.task('_shred-devguide-examples', ['_shred-clean-devguide'], function() {
@ -119,6 +123,7 @@ gulp.task('_shred-clean-devguide', function(cb) {
}); });
gulp.task('_shred-api-examples', ['_shred-clean-api'], function() { gulp.task('_shred-api-examples', ['_shred-clean-api'], function() {
checkAngularProjectPath();
return docShredder.shred( _apiShredOptions); return docShredder.shred( _apiShredOptions);
}); });
@ -232,7 +237,7 @@ function apiSourceWatch(postShredAction) {
console.log('Event type: ' + event.event); // added, changed, or deleted console.log('Event type: ' + event.event); // added, changed, or deleted
console.log('Event path: ' + event.path); // The path of the modified file console.log('Event path: ' + event.path); // The path of the modified file
// need to run just build // need to run just build
buildApiDocs().then(done); Q.all([buildApiDocs('ts'), buildApiDocs('js')]).then(done);
}); });
var examplesPattern = [path.join(ANGULAR_PROJECT_PATH, 'modules/angular2/examples/**/*.*')]; var examplesPattern = [path.join(ANGULAR_PROJECT_PATH, 'modules/angular2/examples/**/*.*')];
watch(examplesPattern, function (event, done) { watch(examplesPattern, function (event, done) {
@ -249,19 +254,34 @@ function apiSourceWatch(postShredAction) {
} }
function buildApiDocs() { // Generate the API docs for the specified language, if not specified then it defaults to ts
function buildApiDocs(targetLanguage) {
var ALLOWED_LANGUAGES = ['ts', 'js'];
checkAngularProjectPath();
try { try {
var dgeni = new Dgeni([require(path.resolve(TOOLS_PATH, 'api-builder/angular.io-package'))]); // Build a specialized package to generate different versions of the API docs
return dgeni.generate().then(function() { var package = new Package('apiDocs', [require(path.resolve(TOOLS_PATH, 'api-builder/angular.io-package'))]);
// Make a copy of the JS API docs to the TS folder package.config(function(targetEnvironments, writeFilesProcessor) {
return gulp.src([path.join(DOCS_PATH, 'js/latest/api/**/*.*'), '!' + path.join(DOCS_PATH, 'js/latest/api/index.jade')]) ALLOWED_LANGUAGES.forEach(function(target) { targetEnvironments.addAllowed(target); });
.pipe(gulp.dest('./public/docs/ts/latest/api')); if (targetLanguage) {
targetEnvironments.activate(targetLanguage);
writeFilesProcessor.outputFolder = targetLanguage + '/latest/api';
}
}); });
var dgeni = new Dgeni([package]);
return dgeni.generate();
} catch(err) { } catch(err) {
console.log(err); console.log(err);
console.log(err.stack); console.log(err.stack);
throw err; throw err;
} }
function copyApiDocsToJsFolder() {
// Make a copy of the JS API docs to the TS folder
return gulp.src([path.join(DOCS_PATH, 'ts/latest/api/**/*.*'), '!' + path.join(DOCS_PATH, 'ts/latest/api/index.jade')])
.pipe(gulp.dest('./public/docs/js/latest/api'));
}
} }
function devGuideExamplesWatch(shredOptions, postShredAction) { function devGuideExamplesWatch(shredOptions, postShredAction) {
@ -443,6 +463,10 @@ function execCommands(cmds, options, cb) {
}); });
} }
function checkAngularProjectPath() {
if (!fs.existsSync(ANGULAR_PROJECT_PATH)) {
throw new Error('API related tasks require the angular2 repo to be at ' + path.resolve(ANGULAR_PROJECT_PATH));
}
}
gulp.task('default', ['help']); gulp.task('default', ['help']);

View File

@ -30,7 +30,7 @@
"canonical-path": "0.0.2", "canonical-path": "0.0.2",
"del": "^1.2.0", "del": "^1.2.0",
"dgeni": "^0.4.0", "dgeni": "^0.4.0",
"dgeni-packages": "^0.11.0", "dgeni-packages": "^0.11.1",
"diff": "^2.1.3", "diff": "^2.1.3",
"fs-extra": "^0.24.0", "fs-extra": "^0.24.0",
"glob": "^5.0.14", "glob": "^5.0.14",

View File

@ -1,12 +1,13 @@
var path = require('canonical-path'); var path = require('canonical-path');
var Package = require('dgeni').Package; var Package = require('dgeni').Package;
var basePackage = require('../docs-package'); var basePackage = require('../docs-package');
var targetPackage = require('../target-package');
var PROJECT_PATH = path.resolve(__dirname, "../../.."); var PROJECT_PATH = path.resolve(__dirname, "../../..");
var PUBLIC_PATH = path.resolve(PROJECT_PATH, 'public'); var PUBLIC_PATH = path.resolve(PROJECT_PATH, 'public');
var DOCS_PATH = path.resolve(PUBLIC_PATH, 'docs'); var DOCS_PATH = path.resolve(PUBLIC_PATH, 'docs');
module.exports = new Package('angular.io', [basePackage]) module.exports = new Package('angular.io', [basePackage, targetPackage])
.factory(require('./services/renderMarkdown')) .factory(require('./services/renderMarkdown'))
.processor(require('./processors/addJadeDataDocsProcessor')) .processor(require('./processors/addJadeDataDocsProcessor'))

View File

@ -1,7 +1,8 @@
var Package = require('dgeni').Package; var Package = require('dgeni').Package;
module.exports = new Package('target', []) module.exports = new Package('target', [require('dgeni-packages/jsdoc')])
.factory(require('./services/targetEnvironments'))
.factory(require('./inline-tag-defs/target')) .factory(require('./inline-tag-defs/target'))
.config(function(inlineTagProcessor, targetInlineTagDef) { .config(function(inlineTagProcessor, targetInlineTagDef) {

View File

@ -5,19 +5,30 @@ var _ = require('lodash');
* @description * @description
* Process inline `target` block tags * Process inline `target` block tags
* (of the form `{@target environment1 environment2}...{@endtarget}`), * (of the form `{@target environment1 environment2}...{@endtarget}`),
* filtering out the blocks that do not match the containing document's * filtering out the blocks that do not match the active `targetEnvironments`.
* `targetEnvironments`.
*/ */
module.exports = function targetInlineTagDef() { module.exports = function targetInlineTagDef(targetEnvironments, log, createDocMessage) {
return { return {
name: 'target', name: 'target',
end: 'endtarget', end: 'endtarget',
handler: function(doc, tagName, tagDescription) { handler: function(doc, tagName, tagDescription) {
var targets = tagDescription && tagDescription.tag.split(' '); var targets = tagDescription && tagDescription.tag.split(' ');
if (!targets || !doc.targetEnvironments || var hasTargets = targets && targets.length;
_.intersection(targets, doc.targetEnvironments).length) {
try {
// Return the contents of this block if any of the following is true:
// * it has no targets
// * there are no targets stored in the targetEnvironments service
// * the block's targets overlap with the active targets in the targetEnvironments service
if (!hasTargets || !targetEnvironments.hasActive() || targetEnvironments.someActive(targets))
{
return tagDescription.content; return tagDescription.content;
} }
} catch(x) {
log.error(createDocMessage('Error processing target inline tag def - ' + x.message, doc));
}
// Otherwise return an empty string
return ''; return '';
} }
}; };

View File

@ -1,24 +1,52 @@
var targetFactory = require('./target'); var mockPackage = require('../mocks/mockPackage');
var Dgeni = require('dgeni');
describe('target inline-tag-def', function() { describe('target inline-tag-def', function() {
it('should filter out content that does not match the doc.targetEnvironments', function() { var dgeni, injector, targetInlineTagDef;
var doc, target, result;
doc = { beforeEach(function() {
targetEnvironments: ['js', 'es6'] dgeni = new Dgeni([mockPackage()]);
}; injector = dgeni.configureInjector();
targetInlineTagDef = injector.get('targetInlineTagDef');
});
target = targetFactory();
result = target.handler(doc, 'target', { it('should filter out content that does not match the targetEnvironments', function() {
var doc = {};
var targetEnvironments = injector.get('targetEnvironments');
targetEnvironments.addAllowed('js', true);
targetEnvironments.addAllowed('es6', true);
targetEnvironments.addAllowed('ts', false);
var result = targetInlineTagDef.handler(doc, 'target', {
tag: 'es6 ts', tag: 'es6 ts',
content: 'abc' content: 'abc'
}); });
expect(result).toEqual('abc'); expect(result).toEqual('abc');
result = target.handler(doc, 'target', { result = targetInlineTagDef.handler(doc, 'target', {
tag: 'ts', tag: 'ts',
content: 'xyz' content: 'xyz'
}); });
expect(result).toEqual(''); expect(result).toEqual('');
}); });
it('should not filter anything if there are no doc nor global target environments', function() {
var doc = {};
var result = targetInlineTagDef.handler(doc, 'target', {
tag: 'es6 ts',
content: 'abc'
});
expect(result).toEqual('abc');
result = targetInlineTagDef.handler(doc, 'target', {
tag: 'ts',
content: 'xyz'
});
expect(result).toEqual('xyz');
});
}); });

View File

@ -0,0 +1,11 @@
var Package = require('dgeni').Package;
module.exports = function mockPackage() {
return new Package('mockPackage', [require('../')])
// provide a mock log service
.factory('log', function() { return require('dgeni/lib/mocks/log')(false); })
.factory('templateEngine', function() { return {}; });
};

View File

@ -0,0 +1,51 @@
module.exports = function targetEnvironments() {
var _targets = Object.create(null);
var _activeCount = 0;
var checkAllowed = function(target) {
if (!(target in _targets)) {
throw new Error('Error accessing target "' + target + '". It is not in the list of allowed targets: ' + Object.keys(_targets));
}
};
var updateActiveCount = function() {
_activeCount = 0;
for(target in _targets) {
if (_targets[target]) _activeCount++;
}
};
return {
addAllowed: function(target, isActive) {
_targets[target] = !!isActive;
updateActiveCount();
},
removeAllowed: function(target) {
delete _targets[target];
updateActiveCount();
},
activate: function(target) {
checkAllowed(target);
_targets[target] = true;
updateActiveCount();
},
deactivate: function(target) {
checkAllowed(target);
_targets[target] = false;
updateActiveCount();
},
isActive: function(target) {
checkAllowed(target);
return _targets[target];
},
hasActive: function() {
return _activeCount > 0;
},
someActive: function(targets) {
for(var i=0,ii=targets.length; i<ii; i++) {
if (this.isActive(targets[i])) return true;
}
return false;
}
};
};

View File

@ -0,0 +1,105 @@
var mockPackage = require('../mocks/mockPackage');
var Dgeni = require('dgeni');
describe('target inline-tag-def', function() {
var dgeni, injector, te;
beforeEach(function() {
dgeni = new Dgeni([mockPackage()]);
injector = dgeni.configureInjector();
te = injector.get('targetEnvironments');
});
describe('addAllowed', function() {
it('should store the target and whether it is active', function() {
te.addAllowed('a', true);
te.addAllowed('b', false);
te.addAllowed('c');
expect(te.isActive('a')).toBe(true);
expect(te.isActive('b')).toBe(false);
expect(te.isActive('c')).toBe(false);
});
});
describe('removeAllowed', function() {
it('should disallow the target', function() {
te.addAllowed('a');
te.addAllowed('b');
te.removeAllowed('b');
expect(te.isActive('a')).toBe(false);
expect(function() { te.isActive('b'); })
.toThrowError('Error accessing target "b". It is not in the list of allowed targets: a');
});
});
describe('activate', function() {
it('should active an already allowed target', function() {
te.addAllowed('a', true);
te.addAllowed('b', false);
te.addAllowed('c');
te.activate('a');
te.activate('b');
te.activate('c');
expect(te.isActive('a')).toBe(true);
expect(te.isActive('b')).toBe(true);
expect(te.isActive('c')).toBe(true);
});
});
describe('deactivate', function() {
it('should deactive an already allowed target', function() {
te.addAllowed('a', true);
te.addAllowed('b', false);
te.addAllowed('c');
te.deactivate('a');
te.deactivate('b');
te.deactivate('c');
expect(te.isActive('a')).toBe(false);
expect(te.isActive('b')).toBe(false);
expect(te.isActive('c')).toBe(false);
});
});
describe('isActive', function() {
it('should return true if the item is allowed and active', function() {
te.addAllowed('a', true);
te.addAllowed('b', false);
expect(te.isActive('a')).toBe(true);
expect(te.isActive('b')).toBe(false);
});
});
describe('hasActive', function() {
it('should return true if there are any active targets', function() {
te.addAllowed('a', true);
te.addAllowed('b', false);
expect(te.hasActive()).toBe(true);
te.deactivate('a');
expect(te.hasActive()).toBe(false);
te.activate('b');
expect(te.hasActive()).toBe(true);
});
});
describe('someActive', function() {
it('should return true if the array of targets passed are all allowed and at least on is active', function() {
te.addAllowed('a', true);
te.addAllowed('b', false);
te.addAllowed('c');
expect(te.someActive(['a'])).toBe(true);
expect(te.someActive(['b'])).toBe(false);
expect(te.someActive(['a', 'b'])).toBe(true);
expect(te.someActive(['b', 'c'])).toBe(false);
expect(te.someActive([])).toBe(false);
expect(function() { te.someActive('d'); })
.toThrowError('Error accessing target "d". It is not in the list of allowed targets: a,b,c');
});
});
});