blob: 5a01cab6842aa2c59cd172f134ebfd2ff6424661 [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.
import * as express from 'express';
import {logger} from '../logger';
import {CompilerProps} from '../properties';
import * as utils from '../utils';
// When it's import profanities from 'profanities'; ts says "Cannot find module 'profanities' or its corresponding type
// declarations."
// Updating profanities to v3 requires ESM modules
// eslint-disable-next-line @typescript-eslint/no-var-requires, unicorn/prefer-module
const profanities = require('profanities');
const FILE_HASH_VERSION = 'Compiler Explorer Config Hasher 2';
/* How long a string to check for possible unusable hashes (Profanities or confusing text)
Note that a Hash might end up being longer than this!
*/
const USABLE_HASH_CHECK_LENGTH = 9; // Quite generous
const MAX_TRIES = 4;
export abstract class StorageBase {
constructor(protected readonly httpRootDir: string, protected readonly compilerProps: CompilerProps) {}
/**
* Encode a buffer as a URL-safe string.
*/
static encodeBuffer(buffer: Buffer): string {
return utils.base32Encode(buffer);
}
static isCleanText(text: string) {
const lowercased = text.toLowerCase();
return !profanities.some(badWord => lowercased.includes(badWord));
}
static getRawConfigHash(config) {
return StorageBase.encodeBuffer(utils.getBinaryHash(JSON.stringify(config), FILE_HASH_VERSION));
}
static getSafeHash(config) {
// Keep rehashing until a usable text is found
let configHash = StorageBase.getRawConfigHash(config);
let tries = 1;
while (!StorageBase.isCleanText(configHash.substr(0, USABLE_HASH_CHECK_LENGTH))) {
// Shake up the hash a bit by adding, or incrementing a nonce value.
config.nonce = tries;
logger.info(`Unusable text found in full hash ${configHash} - Trying again (${tries})`);
if (tries <= MAX_TRIES) {
configHash = StorageBase.getRawConfigHash(config);
++tries;
} else {
logger.warn(`Gave up trying to find clean text for ${configHash}`);
break;
}
}
// And stringify it for the rest of the request
config = JSON.stringify(config);
return {config, configHash};
}
static configFor(req: express.Request) {
if (req.body.config) {
return req.body.config;
} else if (req.body.sessions) {
return req.body;
}
return null;
}
handler(req: express.Request, res: express.Response) {
// Get the desired config and check for profanities in its hash
const origConfig = StorageBase.configFor(req);
if (!origConfig) {
logger.error('No configuration found');
res.status(500);
res.send('Missing config parameter');
return;
}
const {config, configHash} = StorageBase.getSafeHash(origConfig);
this.findUniqueSubhash(configHash)
.then(result => {
logger.info(
`Unique subhash '${result.uniqueSubHash}' ` +
`(${result.alreadyPresent ? 'was already present' : 'newly-created'})`
);
if (result.alreadyPresent) {
return result;
} else {
const storedObject = {
prefix: result.prefix,
uniqueSubHash: result.uniqueSubHash,
fullHash: configHash,
config: config,
};
return this.storeItem(storedObject, req);
}
})
.then(result => {
res.send({url: `${req.protocol}://${req.get('host')}${this.httpRootDir}z/${result.uniqueSubHash}`});
})
.catch(err => {
logger.error(err);
res.status(500);
res.send(err.message);
});
}
abstract storeItem(item, req: express.Request): Promise<any>;
abstract findUniqueSubhash(hash: string): Promise<any>;
abstract expandId(id: string): Promise<{config: string; specialMetadata: any}>;
abstract incrementViewCount(id): Promise<any>;
}