2017-12-07 16:13:19 -08:00
|
|
|
import _ from 'lodash';
|
2017-01-26 21:07:22 -08:00
|
|
|
|
2017-12-07 16:13:19 -08:00
|
|
|
import * as challengeTypes from '../../../utils/challengeTypes.js';
|
Feat: react redux migration (#16200)
* feat: crudely enables test to run solution code against React challenge (and passes!)
* feat: Updates comment
* feat: Adds React 2 and 3, validates challenges in app
* feat: Adds React 4, validates tests
* feat: Adds Peter's migrated challenge seed files for all challenges
* feat: Adds redux, react-redux imports, adds tests for React 7,
* feat: Adds tests for React 08
* fix(challenges): wrap reserved words in <code> and add tests
* feat: complete first two tests for React 9
* feat: modifies tests in React 09
* feat: Adds working tests for React 37, including async setState tests
* feat: Escape hatch to avoid async tests in automated test suite
* feat: Updates React 15 with working tests
* feat: build passes, yay
* feat: Provisions original code string in challenges and adds tests for React Redux 01
* fix(tests): add self-closing tags challenge, other small fixes
* fix(challenge): add react_10, some other stuff
* fix(challenges): update react 22, add react 23
* fix(challenges): react 5 and react 8
* feat: removes dependencies that will break in browser, will replace later
* feat: fix build
* feat: add redux 1
* fix(challenge): add react 24 tests
* feat: partial implemented Redux 2
* feat: migrate redux 3
* feat: Adds React-Redux 04 with working tests under npm test
* feat: Updates automated test runner, just provide all the dependencies. Adds Redux-Thunk.
* feat: Adds working tests for React Redux 07
* feat: redux challenge 4
* feat: migrate redux 5
* feat: redux 6
* feat: migrate Redux test 7
* fix(challenge): add react 25 tests
* feat: Adds tests for React 48, npm test does not pass...
* feat: Migrate Redux test 8
* fix(challenges): skip 26, add react 27 tests
* fix(challenges): add react 28 tests, replace function w/ => throughout, fix linter warnings
* feat: fixes (patches) hard to understand problem with automated test suite
* feat: updates async tests patch
* feat: adds converted tests for React 47
* feat: adds converted tests for React 46
* feat: Partially adds tests for React 43
* docs: adds TO-DO tests for React 43
* feat: migrates tests for React 42
* feat: migrates tests for React 41
* feat: migrates tests for React 39
* feat: Migrates tests for React 38, automated test script fails again!
* feat: migrates tests for React 32
* feat: QAs more React Redux challenge in FCC UI
* feat: Updates tests for React 7
* feat: Migrates React-Redux 3 tests and hardcodes deep-freeze dependency
* feat: migrates React Redux 05 tests
* feat: migrates React Redux 06 tests
* feat: Migrates React Redux 10
* feat: Migrates tests for React 16
* feat: Migrates React 17 tests
* feat: Migrates React 18 tests
* feat: Migrates React 19 tests
* feat: Migrates React 19 tests
* feat: fixing usage of code, replace with editor.getOriginalCode
* feat: Migrates React 21 tests
* feat: Finishes migration of React 09
* fix(challenges): add react 45 tests 💀
* feat: Adds React 11 tests
* feat: Migrates React 50 tests
* feat: Re-enables original code in FCC editor, QAs challenges blocked by original code
* feat: hacks head tail code in editor test environment
* feat: updates React 20 head code
* feat: QAs React Redux 07 in UI
* fix(challenges): add React 29 tests
* fix(challenges): add React 30 tests
* feat: updates async tests
* feat: Migrates React 12, gets ReactDOM challenges working and QAs them
* feat: Migrates React 13 tests
* feat: Migrates tests for React 14 and updates challenge description formatting
* feat: Refactors 2nd test for Redux 02
* feat: Migrates React 33
* feat: Removes React 26 and 43
* feat: Adds React 34 from Kevin
* fix(challenges): add React 31 & 35 tests (thanks Kevin)
* feat: Migrate Redux challenge 10 - pass both UI QA and terminal test
* fix(challenge): add react 40 tests
* feat: Migrates React Redux 02 tests
* feat: Migrates React Redux 08 and fixes async syntax in React challenge
* fix(challenge): add react 49 tests with caveat
* feat: fixes React 49 tests and adds first tests for React Redux 09
* feat: Migrate Redux 11 - pass both terminal test and UI test
* feat: Migrate Redux 12 - passing both UI test and terminal test
* feat: Migrate Redux 13 - passing both terminal and UI tests
* feat: Adding in code tags for previous redux challenges - terminal and UI tests pass
* feat: Migrates React Redux 09 and React 44 (thanks Kevin)
* feat: fix code tag issues - passed UI and terminal tests
* feat: Migrates Redux 14 tests
* feat: Migrates Redux 14
* feat: Migrates Redux 15
* feat: Migrates Redux 17
* feat: Final migration and QA of Redux, except for Redux 9
* feat: migrates React 36 and QAs
* feat: Rewrites Redux 09 and migrates
* feat: refactors pull request and cleans up code
* style(challenges): QA React challenges
* style(challenges): QA react challenges
* fix(challenges): fix react 41 and 45 tests
* style(challenges): QA redux challenges
* style(challenges): QA react and redux challenges
* fix(seed/react): Move head/tail to files
* fix(seed/redux): Move head/tail to file level
* chore(packages): Move jsdom to dev deps
* fix(seed/react/redux): Async funcs
make async func defined
* fix(seed): %s/editor.getUserCode/getUserInput/gc
* fix(Challenges/build): Make sure head/tail is bundled and transformed
* feat(Challenges.react): Add tail to render component
* chore(seed): Disable modern challenge testing for now
We will put these on beta while we update the auto testing framework
2017-12-18 13:04:03 -08:00
|
|
|
import { createPoly, updateFileFromSpec } from '../../../../utils/polyvinyl.js';
|
2017-12-07 16:13:19 -08:00
|
|
|
import { decodeScriptTags } from '../../../../utils/encode-decode.js';
|
|
|
|
|
|
|
|
// turn challengeType to file ext
|
|
|
|
const pathsMap = {
|
|
|
|
[ challengeTypes.html ]: 'html',
|
|
|
|
[ challengeTypes.js ]: 'js',
|
|
|
|
[ challengeTypes.bonfire ]: 'js'
|
|
|
|
};
|
2017-01-26 21:07:22 -08:00
|
|
|
// determine the component to view for each challenge
|
|
|
|
export const viewTypes = {
|
|
|
|
[ challengeTypes.html ]: 'classic',
|
|
|
|
[ challengeTypes.js ]: 'classic',
|
|
|
|
[ challengeTypes.bonfire ]: 'classic',
|
|
|
|
[ challengeTypes.frontEndProject ]: 'project',
|
|
|
|
[ challengeTypes.backEndProject ]: 'project',
|
|
|
|
// might not be used anymore
|
|
|
|
[ challengeTypes.simpleProject ]: 'project',
|
|
|
|
// formally hikes
|
|
|
|
[ challengeTypes.video ]: 'video',
|
|
|
|
[ challengeTypes.step ]: 'step',
|
2017-08-08 15:31:26 -04:00
|
|
|
[ challengeTypes.quiz ]: 'quiz',
|
2017-11-29 15:44:51 -08:00
|
|
|
backend: 'backend',
|
|
|
|
modern: 'modern'
|
2017-01-26 21:07:22 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
// determine the type of submit function to use for the challenge on completion
|
|
|
|
export const submitTypes = {
|
|
|
|
[ challengeTypes.html ]: 'tests',
|
|
|
|
[ challengeTypes.js ]: 'tests',
|
|
|
|
[ challengeTypes.bonfire ]: 'tests',
|
|
|
|
// requires just a button press
|
|
|
|
[ challengeTypes.simpleProject ]: 'project.simple',
|
|
|
|
// requires just a single url
|
|
|
|
// like codepen.com/my-project
|
|
|
|
[ challengeTypes.frontEndProject ]: 'project.frontEnd',
|
|
|
|
// requires two urls
|
|
|
|
// a hosted URL where the app is running live
|
|
|
|
// project code url like GitHub
|
|
|
|
[ challengeTypes.backEndProject ]: 'project.backEnd',
|
|
|
|
// formally hikes
|
|
|
|
[ challengeTypes.video ]: 'video',
|
|
|
|
[ challengeTypes.step ]: 'step',
|
2017-08-08 15:31:26 -04:00
|
|
|
[ challengeTypes.quiz ]: 'quiz',
|
2017-11-29 15:44:51 -08:00
|
|
|
backend: 'backend',
|
|
|
|
modern: 'tests'
|
2017-01-26 21:07:22 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
// determines if a line in a challenge description
|
|
|
|
// has html that should be rendered
|
|
|
|
export const descriptionRegex = /\<blockquote|\<ol|\<h4|\<table/;
|
2016-05-11 23:45:42 -07:00
|
|
|
|
2017-12-07 16:13:19 -08:00
|
|
|
export function arrayToString(seedData = ['']) {
|
|
|
|
seedData = Array.isArray(seedData) ? seedData : [seedData];
|
|
|
|
return seedData.reduce((seed, line) => '' + seed + line + '\n', '\n');
|
|
|
|
}
|
|
|
|
|
|
|
|
export function buildSeed({ challengeSeed = [] } = {}) {
|
|
|
|
return _.flow(
|
|
|
|
arrayToString,
|
|
|
|
decodeScriptTags
|
|
|
|
)(challengeSeed);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getFileKey({ challengeType }) {
|
|
|
|
return 'index' + (pathsMap[challengeType] || 'html');
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getPreFile({ challengeType }) {
|
|
|
|
return {
|
|
|
|
name: 'index',
|
|
|
|
ext: pathsMap[challengeType] || 'html',
|
|
|
|
key: getFileKey({ challengeType })
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export function challengeToFiles(challenge, files) {
|
|
|
|
const previousWork = !!files;
|
|
|
|
files = files || challenge.files || {};
|
|
|
|
if (challenge.type === 'modern') {
|
2017-12-19 09:41:09 -08:00
|
|
|
return _.reduce(challenge.files, (_files, fileSpec) => {
|
|
|
|
const file = _.get(files, fileSpec.key);
|
|
|
|
_files[fileSpec.key] = updateFileFromSpec(fileSpec, file);
|
|
|
|
return _files;
|
2017-12-07 16:13:19 -08:00
|
|
|
}, {});
|
|
|
|
}
|
|
|
|
if (
|
|
|
|
challenge.challengeType !== challengeTypes.html &&
|
|
|
|
challenge.challengeType !== challengeTypes.js &&
|
|
|
|
challenge.challengeType !== challengeTypes.bonfire
|
|
|
|
) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
// classic challenge to modern format
|
|
|
|
const preFile = getPreFile(challenge);
|
|
|
|
const contents = previousWork ?
|
|
|
|
// get previous contents
|
|
|
|
_.property([ preFile.key, 'contents' ])(files) :
|
|
|
|
// otherwise start fresh
|
|
|
|
buildSeed(challenge);
|
|
|
|
return {
|
|
|
|
[preFile.key]: createPoly({
|
|
|
|
...files[preFile.key],
|
|
|
|
...preFile,
|
|
|
|
contents,
|
|
|
|
// make sure head/tail are always fresh from fCC
|
|
|
|
head: arrayToString(challenge.head),
|
|
|
|
tail: arrayToString(challenge.tail)
|
|
|
|
})
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-05-13 20:04:56 -07:00
|
|
|
export function createTests({ tests = [] }) {
|
|
|
|
return tests
|
2016-09-23 10:01:48 -07:00
|
|
|
.map(test => {
|
2016-09-29 11:58:31 -07:00
|
|
|
if (typeof test === 'string') {
|
2016-09-23 10:01:48 -07:00
|
|
|
return {
|
2018-02-06 21:31:33 +05:30
|
|
|
text: ('' + test).split('message: ')
|
|
|
|
.pop().replace(/(\'\);(\s*\};)?)/g, ''),
|
2016-09-23 10:01:48 -07:00
|
|
|
testString: test
|
|
|
|
};
|
|
|
|
}
|
|
|
|
return test;
|
|
|
|
});
|
2016-05-13 20:04:56 -07:00
|
|
|
}
|
2016-05-27 17:11:25 -07:00
|
|
|
|
2017-01-02 15:16:03 +01:00
|
|
|
function logReplacer(value) {
|
|
|
|
if (Array.isArray(value)) {
|
|
|
|
const replaced = value.map(logReplacer);
|
|
|
|
return '[' + replaced.join(', ') + ']';
|
|
|
|
}
|
2017-07-02 10:15:23 -04:00
|
|
|
if (typeof value === 'string' && !(/^\/\//).test(value)) {
|
2017-01-02 15:16:03 +01:00
|
|
|
return '"' + value + '"';
|
|
|
|
}
|
|
|
|
if (typeof value === 'number' && isNaN(value)) {
|
|
|
|
return value.toString();
|
|
|
|
}
|
|
|
|
if (typeof value === 'undefined') {
|
|
|
|
return 'undefined';
|
|
|
|
}
|
|
|
|
if (value === null) {
|
|
|
|
return 'null';
|
|
|
|
}
|
|
|
|
if (typeof value === 'function') {
|
|
|
|
return value.name;
|
|
|
|
}
|
|
|
|
if (typeof value === 'object') {
|
|
|
|
return JSON.stringify(value, null, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2016-05-27 17:11:25 -07:00
|
|
|
export function loggerToStr(args) {
|
|
|
|
args = Array.isArray(args) ? args : [args];
|
|
|
|
return args
|
2017-01-02 15:16:03 +01:00
|
|
|
.map(logReplacer)
|
2016-05-27 17:11:25 -07:00
|
|
|
.reduce((str, arg) => str + arg + '\n', '');
|
|
|
|
}
|
2016-06-01 15:52:08 -07:00
|
|
|
|
2016-07-01 18:44:34 -07:00
|
|
|
export function getNextChallenge(
|
|
|
|
current,
|
|
|
|
entities,
|
|
|
|
{
|
|
|
|
isDev = false,
|
|
|
|
skip = 0
|
|
|
|
} = {}
|
|
|
|
) {
|
2016-07-01 12:34:27 -07:00
|
|
|
const { challenge: challengeMap, block: blockMap } = entities;
|
2016-06-01 15:52:08 -07:00
|
|
|
// find current challenge
|
|
|
|
// find current block
|
|
|
|
// find next challenge in block
|
|
|
|
const currentChallenge = challengeMap[current];
|
2016-06-10 14:01:13 -07:00
|
|
|
if (!currentChallenge) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const block = blockMap[currentChallenge.block];
|
|
|
|
const index = block.challenges.indexOf(currentChallenge.dashedName);
|
2016-07-01 12:34:27 -07:00
|
|
|
// use next challenge name to find challenge in challenge map
|
|
|
|
const nextChallenge = challengeMap[
|
|
|
|
// grab next challenge name in current block
|
|
|
|
// skip is used to skip isComingSoon challenges
|
|
|
|
block.challenges[ index + 1 + skip ]
|
|
|
|
];
|
2016-09-09 13:53:09 -07:00
|
|
|
if (
|
|
|
|
!isDev &&
|
|
|
|
nextChallenge &&
|
|
|
|
(nextChallenge.isComingSoon || nextChallenge.isBeta)
|
|
|
|
) {
|
2016-07-01 12:34:27 -07:00
|
|
|
// if we find a next challenge and it is a coming soon
|
|
|
|
// recur with plus one to skip this challenge
|
2016-07-01 18:44:34 -07:00
|
|
|
return getNextChallenge(current, entities, { isDev, skip: skip + 1 });
|
2016-07-01 12:34:27 -07:00
|
|
|
}
|
|
|
|
return nextChallenge;
|
2016-06-10 14:01:13 -07:00
|
|
|
}
|
|
|
|
|
2016-07-01 18:44:34 -07:00
|
|
|
export function getFirstChallengeOfNextBlock(
|
|
|
|
current,
|
|
|
|
entities,
|
|
|
|
{
|
|
|
|
isDev = false,
|
|
|
|
skip = 0
|
|
|
|
} = {}
|
|
|
|
) {
|
2016-06-10 14:01:13 -07:00
|
|
|
const {
|
|
|
|
challenge: challengeMap,
|
|
|
|
block: blockMap,
|
|
|
|
superBlock: SuperBlockMap
|
2016-07-01 12:34:27 -07:00
|
|
|
} = entities;
|
2016-06-10 14:01:13 -07:00
|
|
|
const currentChallenge = challengeMap[current];
|
|
|
|
if (!currentChallenge) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const block = blockMap[currentChallenge.block];
|
|
|
|
if (!block) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const superBlock = SuperBlockMap[block.superBlock];
|
2016-07-01 12:34:27 -07:00
|
|
|
if (!superBlock) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
// find index of current block
|
2016-06-10 14:01:13 -07:00
|
|
|
const index = superBlock.blocks.indexOf(block.dashedName);
|
2016-07-01 12:34:27 -07:00
|
|
|
|
|
|
|
// find next block name
|
|
|
|
// and pull block object from block map
|
|
|
|
const newBlock = blockMap[
|
|
|
|
superBlock.blocks[ index + 1 + skip ]
|
|
|
|
];
|
2016-06-10 14:01:13 -07:00
|
|
|
if (!newBlock) {
|
|
|
|
return null;
|
|
|
|
}
|
2016-07-01 12:34:27 -07:00
|
|
|
// grab first challenge from next block
|
|
|
|
const nextChallenge = challengeMap[newBlock.challenges[0]];
|
2016-07-01 18:44:34 -07:00
|
|
|
if (isDev || !nextChallenge || !nextChallenge.isComingSoon) {
|
|
|
|
return nextChallenge;
|
|
|
|
}
|
|
|
|
// if first challenge is coming soon, find next challenge here
|
|
|
|
const nextChallenge2 = getNextChallenge(
|
|
|
|
nextChallenge.dashedName,
|
|
|
|
entities,
|
|
|
|
{ isDev }
|
|
|
|
);
|
|
|
|
if (nextChallenge2) {
|
2016-07-01 12:34:27 -07:00
|
|
|
return nextChallenge2;
|
|
|
|
}
|
2016-07-01 18:44:34 -07:00
|
|
|
// whole block is coming soon
|
|
|
|
// skip this block
|
|
|
|
return getFirstChallengeOfNextBlock(
|
|
|
|
current,
|
|
|
|
entities,
|
|
|
|
{ isDev, skip: skip + 1 }
|
|
|
|
);
|
2016-06-10 14:01:13 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
export function getFirstChallengeOfNextSuperBlock(
|
|
|
|
current,
|
2016-07-01 12:34:27 -07:00
|
|
|
entities,
|
|
|
|
superBlocks,
|
2016-07-01 18:44:34 -07:00
|
|
|
{
|
|
|
|
isDev = false,
|
|
|
|
skip = 0
|
|
|
|
} = {}
|
2016-06-10 14:01:13 -07:00
|
|
|
) {
|
|
|
|
const {
|
|
|
|
challenge: challengeMap,
|
|
|
|
block: blockMap,
|
|
|
|
superBlock: SuperBlockMap
|
2016-07-01 12:34:27 -07:00
|
|
|
} = entities;
|
2016-06-10 14:01:13 -07:00
|
|
|
const currentChallenge = challengeMap[current];
|
|
|
|
if (!currentChallenge) {
|
|
|
|
return null;
|
2016-06-01 15:52:08 -07:00
|
|
|
}
|
2016-06-10 14:01:13 -07:00
|
|
|
const block = blockMap[currentChallenge.block];
|
|
|
|
if (!block) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const superBlock = SuperBlockMap[block.superBlock];
|
2016-07-01 12:34:27 -07:00
|
|
|
if (!superBlock) {
|
|
|
|
return null;
|
|
|
|
}
|
2016-06-10 14:01:13 -07:00
|
|
|
const index = superBlocks.indexOf(superBlock.dashedName);
|
2016-07-01 12:34:27 -07:00
|
|
|
const newSuperBlock = SuperBlockMap[superBlocks[ index + 1 + skip]];
|
2016-06-10 14:01:13 -07:00
|
|
|
if (!newSuperBlock) {
|
|
|
|
return null;
|
|
|
|
}
|
2016-07-01 12:34:27 -07:00
|
|
|
const newBlock = blockMap[
|
|
|
|
newSuperBlock.blocks[ 0 ]
|
|
|
|
];
|
|
|
|
if (!newBlock) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const nextChallenge = challengeMap[newBlock.challenges[0]];
|
2016-07-01 18:44:34 -07:00
|
|
|
if (isDev || !nextChallenge || !nextChallenge.isComingSoon) {
|
2016-07-01 12:34:27 -07:00
|
|
|
return nextChallenge;
|
|
|
|
}
|
|
|
|
// coming soon challenge, grab next
|
|
|
|
// non coming soon challenge in same block instead
|
|
|
|
const nextChallengeInBlock = getNextChallenge(
|
|
|
|
nextChallenge.dashedName,
|
2016-07-01 18:44:34 -07:00
|
|
|
entities,
|
|
|
|
{ isDev }
|
2016-07-01 12:34:27 -07:00
|
|
|
);
|
|
|
|
if (nextChallengeInBlock) {
|
|
|
|
return nextChallengeInBlock;
|
|
|
|
}
|
|
|
|
// whole block is coming soon
|
|
|
|
// grab first challenge in next block in newSuperBlock instead
|
|
|
|
const challengeInNextBlock = getFirstChallengeOfNextBlock(
|
|
|
|
nextChallenge.dashedName,
|
2016-07-01 18:44:34 -07:00
|
|
|
entities,
|
|
|
|
{ isDev }
|
2016-07-01 12:34:27 -07:00
|
|
|
);
|
|
|
|
|
|
|
|
if (challengeInNextBlock) {
|
|
|
|
return challengeInNextBlock;
|
|
|
|
}
|
|
|
|
// whole super block is coming soon
|
|
|
|
// skip this super block
|
|
|
|
return getFirstChallengeOfNextSuperBlock(
|
|
|
|
current,
|
|
|
|
entities,
|
|
|
|
superBlocks,
|
2016-07-01 18:44:34 -07:00
|
|
|
{ isDev, skip: skip + 1 }
|
2016-07-01 12:34:27 -07:00
|
|
|
);
|
2016-06-10 14:01:13 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
export function getCurrentBlockName(current, entities) {
|
|
|
|
const { challenge: challengeMap } = entities;
|
|
|
|
const challenge = challengeMap[current];
|
|
|
|
return challenge.block;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getCurrentSuperBlockName(current, entities) {
|
|
|
|
const { challenge: challengeMap, block: blockMap } = entities;
|
|
|
|
const challenge = challengeMap[current];
|
|
|
|
const block = blockMap[challenge.block];
|
|
|
|
return block.superBlock;
|
2016-06-01 15:52:08 -07:00
|
|
|
}
|