Skip to content
This repository was archived by the owner on Oct 2, 2024. It is now read-only.

Commit 58a6d42

Browse files
committed
initial commit
0 parents  commit 58a6d42

10 files changed

+1689
-0
lines changed

.editorconfig

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
# Editor configuration, see http://editorconfig.org
2+
root = true
3+
4+
[*]
5+
charset = utf-8
6+
indent_style = space
7+
indent_size = 2
8+
insert_final_newline = true
9+
trim_trailing_whitespace = true
10+
11+
[*.md]
12+
max_line_length = off
13+
trim_trailing_whitespace = false

.eslintrc.yml

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
extends:
2+
- xo
3+
rules:
4+
indent:
5+
- error
6+
- 2
7+
new-cap: off
8+
space-before-function-paren:
9+
- error
10+
- always
11+
operator-linebreak:
12+
- error
13+
- before

.gitignore

+62
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
# Created by .ignore support plugin (hsz.mobi)
2+
### Node template
3+
# Logs
4+
logs
5+
*.log
6+
npm-debug.log*
7+
yarn-debug.log*
8+
yarn-error.log*
9+
10+
# Runtime data
11+
pids
12+
*.pid
13+
*.seed
14+
*.pid.lock
15+
16+
# Directory for instrumented libs generated by jscoverage/JSCover
17+
lib-cov
18+
19+
# Coverage directory used by tools like istanbul
20+
coverage
21+
22+
# nyc test coverage
23+
.nyc_output
24+
25+
# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
26+
.grunt
27+
28+
# Bower dependency directory (https://bower.io/)
29+
bower_components
30+
31+
# node-waf configuration
32+
.lock-wscript
33+
34+
# Compiled binary addons (http://nodejs.org/api/addons.html)
35+
build/Release
36+
37+
# Dependency directories
38+
node_modules/
39+
jspm_packages/
40+
41+
# Typescript v1 declaration files
42+
typings/
43+
44+
# Optional npm cache directory
45+
.npm
46+
47+
# Optional eslint cache
48+
.eslintcache
49+
50+
# Optional REPL history
51+
.node_repl_history
52+
53+
# Output of 'npm pack'
54+
*.tgz
55+
56+
# Yarn Integrity file
57+
.yarn-integrity
58+
59+
# dotenv environment variables file
60+
.env
61+
62+

.npmignore

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
.*
2+
test/

.travis.yml

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
dist: trusty
2+
sudo: false
3+
language: node_js
4+
cache: yarn
5+
node_js:
6+
- '7'
7+
- '6'
8+

LICENSE

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
Copyright 2017 Christopher Hiller <[email protected]> (https://boneskull.com)
2+
3+
Licensed under the Apache License, Version 2.0 (the "License");
4+
you may not use this file except in compliance with the License.
5+
You may obtain a copy of the License at
6+
7+
http://www.apache.org/licenses/LICENSE-2.0
8+
9+
Unless required by applicable law or agreed to in writing, software
10+
distributed under the License is distributed on an "AS IS" BASIS,
11+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
See the License for the specific language governing permissions and
13+
limitations under the License.

README.md

+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
# egad
2+
3+
> Compile a Git repo full of Handlebars templates
4+
5+
## Basic Usage
6+
7+
```js
8+
const {scaffold} = require('egad');
9+
10+
scaffold('https://github.com/<your>/<repo>', '/path/to/destination/dir', {
11+
some: 'data',
12+
forYour: 'templates'
13+
}, {
14+
// default options
15+
overwrite: true,
16+
offline: false,
17+
remote: 'origin',
18+
branch: 'master'
19+
})
20+
.then(filepaths => {
21+
console.log(`Wrote files: ${filepaths}`);
22+
})
23+
```
24+
25+
The *entire repository* at the given URL will be inspected for Handlebars templates, and those templates will be rendered at the corresponding path under your destination directory. Files *not* containing Handlebars templates will simply be copied.
26+
27+
If destination is omitted, it will default to the current working directory.
28+
29+
## Details
30+
31+
- Repos are stored in the user's XDG cache dir, or whatever XDG cache dir is available, or a temp dir as a last resort.
32+
- If `offline` is `false`, the repo will be either cloned or updated (if a working copy already exists).
33+
- If `offline` is `true` and the repo does not already exist, then you get an rejected `Promise`.
34+
- Use environment variable `DEBUG=egad` to see debug output
35+
- Requires a `git` executable
36+
- **Requires Node.js v6 or greater**
37+
38+
## Known Issues
39+
40+
- Does not copy binary files
41+
- No "errback"-style interface
42+
- How about writing some tests?
43+
44+
## License
45+
46+
© 2017 [Christopher Hiller](https://boneskull.com). Licensed Apache-2.0.

index.js

+174
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,174 @@
1+
'use strict';
2+
3+
const debug = require('debug')('egad');
4+
const promisify = require('es6-promisify');
5+
const fs = require('fs');
6+
const readFile = promisify(fs.readFile);
7+
const stat = promisify(fs.stat);
8+
const writeFile = promisify(fs.writeFile);
9+
const mkdirp = promisify(require('mkdirp'));
10+
const path = require('path');
11+
const xdgBasedir = require('xdg-basedir');
12+
const Walker = require('walker');
13+
const _ = require('lodash/fp');
14+
const render = promisify(require('consolidate').handlebars.render);
15+
const isBinaryPath = require('is-binary-path');
16+
const git = require('simple-git');
17+
const rimraf = promisify(require('rimraf'));
18+
19+
const xdgCachedir = xdgBasedir.cache || require('os')
20+
.tmpdir();
21+
22+
const cachedir = _.partial(path.join, [
23+
xdgCachedir,
24+
'.egad-templates'
25+
]);
26+
const destPath = _.pipe(_.kebabCase, cachedir);
27+
28+
/**
29+
* Downloads (clones) a Git repo
30+
* @param {string} url - URL of Git repo
31+
* @param {Object} [opts] - Options
32+
* @param {string} [opts.dest] - Destination path; default is a user cache dir
33+
* @param {boolean} [opts.offline=false] - If true, just check for existence of
34+
* working copy
35+
* @param {string} [opts.remote=origin] - Git remote
36+
* @param {string} [opts.branch=master] - Git branch
37+
* @returns {Promise.<string>} Destination path
38+
*/
39+
function download (url, opts = {}) {
40+
opts = _.defaults({
41+
dest: destPath(url),
42+
offline: false,
43+
remote: 'origin',
44+
branch: 'master'
45+
}, opts);
46+
debug(`Download target at ${opts.dest}`);
47+
return mkdirp(cachedir())
48+
.then(() => stat(opts.dest))
49+
.then(stats => {
50+
if (!stats.isDirectory()) {
51+
throw new Error(`File exists at ${opts.dest}`);
52+
}
53+
const wc = git(opts.dest);
54+
return new Promise((resolve, reject) => {
55+
debug(`Updating working copy at ${opts.dest}`);
56+
wc.pull(opts.remote, opts.branch, {'--rebase': 'true'}, err => {
57+
if (err) {
58+
return reject(err);
59+
}
60+
resolve();
61+
});
62+
});
63+
})
64+
.catch(err => {
65+
if (opts.offline) {
66+
throw new Error(`No cache of ${url} exists in ${opts.dest}`);
67+
}
68+
debug(err);
69+
return rimraf(opts.dest)
70+
.then(() => {
71+
debug(`Cloning repo ${url}`);
72+
const wc = git();
73+
return new Promise((resolve, reject) => {
74+
wc.clone(url, opts.dest, err => {
75+
if (err) {
76+
return reject(err);
77+
}
78+
resolve();
79+
});
80+
});
81+
});
82+
})
83+
.then(() => opts.dest);
84+
}
85+
86+
/**
87+
* Renders files containing Handlebars templates recursively from one path to
88+
* another.
89+
* @param {string} source - Source directory, containing templates
90+
* @param {string} dest - Destination (output) directory
91+
* @param {Object} [data] - Data for template(s)
92+
* @param {Object} [opts] - Options
93+
* @param {boolean} [opts.overwrite=true] - Set to `false` to avoid overwriting
94+
* existing files
95+
* @returns {Promise.<string[]>} Destination filepaths successfully written to
96+
*/
97+
function generate (source, dest, data = {}, opts = {}) {
98+
opts = _.defaults({
99+
overwrite: true
100+
}, opts);
101+
debug(`Generating from ${source} to ${dest}`);
102+
return mkdirp(dest)
103+
.then(() => new Promise((resolve, reject) => {
104+
const queue = [];
105+
Walker(source)
106+
.filterDir(_.negate(_.endsWith('.git')))
107+
.on('file', sourceFilepath => {
108+
debug(`Trying ${sourceFilepath}`);
109+
if (!isBinaryPath(sourceFilepath)) {
110+
debug(`Queuing ${sourceFilepath}`);
111+
return queue.push(sourceFilepath);
112+
}
113+
debug(`Skipping ${sourceFilepath}; binary`);
114+
})
115+
.on('error', reject)
116+
.on('end', () => resolve(queue));
117+
}))
118+
.then(queue => Promise.all(queue.map(sourceFilepath => {
119+
const destFilepath = path.join(dest,
120+
path.relative(source, sourceFilepath));
121+
return Promise.all([
122+
readFile(sourceFilepath, 'utf8')
123+
.then(str => {
124+
if (/{{([^{}]+)}}/g.test(str)) {
125+
debug(`Rendering template ${sourceFilepath}`);
126+
return render(str, data);
127+
}
128+
return str;
129+
}),
130+
mkdirp(path.dirname(destFilepath))
131+
])
132+
.then(([str]) => writeFile(destFilepath, str, {
133+
encoding: 'utf8',
134+
// This is rather crude
135+
flag: opts.overwrite
136+
? 'w'
137+
: 'wx'
138+
})
139+
.then(() => {
140+
debug(`Wrote ${destFilepath}`);
141+
return destFilepath;
142+
})
143+
.catch(err => {
144+
if (err.code === 'EEXIST') {
145+
debug(`Skipping ${destFilepath}; already exists`);
146+
return;
147+
}
148+
throw err;
149+
}));
150+
})))
151+
.then(_.compact);
152+
}
153+
154+
/**
155+
* Combines download() & generate()
156+
* @param {string} url - Git repo url
157+
* @param {string} [dest] - Destination path; defaults to current working dir
158+
* @param {Object} [data] - Data for template(s)
159+
* @param {Object} [opts] - Options for both `download()` & `generate()`
160+
* @returns {Promise.<string[]>} Destination filepaths successfully written to
161+
*/
162+
function scaffold (url, dest = process.cwd(), data = {}, opts = {}) {
163+
return download(url, opts)
164+
.then(templateDir => {
165+
debug(`${url} cloned into ${templateDir}`);
166+
return generate(templateDir, dest, data, opts);
167+
});
168+
}
169+
170+
module.exports = {
171+
generate,
172+
download,
173+
scaffold
174+
};

package.json

+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
{
2+
"name": "egad",
3+
"version": "0.0.0",
4+
"description": "Compile a Git repo full of Handlebars templates",
5+
"main": "index.js",
6+
"author": "Christopher Hiller <[email protected]> (https://boneskull.com/)",
7+
"license": "Apache-2.0",
8+
"dependencies": {
9+
"consolidate": "^0.14.5",
10+
"debug": "^2.6.6",
11+
"download-git-repo": "^1.0.0",
12+
"es6-promisify": "^5.0.0",
13+
"handlebars": "^4.0.8",
14+
"is-binary-path": "^2.0.0",
15+
"lodash": "^4.17.4",
16+
"mkdirp": "^0.5.1",
17+
"rimraf": "^2.6.1",
18+
"simple-git": "^1.70.0",
19+
"walker": "^1.0.7",
20+
"xdg-basedir": "^3.0.0"
21+
},
22+
"engines": {
23+
"node": ">=6"
24+
},
25+
"keywords": [
26+
"scaffold",
27+
"download",
28+
"git",
29+
"handlebars",
30+
"templates",
31+
"mustache",
32+
"github",
33+
"template",
34+
"hbs",
35+
"scaffolding",
36+
"generate",
37+
"generator"
38+
],
39+
"devDependencies": {
40+
"eslint": "^3.19.0",
41+
"eslint-config-xo": "^0.18.1"
42+
},
43+
"scripts": {
44+
"test": "npm run lint",
45+
"lint": "eslint index.js"
46+
}
47+
}

0 commit comments

Comments
 (0)