Files
openlayers/tasks/generate-externs.js
2014-08-09 08:53:24 +02:00

189 lines
4.7 KiB
JavaScript

var fs = require('fs');
var path = require('path');
var async = require('async');
var fse = require('fs-extra');
var nomnom = require('nomnom');
var generateInfo = require('./generate-info');
/**
* Read the symbols from info file.
* @param {funciton(Error, Array.<string>, Array.<Object>)} callback Called
* with the patterns and symbols (or any error).
*/
function getInfo(callback) {
generateInfo(function(err) {
if (err) {
callback(new Error('Trouble generating info: ' + err.message));
return;
}
var typedefs = require('../build/info.json').typedefs;
var symbols = require('../build/info.json').symbols;
var externs = require('../build/info.json').externs;
callback(null, typedefs, symbols, externs);
});
}
/**
* Generate externs code given a list symbols.
* @param {Array.<Object>} typedefs List of typedefs.
* @param {Array.<Object>} symbols List of symbols.
* @param {Array.<Object>} externs List of externs.
* @param {string|undefined} namespace Target object for exported symbols.
* @return {string} Export code.
*/
function generateExterns(typedefs, symbols, externs) {
var lines = [];
var namespaces = {};
var constructors = {};
function addNamespaces(name) {
var parts = name.split('.');
parts.pop();
var namespace = [];
parts.forEach(function(part) {
namespace.push(part);
var partialNamespace = namespace.join('.');
if (!(partialNamespace in namespaces ||
partialNamespace in constructors)) {
lines.push('/**');
lines.push(' * @type {Object}');
lines.push(' */');
lines.push(nameToJS(partialNamespace) + ';');
lines.push('\n');
}
});
}
function nameToJS(name) {
if (name.indexOf('.') == -1) {
namespaces[name] = true;
name = 'var ' + name;
}
return name;
}
function processSymbol(symbol) {
addNamespaces(symbol.name.split('#')[0]);
var name = symbol.name;
if (name.indexOf('#') > 0) {
name = symbol.name.replace('#', '.prototype.');
var constructor = symbol.name.split('#')[0];
if (!(constructor in constructors)) {
constructors[constructor] = true;
lines.push('/**');
lines.push(' * @constructor');
lines.push(' */');
lines.push(nameToJS(constructor) + ' = function() {};');
lines.push('\n');
}
}
lines.push('/**');
if (symbol.kind == 'class') {
constructors[name] = true;
lines.push(' * @constructor');
}
if (symbol.types) {
lines.push(' * @type {' + symbol.types.join('|') + '}');
}
var args = [];
if (symbol.params) {
symbol.params.forEach(function(param) {
args.push(param.name);
lines.push(' * @param {' +
(param.variable ? '...' : '') +
param.types.join('|') +
(param.optional ? '=' : '') +
'} ' + param.name);
});
}
if (symbol.returns) {
lines.push(' * @return {' + symbol.returns.join('|') + '}');
}
if (symbol.template) {
lines.push(' * @template ' + symbol.template);
}
lines.push(' */');
if (symbol.kind == 'function' || symbol.kind == 'class') {
lines.push(nameToJS(name) + ' = function(' + args.join(', ') + ') {};');
} else {
lines.push(nameToJS(name) + ';');
}
lines.push('\n');
}
externs.forEach(processSymbol);
typedefs.forEach(function(typedef) {
addNamespaces(typedef.name);
lines.push('/**');
lines.push(' * @typedef {' + typedef.types.join('|') + '}');
lines.push(' */');
lines.push(nameToJS(typedef.name) + ';');
lines.push('\n');
});
symbols.forEach(processSymbol);
return lines.join('\n');
}
/**
* Generate the exports code.
*
* @param {function(Error, string)} callback Called with the exports code or any
* error generating it.
*/
function main(callback) {
async.waterfall([
getInfo,
function(typedefs, symbols, externs, done) {
var code, err;
try {
code = generateExterns(typedefs, symbols, externs);
} catch (e) {
err = e;
}
done(err, code);
}
], callback);
}
/**
* If running this module directly, read the config file, call the main
* function, and write the output file.
*/
if (require.main === module) {
var options = nomnom.options({
output: {
position: 0,
required: true,
help: 'Output path for the generated externs file.'
}
}).parse();
async.waterfall([
main,
fse.outputFile.bind(fse, options.output)
], function(err) {
if (err) {
console.error(err.message);
process.exit(1);
} else {
process.exit(0);
}
});
}
/**
* Export main function.
*/
module.exports = main;