* feat(rechallenge): Retool challenge framework * fix(code-storage): should use setContent not updateContent * fix(rechallenge): fix context issue and temporal zone of death * fix(rechallenge): Fix frame sources for user code * fix(polyvinyl): Set should ignore source and transform should keep track of source * fix(rechallenge): Missing return statement causing issues
125 lines
3.0 KiB
JavaScript
125 lines
3.0 KiB
JavaScript
import { Observable } from 'rx';
|
|
import cond from 'lodash/cond';
|
|
import identity from 'lodash/identity';
|
|
import stubTrue from 'lodash/stubTrue';
|
|
import conforms from 'lodash/conforms';
|
|
|
|
import castToObservable from '../../common/app/utils/cast-to-observable.js';
|
|
|
|
const HTML$JSReg = /html|js/;
|
|
|
|
const testHTMLJS = conforms({ ext: (ext) => HTML$JSReg.test(ext) });
|
|
// const testJS = matchesProperty('ext', 'js');
|
|
const passToNext = [ stubTrue, identity ];
|
|
|
|
// Detect if a JS multi-line comment is left open
|
|
const throwIfOpenComments = cond([
|
|
[
|
|
testHTMLJS,
|
|
function _checkForComments({ contents }) {
|
|
const openingComments = contents.match(/\/\*/gi);
|
|
const closingComments = contents.match(/\*\//gi);
|
|
if (
|
|
openingComments &&
|
|
(!closingComments || openingComments.length > closingComments.length)
|
|
) {
|
|
throw new SyntaxError('Unfinished multi-line comment');
|
|
}
|
|
}
|
|
],
|
|
passToNext
|
|
]);
|
|
|
|
|
|
// Nested dollar sign calls breaks browsers
|
|
const nestedJQCallReg = /\$\s*?\(\s*?\$\s*?\)/gi;
|
|
const throwIfNestedJquery = cond([
|
|
[
|
|
testHTMLJS,
|
|
function({ contents }) {
|
|
if (nestedJQCallReg.test(contents)) {
|
|
throw new SyntaxError('Nested jQuery calls breaks browsers');
|
|
}
|
|
}
|
|
],
|
|
passToNext
|
|
]);
|
|
|
|
const functionReg = /function/g;
|
|
const functionCallReg = /function\s*?\(|function\s+\w+\s*?\(/gi;
|
|
// lonely function keywords breaks browsers
|
|
const ThrowIfUnfinishedFunction = cond([
|
|
|
|
[
|
|
testHTMLJS,
|
|
function({ contents }) {
|
|
if (
|
|
functionReg.test(contents) &&
|
|
!functionCallReg.test(contents)
|
|
) {
|
|
throw new SyntaxError(
|
|
'Unsafe or unfinished function declaration'
|
|
);
|
|
}
|
|
}
|
|
],
|
|
passToNext
|
|
]);
|
|
|
|
|
|
// console call stops tests scripts from running
|
|
const unsafeConsoleCallReg = /if\s\(null\)\sconsole\.log\(1\);/gi;
|
|
const throwIfUnsafeConsoleCall = cond([
|
|
[
|
|
testHTMLJS,
|
|
function({ contents }) {
|
|
if (unsafeConsoleCallReg.test(contents)) {
|
|
throw new SyntaxError(
|
|
'`if (null) console.log(1)` detected. This will break tests'
|
|
);
|
|
}
|
|
}
|
|
],
|
|
passToNext
|
|
]);
|
|
|
|
// Code with the URL hyperdev.com should not be allowed to run,
|
|
const goMixReg = /glitch\.(com|me)/gi;
|
|
const throwIfGomixDetected = cond([
|
|
[
|
|
testHTMLJS,
|
|
function({ contents }) {
|
|
if (goMixReg.test(contents)) {
|
|
throw new Error('Glitch.com or Glitch.me should not be in the code');
|
|
}
|
|
}
|
|
],
|
|
passToNext
|
|
]);
|
|
|
|
const validators = [
|
|
throwIfOpenComments,
|
|
throwIfGomixDetected,
|
|
throwIfNestedJquery,
|
|
ThrowIfUnfinishedFunction,
|
|
throwIfUnsafeConsoleCall
|
|
];
|
|
|
|
export default function validate(file) {
|
|
return validators.reduce((obs, validator) => obs.flatMap(file => {
|
|
try {
|
|
return castToObservable(validator(file));
|
|
} catch (err) {
|
|
return Observable.throw(err);
|
|
}
|
|
}), Observable.of(file))
|
|
// if no error has occured map to the original file
|
|
.map(() => file)
|
|
// if err add it to the file
|
|
// and return file
|
|
.catch(err => {
|
|
file.error = err;
|
|
return Observable.just(file);
|
|
});
|
|
}
|