| 
									
										
										
										
											2015-06-02 20:32:10 -07:00
										 |  |  | /* eslint-disable no-process-exit */ | 
					
						
							| 
									
										
										
										
											2016-01-20 09:03:58 -08:00
										 |  |  | require('babel-register'); | 
					
						
							| 
									
										
										
										
											2015-06-02 20:32:10 -07:00
										 |  |  | require('dotenv').load(); | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  | const adler32 = require('adler32'); | 
					
						
							| 
									
										
										
										
											2015-10-20 19:53:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  | const Rx = require('rx'); | 
					
						
							|  |  |  | const _ = require('lodash'); | 
					
						
							| 
									
										
										
										
											2018-05-21 14:56:49 +01:00
										 |  |  | const createDebugger = require('debug'); | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  | const utils = require('../server/utils'); | 
					
						
							| 
									
										
										
										
											2018-06-20 19:20:16 +05:30
										 |  |  | const { getChallenges } = require('@freecodecamp/curriculum'); | 
					
						
							| 
									
										
										
										
											2018-05-21 14:56:49 +01:00
										 |  |  | const { validateChallenge } = require( | 
					
						
							|  |  |  |   './schema/challengeSchema' | 
					
						
							|  |  |  | ); | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  | const app = require('../server/server'); | 
					
						
							| 
									
										
										
										
											2018-02-16 23:18:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  | const log = createDebugger('fcc:seed'); | 
					
						
							|  |  |  | // force logger to always output
 | 
					
						
							|  |  |  | // this may be brittle
 | 
					
						
							|  |  |  | log.enabled = true; | 
					
						
							| 
									
										
										
										
											2015-07-11 19:51:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  | const dasherize = utils.dasherize; | 
					
						
							|  |  |  | const nameify = utils.nameify; | 
					
						
							|  |  |  | const Observable = Rx.Observable; | 
					
						
							|  |  |  | const Challenge = app.models.Challenge; | 
					
						
							| 
									
										
										
										
											2015-06-02 20:32:10 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  | const destroyChallenges = | 
					
						
							| 
									
										
										
										
											2016-03-11 21:45:58 -08:00
										 |  |  |   Observable.fromNodeCallback(Challenge.destroyAll, Challenge); | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  | const createChallenges = | 
					
						
							| 
									
										
										
										
											2016-03-11 21:45:58 -08:00
										 |  |  |   Observable.fromNodeCallback(Challenge.create, Challenge); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  | const Block = app.models.Block; | 
					
						
							|  |  |  | const destroyBlocks = Observable.fromNodeCallback(Block.destroyAll, Block); | 
					
						
							|  |  |  | const createBlocks = Observable.fromNodeCallback(Block.create, Block); | 
					
						
							| 
									
										
										
										
											2017-11-29 15:44:51 -08:00
										 |  |  | const arrToString = arr => | 
					
						
							|  |  |  |   Array.isArray(arr) ? arr.join('\n') : _.toString(arr); | 
					
						
							| 
									
										
										
										
											2016-03-11 21:45:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | Observable.combineLatest( | 
					
						
							|  |  |  |   destroyChallenges(), | 
					
						
							|  |  |  |   destroyBlocks() | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  |   .last() | 
					
						
							|  |  |  |   .flatMap(function() { return Observable.from(getChallenges()); }) | 
					
						
							| 
									
										
										
										
											2015-11-01 17:20:03 -08:00
										 |  |  |   .flatMap(function(challengeSpec) { | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  |     const order = challengeSpec.order; | 
					
						
							|  |  |  |     const blockName = challengeSpec.name; | 
					
						
							|  |  |  |     const superBlock = challengeSpec.superBlock; | 
					
						
							|  |  |  |     const superOrder = challengeSpec.superOrder; | 
					
						
							|  |  |  |     const isBeta = !!challengeSpec.isBeta; | 
					
						
							|  |  |  |     const isComingSoon = !!challengeSpec.isComingSoon; | 
					
						
							|  |  |  |     const fileName = challengeSpec.fileName; | 
					
						
							|  |  |  |     const helpRoom = challengeSpec.helpRoom || 'Help'; | 
					
						
							| 
									
										
										
										
											2018-01-12 11:57:58 +01:00
										 |  |  |     const time = challengeSpec.time; | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  |     const isLocked = !!challengeSpec.isLocked; | 
					
						
							|  |  |  |     const message = challengeSpec.message; | 
					
						
							|  |  |  |     const required = challengeSpec.required || []; | 
					
						
							|  |  |  |     const template = challengeSpec.template; | 
					
						
							| 
									
										
										
										
											2018-02-16 23:18:53 +00:00
										 |  |  |     const isPrivate = !!challengeSpec.isPrivate; | 
					
						
							| 
									
										
										
										
											2016-01-15 00:55:08 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  |     log('parsed %s successfully', blockName); | 
					
						
							| 
									
										
										
										
											2015-08-09 22:14:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 12:14:38 -07:00
										 |  |  |     // challenge file has no challenges...
 | 
					
						
							|  |  |  |     if (challengeSpec.challenges.length === 0) { | 
					
						
							| 
									
										
										
										
											2016-03-11 21:45:58 -08:00
										 |  |  |       return Rx.Observable.just([{ block: 'empty ' + blockName }]); | 
					
						
							| 
									
										
										
										
											2015-08-10 12:14:38 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  |     const block = { | 
					
						
							| 
									
										
										
										
											2016-03-11 21:45:58 -08:00
										 |  |  |       title: blockName, | 
					
						
							|  |  |  |       name: nameify(blockName), | 
					
						
							|  |  |  |       dashedName: dasherize(blockName), | 
					
						
							| 
									
										
										
										
											2018-02-16 23:18:53 +00:00
										 |  |  |       superOrder, | 
					
						
							|  |  |  |       superBlock, | 
					
						
							| 
									
										
										
										
											2016-06-22 14:57:44 -07:00
										 |  |  |       superBlockMessage: message, | 
					
						
							| 
									
										
										
										
											2018-02-16 23:18:53 +00:00
										 |  |  |       order, | 
					
						
							|  |  |  |       time, | 
					
						
							|  |  |  |       isLocked, | 
					
						
							|  |  |  |       isPrivate | 
					
						
							| 
									
										
										
										
											2016-03-11 21:45:58 -08:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return createBlocks(block) | 
					
						
							|  |  |  |       .map(block => { | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  |         log('successfully created %s block', block.name); | 
					
						
							| 
									
										
										
										
											2015-08-09 22:14:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-11 21:45:58 -08:00
										 |  |  |         return challengeSpec.challenges | 
					
						
							|  |  |  |           .map(function(challenge, index) { | 
					
						
							|  |  |  |             challenge.name = nameify(challenge.title); | 
					
						
							| 
									
										
										
										
											2015-11-09 17:27:56 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-15 08:11:24 -05:00
										 |  |  |             challenge.dashedName = dasherize(challenge.title); | 
					
						
							| 
									
										
										
										
											2016-01-30 00:13:41 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-11 21:45:58 -08:00
										 |  |  |             challenge.checksum = adler32.sum( | 
					
						
							|  |  |  |               Buffer( | 
					
						
							|  |  |  |                 challenge.title + | 
					
						
							|  |  |  |                 JSON.stringify(challenge.description) + | 
					
						
							|  |  |  |                 JSON.stringify(challenge.challengeSeed) + | 
					
						
							|  |  |  |                 JSON.stringify(challenge.tests) | 
					
						
							|  |  |  |               ) | 
					
						
							|  |  |  |             ); | 
					
						
							| 
									
										
										
										
											2015-08-09 22:14:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-29 15:44:51 -08:00
										 |  |  |             if (challenge.files) { | 
					
						
							|  |  |  |               challenge.files = _.reduce(challenge.files, (map, file) => { | 
					
						
							|  |  |  |                 map[file.key] = { | 
					
						
							|  |  |  |                   ...file, | 
					
						
							|  |  |  |                   head: arrToString(file.head), | 
					
						
							|  |  |  |                   contents: arrToString(file.contents), | 
					
						
							|  |  |  |                   tail: arrToString(file.tail) | 
					
						
							|  |  |  |                 }; | 
					
						
							|  |  |  |                 return map; | 
					
						
							|  |  |  |               }, {}); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2016-03-11 21:45:58 -08:00
										 |  |  |             challenge.fileName = fileName; | 
					
						
							|  |  |  |             challenge.helpRoom = helpRoom; | 
					
						
							|  |  |  |             challenge.order = order; | 
					
						
							|  |  |  |             challenge.suborder = index + 1; | 
					
						
							| 
									
										
										
										
											2016-06-10 10:21:57 -07:00
										 |  |  |             challenge.block = dasherize(blockName); | 
					
						
							| 
									
										
										
										
											2018-03-23 14:35:29 +00:00
										 |  |  |             challenge.blockId = '' + block.id; | 
					
						
							| 
									
										
										
										
											2016-03-11 21:45:58 -08:00
										 |  |  |             challenge.isBeta = challenge.isBeta || isBeta; | 
					
						
							|  |  |  |             challenge.isComingSoon = challenge.isComingSoon || isComingSoon; | 
					
						
							| 
									
										
										
										
											2016-06-22 14:57:44 -07:00
										 |  |  |             challenge.isLocked = challenge.isLocked || isLocked; | 
					
						
							| 
									
										
										
										
											2018-02-16 23:18:53 +00:00
										 |  |  |             challenge.isPrivate = challenge.isPrivate || isPrivate; | 
					
						
							| 
									
										
										
										
											2016-03-11 21:45:58 -08:00
										 |  |  |             challenge.time = challengeSpec.time; | 
					
						
							|  |  |  |             challenge.superOrder = superOrder; | 
					
						
							|  |  |  |             challenge.superBlock = superBlock | 
					
						
							|  |  |  |               .split('-') | 
					
						
							|  |  |  |               .map(function(word) { | 
					
						
							|  |  |  |                 return _.capitalize(word); | 
					
						
							|  |  |  |               }) | 
					
						
							|  |  |  |               .join(' '); | 
					
						
							| 
									
										
										
										
											2016-07-28 21:20:13 -07:00
										 |  |  |             challenge.required = (challenge.required || []).concat(required); | 
					
						
							| 
									
										
											  
											
												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
										 |  |  |             challenge.template = challenge.template || template; | 
					
						
							| 
									
										
										
										
											2018-03-23 14:35:29 +00:00
										 |  |  |             return _.omit( | 
					
						
							|  |  |  |               challenge, | 
					
						
							|  |  |  |               [ | 
					
						
							|  |  |  |                 'betaSolutions', | 
					
						
							|  |  |  |                 'betaTests', | 
					
						
							|  |  |  |                 'hints', | 
					
						
							|  |  |  |                 'MDNlinks', | 
					
						
							|  |  |  |                 'null', | 
					
						
							|  |  |  |                 'rawSolutions', | 
					
						
							|  |  |  |                 'react', | 
					
						
							|  |  |  |                 'reactRedux', | 
					
						
							|  |  |  |                 'redux', | 
					
						
							|  |  |  |                 'releasedOn', | 
					
						
							|  |  |  |                 'translations', | 
					
						
							|  |  |  |                 'type' | 
					
						
							|  |  |  |               ] | 
					
						
							|  |  |  |             ); | 
					
						
							| 
									
										
										
										
											2016-03-11 21:45:58 -08:00
										 |  |  |           }); | 
					
						
							|  |  |  |       }) | 
					
						
							| 
									
										
										
										
											2018-03-23 14:35:29 +00:00
										 |  |  |       .flatMap(challenges => { | 
					
						
							|  |  |  |         challenges.forEach(challenge => { | 
					
						
							|  |  |  |           const result = validateChallenge(challenge); | 
					
						
							|  |  |  |           if (result.error) { | 
					
						
							|  |  |  |             console.log(result.value); | 
					
						
							|  |  |  |             throw new Error(result.error); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |         return createChallenges(challenges); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2015-10-20 19:53:08 -07:00
										 |  |  |   }) | 
					
						
							|  |  |  |   .subscribe( | 
					
						
							|  |  |  |     function(challenges) { | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  |       log('%s successfully saved', challenges[0].block); | 
					
						
							| 
									
										
										
										
											2015-10-20 19:53:08 -07:00
										 |  |  |     }, | 
					
						
							|  |  |  |     function(err) { throw err; }, | 
					
						
							|  |  |  |     function() { | 
					
						
							| 
									
										
										
										
											2018-01-08 18:38:29 +05:30
										 |  |  |       log('challenge seed completed'); | 
					
						
							| 
									
										
										
										
											2015-10-20 19:53:08 -07:00
										 |  |  |       process.exit(0); | 
					
						
							| 
									
										
										
										
											2015-06-02 20:32:10 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-10-20 19:53:08 -07:00
										 |  |  |   ); |