mirror of
https://github.com/JuanCanham/HackMyResume.git
synced 2024-11-22 16:30:11 +00:00
commit
64db1a654e
@ -48,7 +48,7 @@ To use HackMyResume you'll need to create a valid resume in either
|
|||||||
[FRESH][fresca] or [JSON Resume][6] format. Then you can start using the command
|
[FRESH][fresca] or [JSON Resume][6] format. Then you can start using the command
|
||||||
line tool. There are four basic commands you should be aware of:
|
line tool. There are four basic commands you should be aware of:
|
||||||
|
|
||||||
- `**build**` generates resumes in HTML, Word, Markdown, PDF, and other formats.
|
- **build** generates resumes in HTML, Word, Markdown, PDF, and other formats.
|
||||||
Use it when you need to submit, upload, print, or email resumes in specific
|
Use it when you need to submit, upload, print, or email resumes in specific
|
||||||
formats.
|
formats.
|
||||||
|
|
||||||
@ -58,7 +58,7 @@ formats.
|
|||||||
hackmyresume BUILD r1.json r2.json TO out/rez.html out/rez.md foo/rez.all
|
hackmyresume BUILD r1.json r2.json TO out/rez.html out/rez.md foo/rez.all
|
||||||
```
|
```
|
||||||
|
|
||||||
- `**new**` creates a new resume in FRESH or JSON Resume format.
|
- **new** creates a new resume in FRESH or JSON Resume format.
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# hackmyresume NEW <OUTPUTS> [-f <FORMAT>]
|
# hackmyresume NEW <OUTPUTS> [-f <FORMAT>]
|
||||||
@ -67,7 +67,7 @@ formats.
|
|||||||
hackmyresume NEW r1.json r2.json -f jrs
|
hackmyresume NEW r1.json r2.json -f jrs
|
||||||
```
|
```
|
||||||
|
|
||||||
- `**convert**` converts your source resume between FRESH and JSON Resume
|
- **convert** converts your source resume between FRESH and JSON Resume
|
||||||
formats.
|
formats.
|
||||||
Use it to convert between the two formats to take advantage of tools and
|
Use it to convert between the two formats to take advantage of tools and
|
||||||
services.
|
services.
|
||||||
@ -78,7 +78,7 @@ services.
|
|||||||
hackmyresume CONVERT 1.json 2.json 3.json TO out/1.json out/2.json out/3.json
|
hackmyresume CONVERT 1.json 2.json 3.json TO out/1.json out/2.json out/3.json
|
||||||
```
|
```
|
||||||
|
|
||||||
- `**validate**` validates the specified resume against either the FRESH or JSON
|
- **validate** validates the specified resume against either the FRESH or JSON
|
||||||
Resume schema. Use it to make sure your resume data is sufficient and complete.
|
Resume schema. Use it to make sure your resume data is sufficient and complete.
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "hackmyresume",
|
"name": "hackmyresume",
|
||||||
"version": "1.1.0",
|
"version": "1.2.0",
|
||||||
"description": "Generate polished résumés and CVs in HTML, Markdown, LaTeX, MS Word, PDF, plain text, JSON, XML, YAML, smoke signal, and carrier pigeon.",
|
"description": "Generate polished résumés and CVs in HTML, Markdown, LaTeX, MS Word, PDF, plain text, JSON, XML, YAML, smoke signal, and carrier pigeon.",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
|
19
src/core/default-formats.js
Normal file
19
src/core/default-formats.js
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
(function(){
|
||||||
|
|
||||||
|
var FLUENT = require('../hackmyapi');
|
||||||
|
|
||||||
|
/**
|
||||||
|
Supported resume formats.
|
||||||
|
*/
|
||||||
|
module.exports = [
|
||||||
|
{ name: 'html', ext: 'html', gen: new FLUENT.HtmlGenerator() },
|
||||||
|
{ name: 'txt', ext: 'txt', gen: new FLUENT.TextGenerator() },
|
||||||
|
{ name: 'doc', ext: 'doc', fmt: 'xml', gen: new FLUENT.WordGenerator() },
|
||||||
|
{ name: 'pdf', ext: 'pdf', fmt: 'html', is: false, gen: new FLUENT.HtmlPdfGenerator() },
|
||||||
|
{ name: 'md', ext: 'md', fmt: 'txt', gen: new FLUENT.MarkdownGenerator() },
|
||||||
|
{ name: 'json', ext: 'json', gen: new FLUENT.JsonGenerator() },
|
||||||
|
{ name: 'yml', ext: 'yml', fmt: 'yml', gen: new FLUENT.JsonYamlGenerator() },
|
||||||
|
{ name: 'latex', ext: 'tex', fmt: 'latex', gen: new FLUENT.LaTeXGenerator() }
|
||||||
|
];
|
||||||
|
|
||||||
|
}());
|
13
src/core/default-options.js
Normal file
13
src/core/default-options.js
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
(function(){
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
theme: 'modern',
|
||||||
|
prettify: { // ← See https://github.com/beautify-web/js-beautify#options
|
||||||
|
indent_size: 2,
|
||||||
|
unformatted: ['em','strong'],
|
||||||
|
max_char: 80, // ← See lib/html.js in above-linked repo
|
||||||
|
//wrap_line_length: 120, ← Don't use this
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
}());
|
13
src/core/load-source-resumes.js
Normal file
13
src/core/load-source-resumes.js
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
(function(){
|
||||||
|
|
||||||
|
var FRESHResume = require('../core/fresh-resume');
|
||||||
|
|
||||||
|
module.exports = function loadSourceResumes( src, log, fn ) {
|
||||||
|
return src.map( function( res ) {
|
||||||
|
log( 'Reading '.info + 'SOURCE'.infoBold + ' resume: '.info +
|
||||||
|
res.cyan.bold );
|
||||||
|
return (fn && fn(res)) || (new FRESHResume()).open( res );
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
}());
|
@ -134,7 +134,7 @@ Definition of the Theme class.
|
|||||||
action: 'transform',
|
action: 'transform',
|
||||||
path: absPath,
|
path: absPath,
|
||||||
major: isMajor,
|
major: isMajor,
|
||||||
orgPath: PATH.relative(that.folder, absPath),
|
orgPath: PATH.relative(tplFolder, absPath),
|
||||||
ext: pathInfo.ext.slice(1),
|
ext: pathInfo.ext.slice(1),
|
||||||
title: friendlyName( outFmt ),
|
title: friendlyName( outFmt ),
|
||||||
pre: outFmt,
|
pre: outFmt,
|
||||||
|
@ -161,7 +161,7 @@ Generic template helper definitions for FluentCV.
|
|||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
idx = Math.min( lvl / 2, 4 );
|
idx = Math.min( lvl / 2, 4 );
|
||||||
idx = Math.max( 0, intVal );
|
idx = Math.max( 0, idx );
|
||||||
}
|
}
|
||||||
return idx;
|
return idx;
|
||||||
}
|
}
|
||||||
|
@ -22,16 +22,6 @@ Definition of the HTMLGenerator class.
|
|||||||
the HTML resume prior to saving.
|
the HTML resume prior to saving.
|
||||||
*/
|
*/
|
||||||
onBeforeSave: function( info ) {
|
onBeforeSave: function( info ) {
|
||||||
var cssSrc = PATH.join( info.theme.folder, 'src', '*.css' )
|
|
||||||
, outFolder = PATH.parse( info.outputFile ).dir, that = this;
|
|
||||||
|
|
||||||
info.theme.cssFiles.forEach( function( f ) {
|
|
||||||
var fi = PATH.parse( f.path );
|
|
||||||
FS.copySync( f.path, PATH.join( outFolder, fi.base ), { clobber: true }, function( e ) {
|
|
||||||
throw { fluenterror: that.codes.copyCss, data: [cssSrc,cssDst] };
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
return this.opts.prettify ?
|
return this.opts.prettify ?
|
||||||
HTML.prettyPrint( info.mk, this.opts.prettify ) : info.mk;
|
HTML.prettyPrint( info.mk, this.opts.prettify ) : info.mk;
|
||||||
}
|
}
|
||||||
|
@ -24,7 +24,7 @@ Definition of the HtmlPdfGenerator class.
|
|||||||
*/
|
*/
|
||||||
onBeforeSave: function( info ) {
|
onBeforeSave: function( info ) {
|
||||||
pdf( info.mk, info.outputFile );
|
pdf( info.mk, info.outputFile );
|
||||||
return info.mk;
|
return null; // halt further processing
|
||||||
}
|
}
|
||||||
|
|
||||||
});
|
});
|
||||||
|
@ -38,7 +38,9 @@ Definition of the TemplateGenerator class.
|
|||||||
raw: function( txt ) { return txt; },
|
raw: function( txt ) { return txt; },
|
||||||
xml: function( txt ) { return XML(txt); },
|
xml: function( txt ) { return XML(txt); },
|
||||||
md: function( txt ) { return MD( txt || '' ); },
|
md: function( txt ) { return MD( txt || '' ); },
|
||||||
mdin: function( txt ) { return MD(txt || '' ).replace(/^\s*<p>|<\/p>\s*$/gi, ''); },
|
mdin: function( txt ) {
|
||||||
|
return MD(txt || '' ).replace(/^\s*<p>|<\/p>\s*$/gi, '');
|
||||||
|
},
|
||||||
lower: function( txt ) { return txt.toLowerCase(); },
|
lower: function( txt ) { return txt.toLowerCase(); },
|
||||||
link: function( name, url ) { return url ?
|
link: function( name, url ) { return url ?
|
||||||
'<a href="' + url + '">' + name + '</a>' : name; }
|
'<a href="' + url + '">' + name + '</a>' : name; }
|
||||||
@ -69,24 +71,14 @@ Definition of the TemplateGenerator class.
|
|||||||
},
|
},
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
invoke: function( rez, themeMarkup, cssInfo, opts ) {
|
|
||||||
this.opts = EXTEND( true, {}, _defaultOpts, opts );
|
|
||||||
mk = this.single( rez, themeMarkup, this.format, cssInfo, { } );
|
|
||||||
this.onBeforeSave && (mk = this.onBeforeSave( mk, themeFile, f ));
|
|
||||||
return mk;
|
|
||||||
},
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Default generation method for template-based generators.
|
String-based template generation method.
|
||||||
@method generate
|
@method invoke
|
||||||
@param rez A FreshResume object.
|
@param rez A FreshResume object.
|
||||||
@param f Full path to the output resume file to generate.
|
|
||||||
@param opts Generator options.
|
@param opts Generator options.
|
||||||
|
@returns An array of strings representing generated output files.
|
||||||
*/
|
*/
|
||||||
generate: function( rez, f, opts ) {
|
invoke: function( rez, opts ) {
|
||||||
|
|
||||||
// Carry over options
|
// Carry over options
|
||||||
this.opts = EXTEND( true, { }, _defaultOpts, opts );
|
this.opts = EXTEND( true, { }, _defaultOpts, opts );
|
||||||
@ -96,20 +88,81 @@ Definition of the TemplateGenerator class.
|
|||||||
var theme = themeInfo.theme;
|
var theme = themeInfo.theme;
|
||||||
var tFolder = themeInfo.folder;
|
var tFolder = themeInfo.folder;
|
||||||
var tplFolder = PATH.join( tFolder, 'src' );
|
var tplFolder = PATH.join( tFolder, 'src' );
|
||||||
|
var curFmt = theme.getFormat( this.format );
|
||||||
|
var that = this;
|
||||||
|
|
||||||
|
// "Generate": process individual files within the theme
|
||||||
|
return {
|
||||||
|
files: curFmt.files.map( function( tplInfo ) {
|
||||||
|
return {
|
||||||
|
info: tplInfo,
|
||||||
|
data: tplInfo.action === 'transform' ?
|
||||||
|
transform.call( that, rez, tplInfo, theme ) : undefined
|
||||||
|
};
|
||||||
|
}).filter(function(item){ return item !== null; }),
|
||||||
|
themeInfo: themeInfo
|
||||||
|
};
|
||||||
|
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
File-based template generation method.
|
||||||
|
@method generate
|
||||||
|
@param rez A FreshResume object.
|
||||||
|
@param f Full path to the output resume file to generate.
|
||||||
|
@param opts Generator options.
|
||||||
|
*/
|
||||||
|
generate: function( rez, f, opts ) {
|
||||||
|
|
||||||
|
// Call the generation method
|
||||||
|
var genInfo = this.invoke( rez, opts );
|
||||||
|
|
||||||
|
// Carry over options
|
||||||
|
this.opts = EXTEND( true, { }, _defaultOpts, opts );
|
||||||
|
|
||||||
|
// Load the theme
|
||||||
|
var themeInfo = genInfo.themeInfo;
|
||||||
|
var theme = themeInfo.theme;
|
||||||
|
var tFolder = themeInfo.folder;
|
||||||
|
var tplFolder = PATH.join( tFolder, 'src' );
|
||||||
var outFolder = PATH.parse(f).dir;
|
var outFolder = PATH.parse(f).dir;
|
||||||
var curFmt = theme.getFormat( this.format );
|
var curFmt = theme.getFormat( this.format );
|
||||||
var that = this;
|
var that = this;
|
||||||
|
|
||||||
// "Generate": process individual files within the theme
|
// "Generate": process individual files within the theme
|
||||||
curFmt.files.forEach(function(tplInfo){
|
genInfo.files.forEach(function( file ){
|
||||||
if( tplInfo.action === 'transform' ) {
|
|
||||||
transform.call( that, rez, f, tplInfo, theme, outFolder );
|
var thisFilePath;
|
||||||
|
|
||||||
|
if( file.info.action === 'transform' ) {
|
||||||
|
thisFilePath = PATH.join( outFolder, file.info.orgPath );
|
||||||
|
try {
|
||||||
|
if( that.onBeforeSave ) {
|
||||||
|
file.data = that.onBeforeSave({
|
||||||
|
theme: theme,
|
||||||
|
outputFile: (file.info.major ? f : thisFilePath),
|
||||||
|
mk: file.data
|
||||||
|
});
|
||||||
|
if( !file.data ) return; // PDF etc
|
||||||
}
|
}
|
||||||
else if( tplInfo.action === null && theme.explicit ) {
|
var fileName = file.info.major ? f : thisFilePath;
|
||||||
var thisFilePath = PATH.join(outFolder, tplInfo.orgPath);
|
MKDIRP.sync( PATH.dirname( fileName ) );
|
||||||
|
FS.writeFileSync( fileName, file.data,
|
||||||
|
{ encoding: 'utf8', flags: 'w' } );
|
||||||
|
that.onAfterSave && that.onAfterSave(
|
||||||
|
{ outputFile: fileName, mk: file.data } );
|
||||||
|
}
|
||||||
|
catch(ex) {
|
||||||
|
console.log(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if( file.info.action === null/* && theme.explicit*/ ) {
|
||||||
|
thisFilePath = PATH.join( outFolder, file.info.orgPath );
|
||||||
try {
|
try {
|
||||||
MKDIRP.sync( PATH.dirname(thisFilePath) );
|
MKDIRP.sync( PATH.dirname(thisFilePath) );
|
||||||
FS.copySync( tplInfo.path, thisFilePath );
|
FS.copySync( file.info.path, thisFilePath );
|
||||||
}
|
}
|
||||||
catch(ex) {
|
catch(ex) {
|
||||||
console.log(ex);
|
console.log(ex);
|
||||||
@ -142,8 +195,10 @@ Definition of the TemplateGenerator class.
|
|||||||
*/
|
*/
|
||||||
single: function( json, jst, format, cssInfo, opts, theme ) {
|
single: function( json, jst, format, cssInfo, opts, theme ) {
|
||||||
this.opts.freezeBreaks && ( jst = freeze(jst) );
|
this.opts.freezeBreaks && ( jst = freeze(jst) );
|
||||||
|
|
||||||
var eng = require( '../eng/' + theme.engine + '-generator' );
|
var eng = require( '../eng/' + theme.engine + '-generator' );
|
||||||
var result = eng.generate( json, jst, format, cssInfo, opts, theme );
|
var result = eng.generate( json, jst, format, cssInfo, opts, theme );
|
||||||
|
|
||||||
this.opts.freezeBreaks && ( result = unfreeze(result) );
|
this.opts.freezeBreaks && ( result = unfreeze(result) );
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@ -188,18 +243,15 @@ Definition of the TemplateGenerator class.
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
function transform( rez, tplInfo, theme ) {
|
||||||
Transform a single subfile.
|
|
||||||
*/
|
|
||||||
function transform( rez, f, tplInfo, theme, outFolder ) {
|
|
||||||
var cssInfo = { file: tplInfo.css ? tplInfo.cssPath : null, data: tplInfo.css || null };
|
|
||||||
var mk = this.single( rez, tplInfo.data, this.format, cssInfo, this.opts, theme );
|
|
||||||
this.onBeforeSave && (mk = this.onBeforeSave( { mk: mk, theme: theme, outputFile: f } ));
|
|
||||||
var thisFilePath = PATH.join( outFolder, tplInfo.orgPath );
|
|
||||||
try {
|
try {
|
||||||
MKDIRP.sync( PATH.dirname( tplInfo.major ? f : thisFilePath) );
|
var cssInfo = {
|
||||||
FS.writeFileSync( tplInfo.major ? f : thisFilePath, mk, { encoding: 'utf8', flags: 'w' } );
|
file: tplInfo.css ? tplInfo.cssPath : null,
|
||||||
this.onAfterSave && (mk = this.onAfterSave( { outputFile: (tplInfo.major ? f : thisFilePath), mk: mk } ));
|
data: tplInfo.css || null
|
||||||
|
};
|
||||||
|
|
||||||
|
return this.single( rez, tplInfo.data, this.format, cssInfo, this.opts,
|
||||||
|
theme );
|
||||||
}
|
}
|
||||||
catch(ex) {
|
catch(ex) {
|
||||||
console.log(ex);
|
console.log(ex);
|
||||||
|
323
src/hackmycmd.js
323
src/hackmycmd.js
@ -7,279 +7,9 @@ Internal resume generation logic for HackMyResume.
|
|||||||
(function() {
|
(function() {
|
||||||
module.exports = function () {
|
module.exports = function () {
|
||||||
|
|
||||||
var path = require( 'path' )
|
var unused = require('./utils/string')
|
||||||
, extend = require( './utils/extend' )
|
, PATH = require('path');
|
||||||
, unused = require('./utils/string')
|
|
||||||
, FS = require('fs')
|
|
||||||
, _ = require('underscore')
|
|
||||||
, FLUENT = require('./hackmyapi')
|
|
||||||
, PATH = require('path')
|
|
||||||
, MKDIRP = require('mkdirp')
|
|
||||||
//, COLORS = require('colors')
|
|
||||||
, rez, _log, _err;
|
|
||||||
|
|
||||||
/**
|
|
||||||
Given a source JSON resume, a destination resume path, and a theme file,
|
|
||||||
generate 0..N resumes in the desired formats.
|
|
||||||
@param src Path to the source JSON resume file: "rez/resume.json".
|
|
||||||
@param dst An array of paths to the target resume file(s).
|
|
||||||
@param theme Friendly name of the resume theme. Defaults to "modern".
|
|
||||||
@param logger Optional logging override.
|
|
||||||
*/
|
|
||||||
function generate( src, dst, opts, logger, errHandler ) {
|
|
||||||
|
|
||||||
_log = logger || console.log;
|
|
||||||
_err = errHandler || error;
|
|
||||||
|
|
||||||
//_opts = extend( true, _opts, opts );
|
|
||||||
_opts.theme = (opts.theme && opts.theme.toLowerCase().trim())|| 'modern';
|
|
||||||
_opts.prettify = opts.prettify === true ? _opts.prettify : false;
|
|
||||||
|
|
||||||
// Load input resumes...
|
|
||||||
if(!src || !src.length) { throw { fluenterror: 3 }; }
|
|
||||||
var sheets = loadSourceResumes( src );
|
|
||||||
|
|
||||||
// Merge input resumes...
|
|
||||||
var msg = '';
|
|
||||||
rez = _.reduceRight( sheets, function( a, b, idx ) {
|
|
||||||
msg += ((idx == sheets.length - 2) ?
|
|
||||||
'Merging '.gray+ a.imp.fileName : '') + ' onto '.gray + b.imp.fileName;
|
|
||||||
return extend( true, b, a );
|
|
||||||
});
|
|
||||||
msg && _log(msg);
|
|
||||||
|
|
||||||
// Verify the specified theme name/path
|
|
||||||
var relativeThemeFolder = '../node_modules/fluent-themes/themes';
|
|
||||||
var tFolder = PATH.resolve( __dirname, relativeThemeFolder, _opts.theme);
|
|
||||||
var exists = require('./utils/file-exists');
|
|
||||||
if (!exists( tFolder )) {
|
|
||||||
tFolder = PATH.resolve( _opts.theme );
|
|
||||||
if (!exists( tFolder )) {
|
|
||||||
throw { fluenterror: 1, data: _opts.theme };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Load the theme
|
|
||||||
var theTheme = new FLUENT.Theme().open( tFolder );
|
|
||||||
_opts.themeObj = theTheme;
|
|
||||||
_log( 'Applying '.info + theTheme.name.toUpperCase().infoBold +
|
|
||||||
(' theme (' +Object.keys(theTheme.formats).length + ' formats)').info);
|
|
||||||
|
|
||||||
// Expand output resumes... (can't use map() here)
|
|
||||||
var targets = [], that = this;
|
|
||||||
( (dst && dst.length && dst) || ['resume.all'] ).forEach( function(t) {
|
|
||||||
|
|
||||||
var to = path.resolve(t),
|
|
||||||
pa = path.parse(to),
|
|
||||||
fmat = pa.ext || '.all';
|
|
||||||
|
|
||||||
targets.push.apply(targets, fmat === '.all' ?
|
|
||||||
Object.keys( theTheme.formats ).map(function(k){
|
|
||||||
var z = theTheme.formats[k];
|
|
||||||
return { file: to.replace(/all$/g,z.outFormat), fmt: z };
|
|
||||||
}) : [{ file: to, fmt: theTheme.getFormat( fmat.slice(1) ) }]);
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
// Run the transformation!
|
|
||||||
var finished = targets.map( function(t) { return single(t, theTheme); });
|
|
||||||
|
|
||||||
// Don't send the client back empty-handed
|
|
||||||
return { sheet: rez, targets: targets, processed: finished };
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
Generate a single resume of a specific format.
|
|
||||||
@param f Full path to the destination resume to generate, for example,
|
|
||||||
"/foo/bar/resume.pdf" or "c:\foo\bar\resume.txt".
|
|
||||||
*/
|
|
||||||
function single( targInfo, theme ) {
|
|
||||||
try {
|
|
||||||
var f = targInfo.file
|
|
||||||
, fType = targInfo.fmt.outFormat
|
|
||||||
, fName = path.basename(f, '.' + fType)
|
|
||||||
, theFormat;
|
|
||||||
|
|
||||||
// If targInfo.fmt.files exists, this theme has an explicit "files"
|
|
||||||
// section in its theme.json file.
|
|
||||||
if( targInfo.fmt.files && targInfo.fmt.files.length ) {
|
|
||||||
|
|
||||||
_log( 'Generating '.useful +
|
|
||||||
targInfo.fmt.outFormat.toUpperCase().useful.bold +
|
|
||||||
' resume: '.useful + path.relative(process.cwd(), f ).replace(/\\/g,'/').useful.bold);
|
|
||||||
|
|
||||||
theFormat = _fmts.filter(
|
|
||||||
function(fmt) { return fmt.name === targInfo.fmt.outFormat; })[0];
|
|
||||||
MKDIRP.sync( path.dirname( f ) ); // Ensure dest folder exists;
|
|
||||||
theFormat.gen.generate( rez, f, _opts );
|
|
||||||
|
|
||||||
// targInfo.fmt.files.forEach( function( form ) {
|
|
||||||
//
|
|
||||||
// if( form.action === 'transform' ) {
|
|
||||||
// var theFormat = _fmts.filter( function( fmt ) {
|
|
||||||
// return fmt.name === targInfo.fmt.outFormat;
|
|
||||||
// })[0];
|
|
||||||
// MKDIRP.sync( path.dirname( f ) ); // Ensure dest folder exists;
|
|
||||||
// theFormat.gen.generate( rez, f, _opts );
|
|
||||||
// }
|
|
||||||
// else if( form.action === null ) {
|
|
||||||
// // Copy the file
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// });
|
|
||||||
|
|
||||||
}
|
|
||||||
// Otherwise the theme has no files section
|
|
||||||
else {
|
|
||||||
_log( 'Generating '.useful +
|
|
||||||
targInfo.fmt.outFormat.toUpperCase().useful.bold +
|
|
||||||
' resume: '.useful + path.relative(process.cwd(), f ).replace(/\\/g,'/').useful.bold);
|
|
||||||
|
|
||||||
theFormat = _fmts.filter(
|
|
||||||
function(fmt) { return fmt.name === targInfo.fmt.outFormat; })[0];
|
|
||||||
MKDIRP.sync( path.dirname( f ) ); // Ensure dest folder exists;
|
|
||||||
theFormat.gen.generate( rez, f, _opts );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch( ex ) {
|
|
||||||
_err( ex );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
Handle an exception.
|
|
||||||
*/
|
|
||||||
function error( ex ) {
|
|
||||||
throw ex;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
Validate 1 to N resumes in either FRESH or JSON Resume format.
|
|
||||||
*/
|
|
||||||
function validate( src, unused, opts, logger ) {
|
|
||||||
_log = logger || console.log;
|
|
||||||
if( !src || !src.length ) { throw { fluenterror: 6 }; }
|
|
||||||
var isValid = true;
|
|
||||||
|
|
||||||
var validator = require('is-my-json-valid');
|
|
||||||
var schemas = {
|
|
||||||
fresh: require('FRESCA'),
|
|
||||||
jars: require('./core/resume.json')
|
|
||||||
};
|
|
||||||
|
|
||||||
// Load input resumes...
|
|
||||||
var sheets = loadSourceResumes(src, function( res ) {
|
|
||||||
try {
|
|
||||||
return {
|
|
||||||
file: res,
|
|
||||||
raw: FS.readFileSync( res, 'utf8' )
|
|
||||||
};
|
|
||||||
}
|
|
||||||
catch( ex ) {
|
|
||||||
throw ex;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
sheets.forEach( function( rep ) {
|
|
||||||
|
|
||||||
var rez;
|
|
||||||
try {
|
|
||||||
rez = JSON.parse( rep.raw );
|
|
||||||
}
|
|
||||||
catch( ex ) {
|
|
||||||
_log('Validating '.info + rep.file.infoBold +
|
|
||||||
' against FRESH/JRS schema: '.info + 'ERROR!'.error.bold);
|
|
||||||
|
|
||||||
if (ex instanceof SyntaxError) {
|
|
||||||
// Invalid JSON
|
|
||||||
_log( '--> '.bold.red + rep.file.toUpperCase().red +
|
|
||||||
' contains invalid JSON. Unable to validate.'.red );
|
|
||||||
_log( (' INTERNAL: ' + ex).red );
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
|
|
||||||
_log(('ERROR: ' + ex.toString()).red.bold);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var isValid = false;
|
|
||||||
var style = 'useful';
|
|
||||||
var errors = [];
|
|
||||||
var fmt = rez.meta &&
|
|
||||||
(rez.meta.format === 'FRESH@0.1.0') ? 'fresh':'jars';
|
|
||||||
|
|
||||||
try {
|
|
||||||
|
|
||||||
var validate = validator( schemas[ fmt ], { // Note [1]
|
|
||||||
formats: {
|
|
||||||
date: /^\d{4}(?:-(?:0[0-9]{1}|1[0-2]{1})(?:-[0-9]{2})?)?$/
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
isValid = validate( rez );
|
|
||||||
if( !isValid ) {
|
|
||||||
style = 'warn';
|
|
||||||
errors = validate.errors;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
catch(ex) {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
_log( 'Validating '.info + rep.file.infoBold + ' against '.info +
|
|
||||||
fmt.replace('jars','JSON Resume').toUpperCase().infoBold +
|
|
||||||
' schema: '.info + (isValid ? 'VALID!' : 'INVALID')[style].bold );
|
|
||||||
|
|
||||||
errors.forEach(function(err,idx) {
|
|
||||||
_log( '--> '.bold.yellow +
|
|
||||||
(err.field.replace('data.','resume.').toUpperCase() + ' ' +
|
|
||||||
err.message).yellow );
|
|
||||||
});
|
|
||||||
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
Convert between FRESH and JRS formats.
|
|
||||||
*/
|
|
||||||
function convert( src, dst, opts, logger ) {
|
|
||||||
_log = logger || console.log;
|
|
||||||
if( !src || !src.length ) { throw { fluenterror: 6 }; }
|
|
||||||
if( !dst || !dst.length ) {
|
|
||||||
if( src.length === 1 ) { throw { fluenterror: 5 }; }
|
|
||||||
else if( src.length === 2 ) { dst = [ src[1] ]; src = [ src[0] ]; }
|
|
||||||
else { throw { fluenterror: 5 }; }
|
|
||||||
}
|
|
||||||
if( src && dst && src.length && dst.length && src.length !== dst.length ) {
|
|
||||||
throw { fluenterror: 7 };
|
|
||||||
}
|
|
||||||
var sheets = loadSourceResumes( src );
|
|
||||||
sheets.forEach(function(sheet, idx){
|
|
||||||
var sourceFormat = sheet.imp.orgFormat === 'JRS' ? 'JRS' : 'FRESH';
|
|
||||||
var targetFormat = sourceFormat === 'JRS' ? 'FRESH' : 'JRS';
|
|
||||||
_log( 'Converting '.useful + sheet.imp.fileName.useful.bold + (' (' +
|
|
||||||
sourceFormat + ') to ').useful + dst[0].useful.bold +
|
|
||||||
(' (' + targetFormat + ').').useful );
|
|
||||||
sheet.saveAs( dst[idx], targetFormat );
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
Create a new empty resume in either FRESH or JRS format.
|
|
||||||
*/
|
|
||||||
function create( src, dst, opts, logger ) {
|
|
||||||
_log = logger || console.log;
|
|
||||||
dst = src || ['resume.json'];
|
|
||||||
dst.forEach( function( t ) {
|
|
||||||
var safeFormat = opts.format.toUpperCase();
|
|
||||||
_log('Creating new '.useful +safeFormat.useful.bold +
|
|
||||||
' resume: '.useful + t.useful.bold);
|
|
||||||
MKDIRP.sync( path.dirname( t ) ); // Ensure dest folder exists;
|
|
||||||
FLUENT[ safeFormat + 'Resume' ].default().save( t );
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Display help documentation.
|
Display help documentation.
|
||||||
@ -289,55 +19,22 @@ Internal resume generation logic for HackMyResume.
|
|||||||
.useful.bold );
|
.useful.bold );
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadSourceResumes( src, fn ) {
|
|
||||||
return src.map( function( res ) {
|
|
||||||
_log( 'Reading '.info + 'SOURCE'.infoBold + ' resume: '.info +
|
|
||||||
res.cyan.bold );
|
|
||||||
return (fn && fn(res)) || (new FLUENT.FRESHResume()).open( res );
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
Supported resume formats.
|
|
||||||
*/
|
|
||||||
var _fmts = [
|
|
||||||
{ name: 'html', ext: 'html', gen: new FLUENT.HtmlGenerator() },
|
|
||||||
{ name: 'txt', ext: 'txt', gen: new FLUENT.TextGenerator() },
|
|
||||||
{ name: 'doc', ext: 'doc', fmt: 'xml', gen: new FLUENT.WordGenerator() },
|
|
||||||
{ name: 'pdf', ext: 'pdf', fmt: 'html', is: false, gen: new FLUENT.HtmlPdfGenerator() },
|
|
||||||
{ name: 'md', ext: 'md', fmt: 'txt', gen: new FLUENT.MarkdownGenerator() },
|
|
||||||
{ name: 'json', ext: 'json', gen: new FLUENT.JsonGenerator() },
|
|
||||||
{ name: 'yml', ext: 'yml', fmt: 'yml', gen: new FLUENT.JsonYamlGenerator() },
|
|
||||||
{ name: 'latex', ext: 'tex', fmt: 'latex', gen: new FLUENT.LaTeXGenerator() }
|
|
||||||
];
|
|
||||||
|
|
||||||
/**
|
|
||||||
Default HackMyResume options.
|
|
||||||
*/
|
|
||||||
var _opts = {
|
|
||||||
theme: 'modern',
|
|
||||||
prettify: { // ← See https://github.com/beautify-web/js-beautify#options
|
|
||||||
indent_size: 2,
|
|
||||||
unformatted: ['em','strong'],
|
|
||||||
max_char: 80, // ← See lib/html.js in above-linked repo
|
|
||||||
//wrap_line_length: 120, ← Don't use this
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
Internal module interface. Used by FCV Desktop and HMR.
|
Internal module interface. Used by FCV Desktop and HMR.
|
||||||
*/
|
*/
|
||||||
return {
|
return {
|
||||||
verbs: {
|
verbs: {
|
||||||
build: generate,
|
generate: require('./verbs/generate'),
|
||||||
validate: validate,
|
build: require('./verbs/generate'),
|
||||||
convert: convert,
|
validate: require('./verbs/validate'),
|
||||||
new: create,
|
convert: require('./verbs/convert'),
|
||||||
|
create: require('./verbs/create'),
|
||||||
|
new: require('./verbs/create'),
|
||||||
help: help
|
help: help
|
||||||
},
|
},
|
||||||
lib: require('./hackmyapi'),
|
lib: require('./hackmyapi'),
|
||||||
options: _opts,
|
options: require('./core/default-options'),
|
||||||
formats: _fmts
|
formats: require('./core/default-formats')
|
||||||
};
|
};
|
||||||
|
|
||||||
}();
|
}();
|
||||||
|
@ -68,7 +68,7 @@ function main() {
|
|||||||
// Massage inputs and outputs
|
// Massage inputs and outputs
|
||||||
var src = a._.slice(1, splitAt === -1 ? undefined : splitAt );
|
var src = a._.slice(1, splitAt === -1 ? undefined : splitAt );
|
||||||
var dst = splitAt === -1 ? [] : a._.slice( splitAt + 1 );
|
var dst = splitAt === -1 ? [] : a._.slice( splitAt + 1 );
|
||||||
( splitAt === -1 ) && dst.push( src.pop() ); // Allow omitting TO keyword
|
( splitAt === -1 ) && src.length > 1 && dst.push( src.pop() ); // Allow omitting TO keyword
|
||||||
var parms = [ src, dst, opts, logMsg ];
|
var parms = [ src, dst, opts, logMsg ];
|
||||||
|
|
||||||
// Invoke the action
|
// Invoke the action
|
||||||
@ -108,9 +108,10 @@ function handleError( ex ) {
|
|||||||
}).join(', '.guide) + ").\n\n".guide + FS.readFileSync( PATH.join(__dirname, 'use.txt'), 'utf8' ).info.bold;
|
}).join(', '.guide) + ").\n\n".guide + FS.readFileSync( PATH.join(__dirname, 'use.txt'), 'utf8' ).info.bold;
|
||||||
break;
|
break;
|
||||||
//case 4: msg = title + '\n' + ; break;
|
//case 4: msg = title + '\n' + ; break;
|
||||||
case 5: msg = 'Please '.guide + 'specify the output resume file'.guide.bold + ' that should be created in the new format.'.guide; break;
|
case 5: msg = 'Please '.guide + 'specify the output resume file'.guide.bold + ' that should be created.'.guide; break;
|
||||||
case 6: msg = 'Please '.guide + 'specify a valid input resume'.guide.bold + ' in either FRESH or JSON Resume format.'.guide; break;
|
case 6: msg = 'Please '.guide + 'specify a valid input resume'.guide.bold + ' in either FRESH or JSON Resume format.'.guide; break;
|
||||||
case 7: msg = 'Please '.guide + 'specify an output file name'.guide.bold + ' for every input file you wish to convert.'.guide; break;
|
case 7: msg = 'Please '.guide + 'specify an output file name'.guide.bold + ' for every input file you wish to convert.'.guide; break;
|
||||||
|
case 8: msg = 'Please '.guide + 'specify the filename of the resume'.guide.bold + ' to create.'.guide; break;
|
||||||
}
|
}
|
||||||
exitCode = ex.fluenterror;
|
exitCode = ex.fluenterror;
|
||||||
|
|
||||||
|
30
src/verbs/convert.js
Normal file
30
src/verbs/convert.js
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
(function(){
|
||||||
|
|
||||||
|
var loadSourceResumes = require('../core/load-source-resumes');
|
||||||
|
|
||||||
|
/**
|
||||||
|
Convert between FRESH and JRS formats.
|
||||||
|
*/
|
||||||
|
module.exports = function convert( src, dst, opts, logger ) {
|
||||||
|
var _log = logger || console.log;
|
||||||
|
if( !src || !src.length ) { throw { fluenterror: 6 }; }
|
||||||
|
if( !dst || !dst.length ) {
|
||||||
|
if( src.length === 1 ) { throw { fluenterror: 5 }; }
|
||||||
|
else if( src.length === 2 ) { dst = [ src[1] ]; src = [ src[0] ]; }
|
||||||
|
else { throw { fluenterror: 5 }; }
|
||||||
|
}
|
||||||
|
if( src && dst && src.length && dst.length && src.length !== dst.length ) {
|
||||||
|
throw { fluenterror: 7 };
|
||||||
|
}
|
||||||
|
var sheets = loadSourceResumes( src, _log );
|
||||||
|
sheets.forEach(function(sheet, idx){
|
||||||
|
var sourceFormat = sheet.imp.orgFormat === 'JRS' ? 'JRS' : 'FRESH';
|
||||||
|
var targetFormat = sourceFormat === 'JRS' ? 'FRESH' : 'JRS';
|
||||||
|
_log( 'Converting '.useful + sheet.imp.fileName.useful.bold + (' (' +
|
||||||
|
sourceFormat + ') to ').useful + dst[0].useful.bold +
|
||||||
|
(' (' + targetFormat + ').').useful );
|
||||||
|
sheet.saveAs( dst[idx], targetFormat );
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
}());
|
22
src/verbs/create.js
Normal file
22
src/verbs/create.js
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
(function(){
|
||||||
|
|
||||||
|
var FLUENT = require('../hackmyapi')
|
||||||
|
, MKDIRP = require('mkdirp')
|
||||||
|
, PATH = require('path');
|
||||||
|
|
||||||
|
/**
|
||||||
|
Create a new empty resume in either FRESH or JRS format.
|
||||||
|
*/
|
||||||
|
module.exports = function create( src, dst, opts, logger ) {
|
||||||
|
var _log = logger || console.log;
|
||||||
|
if( !src || !src.length ) throw { fluenterror: 8 };
|
||||||
|
src.forEach( function( t ) {
|
||||||
|
var safeFormat = opts.format.toUpperCase();
|
||||||
|
_log('Creating new '.useful +safeFormat.useful.bold +
|
||||||
|
' resume: '.useful + t.useful.bold);
|
||||||
|
MKDIRP.sync( PATH.dirname( t ) ); // Ensure dest folder exists;
|
||||||
|
FLUENT[ safeFormat + 'Resume' ].default().save( t );
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
}());
|
149
src/verbs/generate.js
Normal file
149
src/verbs/generate.js
Normal file
@ -0,0 +1,149 @@
|
|||||||
|
(function() {
|
||||||
|
|
||||||
|
var PATH = require('path')
|
||||||
|
, MKDIRP = require('mkdirp')
|
||||||
|
, _opts = require('../core/default-options')
|
||||||
|
, FluentTheme = require('../core/theme')
|
||||||
|
, loadSourceResumes = require('../core/load-source-resumes')
|
||||||
|
, _ = require('underscore')
|
||||||
|
, _fmts = require('../core/default-formats')
|
||||||
|
, _err, _log, rez;
|
||||||
|
|
||||||
|
/**
|
||||||
|
Handle an exception.
|
||||||
|
*/
|
||||||
|
function error( ex ) {
|
||||||
|
throw ex;
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports =
|
||||||
|
|
||||||
|
/**
|
||||||
|
Given a source JSON resume, a destination resume path, and a theme file,
|
||||||
|
generate 0..N resumes in the desired formats.
|
||||||
|
@param src Path to the source JSON resume file: "rez/resume.json".
|
||||||
|
@param dst An array of paths to the target resume file(s).
|
||||||
|
@param theme Friendly name of the resume theme. Defaults to "modern".
|
||||||
|
@param logger Optional logging override.
|
||||||
|
*/
|
||||||
|
function generate( src, dst, opts, logger, errHandler ) {
|
||||||
|
|
||||||
|
_log = logger || console.log;
|
||||||
|
_err = errHandler || error;
|
||||||
|
|
||||||
|
//_opts = extend( true, _opts, opts );
|
||||||
|
_opts.theme = (opts.theme && opts.theme.toLowerCase().trim())|| 'modern';
|
||||||
|
_opts.prettify = opts.prettify === true ? _opts.prettify : false;
|
||||||
|
|
||||||
|
// Load input resumes...
|
||||||
|
if( !src || !src.length ) { throw { fluenterror: 3 }; }
|
||||||
|
var sheets = loadSourceResumes( src, _log );
|
||||||
|
|
||||||
|
// Merge input resumes...
|
||||||
|
var msg = '';
|
||||||
|
rez = _.reduceRight( sheets, function( a, b, idx ) {
|
||||||
|
msg += ((idx == sheets.length - 2) ?
|
||||||
|
'Merging '.gray+ a.imp.fileName : '') + ' onto '.gray + b.imp.fileName;
|
||||||
|
return extend( true, b, a );
|
||||||
|
});
|
||||||
|
msg && _log(msg);
|
||||||
|
|
||||||
|
// Verify the specified theme name/path
|
||||||
|
var relativeThemeFolder = '../../node_modules/fluent-themes/themes';
|
||||||
|
var tFolder = PATH.resolve( __dirname, relativeThemeFolder, _opts.theme);
|
||||||
|
var exists = require('../utils/file-exists');
|
||||||
|
if (!exists( tFolder )) {
|
||||||
|
tFolder = PATH.resolve( _opts.theme );
|
||||||
|
if (!exists( tFolder )) {
|
||||||
|
throw { fluenterror: 1, data: _opts.theme };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load the theme
|
||||||
|
var theTheme = new FluentTheme().open( tFolder );
|
||||||
|
_opts.themeObj = theTheme;
|
||||||
|
_log( 'Applying '.info + theTheme.name.toUpperCase().infoBold +
|
||||||
|
(' theme (' + Object.keys(theTheme.formats).length + ' formats)').info);
|
||||||
|
|
||||||
|
// Expand output resumes... (can't use map() here)
|
||||||
|
var targets = [], that = this;
|
||||||
|
( (dst && dst.length && dst) || ['resume.all'] ).forEach( function(t) {
|
||||||
|
|
||||||
|
var to = PATH.resolve(t),
|
||||||
|
pa = PATH.parse(to),
|
||||||
|
fmat = pa.ext || '.all';
|
||||||
|
|
||||||
|
targets.push.apply(targets, fmat === '.all' ?
|
||||||
|
Object.keys( theTheme.formats ).map(function(k){
|
||||||
|
var z = theTheme.formats[k];
|
||||||
|
return { file: to.replace(/all$/g,z.outFormat), fmt: z };
|
||||||
|
}) : [{ file: to, fmt: theTheme.getFormat( fmat.slice(1) ) }]);
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
// Run the transformation!
|
||||||
|
var finished = targets.map( function(t) { return single(t, theTheme); });
|
||||||
|
|
||||||
|
// Don't send the client back empty-handed
|
||||||
|
return { sheet: rez, targets: targets, processed: finished };
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
Generate a single resume of a specific format.
|
||||||
|
@param f Full path to the destination resume to generate, for example,
|
||||||
|
"/foo/bar/resume.pdf" or "c:\foo\bar\resume.txt".
|
||||||
|
*/
|
||||||
|
function single( targInfo, theme ) {
|
||||||
|
try {
|
||||||
|
var f = targInfo.file
|
||||||
|
, fType = targInfo.fmt.outFormat
|
||||||
|
, fName = PATH.basename(f, '.' + fType)
|
||||||
|
, theFormat;
|
||||||
|
|
||||||
|
// If targInfo.fmt.files exists, this theme has an explicit "files"
|
||||||
|
// section in its theme.json file.
|
||||||
|
if( targInfo.fmt.files && targInfo.fmt.files.length ) {
|
||||||
|
|
||||||
|
_log( 'Generating '.useful +
|
||||||
|
targInfo.fmt.outFormat.toUpperCase().useful.bold +
|
||||||
|
' resume: '.useful + PATH.relative(process.cwd(), f ).replace(/\\/g,'/').useful.bold);
|
||||||
|
|
||||||
|
theFormat = _fmts.filter(
|
||||||
|
function(fmt) { return fmt.name === targInfo.fmt.outFormat; })[0];
|
||||||
|
MKDIRP.sync( PATH.dirname( f ) ); // Ensure dest folder exists;
|
||||||
|
theFormat.gen.generate( rez, f, _opts );
|
||||||
|
|
||||||
|
// targInfo.fmt.files.forEach( function( form ) {
|
||||||
|
//
|
||||||
|
// if( form.action === 'transform' ) {
|
||||||
|
// var theFormat = _fmts.filter( function( fmt ) {
|
||||||
|
// return fmt.name === targInfo.fmt.outFormat;
|
||||||
|
// })[0];
|
||||||
|
// MKDIRP.sync( PATH.dirname( f ) ); // Ensure dest folder exists;
|
||||||
|
// theFormat.gen.generate( rez, f, _opts );
|
||||||
|
// }
|
||||||
|
// else if( form.action === null ) {
|
||||||
|
// // Copy the file
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// });
|
||||||
|
|
||||||
|
}
|
||||||
|
// Otherwise the theme has no files section
|
||||||
|
else {
|
||||||
|
_log( 'Generating '.useful +
|
||||||
|
targInfo.fmt.outFormat.toUpperCase().useful.bold +
|
||||||
|
' resume: '.useful + PATH.relative(process.cwd(), f ).replace(/\\/g,'/').useful.bold);
|
||||||
|
|
||||||
|
theFormat = _fmts.filter(
|
||||||
|
function(fmt) { return fmt.name === targInfo.fmt.outFormat; })[0];
|
||||||
|
MKDIRP.sync( PATH.dirname( f ) ); // Ensure dest folder exists;
|
||||||
|
theFormat.gen.generate( rez, f, _opts );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch( ex ) {
|
||||||
|
_err( ex );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}());
|
97
src/verbs/validate.js
Normal file
97
src/verbs/validate.js
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
(function() {
|
||||||
|
|
||||||
|
var FS = require('fs');
|
||||||
|
var loadSourceResumes = require('../core/load-source-resumes');
|
||||||
|
|
||||||
|
module.exports =
|
||||||
|
|
||||||
|
/**
|
||||||
|
Validate 1 to N resumes in either FRESH or JSON Resume format.
|
||||||
|
*/
|
||||||
|
function validate( src, unused, opts, logger ) {
|
||||||
|
var _log = logger || console.log;
|
||||||
|
if( !src || !src.length ) { throw { fluenterror: 6 }; }
|
||||||
|
var isValid = true;
|
||||||
|
|
||||||
|
var validator = require('is-my-json-valid');
|
||||||
|
var schemas = {
|
||||||
|
fresh: require('FRESCA'),
|
||||||
|
jars: require('../core/resume.json')
|
||||||
|
};
|
||||||
|
|
||||||
|
// Load input resumes...
|
||||||
|
var sheets = loadSourceResumes(src, _log, function( res ) {
|
||||||
|
try {
|
||||||
|
return {
|
||||||
|
file: res,
|
||||||
|
raw: FS.readFileSync( res, 'utf8' )
|
||||||
|
};
|
||||||
|
}
|
||||||
|
catch( ex ) {
|
||||||
|
throw ex;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
sheets.forEach( function( rep ) {
|
||||||
|
|
||||||
|
var rez;
|
||||||
|
try {
|
||||||
|
rez = JSON.parse( rep.raw );
|
||||||
|
}
|
||||||
|
catch( ex ) { // Note [1]
|
||||||
|
_log('Validating '.info + rep.file.infoBold +
|
||||||
|
' against FRESH/JRS schema: '.info + 'ERROR!'.error.bold);
|
||||||
|
|
||||||
|
if (ex instanceof SyntaxError) {
|
||||||
|
// Invalid JSON
|
||||||
|
_log( '--> '.bold.red + rep.file.toUpperCase().red +
|
||||||
|
' contains invalid JSON. Unable to validate.'.red );
|
||||||
|
_log( (' INTERNAL: ' + ex).red );
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
|
||||||
|
_log(('ERROR: ' + ex.toString()).red.bold);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var isValid = false;
|
||||||
|
var style = 'useful';
|
||||||
|
var errors = [];
|
||||||
|
var fmt = rez.meta &&
|
||||||
|
(rez.meta.format === 'FRESH@0.1.0') ? 'fresh':'jars';
|
||||||
|
|
||||||
|
try {
|
||||||
|
|
||||||
|
var validate = validator( schemas[ fmt ], { // Note [1]
|
||||||
|
formats: {
|
||||||
|
date: /^\d{4}(?:-(?:0[0-9]{1}|1[0-2]{1})(?:-[0-9]{2})?)?$/
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
isValid = validate( rez );
|
||||||
|
if( !isValid ) {
|
||||||
|
style = 'warn';
|
||||||
|
errors = validate.errors;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
catch(ex) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_log( 'Validating '.info + rep.file.infoBold + ' against '.info +
|
||||||
|
fmt.replace('jars','JSON Resume').toUpperCase().infoBold +
|
||||||
|
' schema: '.info + (isValid ? 'VALID!' : 'INVALID')[style].bold );
|
||||||
|
|
||||||
|
errors.forEach(function(err,idx) {
|
||||||
|
_log( '--> '.bold.yellow +
|
||||||
|
(err.field.replace('data.','resume.').toUpperCase() + ' ' +
|
||||||
|
err.message).yellow );
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
}());
|
76
tests/test-cli.js
Normal file
76
tests/test-cli.js
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
|
||||||
|
var chai = require('chai')
|
||||||
|
, expect = chai.expect
|
||||||
|
, should = chai.should()
|
||||||
|
, path = require('path')
|
||||||
|
, _ = require('underscore')
|
||||||
|
, FRESHResume = require('../src/core/fresh-resume')
|
||||||
|
, FCMD = require( '../src/hackmycmd')
|
||||||
|
, validator = require('is-my-json-valid')
|
||||||
|
, COLORS = require('colors');
|
||||||
|
|
||||||
|
chai.config.includeStack = false;
|
||||||
|
|
||||||
|
describe('Testing CLI interface', function () {
|
||||||
|
|
||||||
|
var _sheet;
|
||||||
|
|
||||||
|
function logMsg() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
COLORS.setTheme({
|
||||||
|
title: ['white','bold'],
|
||||||
|
info: process.platform === 'win32' ? 'gray' : ['white','dim'],
|
||||||
|
infoBold: ['white','dim'],
|
||||||
|
warn: 'yellow',
|
||||||
|
error: 'red',
|
||||||
|
guide: 'yellow',
|
||||||
|
status: 'gray',//['white','dim'],
|
||||||
|
useful: 'green',
|
||||||
|
});
|
||||||
|
|
||||||
|
var opts = {
|
||||||
|
//theme: 'compact',
|
||||||
|
format: 'FRESH',
|
||||||
|
prettify: true,
|
||||||
|
silent: true
|
||||||
|
};
|
||||||
|
|
||||||
|
var opts2 = {
|
||||||
|
format: 'JRS',
|
||||||
|
prettify: true,
|
||||||
|
silent: true
|
||||||
|
};
|
||||||
|
|
||||||
|
run( 'new', ['tests/sandbox/new-fresh-resume.json'], [], opts, ' (FRESH format)' );
|
||||||
|
run( 'new', ['tests/sandbox/new-jrs-resume.json'], [], opts2, ' (JRS format)' );
|
||||||
|
run( 'new', ['tests/sandbox/new-1.json', 'tests/sandbox/new-2.json', 'tests/sandbox/new-3.json'], [], opts, ' (multiple FRESH resumes)' );
|
||||||
|
run( 'new', ['tests/sandbox/new-jrs-1.json', 'tests/sandbox/new-jrs-2.json', 'tests/sandbox/new-jrs-3.json'], [], opts, ' (multiple JRS resumes)' );
|
||||||
|
run( 'new', ['tests/sandbox/new-jrs-resume.json'], [], opts2, ' (JRS format)' );
|
||||||
|
fail( 'new', [], [], opts, " (when a filename isn't specified)" );
|
||||||
|
|
||||||
|
run( 'validate', ['node_modules/jane-q-fullstacker/resume/jane-resume.json'], [], opts, ' (FRESH format)' );
|
||||||
|
run( 'validate', ['tests/sandbox/new-fresh-resume.json'], [], opts, ' (FRESH format)' );
|
||||||
|
|
||||||
|
function run( verb, src, dst, opts, msg ) {
|
||||||
|
msg = msg || '.';
|
||||||
|
it( 'The ' + verb.toUpperCase() + ' command should SUCCEED' + msg, function () {
|
||||||
|
function runIt() {
|
||||||
|
FCMD.verbs[verb]( src, dst, opts, opts.silent ? logMsg : null );
|
||||||
|
}
|
||||||
|
runIt.should.not.Throw();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function fail( verb, src, dst, opts, msg ) {
|
||||||
|
msg = msg || '.';
|
||||||
|
it( 'The ' + verb.toUpperCase() + ' command should FAIL' + msg, function () {
|
||||||
|
function runIt() {
|
||||||
|
FCMD.verbs[verb]( src, dst, opts, logMsg );
|
||||||
|
}
|
||||||
|
runIt.should.Throw();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
@ -35,9 +35,9 @@ describe('Testing themes', function () {
|
|||||||
theme: themeName,
|
theme: themeName,
|
||||||
format: 'FRESH',
|
format: 'FRESH',
|
||||||
prettify: true,
|
prettify: true,
|
||||||
silent: false
|
silent: true
|
||||||
};
|
};
|
||||||
FCMD.verbs.build( src, dst, opts );
|
FCMD.verbs.build( src, dst, opts, function() { } );
|
||||||
}
|
}
|
||||||
tryOpen.should.not.Throw();
|
tryOpen.should.not.Throw();
|
||||||
});
|
});
|
||||||
|
Loading…
Reference in New Issue
Block a user