blob: 0196480c385e05e754b36080f8f7bbd329e83a64 [file] [log] [blame] [raw]
// Copyright (c) 2023, 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.
import * as Sentry from '@sentry/node';
import bodyParser from 'body-parser';
import express from 'express';
import _ from 'underscore';
import {CompilerInfo} from '../../types/compiler.interfaces.js';
import {Language, LanguageKey} from '../../types/languages.interfaces.js';
import {assert, unwrap} from '../assert.js';
import {ClientStateNormalizer} from '../clientstate-normalizer.js';
import {isString, unique} from '../common-utils.js';
import {logger} from '../logger.js';
import {ClientOptionsHandler} from '../options-handler.js';
import {PropertyGetter} from '../properties.interfaces.js';
import {BaseShortener, getShortenerTypeByKey} from '../shortener/index.js';
import {StorageBase} from '../storage/index.js';
import * as utils from '../utils.js';
import {withAssemblyDocumentationProviders} from './assembly-documentation.js';
import {CompileHandler} from './compile.js';
import {FormattingHandler} from './formatting.js';
import {getSiteTemplates} from './site-templates.js';
function methodNotAllowed(req: express.Request, res: express.Response) {
res.send('Method Not Allowed');
return res.status(405).end();
}
export class ApiHandler {
public compilers: CompilerInfo[] = [];
public languages: Partial<Record<LanguageKey, Language>> = {};
private usedLangIds: LanguageKey[] = [];
private options: ClientOptionsHandler | null = null;
public readonly handle: express.Router;
public readonly shortener: BaseShortener;
private release = {
gitReleaseName: '',
releaseBuildNumber: '',
};
constructor(
compileHandler: CompileHandler,
ceProps: PropertyGetter,
private readonly storageHandler: StorageBase,
urlShortenService: string,
) {
this.handle = express.Router();
const cacheHeader = `public, max-age=${ceProps('apiMaxAgeSecs', 24 * 60 * 60)}`;
this.handle.use((req, res, next) => {
res.header({
'Access-Control-Allow-Origin': '*',
'Access-Control-Allow-Headers': 'Origin, X-Requested-With, Content-Type, Accept',
'Cache-Control': cacheHeader,
});
next();
});
this.handle.route('/compilers').get(this.handleCompilers.bind(this)).all(methodNotAllowed);
this.handle.route('/compilers/:language').get(this.handleCompilers.bind(this)).all(methodNotAllowed);
this.handle.route('/languages').get(this.handleLanguages.bind(this)).all(methodNotAllowed);
this.handle.route('/libraries/:language').get(this.handleLangLibraries.bind(this)).all(methodNotAllowed);
this.handle.route('/libraries').get(this.handleAllLibraries.bind(this)).all(methodNotAllowed);
// Binding for assembly documentation
withAssemblyDocumentationProviders(this.handle);
// Legacy binding for old clients.
this.handle
.route('/asm/:opcode')
.get((req, res) => res.redirect(`amd64/${req.params.opcode}`))
.all(methodNotAllowed);
const maxUploadSize = ceProps('maxUploadSize', '1mb');
const textParser = bodyParser.text({limit: ceProps('bodyParserLimit', maxUploadSize), type: () => true});
this.handle
.route('/compiler/:compiler/compile')
.post(textParser, compileHandler.handle.bind(compileHandler))
.all(methodNotAllowed);
this.handle
.route('/compiler/:compiler/cmake')
.post(compileHandler.handleCmake.bind(compileHandler))
.all(methodNotAllowed);
this.handle
.route('/popularArguments/:compiler')
.post(compileHandler.handlePopularArguments.bind(compileHandler))
.get(compileHandler.handlePopularArguments.bind(compileHandler))
.all(methodNotAllowed);
this.handle
.route('/optimizationArguments/:compiler')
.post(compileHandler.handleOptimizationArguments.bind(compileHandler))
.get(compileHandler.handleOptimizationArguments.bind(compileHandler))
.all(methodNotAllowed);
const formatHandler = new FormattingHandler(ceProps);
this.handle
.route('/format/:tool')
.post((req, res) => formatHandler.handle(req, res))
.all(methodNotAllowed);
this.handle
.route('/formats')
.get((req, res) => {
const all = formatHandler.getFormatterInfo();
res.send(all);
})
.all(methodNotAllowed);
this.handle
.route('/siteTemplates')
.get((req, res) => {
res.send(getSiteTemplates());
})
.all(methodNotAllowed);
this.handle.route('/shortlinkinfo/:id').get(this.shortlinkInfoHandler.bind(this)).all(methodNotAllowed);
const shortenerType = getShortenerTypeByKey(urlShortenService);
this.shortener = new shortenerType(storageHandler);
this.handle.route('/shortener').post(this.shortener.handle.bind(this.shortener)).all(methodNotAllowed);
this.handle.route('/version').get(this.handleReleaseName.bind(this)).all(methodNotAllowed);
this.handle.route('/releaseBuild').get(this.handleReleaseBuild.bind(this)).all(methodNotAllowed);
}
shortlinkInfoHandler(req: express.Request, res: express.Response, next: express.NextFunction) {
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.send(normalizer.normalized);
} else {
res.send(config);
}
})
.catch(err => {
logger.warn(`Exception thrown when expanding ${id}: `, err);
logger.warn('Exception value:', err);
Sentry.captureException(err);
next({
statusCode: 404,
message: `ID "${id}" could not be found`,
});
});
}
handleLanguages(req: express.Request, res: express.Response) {
const availableLanguages = this.usedLangIds.map(val => {
const lang = this.languages[val];
const newLangObj: Language = Object.assign({}, lang);
if (this.options) {
newLangObj.defaultCompiler = this.options.options.defaultCompiler[unwrap(lang).id];
}
return newLangObj;
});
this.outputList(availableLanguages, 'Id', req, res);
}
filterCompilerProperties(list: CompilerInfo[] | Language[], selectedFields: string[]) {
return list.map(compiler => {
return _.pick(compiler, selectedFields);
});
}
outputList(list: CompilerInfo[] | Language[], title: string, req: express.Request, res: express.Response) {
if (req.accepts(['text', 'json']) === 'json') {
if (req.query.fields === 'all') {
res.send(list);
} else {
const defaultfields = [
'id',
'name',
'lang',
'compilerType',
'semver',
'extensions',
'monaco',
'instructionSet',
];
if (req.query.fields) {
assert(isString(req.query.fields));
const filteredList = this.filterCompilerProperties(list, req.query.fields.split(','));
res.send(filteredList);
} else {
const filteredList = this.filterCompilerProperties(list, defaultfields);
res.send(filteredList);
}
}
return;
}
const maxLength = Math.max(
...list
.map(item => item.id)
.concat([title])
.map(item => item.length),
);
res.set('Content-Type', 'text/plain');
res.send(
utils.padRight(title, maxLength) +
' | Name\n' +
list.map(lang => utils.padRight(lang.id, maxLength) + ' | ' + lang.name).join('\n'),
);
}
getLibrariesAsArray(languageId: LanguageKey) {
const libsForLanguageObj = unwrap(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 => {
const 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,
};
});
}
handleLangLibraries(req: express.Request, res: express.Response, next: express.NextFunction) {
if (this.options) {
if (req.params.language) {
res.send(this.getLibrariesAsArray(req.params.language as LanguageKey));
} else {
next({
statusCode: 404,
message: 'Language is required',
});
}
} else {
next({
statusCode: 500,
message: 'Internal error',
});
}
}
handleAllLibraries(req: express.Request, res: express.Response, next: express.NextFunction) {
if (this.options) {
res.send(this.options.options.libs);
} else {
next({
statusCode: 500,
message: 'Internal error',
});
}
}
handleCompilers(req: express.Request, res: express.Response) {
let filteredCompilers = this.compilers;
if (req.params.language) {
filteredCompilers = this.compilers.filter(compiler => compiler.lang === req.params.language);
}
this.outputList(filteredCompilers, 'Compiler Name', req, res);
}
handleReleaseName(req: express.Request, res: express.Response) {
res.send(this.release.gitReleaseName);
}
handleReleaseBuild(req: express.Request, res: express.Response) {
res.send(this.release.releaseBuildNumber);
}
setCompilers(compilers: CompilerInfo[]) {
this.compilers = compilers;
this.usedLangIds = unique(this.compilers.map(compiler => compiler.lang));
}
setLanguages(languages: Partial<Record<LanguageKey, Language>>) {
this.languages = languages;
}
setOptions(options: ClientOptionsHandler) {
this.options = options;
}
setReleaseInfo(gitReleaseName: string | undefined, releaseBuildNumber: string | undefined) {
this.release = {
gitReleaseName: gitReleaseName || '',
releaseBuildNumber: releaseBuildNumber || '',
};
}
}