From 95b593a9dc3d83544a810e918996255925fe6989 Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Wed, 2 Mar 2016 16:47:52 -0800 Subject: [PATCH 01/15] Add data module, basic tests, YAML dependency --- package.json | 3 ++- src/data.js | 20 ++++++++++++++++++++ test/data.js | 16 ++++++++++++++++ test/fixtures/data/sample-data.yaml | 4 ++++ 4 files changed, 42 insertions(+), 1 deletion(-) create mode 100644 src/data.js create mode 100644 test/data.js create mode 100644 test/fixtures/data/sample-data.yaml diff --git a/package.json b/package.json index 1834f5a..f8c0e7f 100644 --- a/package.json +++ b/package.json @@ -20,7 +20,8 @@ "dependencies": { "bluebird": "^3.3.1", "globby": "^4.0.0", - "handlebars": "^4.0.5" + "handlebars": "^4.0.5", + "js-yaml": "^3.5.3" }, "devDependencies": { "babel-cli": "^6.5.1", diff --git a/src/data.js b/src/data.js new file mode 100644 index 0000000..4091ae1 --- /dev/null +++ b/src/data.js @@ -0,0 +1,20 @@ +import yaml from 'js-yaml'; +import { readFilesKeyed } from './utils'; + +/** + * Take glob in dataOpts and parse all data from matching files as YAML. + * + * @param {glob} dataOpts + * @return {Promise} resolving to object of parsed data + */ +function prepareData (dataOpts) { + const parsedData = new Object(); + return readFilesKeyed(dataOpts).then(fileData => { + Object.keys(fileData).forEach(fileKey => { + parsedData[fileKey] = yaml.safeLoad(fileData[fileKey]); + }); + return parsedData; + }); +} + +export default prepareData; diff --git a/test/data.js b/test/data.js new file mode 100644 index 0000000..5f0c13e --- /dev/null +++ b/test/data.js @@ -0,0 +1,16 @@ +/* global describe, it */ +var chai = require('chai'); +var expect = chai.expect; +var prepareData = require('../dist/data'); + +describe ('data', () => { + describe ('data parsing', () => { + it ('should parse YAML data from files', done => { + prepareData(`${__dirname}/fixtures/data/*.yaml`).then(dataObj => { + expect(dataObj).to.contain.keys('sample-data'); + expect(dataObj['sample-data'].foo).to.be.an('Array'); + done(); + }); + }); + }); +}); diff --git a/test/fixtures/data/sample-data.yaml b/test/fixtures/data/sample-data.yaml new file mode 100644 index 0000000..a3b49d9 --- /dev/null +++ b/test/fixtures/data/sample-data.yaml @@ -0,0 +1,4 @@ +foo: + - bar + - baz +elfin: 'small things' From 276c8547dd3eee52cb4b8c5c489f2f7bfd41714b Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Thu, 3 Mar 2016 14:28:09 -0800 Subject: [PATCH 02/15] Convert module functions to not be fat-arrow --- src/utils.js | 21 ++++++++++++++++----- 1 file changed, 16 insertions(+), 5 deletions(-) diff --git a/src/utils.js b/src/utils.js index 7d8ca98..b5bf74d 100644 --- a/src/utils.js +++ b/src/utils.js @@ -5,11 +5,22 @@ import {readFile as readFileCB} from 'fs'; var readFile = Promise.promisify(readFileCB); /* Helper functions */ -const basename = filepath => path.basename(filepath, path.extname(filepath)); -const dirname = filepath => path.normalize(path.dirname(filepath)); -const parentDirname = filepath => dirname(filepath).split(path.sep).pop(); -const removeNumbers = str => str.replace(/^[0-9|\.\-]+/, ''); -const getFiles = glob => globby(glob, {nodir: true }); + +function basename (filepath) { + return path.basename(filepath, path.extname(filepath)); +} +function dirname (filepath) { + return path.normalize(path.dirname(filepath)); +} +function parentDirname (filepath) { + return dirname(filepath).split(path.sep).pop(); +} +function removeNumbers (str) { + return str.replace(/^[0-9|\.\-]+/, ''); +} +function getFiles (glob) { + return globby(glob, {nodir: true }); +} /** * Utility function to test if a value COULD be a glob. A single string or From 5f9b4260a25ac9cf22581a7d6177658a0df1b67f Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Thu, 3 Mar 2016 14:29:15 -0800 Subject: [PATCH 03/15] --amend --- src/utils.js | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/src/utils.js b/src/utils.js index b5bf74d..3ff40fd 100644 --- a/src/utils.js +++ b/src/utils.js @@ -18,8 +18,12 @@ function parentDirname (filepath) { function removeNumbers (str) { return str.replace(/^[0-9|\.\-]+/, ''); } +/** + * @param {glob} glob + * @return {Promise} resolving to {Array} of files matching glob + */ function getFiles (glob) { - return globby(glob, {nodir: true }); + return globby(glob, { nodir: true }); } /** From 361e64f7c682cbf8d4cae0a81fcb52592fc31552 Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Thu, 3 Mar 2016 15:21:45 -0800 Subject: [PATCH 04/15] Add ability to pass function to run on file contents --- src/utils.js | 20 ++++++++++++++------ test/utils.js | 9 +++++++++ 2 files changed, 23 insertions(+), 6 deletions(-) diff --git a/src/utils.js b/src/utils.js index 3ff40fd..ae1e6fd 100644 --- a/src/utils.js +++ b/src/utils.js @@ -43,16 +43,24 @@ function isGlob (candidate) { } /** - * Take a glob; read the files. Return a Promise that ultimately resolves - * to an Array of objects: - * [{ path: original filepath, - * contents: utf-8 file contents}...] + * Take a glob; read the files, optionally running a `contentFn` over + * the contents of the file. + * + * @param {glob} + * @param {object} opts + * @return {Promise} resolving to {Array} of file objects */ -function readFiles (glob) { +function readFiles (glob, opts = {}) { + opts = Object.assign({}, { + contentFn: (content, path) => content + }, opts); return getFiles(glob).then(paths => { var fileReadPromises = paths.map(path => { return readFile(path, 'utf-8') - .then(contents => ({ path, contents })); + .then(contents => { + contents = opts.contentFn(contents, path); + return {path, contents}; + }); }); return Promise.all(fileReadPromises); }); diff --git a/test/utils.js b/test/utils.js index 9ee3054..732bfad 100644 --- a/test/utils.js +++ b/test/utils.js @@ -65,6 +65,15 @@ describe ('utils', () => { done(); }); }); + it ('should run passed function over content', done => { + var glob = path.join(__dirname, 'fixtures/helpers/*.js'); + utils.readFiles(glob, { contentFn: (content, path) => 'foo' }) + .then(allFileData => { + expect(allFileData).to.have.length.of(3); + expect(allFileData[0].contents).to.equal('foo'); + done(); + }); + }); it ('should be able to key files by getName', done => { var glob = path.join(__dirname, 'fixtures/helpers/*.js'); utils.readFilesKeyed(glob).then(allFileData => { From f9ba02d23750d081d613c21d7689fe68a4d6b13d Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Fri, 4 Mar 2016 09:20:41 -0800 Subject: [PATCH 05/15] Refactor keyname and add tests --- src/utils.js | 8 ++++---- test/utils.js | 17 ++++++++++++++++- 2 files changed, 20 insertions(+), 5 deletions(-) diff --git a/src/utils.js b/src/utils.js index 0eb9d53..ea9b621 100644 --- a/src/utils.js +++ b/src/utils.js @@ -15,7 +15,7 @@ function dirname (filepath) { function parentDirname (filepath) { return dirname(filepath).split(path.sep).pop(); } -function removeNumbers (str) { +function removeLeadingNumbers (str) { return str.replace(/^[0-9|\.\-]+/, ''); } /** @@ -91,12 +91,12 @@ function readFilesKeyed (glob, preserveNumbers = false) { * - unless preserveNumbers, remove numbers from the string as well * * @param {String} str filepath - * @param {Boolean} preserveNumbers + * @param {Object} options * @return {String} */ -function keyname (str, preserveNumbers = false) { +function keyname (str, { stripNumbers = true } = {}) { const name = basename(str).replace(/\s/g, '-'); - return (preserveNumbers) ? name : removeNumbers(name); + return (stripNumbers) ? removeLeadingNumbers(name) : name; } /** diff --git a/test/utils.js b/test/utils.js index dc5215a..40edec2 100644 --- a/test/utils.js +++ b/test/utils.js @@ -56,6 +56,21 @@ describe ('utils', () => { expect(badGlobs.every(glob => utils.isGlob(glob))).to.be.false; }); }); + describe('keyname', () => { + it ('should strip leading numbers by default', () => { + var result = utils.keyname('foo/01-bar.baz'); + expect(result).not.to.contain('01-'); + }); + it ('should strip parent directories and extensions', () => { + var result = utils.keyname('foo/01-bar.baz'); + expect(result).not.to.contain('foo'); + expect(result).not.to.contain('baz'); + }); + it ('should accept option to retain leading numbers', () => { + var result = utils.keyname('foo/01-bar.baz', { stripNumbers: false }); + expect(result).to.contain('01-'); + }); + }); describe('readFiles', () => { it ('should read files from a glob', done => { var glob = path.join(__dirname, 'fixtures/helpers/*.js'); @@ -91,7 +106,7 @@ describe ('utils', () => { }); }); describe('merge()', () => { - it ('works', () => { + it ('merges objects correctly', () => { var actual = utils.merge( {a: 1, c: {d: 3}}, {a: 2, b: 1, c: {e: 4}} From 184451d1d2a6db7d0e458c347542118e9d0a2d74 Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Fri, 4 Mar 2016 09:21:21 -0800 Subject: [PATCH 06/15] Refactor keyname and add tests --- src/utils.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/utils.js b/src/utils.js index ea9b621..d4b7ef3 100644 --- a/src/utils.js +++ b/src/utils.js @@ -73,11 +73,11 @@ function readFiles (glob, opts = {}) { * through keyname(). * */ -function readFilesKeyed (glob, preserveNumbers = false) { +function readFilesKeyed (glob, options = {}) { return readFiles(glob).then(allFileData => { const keyedFileData = new Object(); for (var aFile of allFileData) { - keyedFileData[keyname(aFile.path, preserveNumbers)] = aFile.contents; + keyedFileData[keyname(aFile.path, options)] = aFile.contents; } return keyedFileData; }); From 2021246ed3715bc5bb5077fa197a624788dd030d Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Fri, 4 Mar 2016 09:36:57 -0800 Subject: [PATCH 07/15] Refactor readFiles to be more generic --- src/utils.js | 31 ++++++++++++++++++------------- test/utils.js | 11 ++++++++++- 2 files changed, 28 insertions(+), 14 deletions(-) diff --git a/src/utils.js b/src/utils.js index d4b7ef3..0845ab9 100644 --- a/src/utils.js +++ b/src/utils.js @@ -46,23 +46,28 @@ function isGlob (candidate) { * Take a glob; read the files, optionally running a `contentFn` over * the contents of the file. * - * @param {glob} - * @param {object} opts - * @return {Promise} resolving to {Array} of file objects + * @param {glob} glob of files to read + * @param {Object} Options: + * - {Function} contentFn(content, path): optional function to run over content + * in files; defaults to a no-op + * - {String} encoding + * + * @return {Promise} resolving to Array of Objects: + * - {String} path + * - {String || Mixed} contents: contents of file after contentFn */ -function readFiles (glob, opts = {}) { - opts = Object.assign({}, { - contentFn: (content, path) => content - }, opts); +function readFiles (glob, { + contentFn = (content, path) => content, + encoding = 'utf-8' +} = {}) { return getFiles(glob).then(paths => { - var fileReadPromises = paths.map(path => { - return readFile(path, 'utf-8') + return Promise.all(paths.map(path => { + return readFile(path, encoding) .then(contents => { - contents = opts.contentFn(contents, path); - return {path, contents}; + contents = contentFn(contents, path); + return { path, contents }; }); - }); - return Promise.all(fileReadPromises); + })); }); } diff --git a/test/utils.js b/test/utils.js index 40edec2..b79a1be 100644 --- a/test/utils.js +++ b/test/utils.js @@ -89,7 +89,9 @@ describe ('utils', () => { done(); }); }); - it ('should be able to key files by getName', done => { + }); + describe('readFilesKeyed', () => { + it ('should be able to key files by keyname', done => { var glob = path.join(__dirname, 'fixtures/helpers/*.js'); utils.readFilesKeyed(glob).then(allFileData => { expect(allFileData).to.be.an('object'); @@ -97,6 +99,13 @@ describe ('utils', () => { done(); }); }); + it ('should accept an option to preserve leading numbers', done => { + var glob = path.join(__dirname, 'fixtures/data/*.yaml'); + utils.readFilesKeyed(glob, { stripNumbers: false }).then(allFileData => { + expect(allFileData).to.be.an('object'); + done(); + }); + }); }); describe('parent directory (parentDirname)', () => { it ('should derive correct parent dirname of files', () => { From 9f4fc3b36ff4f84baa434228d3c3aed6955c6986 Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Fri, 4 Mar 2016 09:53:41 -0800 Subject: [PATCH 08/15] Refactor readFilesKeyed; tests --- src/utils.js | 18 ++++++++++++++---- test/utils.js | 21 +++++++++++++++++++++ 2 files changed, 35 insertions(+), 4 deletions(-) diff --git a/src/utils.js b/src/utils.js index 0845ab9..3aa625a 100644 --- a/src/utils.js +++ b/src/utils.js @@ -75,14 +75,24 @@ function readFiles (glob, { * Read the files from a glob, but then instead of resolving the * Promise with an Array of objects (@see readFiles), resolve with a * single object; each file's contents is keyed by its filename run - * through keyname(). + * through optional keyFn(filePath, options) (default: keyname). + * Will pass other options on to readFiles and keyFn * + * @param {glob} + * @param {Object} options (all optional): + * - keyFn + * - contentFn + * - stripNumbers + * @return {Promise} resolving to {Object} of keyed file contents */ function readFilesKeyed (glob, options = {}) { - return readFiles(glob).then(allFileData => { + const { + keyFn = keyname + } = options; + return readFiles(glob, options).then(allFileData => { const keyedFileData = new Object(); for (var aFile of allFileData) { - keyedFileData[keyname(aFile.path, options)] = aFile.contents; + keyedFileData[keyFn(aFile.path, options)] = aFile.contents; } return keyedFileData; }); @@ -93,7 +103,7 @@ function readFilesKeyed (glob, options = {}) { * partials, etc, based on a filepath: * - replace whitespace characters with `-` * - use only the basename, no extension - * - unless preserveNumbers, remove numbers from the string as well + * - unless stripNumbers option false, remove numbers from the string as well * * @param {String} str filepath * @param {Object} options diff --git a/test/utils.js b/test/utils.js index b79a1be..65d51a2 100644 --- a/test/utils.js +++ b/test/utils.js @@ -106,6 +106,27 @@ describe ('utils', () => { done(); }); }); + it ('should accept a function to derive keys', done => { + var glob = path.join(__dirname, 'fixtures/data/*.yaml'); + utils.readFilesKeyed(glob, { keyFn: (path, options) => 'foo' + path }) + .then(allFileData => { + expect(Object.keys(allFileData)[0]).to.contain('foo'); + done(); + }); + }); + it ('should pass contentFn through to readFiles', done => { + var glob = path.join(__dirname, 'fixtures/data/*.yaml'); + utils.readFilesKeyed(glob, { + keyFn: (path, options) => 'foo' + path, + contentFn: (content, path) => 'foo' + }).then(allFileData => { + for (var fileKey in allFileData) { + expect(fileKey).to.contain('foo'); + expect(allFileData[fileKey]).to.equal('foo'); + } + done(); + }); + }); }); describe('parent directory (parentDirname)', () => { it ('should derive correct parent dirname of files', () => { From 0321b6de4c261d2e49a8d5459fdfbcef6713db5e Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Fri, 4 Mar 2016 09:54:30 -0800 Subject: [PATCH 09/15] --amend --- src/utils.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/utils.js b/src/utils.js index 3aa625a..ada0591 100644 --- a/src/utils.js +++ b/src/utils.js @@ -75,7 +75,7 @@ function readFiles (glob, { * Read the files from a glob, but then instead of resolving the * Promise with an Array of objects (@see readFiles), resolve with a * single object; each file's contents is keyed by its filename run - * through optional keyFn(filePath, options) (default: keyname). + * through optional `keyFn(filePath, options)`` (default: keyname). * Will pass other options on to readFiles and keyFn * * @param {glob} @@ -87,7 +87,7 @@ function readFiles (glob, { */ function readFilesKeyed (glob, options = {}) { const { - keyFn = keyname + keyFn = (path, options) => keyname(path, options) } = options; return readFiles(glob, options).then(allFileData => { const keyedFileData = new Object(); From 608509a9377030443cd8500619c154e4d1f0febd Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Fri, 4 Mar 2016 09:58:18 -0800 Subject: [PATCH 10/15] Refactor prepareData; tests --- src/data.js | 10 +++------- test/data.js | 2 +- test/fixtures/data/05-another-data.yaml | 8 ++++++++ 3 files changed, 12 insertions(+), 8 deletions(-) create mode 100644 test/fixtures/data/05-another-data.yaml diff --git a/src/data.js b/src/data.js index 4091ae1..c74d55b 100644 --- a/src/data.js +++ b/src/data.js @@ -7,13 +7,9 @@ import { readFilesKeyed } from './utils'; * @param {glob} dataOpts * @return {Promise} resolving to object of parsed data */ -function prepareData (dataOpts) { - const parsedData = new Object(); - return readFilesKeyed(dataOpts).then(fileData => { - Object.keys(fileData).forEach(fileKey => { - parsedData[fileKey] = yaml.safeLoad(fileData[fileKey]); - }); - return parsedData; +function prepareData (dataGlob) { + return readFilesKeyed(dataGlob, { + contentFn: yaml.safeLoad }); } diff --git a/test/data.js b/test/data.js index 5f0c13e..e177905 100644 --- a/test/data.js +++ b/test/data.js @@ -7,7 +7,7 @@ describe ('data', () => { describe ('data parsing', () => { it ('should parse YAML data from files', done => { prepareData(`${__dirname}/fixtures/data/*.yaml`).then(dataObj => { - expect(dataObj).to.contain.keys('sample-data'); + expect(dataObj).to.contain.keys('sample-data', 'another-data'); expect(dataObj['sample-data'].foo).to.be.an('Array'); done(); }); diff --git a/test/fixtures/data/05-another-data.yaml b/test/fixtures/data/05-another-data.yaml new file mode 100644 index 0000000..b3880c9 --- /dev/null +++ b/test/fixtures/data/05-another-data.yaml @@ -0,0 +1,8 @@ +ding: + - dong + - dell + - 'cat is in the well' + - 5 +forestry: + fob: 'key' + bork: 'bing' From 612131ee5333a4c79a1c976394cf7b15132e4057 Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Fri, 4 Mar 2016 10:03:40 -0800 Subject: [PATCH 11/15] Add minor comment --- src/data.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/data.js b/src/data.js index c74d55b..d9afbd3 100644 --- a/src/data.js +++ b/src/data.js @@ -4,8 +4,8 @@ import { readFilesKeyed } from './utils'; /** * Take glob in dataOpts and parse all data from matching files as YAML. * - * @param {glob} dataOpts - * @return {Promise} resolving to object of parsed data + * @param {glob} dataGlob + * @return {Promise} resolving to {Object} of keyed parsed YAML data */ function prepareData (dataGlob) { return readFilesKeyed(dataGlob, { From f733af888c81f73571ddba49eebf2b85475ec30f Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Fri, 4 Mar 2016 10:21:41 -0800 Subject: [PATCH 12/15] Add data opts and translation; tests --- src/options.js | 9 +++++++++ test/options.js | 8 +++++++- 2 files changed, 16 insertions(+), 1 deletion(-) diff --git a/src/options.js b/src/options.js index ce0cb73..b5cba63 100644 --- a/src/options.js +++ b/src/options.js @@ -1,7 +1,12 @@ import Handlebars from 'handlebars'; +import yaml from 'js-yaml'; import { merge } from './utils'; const defaults = { + data: { + src: ['src/data/**/*.yaml'], + parseFn: yaml.safeLoad + }, templates: { handlebars: Handlebars, helpers : {}, @@ -32,6 +37,7 @@ function mergeDefaults (options = {}) { function translateOptions (options = {}) { /* eslint-disable prefer-const */ const { + data, handlebars, helpers, layouts, @@ -40,6 +46,9 @@ function translateOptions (options = {}) { } = options; const result = { + data: { + src: data + }, templates: { handlebars, helpers, diff --git a/test/options.js b/test/options.js index 26c8467..e904cf6 100644 --- a/test/options.js +++ b/test/options.js @@ -23,10 +23,14 @@ describe ('drizzle-builder', () => { }); it ('should translate template options', () => { var opts = parseOptions({ + data: 'foo/bar/baz.yml', layoutIncludes: 'a path', views: 'a path to views' }); expect(opts).to.be.an('object'); + expect(opts.data).to.be.an('object'); + expect(opts.data.src).to.be.a('string'); + expect(opts.data.src).to.equal('foo/bar/baz.yml'); expect(opts.views).not.to.be; expect(opts.layoutIncludes).not.to.be; expect(opts.templates).to.be.an('object'); @@ -41,11 +45,13 @@ describe ('drizzle-builder', () => { it ('should provide default templating options', () => { var opts = parseOptions(); - expect(opts).to.contain.keys('templates'); + expect(opts).to.contain.keys('templates', 'data'); expect(opts.templates).to.be.an('object'); expect(opts.templates).to.have.keys('handlebars', 'helpers', 'layouts', 'pages', 'partials'); expect(opts.templates.handlebars).to.be.an('object'); + expect(opts.data).to.have.keys('src', 'parseFn'); + expect(opts.data.parseFn).to.be.a('function'); }); }); }); From bd930673fc08d1050a1ba3c2441247b3754b28bb Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Fri, 4 Mar 2016 13:15:55 -0800 Subject: [PATCH 13/15] Expand tests on parsing; remove data module --- src/data.js | 16 ---------------- src/options.js | 2 +- test/data.js | 16 ---------------- test/fixtures/data/data-as-json.json | 4 ++++ 4 files changed, 5 insertions(+), 33 deletions(-) delete mode 100644 src/data.js delete mode 100644 test/data.js create mode 100644 test/fixtures/data/data-as-json.json diff --git a/src/data.js b/src/data.js deleted file mode 100644 index d9afbd3..0000000 --- a/src/data.js +++ /dev/null @@ -1,16 +0,0 @@ -import yaml from 'js-yaml'; -import { readFilesKeyed } from './utils'; - -/** - * Take glob in dataOpts and parse all data from matching files as YAML. - * - * @param {glob} dataGlob - * @return {Promise} resolving to {Object} of keyed parsed YAML data - */ -function prepareData (dataGlob) { - return readFilesKeyed(dataGlob, { - contentFn: yaml.safeLoad - }); -} - -export default prepareData; diff --git a/src/options.js b/src/options.js index b5cba63..dfa2a05 100644 --- a/src/options.js +++ b/src/options.js @@ -5,7 +5,7 @@ import { merge } from './utils'; const defaults = { data: { src: ['src/data/**/*.yaml'], - parseFn: yaml.safeLoad + parseFn: (contents, path) => yaml.safeLoad(contents) }, templates: { handlebars: Handlebars, diff --git a/test/data.js b/test/data.js deleted file mode 100644 index e177905..0000000 --- a/test/data.js +++ /dev/null @@ -1,16 +0,0 @@ -/* global describe, it */ -var chai = require('chai'); -var expect = chai.expect; -var prepareData = require('../dist/data'); - -describe ('data', () => { - describe ('data parsing', () => { - it ('should parse YAML data from files', done => { - prepareData(`${__dirname}/fixtures/data/*.yaml`).then(dataObj => { - expect(dataObj).to.contain.keys('sample-data', 'another-data'); - expect(dataObj['sample-data'].foo).to.be.an('Array'); - done(); - }); - }); - }); -}); diff --git a/test/fixtures/data/data-as-json.json b/test/fixtures/data/data-as-json.json new file mode 100644 index 0000000..2f97f6f --- /dev/null +++ b/test/fixtures/data/data-as-json.json @@ -0,0 +1,4 @@ +{ + "foo" : "bar", + "fortunately": 5 +} From cc5895f2911f13b7db0b374478bbf9bf8427f331 Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Fri, 4 Mar 2016 13:54:38 -0800 Subject: [PATCH 14/15] Refactor data-parsing; tests for integration --- src/index.js | 36 +++++++++++++++--------------------- src/options.js | 17 ++++++++++++----- test/index.js | 26 +++++++++++++++----------- 3 files changed, 42 insertions(+), 37 deletions(-) diff --git a/src/index.js b/src/index.js index 9049076..fc8f75e 100644 --- a/src/index.js +++ b/src/index.js @@ -2,33 +2,27 @@ var parseOptions = require('./options'); var prepareTemplates = require('./template').prepareTemplates; var utils = require('./utils'); - +/** + * Build a data/context object for use by the builder + * @TODO This may move into its own module if it seems appropriate + * + * @param {Object} options + * @return {Promise} resolving to {Object} of keyed file data + */ +function prepareData (options) { + // Data data + return utils.readFilesKeyed(options.data.src, { + contentFn: options.data.parseFn + }); +} /** * Build the drizzle output * - * @return {Promise}; resolves to options {object} (for now) + * @return {Promise}; resolves to [dataObj, Handlebars] for now */ function drizzle (options) { const opts = parseOptions(options); - - // const buildData = new Object(); - // const readLayouts = utils.readFilesKeyed(opts.templates.layouts) - // .then(fileData => buildData.layouts = fileData); - // const readDocs = utils.readFilesKeyed(opts.docs) - // .then(fileData => { - // for (var file in fileData) { - // fileData[file].name = utils.toTitleCase(file); - // fileData[file].content = 'todo'; // markdown file.content - // } - // return fileData; - // }); - // const readData = utils.readFilesKeyed(opts.data).then(fileData => { - // for (var file in fileData) { - // fileData[file].contents = 'todo'; // yaml load contents - // } - // return fileData; - // }); - return prepareTemplates(opts).then(handlebars => opts); + return Promise.all([prepareData(opts), prepareTemplates(opts)]); } export default drizzle; diff --git a/src/options.js b/src/options.js index dfa2a05..eb5400c 100644 --- a/src/options.js +++ b/src/options.js @@ -35,7 +35,6 @@ function mergeDefaults (options = {}) { * @return {object} User options */ function translateOptions (options = {}) { - /* eslint-disable prefer-const */ const { data, handlebars, @@ -46,9 +45,6 @@ function translateOptions (options = {}) { } = options; const result = { - data: { - src: data - }, templates: { handlebars, helpers, @@ -57,8 +53,19 @@ function translateOptions (options = {}) { partials } }; + // @TODO: Is there are more concise way of handling this? + // If you use the pattern above, an object value for `data` + // will get improperly nested/trounced + if (data) { + if (typeof data === 'string') { + result.data = { + src: data + }; + } else { + result.data = data; + } + } return result; - /* eslint-enable prefer-const */ } const parseOptions = options => mergeDefaults(translateOptions(options)); diff --git a/test/index.js b/test/index.js index c006ff1..b4d5bef 100644 --- a/test/index.js +++ b/test/index.js @@ -2,20 +2,24 @@ var chai = require('chai'); var expect = chai.expect; var builder = require('../dist/'); +var path = require('path'); -const options = { - templates: { - partials: `${__dirname}/fixtures/partials/*`, - helpers: `${__dirname}/fixtures/helpers/*.js` - } -}; describe ('drizzle builder integration', () => { - it ('should return opts used for building', () => { - builder(options).then(opts => { - expect(opts).to.be.an.object; - expect(opts.templates).to.be.an.object; - expect(opts.templates.handlebars).to.be.an.object; + const options = { + data: { + src: path.join(__dirname, 'fixtures/data/*.yaml') + }, + templates: { + helpers: path.join(__dirname, 'fixtures/helpers/**/*.js'), + partials: path.join(__dirname, 'fixtures/partials/*.hbs') + } + }; + it ('should return data and context', done => { + builder(options).then(drizzleData => { + expect(drizzleData[0]).to.contain.keys('another-data', 'sample-data'); + expect(drizzleData[0]['another-data']).to.be.an('object'); + done(); }); }); }); From 38376cfa759c2dd6a98c80f84f30c6890777f8ed Mon Sep 17 00:00:00 2001 From: Lyza Danger Gardner Date: Fri, 4 Mar 2016 15:03:03 -0800 Subject: [PATCH 15/15] Comment some utility functions with examples --- src/utils.js | 19 +++++++++++++++++++ 1 file changed, 19 insertions(+) diff --git a/src/utils.js b/src/utils.js index ada0591..b4fad9a 100644 --- a/src/utils.js +++ b/src/utils.js @@ -6,15 +6,34 @@ var readFile = Promise.promisify(readFileCB); /* Helper functions */ +/** + * Return extension-less basename of filepath + * @param {String} filepath + * @example basename('foo/bar/baz.txt'); // -> 'baz' + */ function basename (filepath) { return path.basename(filepath, path.extname(filepath)); } + +/** + * Return normalized (no '..', '.') full dirname of filepath + * @param {String} filepath + * @example dirname('../ding/foo.txt'); // -> '/Users/shiela/ding/' + */ function dirname (filepath) { return path.normalize(path.dirname(filepath)); } + +/** + * Return the name of this files immediate parent directory + * @param {String} filepath + * @example basename('foo/bar/baz.txt'); // -> 'bar' + */ function parentDirname (filepath) { return dirname(filepath).split(path.sep).pop(); } + + function removeLeadingNumbers (str) { return str.replace(/^[0-9|\.\-]+/, ''); }