295 lines
12 KiB
JavaScript
Executable file
295 lines
12 KiB
JavaScript
Executable file
#!/usr/bin/env node
|
|
/** section: github, internal
|
|
* class ApiGenerator
|
|
*
|
|
* Copyright 2012 Cloud9 IDE, Inc.
|
|
*
|
|
* This product includes software developed by
|
|
* Cloud9 IDE, Inc (http://c9.io).
|
|
*
|
|
* Author: Mike de Boer <mike@c9.io>
|
|
**/
|
|
|
|
"use strict";
|
|
|
|
var Fs = require("fs");
|
|
var Path = require("path");
|
|
|
|
var Optimist = require("optimist");
|
|
var Util = require("./util");
|
|
|
|
var IndexTpl = Fs.readFileSync(__dirname + "/templates/index.js.tpl", "utf8");
|
|
var SectionTpl = Fs.readFileSync(__dirname + "/templates/section.js.tpl", "utf8");
|
|
var HandlerTpl = Fs.readFileSync(__dirname + "/templates/handler.js.tpl", "utf8");
|
|
var AfterRequestTpl = Fs.readFileSync(__dirname + "/templates/after_request.js.tpl", "utf8");
|
|
var TestSectionTpl = Fs.readFileSync(__dirname + "/templates/test_section.js.tpl", "utf8");
|
|
var TestHandlerTpl = Fs.readFileSync(__dirname + "/templates/test_handler.js.tpl", "utf8");
|
|
|
|
var main = module.exports = function(versions, tests, restore) {
|
|
Util.log("Generating for versions", Object.keys(versions));
|
|
|
|
Object.keys(versions).forEach(function(version) {
|
|
var dir = Path.join(__dirname, "api", version);
|
|
|
|
// If we're in restore mode, move .bak file back to their original position
|
|
// and short-circuit.
|
|
if (restore) {
|
|
var bakRE = /\.bak$/;
|
|
var files = Fs.readdirSync(dir).filter(function(file) {
|
|
return bakRE.test(file);
|
|
}).forEach(function(file) {
|
|
var from = Path.join(dir, file);
|
|
var to = Path.join(dir, file.replace(/\.bak$/, ""));
|
|
Fs.renameSync(from, to);
|
|
Util.log("Restored '" + file + "' (" + version + ")");
|
|
});
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
var routes = versions[version];
|
|
var defines = routes.defines;
|
|
delete routes.defines;
|
|
var headers = defines["response-headers"];
|
|
// cast header names to lowercase.
|
|
if (headers && headers.length)
|
|
headers = headers.map(function(header) { return header.toLowerCase(); });
|
|
var sections = {};
|
|
var testSections = {};
|
|
|
|
function createComment(paramsStruct, section, funcName, indent) {
|
|
var params = Object.keys(paramsStruct);
|
|
var comment = [
|
|
indent + "/** section: github",
|
|
indent + " * " + section + "#" + funcName + "(msg, callback) -> null",
|
|
indent + " * - msg (Object): Object that contains the parameters and their values to be sent to the server.",
|
|
indent + " * - callback (Function): function to call when the request is finished " +
|
|
"with an error as first argument and result data as second argument.",
|
|
indent + " *",
|
|
indent + " * ##### Params on the `msg` object:",
|
|
indent + " *"
|
|
];
|
|
comment.push(indent + " * - headers (Object): Optional. Key/ value pair "
|
|
+ "of request headers to pass along with the HTTP request. Valid headers are: "
|
|
+ "'" + defines["request-headers"].join("', '") + "'.");
|
|
if (!params.length)
|
|
comment.push(indent + " * No other params, simply pass an empty Object literal `{}`");
|
|
var paramName, def, line;
|
|
for (var i = 0, l = params.length; i < l; ++i) {
|
|
paramName = params[i];
|
|
if (paramName.charAt(0) == "$") {
|
|
paramName = paramName.substr(1);
|
|
if (!defines.params[paramName]) {
|
|
Util.log("Invalid variable parameter name substitution; param '" +
|
|
paramName + "' not found in defines block", "fatal");
|
|
process.exit(1);
|
|
}
|
|
else
|
|
def = defines.params[paramName];
|
|
}
|
|
else
|
|
def = paramsStruct[paramName];
|
|
|
|
line = indent + " * - " + paramName + " (" + (def.type || "mixed") + "): " +
|
|
(def.required ? "Required. " : "Optional. ");
|
|
if (def.description)
|
|
line += def.description;
|
|
if (def.validation)
|
|
line += " Validation rule: ` " + def.validation + " `.";
|
|
|
|
comment.push(line);
|
|
}
|
|
|
|
return comment.join("\n") + "\n" + indent + " **/";
|
|
}
|
|
|
|
function getParams(paramsStruct, indent) {
|
|
var params = Object.keys(paramsStruct);
|
|
if (!params.length)
|
|
return "{}";
|
|
var values = [];
|
|
var paramName, def;
|
|
for (var i = 0, l = params.length; i < l; ++i) {
|
|
paramName = params[i];
|
|
if (paramName.charAt(0) == "$") {
|
|
paramName = paramName.substr(1);
|
|
if (!defines.params[paramName]) {
|
|
Util.log("Invalid variable parameter name substitution; param '" +
|
|
paramName + "' not found in defines block", "fatal");
|
|
process.exit(1);
|
|
}
|
|
else
|
|
def = defines.params[paramName];
|
|
}
|
|
else
|
|
def = paramsStruct[paramName];
|
|
|
|
values.push(indent + " " + paramName + ": \"" + def.type + "\"");
|
|
}
|
|
return "{\n" + values.join(",\n") + "\n" + indent + "}";
|
|
}
|
|
|
|
function prepareApi(struct, baseType) {
|
|
if (!baseType)
|
|
baseType = "";
|
|
|
|
Object.keys(struct).forEach(function(routePart) {
|
|
var block = struct[routePart];
|
|
if (!block)
|
|
return;
|
|
var messageType = baseType + "/" + routePart;
|
|
if (block.url && block.params) {
|
|
// we ended up at an API definition part!
|
|
var parts = messageType.split("/");
|
|
var section = Util.toCamelCase(parts[1].toLowerCase());
|
|
if (!block.method) {
|
|
throw new Error("No HTTP method specified for " + messageType +
|
|
"in section " + section);
|
|
}
|
|
|
|
parts.splice(0, 2);
|
|
var funcName = Util.toCamelCase(parts.join("-"));
|
|
var comment = createComment(block.params, section, funcName, " ");
|
|
|
|
// add the handler to the sections
|
|
if (!sections[section])
|
|
sections[section] = [];
|
|
|
|
var afterRequest = "";
|
|
if (headers && headers.length) {
|
|
afterRequest = AfterRequestTpl.replace("<%headers%>", "\"" +
|
|
headers.join("\", \"") + "\"");
|
|
}
|
|
sections[section].push(HandlerTpl
|
|
.replace("<%funcName%>", funcName)
|
|
.replace("<%comment%>", comment)
|
|
.replace("<%afterRequest%>", afterRequest)
|
|
);
|
|
|
|
// add test to the testSections
|
|
if (!testSections[section])
|
|
testSections[section] = [];
|
|
testSections[section].push(TestHandlerTpl
|
|
.replace("<%name%>", block.method + " " + block.url + " (" + funcName + ")")
|
|
.replace("<%funcName%>", section + "." + funcName)
|
|
.replace("<%params%>", getParams(block.params, " "))
|
|
);
|
|
}
|
|
else {
|
|
// recurse into this block next:
|
|
prepareApi(block, messageType);
|
|
}
|
|
});
|
|
}
|
|
|
|
Util.log("Converting routes to functions");
|
|
prepareApi(routes);
|
|
|
|
Util.log("Writing files to version dir");
|
|
var sectionNames = Object.keys(sections);
|
|
|
|
Util.log("Writing index.js file for version " + version);
|
|
Fs.writeFileSync(Path.join(dir, "index.js"),
|
|
IndexTpl
|
|
.replace("<%name%>", defines.constants.name)
|
|
.replace("<%description%>", defines.constants.description)
|
|
.replace("<%scripts%>", "\"" + sectionNames.join("\", \"") + "\""),
|
|
"utf8");
|
|
|
|
Object.keys(sections).forEach(function(section) {
|
|
var def = sections[section];
|
|
Util.log("Writing '" + section + ".js' file for version " + version);
|
|
Fs.writeFileSync(Path.join(dir, section + ".js"), SectionTpl
|
|
.replace(/<%sectionName%>/g, section)
|
|
.replace("<%sectionBody%>", def.join("\n")),
|
|
"utf8"
|
|
);
|
|
|
|
// When we don't need to generate tests, bail out here.
|
|
if (!tests)
|
|
return;
|
|
|
|
def = testSections[section];
|
|
// test if previous tests already contained implementations by checking
|
|
// if the difference in character count between the current test file
|
|
// and the newly generated one is more than twenty characters.
|
|
var body = TestSectionTpl
|
|
.replace("<%version%>", version.replace("v", ""))
|
|
.replace(/<%sectionName%>/g, section)
|
|
.replace("<%testBody%>", def.join("\n\n"));
|
|
var path = Path.join(dir, section + "Test.js");
|
|
if (Fs.existsSync(path) && Math.abs(Fs.readFileSync(path, "utf8").length - body.length) >= 20) {
|
|
Util.log("Moving old test file to '" + path + ".bak' to preserve tests " +
|
|
"that were already implemented. \nPlease be sure te check this file " +
|
|
"and move all implemented tests back into the newly generated test!", "error");
|
|
Fs.renameSync(path, path + ".bak");
|
|
}
|
|
|
|
Util.log("Writing test file for " + section + ", version " + version);
|
|
Fs.writeFileSync(path, body, "utf8");
|
|
});
|
|
});
|
|
};
|
|
|
|
if (!module.parent) {
|
|
var argv = Optimist
|
|
.wrap(80)
|
|
.usage("Generate the implementation of the node-github module, including "
|
|
+ "unit-test scaffolds.\nUsage: $0 [-r] [-v VERSION]")
|
|
.alias("r", "restore")
|
|
.describe("r", "Restore .bak files, generated by a previous run, to the original")
|
|
.alias("v", "version")
|
|
.describe("v", "Semantic version number of the API to generate. Example: '3.0.0'")
|
|
.alias("t", "tests")
|
|
.describe("t", "Also generate unit test scaffolds")
|
|
.alias("h", "help")
|
|
.describe("h", "Display this usage information")
|
|
.boolean(["r", "t", "h"])
|
|
.argv;
|
|
|
|
if (argv.help) {
|
|
Util.log(Optimist.help());
|
|
process.exit();
|
|
}
|
|
|
|
var baseDir = Path.join(__dirname, "api");
|
|
var availVersions = {};
|
|
Fs.readdirSync(baseDir).forEach(function(version) {
|
|
var path = Path.join(baseDir, version, "routes.json");
|
|
if (!Fs.existsSync(path))
|
|
return;
|
|
var routes;
|
|
try {
|
|
routes = JSON.parse(Fs.readFileSync(path, "utf8"));
|
|
}
|
|
catch (ex) {
|
|
return;
|
|
}
|
|
if (!routes.defines)
|
|
return;
|
|
availVersions[version] = routes;
|
|
});
|
|
|
|
if (!Object.keys(availVersions).length) {
|
|
Util.log("No versions available to generate.", "fatal");
|
|
process.exit(1);
|
|
}
|
|
var versions = {};
|
|
if (argv.version) {
|
|
if (argv.version.charAt(0) != "v")
|
|
argv.version = argv.v = "v" + argv.version;
|
|
if (!availVersions[argv.version]) {
|
|
Util.log("Version '" + argv.version + "' is not available", "fatal");
|
|
process.exit(1);
|
|
}
|
|
versions[argv.version] = availVersions[argv.version];
|
|
}
|
|
if (!Object.keys(versions).length) {
|
|
Util.log("No versions specified via the command line, generating for all available versions.");
|
|
versions = availVersions;
|
|
}
|
|
|
|
Util.log("Starting up...");
|
|
main(versions, argv.tests, argv.restore);
|
|
}
|