Generate metadata for boolean defines in addition to exportable symbols
This commit is contained in:
@@ -10,6 +10,7 @@
|
|||||||
"includePattern": "\\.js$"
|
"includePattern": "\\.js$"
|
||||||
},
|
},
|
||||||
"plugins": [
|
"plugins": [
|
||||||
|
"buildcfg/jsdoc/symbols/define-plugin",
|
||||||
"buildcfg/jsdoc/symbols/todo-plugin"
|
"buildcfg/jsdoc/symbols/todo-plugin"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
34
buildcfg/jsdoc/symbols/define-plugin.js
Normal file
34
buildcfg/jsdoc/symbols/define-plugin.js
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
/**
|
||||||
|
* @fileoverview This plugin extracts info from boolean defines. This only
|
||||||
|
* handles boolean defines with the default value in the description. Default
|
||||||
|
* is assumed to be provided with something like "default is `true`" (case
|
||||||
|
* insensitive, with or without ticks).
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
var DEFAULT_VALUE = /default\s+is\s+`?(true|false)`?/i;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Hook to define new tags.
|
||||||
|
* @param {Object} dictionary The tag dictionary.
|
||||||
|
*/
|
||||||
|
exports.defineTags = function(dictionary) {
|
||||||
|
|
||||||
|
dictionary.defineTag('define', {
|
||||||
|
canHaveType: true,
|
||||||
|
mustHaveValue: true,
|
||||||
|
onTagged: function(doclet, tag) {
|
||||||
|
var types = tag.value.type.names;
|
||||||
|
if (types.length === 1 && types[0] === 'boolean') {
|
||||||
|
var match = tag.value.description.match(DEFAULT_VALUE);
|
||||||
|
if (match) {
|
||||||
|
doclet.define = {
|
||||||
|
default: match[1] === 'true'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
};
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
/**
|
/**
|
||||||
* @fileoverview Generates JSON output based on doclets with the "api" tag.
|
* @fileoverview Generates JSON output based on exportable symbols (those with
|
||||||
|
* an api tag) and boolean defines (with a define tag and a default value).
|
||||||
*/
|
*/
|
||||||
var assert = require('assert');
|
var assert = require('assert');
|
||||||
var fs = require('fs');
|
var fs = require('fs');
|
||||||
@@ -14,16 +15,18 @@ var path = require('path');
|
|||||||
exports.publish = function(data, opts) {
|
exports.publish = function(data, opts) {
|
||||||
var cwd = process.cwd();
|
var cwd = process.cwd();
|
||||||
|
|
||||||
// get all doclets with the "api" property, but no enums, typedefs and events.
|
// get all doclets with the "api" property or define (excluding enums,
|
||||||
|
// typedefs and events)
|
||||||
var docs = data(
|
var docs = data(
|
||||||
{api: {isString: true}},
|
[{define: {isObject: true}}, {api: {isString: true}}],
|
||||||
{isEnum: {'!is': true}},
|
{isEnum: {'!is': true}},
|
||||||
{kind: {'!is': 'typedef'}},
|
{kind: {'!is': 'typedef'}},
|
||||||
{kind: {'!is': 'event'}}
|
{kind: {'!is': 'event'}}).get();
|
||||||
).get();
|
|
||||||
|
|
||||||
// get symbols data, filter out those that are members of private classes
|
// get symbols data, filter out those that are members of private classes
|
||||||
var symbols = docs.filter(function(doc) {
|
var symbols = [];
|
||||||
|
var defines = [];
|
||||||
|
docs.filter(function(doc) {
|
||||||
var include = true;
|
var include = true;
|
||||||
var constructor = doc.memberof;
|
var constructor = doc.memberof;
|
||||||
if (constructor && constructor.substr(-1) === '_') {
|
if (constructor && constructor.substr(-1) === '_') {
|
||||||
@@ -32,14 +35,23 @@ exports.publish = function(data, opts) {
|
|||||||
include = false;
|
include = false;
|
||||||
}
|
}
|
||||||
return include;
|
return include;
|
||||||
}).map(function(doc) {
|
}).forEach(function(doc) {
|
||||||
return {
|
if (doc.define) {
|
||||||
name: doc.longname,
|
defines.push({
|
||||||
extends: doc.augments,
|
name: doc.longname,
|
||||||
path: path.join(doc.meta.path, doc.meta.filename)
|
path: path.join(doc.meta.path, doc.meta.filename),
|
||||||
};
|
default: doc.define.default
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
symbols.push({
|
||||||
|
name: doc.longname,
|
||||||
|
extends: doc.augments,
|
||||||
|
path: path.join(doc.meta.path, doc.meta.filename)
|
||||||
|
});
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
process.stdout.write(JSON.stringify({symbols: symbols}, null, 2));
|
process.stdout.write(
|
||||||
|
JSON.stringify({symbols: symbols, defines: defines}, null, 2));
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|||||||
23
src/ol/ol.js
23
src/ol/ol.js
@@ -4,7 +4,7 @@ goog.provide('ol');
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @define {boolean} Assume touch.
|
* @define {boolean} Assume touch. Default is `false`.
|
||||||
*/
|
*/
|
||||||
ol.ASSUME_TOUCH = false;
|
ol.ASSUME_TOUCH = false;
|
||||||
|
|
||||||
@@ -64,19 +64,20 @@ ol.DRAG_BOX_HYSTERESIS_PIXELS = 8;
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @define {boolean} Whether to enable canvas.
|
* @define {boolean} Whether to enable canvas. Default is `true`.
|
||||||
*/
|
*/
|
||||||
ol.ENABLE_CANVAS = true;
|
ol.ENABLE_CANVAS = true;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @define {boolean} Whether to enable DOM.
|
* @define {boolean} Whether to enable DOM. Default is `true`.
|
||||||
*/
|
*/
|
||||||
ol.ENABLE_DOM = true;
|
ol.ENABLE_DOM = true;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @define {boolean} Whether to enable rendering of image layers.
|
* @define {boolean} Whether to enable rendering of image layers. Default is
|
||||||
|
* `true`.
|
||||||
*/
|
*/
|
||||||
ol.ENABLE_IMAGE = true;
|
ol.ENABLE_IMAGE = true;
|
||||||
|
|
||||||
@@ -84,37 +85,39 @@ ol.ENABLE_IMAGE = true;
|
|||||||
/**
|
/**
|
||||||
* @define {boolean} Enable named colors.
|
* @define {boolean} Enable named colors.
|
||||||
* Enabling named colors adds about 3KB uncompressed / 1.5KB compressed to the
|
* Enabling named colors adds about 3KB uncompressed / 1.5KB compressed to the
|
||||||
* final build size.
|
* final build size. Default is `true`.
|
||||||
*/
|
*/
|
||||||
ol.ENABLE_NAMED_COLORS = true;
|
ol.ENABLE_NAMED_COLORS = true;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @define {boolean} Enable Proj4js.
|
* @define {boolean} Enable Proj4js. Default is `true`.
|
||||||
*/
|
*/
|
||||||
ol.ENABLE_PROJ4JS = true;
|
ol.ENABLE_PROJ4JS = true;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @define {boolean} Whether to enable rendering of tile layers.
|
* @define {boolean} Whether to enable rendering of tile layers. Default is
|
||||||
|
* `true`.
|
||||||
*/
|
*/
|
||||||
ol.ENABLE_TILE = true;
|
ol.ENABLE_TILE = true;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @define {boolean} Whether to enable rendering of vector layers.
|
* @define {boolean} Whether to enable rendering of vector layers. Default is
|
||||||
|
* `true`.
|
||||||
*/
|
*/
|
||||||
ol.ENABLE_VECTOR = true;
|
ol.ENABLE_VECTOR = true;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @define {boolean} Whether to enable WebGL.
|
* @define {boolean} Whether to enable WebGL. Default is `true`.
|
||||||
*/
|
*/
|
||||||
ol.ENABLE_WEBGL = true;
|
ol.ENABLE_WEBGL = true;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @define {boolean} Whether to support legacy IE (7-8).
|
* @define {boolean} Whether to support legacy IE (7-8). Default is `false`.
|
||||||
*/
|
*/
|
||||||
ol.LEGACY_IE_SUPPORT = false;
|
ol.LEGACY_IE_SUPPORT = false;
|
||||||
|
|
||||||
|
|||||||
@@ -7,31 +7,40 @@ var fse = require('fs-extra');
|
|||||||
var walk = require('walk').walk;
|
var walk = require('walk').walk;
|
||||||
|
|
||||||
var sourceDir = path.join(__dirname, '..', 'src', 'ol');
|
var sourceDir = path.join(__dirname, '..', 'src', 'ol');
|
||||||
var destPath = path.join(__dirname, '..', 'build', 'symbols.json');
|
var infoPath = path.join(__dirname, '..', 'build', 'symbols.json');
|
||||||
var jsdoc = path.join(__dirname, '..', 'node_modules', '.bin', 'jsdoc');
|
var jsdoc = path.join(__dirname, '..', 'node_modules', '.bin', 'jsdoc');
|
||||||
var jsdocConfig = path.join(
|
var jsdocConfig = path.join(
|
||||||
__dirname, '..', 'buildcfg', 'jsdoc', 'symbols', 'conf.json');
|
__dirname, '..', 'buildcfg', 'jsdoc', 'symbols', 'conf.json');
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Read symbols from dest file.
|
* Create a new metadata object.
|
||||||
* @param {function(Error, Array, Date)} callback Callback called with any
|
* @return {Object} New metadata.
|
||||||
* error, the symbols array, and the mtime of the symbols file.
|
|
||||||
*/
|
*/
|
||||||
function readSymbols(callback) {
|
function createInfo() {
|
||||||
fs.stat(destPath, function(err, stats) {
|
return {symbols: [], defines: []};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read symbols & defines metadata from info file.
|
||||||
|
* @param {function(Error, Object, Date)} callback Callback called with any
|
||||||
|
* error, the metadata, and the mtime of the info file.
|
||||||
|
*/
|
||||||
|
function readInfo(callback) {
|
||||||
|
fs.stat(infoPath, function(err, stats) {
|
||||||
if (err) {
|
if (err) {
|
||||||
if (err.code === 'ENOENT') {
|
if (err.code === 'ENOENT') {
|
||||||
callback(null, [], new Date(0));
|
callback(null, createInfo(), new Date(0));
|
||||||
} else {
|
} else {
|
||||||
callback(err);
|
callback(err);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
fs.readFile(destPath, function(err, data) {
|
fs.readFile(infoPath, function(err, data) {
|
||||||
if (err) {
|
if (err) {
|
||||||
callback(err);
|
callback(err);
|
||||||
} else {
|
} else {
|
||||||
callback(null, JSON.parse(String(data)).symbols, stats.mtime);
|
callback(null, JSON.parse(String(data)), stats.mtime);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -50,13 +59,13 @@ function makeUnique(array) {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate a list of .js paths in the source directory that are newer than
|
* Generate a list of .js paths in the source directory that are newer than
|
||||||
* the symbols file.
|
* the info file.
|
||||||
* @param {Array} symbols Array of symbol metadata.
|
* @param {Object} info Symbol and defines metadata.
|
||||||
* @param {Date} date Modification time of symbols file.
|
* @param {Date} date Modification time of info file.
|
||||||
* @param {function(Error, Array, Array.<string>)} callback Callback called with
|
* @param {function(Error, Object, Array.<string>)} callback Called with any
|
||||||
* any error, the symbols array, and the array of newer source paths.
|
* error, the info object, and the array of newer source paths.
|
||||||
*/
|
*/
|
||||||
function getNewer(symbols, date, callback) {
|
function getNewer(info, date, callback) {
|
||||||
var allPaths = [];
|
var allPaths = [];
|
||||||
var newerPaths = [];
|
var newerPaths = [];
|
||||||
|
|
||||||
@@ -77,7 +86,7 @@ function getNewer(symbols, date, callback) {
|
|||||||
walker.on('end', function() {
|
walker.on('end', function() {
|
||||||
// prune symbols if file no longer exists or has been modified
|
// prune symbols if file no longer exists or has been modified
|
||||||
var lookup = {};
|
var lookup = {};
|
||||||
symbols.forEach(function(symbol) {
|
info.symbols.forEach(function(symbol) {
|
||||||
lookup[symbol.name] = symbol;
|
lookup[symbol.name] = symbol;
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -100,7 +109,7 @@ function getNewer(symbols, date, callback) {
|
|||||||
|
|
||||||
var dirtyPaths = [];
|
var dirtyPaths = [];
|
||||||
|
|
||||||
symbols = symbols.filter(function(symbol) {
|
info.symbols = info.symbols.filter(function(symbol) {
|
||||||
var dirty = allPaths.indexOf(symbol.path) < 0;
|
var dirty = allPaths.indexOf(symbol.path) < 0;
|
||||||
if (!dirty) {
|
if (!dirty) {
|
||||||
// confirm that symbol and all parent paths are not newer
|
// confirm that symbol and all parent paths are not newer
|
||||||
@@ -116,21 +125,30 @@ function getNewer(symbols, date, callback) {
|
|||||||
return !dirty;
|
return !dirty;
|
||||||
});
|
});
|
||||||
|
|
||||||
callback(null, symbols, makeUnique(newerPaths.concat(dirtyPaths)));
|
info.defines = info.defines.filter(function(define) {
|
||||||
|
var dirty = allPaths.indexOf(define.path) < 0 ||
|
||||||
|
newerPaths.indexOf(define.path) >= 0;
|
||||||
|
if (dirty) {
|
||||||
|
dirtyPaths.push(define.path);
|
||||||
|
}
|
||||||
|
return !dirty;
|
||||||
|
});
|
||||||
|
|
||||||
|
callback(null, info, makeUnique(newerPaths.concat(dirtyPaths)));
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Spawn JSDoc.
|
* Spawn JSDoc.
|
||||||
* @param {Array} symbols Array of symbol metadata.
|
* @param {Object} info Symbol and defines metadata.
|
||||||
* @param {Array.<string>} newerSources Paths to newer source files.
|
* @param {Array.<string>} newerSources Paths to newer source files.
|
||||||
* @param {function(Error, Array, string)} callback Callback called with any
|
* @param {function(Error, Array, string)} callback Callback called with any
|
||||||
* error, existing symbols, and the JSDoc output.
|
* error, existing metadata, and the JSDoc output (new metadata).
|
||||||
*/
|
*/
|
||||||
function spawnJSDoc(symbols, newerSources, callback) {
|
function spawnJSDoc(info, newerSources, callback) {
|
||||||
if (newerSources.length === 0) {
|
if (newerSources.length === 0) {
|
||||||
callback(null, symbols, JSON.stringify({symbols: []}));
|
callback(null, info, JSON.stringify(createInfo()));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -150,12 +168,49 @@ function spawnJSDoc(symbols, newerSources, callback) {
|
|||||||
if (code) {
|
if (code) {
|
||||||
callback(new Error(errors || 'JSDoc failed with no output'));
|
callback(new Error(errors || 'JSDoc failed with no output'));
|
||||||
} else {
|
} else {
|
||||||
callback(null, symbols, output);
|
callback(null, info, output);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse the JSDoc output.
|
||||||
|
* @param {Object} info Existing metadata.
|
||||||
|
* @param {string} output JSDoc output
|
||||||
|
* @param {function(Error, Object, Object)} callback Called with any error,
|
||||||
|
* existing metadata, and new metadata.
|
||||||
|
*/
|
||||||
|
function parseOutput(info, output, callback) {
|
||||||
|
if (!output) {
|
||||||
|
callback(new Error('Expected JSON output'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var newInfo;
|
||||||
|
try {
|
||||||
|
newInfo = JSON.parse(String(output));
|
||||||
|
} catch (err) {
|
||||||
|
callback(new Error('Failed to parse output as JSON: ' + output));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Array.isArray(newInfo.symbols)) {
|
||||||
|
callback(new Error('Expected symbols array: ' + output));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!Array.isArray(newInfo.defines)) {
|
||||||
|
callback(new Error('Expected defines array: ' + output));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
process.nextTick(function() {
|
||||||
|
callback(null, info, newInfo);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Given the path to a source file, get the list of provides.
|
* Given the path to a source file, get the list of provides.
|
||||||
* @param {string} srcPath Path to source file.
|
* @param {string} srcPath Path to source file.
|
||||||
@@ -182,85 +237,67 @@ var getProvides = async.memoize(function(srcPath, callback) {
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Add provides to a symbol.
|
* Add provides data to new symbols.
|
||||||
* @param {Object} symbol Symbol object.
|
* @param {Object} info Existing symbols and defines metadata.
|
||||||
* @param {function(Error, Object)} callback Called with the augmented symbol or
|
* @param {Object} newInfo New metadata.
|
||||||
* any error.
|
* @param {function(Error, Object)} callback Updated metadata.
|
||||||
*/
|
*/
|
||||||
function addProvides(symbol, callback) {
|
function addSymbolProvides(info, newInfo, callback) {
|
||||||
getProvides(symbol.path, function(err, provides) {
|
|
||||||
if (err) {
|
function addProvides(symbol, callback) {
|
||||||
callback(err);
|
getProvides(symbol.path, function(err, provides) {
|
||||||
return;
|
if (err) {
|
||||||
}
|
callback(err);
|
||||||
symbol.provides = provides;
|
return;
|
||||||
callback(null, symbol);
|
}
|
||||||
|
symbol.provides = provides;
|
||||||
|
callback(null, symbol);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async.map(newInfo.symbols, addProvides, function(err, newSymbols) {
|
||||||
|
newInfo.symbols = newSymbols;
|
||||||
|
callback(err, info, newInfo);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Parse JSDoc output and add provides data to each symbol.
|
* Write symbol and define metadata to the info file.
|
||||||
* @param {Array} symbols Existing symbols.
|
* @param {Object} info Existing metadata.
|
||||||
* @param {string} output Output from JSDoc.
|
* @param {Object} newInfo New meatadat.
|
||||||
* @param {function(Error, Array)} callback Concatenated symbols.
|
|
||||||
*/
|
|
||||||
function addAllProvides(symbols, output, callback) {
|
|
||||||
if (!output) {
|
|
||||||
callback(new Error('Expected JSON output'));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var data;
|
|
||||||
try {
|
|
||||||
data = JSON.parse(String(output));
|
|
||||||
} catch (err) {
|
|
||||||
callback(new Error('Failed to parse output as JSON: ' + output));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!data || !Array.isArray(data.symbols)) {
|
|
||||||
callback(new Error('Expected symbols array: ' + output));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
async.map(data.symbols, addProvides, function(err, newSymbols) {
|
|
||||||
callback(err, symbols, newSymbols);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Write symbol metadata to the symbols file.
|
|
||||||
* @param {Array} symbols Existing symbols.
|
|
||||||
* @param {Array} newSymbols New symbols.
|
|
||||||
* @param {function(Error)} callback Callback.
|
* @param {function(Error)} callback Callback.
|
||||||
*/
|
*/
|
||||||
function writeSymbols(symbols, newSymbols, callback) {
|
function writeInfo(info, newInfo, callback) {
|
||||||
|
|
||||||
symbols = symbols.concat(newSymbols).sort(function(a, b) {
|
info.symbols = info.symbols.concat(newInfo.symbols).sort(function(a, b) {
|
||||||
return a.name < b.name ? -1 : 1;
|
return a.name < b.name ? -1 : 1;
|
||||||
});
|
});
|
||||||
|
|
||||||
var str = JSON.stringify({symbols: symbols}, null, ' ');
|
info.defines = info.defines.concat(newInfo.defines).sort(function(a, b) {
|
||||||
fse.outputFile(destPath, str, callback);
|
return a.name < b.name ? -1 : 1;
|
||||||
|
});
|
||||||
|
|
||||||
|
var str = JSON.stringify(info, null, ' ');
|
||||||
|
fse.outputFile(infoPath, str, callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Determine which source files have been changed, run JSDoc against those,
|
* Determine which source files have been changed, run JSDoc against those, and
|
||||||
* write out exported symbols, and clean up the build dir.
|
* write out updated info.
|
||||||
*
|
*
|
||||||
* @param {function(Error)} callback Called when the symbols file has been
|
* @param {function(Error)} callback Called when the info file has been written
|
||||||
* written (or if an error occurs).
|
* (or an error occurs).
|
||||||
*/
|
*/
|
||||||
function main(callback) {
|
function main(callback) {
|
||||||
async.waterfall([
|
async.waterfall([
|
||||||
readSymbols,
|
readInfo,
|
||||||
getNewer,
|
getNewer,
|
||||||
spawnJSDoc,
|
spawnJSDoc,
|
||||||
addAllProvides,
|
parseOutput,
|
||||||
writeSymbols
|
addSymbolProvides,
|
||||||
|
writeInfo
|
||||||
], callback);
|
], callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user