blob: 9bb34e4a5b3aad630f3cada42eadeefa8b7219c6 [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 logger = require('../logger').logger,
hash = require('../utils').getBinaryHash,
/***
* @type {string[]}
*/
profanities = require('profanities');
const FILE_HASH_VERSION = 'Compiler Explorer Config Hasher';
const USABLE_NAME_MAX_LENGTH = 9; // Quite generous
const ILLEGIBLE_LETERS = ['l', 'I', 'O', '0', '1']; // L i and o are allowed. Readable if the others are not present
const MAX_TRIES = 4;
class StorageBase {
constructor(httpRootDir, compilerProps) {
this.compilerProps = compilerProps;
this.httpRootDir = httpRootDir;
}
/**
* Encode a buffer as a URL-safe string.
* @param {Buffer} buffer
* @returns {string}
*/
static safe64Encoded(buffer) {
return buffer.toString('base64')
.replace(/\+/g, '-')
.replace(/\//g, '_')
.replace(/=+$/, '');
}
static isLegibleText(text) {
return !ILLEGIBLE_LETERS.some(letter => text.includes(letter));
}
static isCleanText(text) {
return !profanities.some(badWord => text.includes(badWord));
}
static isUsableText(text) {
const lowercasedText = text.toLowerCase();
return StorageBase.isLegibleText(text) && StorageBase.isCleanText(lowercasedText);
}
static getSafeHash(config) {
// Keep rehashing until a usable text is found
let configHash = StorageBase.safe64Encoded(hash(JSON.stringify(config), FILE_HASH_VERSION));
let tries = 1;
while (!StorageBase.isUsableText(configHash.substr(0, USABLE_NAME_MAX_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.safe64Encoded(hash(JSON.stringify(config), FILE_HASH_VERSION));
++tries;
} else {
logger.warn(`Gave up trying to find usable text for ${configHash}`);
break;
}
}
// And stringify it for the rest of the request
config = JSON.stringify(config);
return {config, configHash};
}
static configFor(req) {
if (req.body.config) {
return req.body.config;
} else if (req.body.sessions) {
return req.body;
}
return null;
}
handler(req, res) {
// 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.end("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) {
const storedObject = {
prefix: result.prefix,
uniqueSubHash: result.uniqueSubHash,
fullHash: configHash,
config: config
};
return this.storeItem(storedObject, req);
} else {
return result;
}
})
.then(result => {
res.set('Content-Type', 'application/json');
const shortlink = `${req.protocol}://${req.get('host')}${this.httpRootDir}z/${result.uniqueSubHash}`;
res.send(JSON.stringify({url: shortlink}));
})
.catch(err => {
logger.error(err);
res.status(500);
res.end(err.message);
});
}
storeItem(item) {
logger.error('Trying to store item from base storage' + item);
return Promise.reject();
}
findUniqueSubhash(hash) {
logger.error(`Trying to find unique subhash from base storage ${hash}`);
return Promise.reject();
}
expandId(id) {
logger.error(`Trying to expand from base storage ${id}`);
return Promise.reject();
}
incrementViewCount(id) {
logger.error(`Trying to increment view count from base storage ${id}`);
return Promise.reject();
}
}
function storageFactory(storageSolution, compilerProps, awsProps, httpRootDir) {
const storage = require(`./storage-${storageSolution}`);
return new storage(httpRootDir, compilerProps, awsProps);
}
module.exports = {
StorageBase: StorageBase,
storageFactory: storageFactory
};