blob: d06e0c444804458a4fa2e006864b8f39d38b67a4 [file] [log] [blame] [raw]
// Copyright (c) 2017, Matt Godbolt
// 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 express = require('express'),
_ = require('underscore'),
AsmDocsApi = require('./asm-docs-api'),
FormatterHandler = require('./formatting'),
utils = require('../utils'),
logger = require('../logger').logger,
clientStateNormalizer = require('../clientstate-normalizer').ClientStateNormalizer;
class ApiHandler {
constructor(compileHandler, ceProps, storageHandler) {
this.compilers = [];
this.languages = [];
this.usedLangIds = [];
this.options = null;
this.storageHandler = storageHandler;
this.handle = express.Router();
this.handle.use((req, res, next) => {
res.header("Access-Control-Allow-Origin", "*");
res.header("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept");
next();
});
this.handle.get('/compilers', this.handleCompilers.bind(this));
this.handle.get('/compilers/:language', this.handleCompilers.bind(this));
this.handle.get('/languages', this.handleLanguages.bind(this));
this.handle.get('/libraries/:language', this.handleLibraries.bind(this));
const asmDocsHandler = new AsmDocsApi.Handler();
this.handle.get('/asm/:opcode', asmDocsHandler.handle.bind(asmDocsHandler));
this.handle.post('/compiler/:compiler/compile', compileHandler.handle.bind(compileHandler));
const formatter = new FormatterHandler(ceProps);
this.handle.post('/format/:tool', formatter.formatHandler.bind(formatter));
this.handle.get('/formats', formatter.listHandler.bind(formatter));
this.handle.get('/shortlinkinfo/:id', this.shortlinkInfoHandler.bind(this));
}
shortlinkInfoHandler(req, res, next) {
const id = req.params.id;
this.storageHandler.expandId(id)
.then(result => {
const config = JSON.parse(result.config);
if (config.content) {
const normalizer = new clientStateNormalizer();
normalizer.fromGoldenLayout(config);
res.set('Content-Type', 'application/json');
res.end(JSON.stringify(normalizer.normalized));
} else {
res.set('Content-Type', 'application/json');
res.end(JSON.stringify(config));
}
})
.catch(err => {
logger.warn(`Exception thrown when expanding ${id}: `, err);
next({
statusCode: 404,
message: `ID "${id}" could not be found`
});
});
}
handleLanguages(req, res) {
const availableLanguages = this.usedLangIds.map(val => {
let lang = this.languages[val];
let newLangObj = Object.assign({}, lang);
if (this.options) {
newLangObj.defaultCompiler = this.options.options.defaultCompiler[lang.id];
}
return newLangObj;
});
if (req.accepts(['text', 'json']) === 'json') {
res.set('Content-Type', 'application/json');
res.end(JSON.stringify(availableLanguages));
} else {
res.set('Content-Type', 'text/plain');
const title = 'Id';
const maxLength = _.max(_.pluck(_.pluck(availableLanguages, 'id').concat([title]), 'length'));
res.write(utils.padRight(title, maxLength) + ' | Name\n');
res.end(availableLanguages.map(lang => utils.padRight(lang.id, maxLength) + ' | ' + lang.name)
.join("\n"));
}
}
getLibrariesAsArray(languageId) {
const libsForLanguageObj = this.options.options.libs[languageId];
if (!libsForLanguageObj) return [];
return Object.keys(libsForLanguageObj).map((key) => {
const language = libsForLanguageObj[key];
const versionArr = Object.keys(language.versions).map((key) => {
let versionObj = Object.assign({}, language.versions[key]);
versionObj.id = key;
return versionObj;
});
return {
id: key,
name: language.name,
description: language.description,
url: language.url,
versions: versionArr
};
});
}
handleLibraries(req, res, next) {
if (this.options) {
if (req.params.language) {
res.set('Content-Type', 'application/json');
const libsForLanguageArr = this.getLibrariesAsArray(req.params.language);
res.end(JSON.stringify(libsForLanguageArr));
} else {
next({
statusCode: 404,
message: "Language is required"
});
}
} else {
next({
statusCode: 500,
message: "Internal error"
});
}
}
handleCompilers(req, res) {
let filteredCompilers = this.compilers;
if (req.params.language) {
filteredCompilers = this.compilers.filter((val) => val.lang === req.params.language);
}
if (req.accepts(['text', 'json']) === 'json') {
res.set('Content-Type', 'application/json');
res.end(JSON.stringify(filteredCompilers));
} else {
res.set('Content-Type', 'text/plain');
const title = 'Compiler Name';
const maxLength = _.max(_.pluck(_.pluck(filteredCompilers, 'id').concat([title]), 'length'));
res.write(utils.padRight(title, maxLength) + ' | Description\n');
res.end(
filteredCompilers.map(compiler => utils.padRight(compiler.id, maxLength) + ' | ' + compiler.name)
.join("\n"));
}
}
setCompilers(compilers) {
this.compilers = compilers;
this.usedLangIds = _.uniq(_.pluck(this.compilers, 'lang'));
}
setLanguages(languages) {
this.languages = languages;
}
setOptions(options) {
this.options = options;
}
}
module.exports.Handler = ApiHandler;