blob: ff3660b3fb190ab67b3022c80890a33f88b2d0d4 [file] [log] [blame] [raw]
// Copyright (c) 2018, Compiler Explorer Authors
//
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// * Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.
const exec = require('../exec'),
_ = require('underscore'),
logger = require('../logger').logger;
class Formatter {
constructor(ceProps) {
this.tools = {};
this.ceProps = ceProps;
const formatters = _.compact(ceProps('formatters').split(':'));
_.each(formatters, this.fetchToolInfo.bind(this));
}
fetchToolInfo(formatter) {
const exe = this.ceProps(`formatter.${formatter}.exe`);
if (!exe) {
logger.warn(`Formatter ${formatter} does not have a valid exe. Skipping...`);
return;
}
const versionArg = this.ceProps(`formatter.${formatter}.version`, '--version');
const versionRe = this.ceProps(`formatter.${formatter}.versionRe`, '.*');
this.run(exe, [versionArg], {})
.then(result => {
const match = result.stdout.match(versionRe);
this.tools[formatter] = {
exe: exe,
version: match ? match[0] : result.stdout,
name: this.ceProps(`formatter.${formatter}.name`, exe),
styles: this.ceProps(`formatter.${formatter}.styles`, "").split(':')
};
});
}
supportsStyle(tool, style) {
return tool.styles.includes(style);
}
formatHandler(req, res) {
res.set('Content-Type', 'application/json');
let requestedTool = this.tools[req.params.tool];
if (!requestedTool) {
res.status(422); // Unprocessable Entity
return res.end(JSON.stringify({
exit: 2,
answer: "Tool not supported"
}));
}
// Only clang supported for now
const options = req.body || {};
if (!options.source) {
return res.end(JSON.stringify({
exit: 0,
answer: "",
empty: true
}));
}
let style = options.base || "Google";
// Hardcoded supported clang-format base styles.
// Will need a bit of work if we want to support other tools!
if (this.supportsStyle(requestedTool, style)) {
res.status(422); // Unprocessable Entity
return res.end(JSON.stringify({
exit: 3,
answer: "Base style not supported"
}));
}
this.formatCode(requestedTool, [`-style=${style}`], {input: options.source})
.then(result => {
return res.end(JSON.stringify({
exit: result.code,
answer: result.stdout || "",
}));
}).catch(ex => {
res.status(500);
return res.end(JSON.stringify({
exit: 1,
thrown: true,
answer: ex.message || "Internal server error"
}));
});
}
run(exe, args, options) {
return exec.execute(exe, args, options);
}
formatCode(tool, args, options) {
return this.run(tool.exe, args, options);
}
listHandler(req, res) {
res.setHeader('Content-Type', 'application/json');
const response = [];
_.each(this.tools, tool => {
response.push({name: tool.name, version: tool.version});
});
return res.end(JSON.stringify(response));
}
}
module.exports = Formatter;