462 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			462 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /* eslint-disable no-inline-comments */
 | |
| import fs from 'fs-extra';
 | |
| import path from 'path';
 | |
| import _ from 'lodash';
 | |
| import { dasherize } from './utils';
 | |
| 
 | |
| const jsonLinePrefix = '//--JSON:';
 | |
| const paragraphBreak = '<!--break-->';
 | |
| 
 | |
| class ChallengeFile {
 | |
|   constructor(dir, name, suffix) {
 | |
|     this.dir = dir;
 | |
|     this.name = name;
 | |
|     this.suffix = suffix;
 | |
|   }
 | |
| 
 | |
|   filePath() {
 | |
|     return path.join(this.dir, this.name + this.suffix);
 | |
|   }
 | |
| 
 | |
|   write(contents) {
 | |
|     if (_.isArray(contents)) {
 | |
|       contents = contents.join('\n');
 | |
|     }
 | |
|     fs.writeFile(this.filePath(), contents, err => {
 | |
|       if (err) {
 | |
|         throw err;
 | |
|       }
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   readChunks() {
 | |
|     // todo: make this work async
 | |
|     // todo: make sure it works with encodings
 | |
|     let data = fs.readFileSync(this.filePath());
 | |
|     let lines = data.toString().split(/(?:\r\n|\r|\n)/g);
 | |
| 
 | |
|     function removeLeadingEmptyLines(array) {
 | |
|       let emptyString = /^\s*$/;
 | |
|       while (array && Array.isArray(array) && emptyString.test(array[0])) {
 | |
|         array.shift();
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     let chunkEnd = /(<!|\/\*)--end--/;
 | |
|     let index = 0;
 | |
| 
 | |
|     function endOfFile() {
 | |
|       return index === lines.length;
 | |
|     }
 | |
| 
 | |
|     function nextLine() {
 | |
|       if (endOfFile()) {
 | |
|         return null;
 | |
|       }
 | |
|       return lines[index++].toString();
 | |
|     }
 | |
| 
 | |
|     function readDescription() {
 | |
|       let description = [];
 | |
|       let currentParagraph = [];
 | |
| 
 | |
|       function pushParagraph() {
 | |
|         removeLeadingEmptyLines(currentParagraph);
 | |
|         description.push(currentParagraph.join('\n'));
 | |
|         currentParagraph = [];
 | |
|       }
 | |
| 
 | |
|       while (!endOfFile()) {
 | |
|         let line = nextLine();
 | |
|         if (chunkEnd.test(line)) {
 | |
|           pushParagraph();
 | |
|           return description;
 | |
|         } else if (line === paragraphBreak) {
 | |
|           pushParagraph();
 | |
|         } else {
 | |
|           currentParagraph.push(line);
 | |
|         }
 | |
|       }
 | |
|       throw `Unexpected end of the file while reading a description.
 | |
|         ${this.filePath()}`;
 | |
|     }
 | |
| 
 | |
|     function readTranslations() {
 | |
|       let translations = {};
 | |
|       let langChunk = /<!--(\w+(-\w+)*)-->/;
 | |
| 
 | |
|       while (!endOfFile()) {
 | |
|         let line = nextLine();
 | |
|         if (chunkEnd.test(line)) {
 | |
|           return translations;
 | |
|         } else if (langChunk.test(line)) {
 | |
|           let langCode = line.match(langChunk)[1];
 | |
|           translations[langCode] = readProperties();
 | |
|           line = nextLine();
 | |
|           if (!chunkEnd.test(line)) {
 | |
|             throw `Expected --end--:
 | |
|               ${line}`;
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       throw `Unexpected end of the file while reading translations.
 | |
|         ${this.filePath()}`;
 | |
|     }
 | |
| 
 | |
|     function readFiles() {
 | |
|       let files = {};
 | |
|       let fileChunk = /<!--(\w+)\.(\w+)-->/;
 | |
|       while (!endOfFile()) {
 | |
|         let line = nextLine();
 | |
|         if (chunkEnd.test(line)) {
 | |
|           return files;
 | |
|         } else if (fileChunk.test(line)) {
 | |
|           let name = line.match(fileChunk)[1];
 | |
|           let ext = line.match(fileChunk)[2];
 | |
|           let key = name + ext;
 | |
|           files[key] = {};
 | |
|           files[key].key = key;
 | |
|           files[key].ext = ext;
 | |
|           files[key].name = name;
 | |
|           Object.assign(files[key], readProperties());
 | |
|           line = nextLine();
 | |
|           if (!chunkEnd.test(line)) {
 | |
|             throw `Expected --end--:
 | |
|               ${line}`;
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       throw `Unexpected end of the file while reading files.
 | |
|         ${this.filePath()}`;
 | |
|     }
 | |
| 
 | |
|     function readProperty() {
 | |
|       let property = [];
 | |
|       while (!endOfFile()) {
 | |
|         let line = nextLine();
 | |
|         if (chunkEnd.test(line)) {
 | |
|           removeLeadingEmptyLines(property);
 | |
|           return property;
 | |
|         } else if (line.startsWith(jsonLinePrefix)) {
 | |
|           line = JSON.parse(line.slice(jsonLinePrefix.length));
 | |
|           property.push(line);
 | |
|         } else {
 | |
|           property.push(line);
 | |
|         }
 | |
|       }
 | |
|       throw `Unexpected end of the file while reading a property.
 | |
|         ${this.filePath()}`;
 | |
|     }
 | |
| 
 | |
|     function readProperties() {
 | |
|       let properties = {};
 | |
|       let chunkStart = /(<!|\/\*)--(\w+)--/;
 | |
| 
 | |
|       while (!endOfFile()) {
 | |
|         let line = nextLine();
 | |
|         if (chunkEnd.test(line)) {
 | |
|           index--; // must to do step back
 | |
|           return properties;
 | |
|         } else if (chunkStart.test(line)) {
 | |
|           let chunkName = line.match(chunkStart)[2];
 | |
|           if (chunkName === 'description') {
 | |
|             properties[chunkName] = readDescription();
 | |
|           } else if (chunkName === 'translations') {
 | |
|             properties[chunkName] = readTranslations();
 | |
|           } else if (chunkName === 'files') {
 | |
|             properties[chunkName] = readFiles();
 | |
|           } else {
 | |
|             properties[chunkName] = readProperty();
 | |
|             if (chunkName === 'releasedOn' || chunkName === 'title') {
 | |
|               properties[chunkName] = properties[chunkName].join('\n');
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       return properties;
 | |
|     }
 | |
| 
 | |
|     let chunks = readProperties();
 | |
| 
 | |
|     if (!endOfFile()) {
 | |
|       throw `Unexpected content of file: ${this.filePath()}`;
 | |
|     }
 | |
| 
 | |
|     // hack to deal with solutions field being an array of a single string
 | |
|     // instead of an array of lines like some other fields
 | |
|     if (chunks.solutions) {
 | |
|       chunks.solutions = [chunks.solutions.join('\n')];
 | |
|       removeLeadingEmptyLines(chunks.solutions);
 | |
|     }
 | |
| 
 | |
|     // console.log(JSON.stringify(chunks, null, 2));
 | |
|     return chunks;
 | |
|   }
 | |
| }
 | |
| 
 | |
| export { ChallengeFile };
 | |
| 
 | |
| class UnpackedChallenge {
 | |
|   constructor(targetDir, challengeJson, index) {
 | |
|     this.targetDir = targetDir;
 | |
|     this.index = index;
 | |
| 
 | |
|     // todo: merge challengeJson properties into this object?
 | |
|     this.challenge = challengeJson;
 | |
| 
 | |
|     // extract names of block and superblock from path
 | |
|     // note: this is a bit redundant with the
 | |
|     // fileName,superBlock,superOrder fields
 | |
|     // that getChallenges() adds to the challenge JSON
 | |
|     let targetDirPath = path.parse(targetDir);
 | |
|     let parentDirPath = path.parse(targetDirPath.dir);
 | |
|     // superBlockName e.g. "03-front-end-libraries"
 | |
|     this.superBlockName = parentDirPath.base;
 | |
|     // challengeBlockName e.g. "bootstrap"
 | |
|     this.challengeBlockName = targetDirPath.base;
 | |
|   }
 | |
| 
 | |
|   unpack() {
 | |
|     this.challengeFile().write(this.unpackedHTML());
 | |
|   }
 | |
| 
 | |
|   challengeFile() {
 | |
|     return new ChallengeFile(this.targetDir, this.baseName(), '.html');
 | |
|   }
 | |
| 
 | |
|   baseName() {
 | |
|     // eslint-disable-next-line no-nested-ternary
 | |
|     let prefix =
 | |
|       (this.index < 10 ? '00' : this.index < 100 ? '0' : '') + this.index;
 | |
|     return `${prefix}-${dasherize(this.challenge.title)}-${this.challenge.id}`;
 | |
|   }
 | |
| 
 | |
|   expandedDescription(description) {
 | |
|     let out = [];
 | |
|     description.forEach(part => {
 | |
|       if (_.isString(part)) {
 | |
|         out.push(part.toString());
 | |
|         out.push(paragraphBreak);
 | |
|       } else {
 | |
|         // Descriptions are weird since sometimes they're text and sometimes
 | |
|         // they're "steps" which appear one at a time with optional pix and
 | |
|         // captions and links, or "questions" with choices and explanations...
 | |
|         // For now we preserve non-string descriptions via JSON but this is
 | |
|         // not a great solution.
 | |
|         // It would be better if "steps" and "description" were separate fields.
 | |
|         // For the record, the (unnamed) fields in step are:
 | |
|         // 0: image URL
 | |
|         // 1: caption
 | |
|         // 2: text
 | |
|         // 3: link URL
 | |
|         out.push(jsonLinePrefix + JSON.stringify(part));
 | |
|       }
 | |
|     });
 | |
| 
 | |
|     if (out[out.length - 1] === paragraphBreak) {
 | |
|       out.pop();
 | |
|     }
 | |
|     return out;
 | |
|   }
 | |
| 
 | |
|   expandedTests(tests) {
 | |
|     if (!tests) {
 | |
|       return [];
 | |
|     }
 | |
|     let out = [];
 | |
|     tests.forEach(test => {
 | |
|       if (_.isString(test)) {
 | |
|         out.push(test);
 | |
|       } else {
 | |
|         // todo: figure out what to do about these id-title challenge links
 | |
|         out.push(jsonLinePrefix + JSON.stringify(test));
 | |
|       }
 | |
|     });
 | |
|     return out;
 | |
|   }
 | |
| 
 | |
|   unpackedHTML() {
 | |
|     let text = [];
 | |
|     text.push('<html>');
 | |
|     text.push('<head>');
 | |
|     text.push('<link rel="stylesheet" href="../../../unpacked.css">');
 | |
|     text.push('<!-- shim to enable running the tests in-browser -->');
 | |
|     text.push('<script src="../../unpacked-bundle.js"></script>');
 | |
|     text.push('</head>');
 | |
|     text.push('<body>');
 | |
|     text.push(`<h1>${this.challenge.title}</h1>`);
 | |
|     text.push(`<p>This is the <b>unpacked</b> version of
 | |
|         <code>${this.superBlockName}/${this.challengeBlockName}</code>
 | |
|         (challenge id <code>${this.challenge.id}</code>).</p>`);
 | |
|     text.push('<p>Open the JavaScript console to see test results.</p>');
 | |
| 
 | |
|     text.push(`<p>Edit this HTML file (between <!-- marks only!)
 | |
|        and run <code>npm run repack</code>
 | |
|        to incorporate your changes into the challenge database.</p>`);
 | |
| 
 | |
|     text.push('<h2>Title</h2>');
 | |
|     text.push('<!--title-->');
 | |
|     text.push(this.challenge.title);
 | |
|     text.push('<!--end-->');
 | |
| 
 | |
|     text.push('');
 | |
|     text.push('<h2>Description</h2>');
 | |
|     text.push('<div class="unpacked description">');
 | |
|     text.push('<!--description-->');
 | |
|     if (this.challenge.description.length) {
 | |
|       text.push(
 | |
|         this.expandedDescription(this.challenge.description).join('\n')
 | |
|       );
 | |
|     }
 | |
|     text.push('<!--end-->');
 | |
|     text.push('</div>');
 | |
|     text.push('');
 | |
|     text.push('<h2>Translations</h2>');
 | |
|     text.push(`
 | |
|       <p>Format of translation unit:</p>
 | |
|         <p>
 | |
|           <code>
 | |
|             <!--<em>language-code</em>--><br>
 | |
|             <!--<em>title</em>--><br>
 | |
|             <em>Title</em><br>
 | |
|             <!--<em>end</em>--><br>
 | |
|             <!--<em>description</em>--><br>
 | |
|             <em>Description</em><br>
 | |
|             <!--<em>end</em>--><br>
 | |
|             <!--<em>end</em>--><br>
 | |
|           </code>
 | |
|         </p>`);
 | |
|     text.push('<div class="unpacked">');
 | |
|     text.push('<!--translations-->');
 | |
|     if (this.challenge.hasOwnProperty('translations')) {
 | |
|       const translations = this.challenge.translations;
 | |
|       const keys = Object.keys(translations);
 | |
|       if (keys) {
 | |
|         keys.forEach(lang => {
 | |
|           text.push(`<h2>${lang}</h2>`);
 | |
|           text.push(`<!--${lang}-->`);
 | |
|           const translation = translations[lang];
 | |
|           if (translation.title) {
 | |
|             text.push('<h3>Title</h3>');
 | |
|             text.push('<!--title-->');
 | |
|             text.push(translation.title);
 | |
|             text.push('<!--end-->');
 | |
|           }
 | |
|           if (translation.description && translation.description.length) {
 | |
|             text.push('<h3>Description</h3>');
 | |
|             text.push('<!--description-->');
 | |
|             text.push(
 | |
|               this.expandedDescription(translation.description).join('\n')
 | |
|             );
 | |
|             text.push('<!--end-->');
 | |
|           }
 | |
|           text.push('<!--end-->');
 | |
|         });
 | |
|       }
 | |
|     }
 | |
|     text.push('<!--end-->');
 | |
|     text.push('</div>');
 | |
| 
 | |
|     text.push('');
 | |
|     text.push('<h2>Files</h2>');
 | |
|     text.push(`
 | |
|       <p>Format of file:</p>
 | |
|         <p>
 | |
|           <code>
 | |
|             <!--<em>name.ext</em>--><br>
 | |
|             <!--<em>contents</em>--><br>
 | |
|             <em>Contents</em><br>
 | |
|             <!--<em>end</em>--><br>
 | |
|             <!--<em>head</em>--><br>
 | |
|             <em>Head</em><br>
 | |
|             <!--<em>end</em>--><br>
 | |
|             <!--<em>tail</em>--><br>
 | |
|             <em>Tail</em><br>
 | |
|             <!--<em>end</em>--><br>
 | |
|             <!--<em>end</em>--><br>
 | |
|           </code>
 | |
|         </p>`);
 | |
|     text.push('<div class="unpacked">');
 | |
|     text.push('<!--files-->');
 | |
|     if (this.challenge.files) {
 | |
|       Object.keys(this.challenge.files).forEach(key => {
 | |
|         let file = this.challenge.files[key];
 | |
|         let { contents, head, tail } = file;
 | |
|         text.push(`<h3>${file.name + '.' + file.ext}</h3>`);
 | |
|         text.push(`<!--${file.name + '.' + file.ext}-->`);
 | |
|         text.push('<h4>Contents</h4>');
 | |
|         text.push('<pre>');
 | |
|         text.push(_.escape(contents.join('\n')));
 | |
|         text.push('</pre>');
 | |
|         text.push('<pre style="display: none;">');
 | |
|         text.push('<!--contents-->');
 | |
|         text.push(contents.join('\n'));
 | |
|         text.push('<!--end-->');
 | |
|         text.push('</pre>');
 | |
| 
 | |
|         text.push('<h4>Head</h4>');
 | |
|         text.push('<pre>');
 | |
|         text.push(_.escape(head.join('\n')));
 | |
|         text.push('</pre>');
 | |
|         text.push('<pre style="display: none;">');
 | |
|         text.push('<!--head-->');
 | |
|         text.push(head.join('\n'));
 | |
|         text.push('<!--end-->');
 | |
|         text.push('</pre>');
 | |
| 
 | |
|         text.push('<h4>Tail</h4>');
 | |
|         text.push('<pre>');
 | |
|         text.push(_.escape(tail.join('\n')));
 | |
|         text.push('</pre>');
 | |
|         text.push('<pre style="display: none;">');
 | |
|         text.push('<!--tail-->');
 | |
|         text.push(tail.join('\n'));
 | |
|         text.push('<!--end-->');
 | |
|         text.push('</pre>');
 | |
|         text.push('<!--end-->');
 | |
|       });
 | |
|     }
 | |
|     text.push('<!--end-->');
 | |
|     text.push('</div>');
 | |
| 
 | |
|     text.push('');
 | |
|     text.push('<h2>Solution</h2>');
 | |
|     text.push(
 | |
|       '<!--solutions--><script class="unpacked solution" id="solution">'
 | |
|     );
 | |
|     // Note: none of the challenges have more than one solution
 | |
|     // todo: should we deal with multiple solutions or not?
 | |
|     if (this.challenge.solutions && this.challenge.solutions.length > 0) {
 | |
|       let solution = this.challenge.solutions[0];
 | |
|       text.push(solution);
 | |
|     }
 | |
|     text.push('</script><!--end-->');
 | |
| 
 | |
|     text.push('');
 | |
|     text.push('<h2>Tests</h2>');
 | |
|     text.push('<script class="unpacked tests">');
 | |
|     text.push(
 | |
|       `test(\'${this.challenge.title} challenge tests\', ` + 'function(t) {'
 | |
|     );
 | |
|     text.push('let assert = addAssertsToTapTest(t);');
 | |
|     text.push("let code = document.getElementById('solution').innerText;");
 | |
|     text.push(
 | |
|       't.plan(' +
 | |
|         (this.challenge.tests ? this.challenge.tests.length : 0) +
 | |
|         ');'
 | |
|     );
 | |
|     text.push('/*--tests--*/');
 | |
|     text.push(this.expandedTests(this.challenge.tests).join('\n'));
 | |
|     text.push('/*--end--*/');
 | |
|     text.push('});');
 | |
|     text.push('</script>');
 | |
| 
 | |
|     text.push('');
 | |
|     text.push('</body>');
 | |
|     text.push('</html>');
 | |
|     return text;
 | |
|   }
 | |
| }
 | |
| 
 | |
| export { UnpackedChallenge };
 |