Merge pull request #903 from embark-framework/bug_fix/template-downloader-branch-names
make template downloader more robust
This commit is contained in:
commit
207c5b12cb
|
@ -94,7 +94,7 @@ class Cmd {
|
||||||
.description(__('New Application'))
|
.description(__('New Application'))
|
||||||
.option('--simple', __('create a barebones project meant only for contract development'))
|
.option('--simple', __('create a barebones project meant only for contract development'))
|
||||||
.option('--locale [locale]', __('language to use (default: en)'))
|
.option('--locale [locale]', __('language to use (default: en)'))
|
||||||
.option('--template [name/url]', __('download a template using a known name or a GitHub repository URL'))
|
.option('--template <name/url>', __('download a template using a known name or a git host URL'))
|
||||||
.action(function(name, options) {
|
.action(function(name, options) {
|
||||||
i18n.setOrDetectLocale(options.locale);
|
i18n.setOrDetectLocale(options.locale);
|
||||||
if (name === undefined) {
|
if (name === undefined) {
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
let fs = require('../core/fs.js');
|
let fs = require('../core/fs.js');
|
||||||
|
let hostedGitInfo = require('hosted-git-info');
|
||||||
let utils = require('./utils.js');
|
let utils = require('./utils.js');
|
||||||
|
|
||||||
class TemplateGenerator {
|
class TemplateGenerator {
|
||||||
|
@ -17,9 +18,17 @@ class TemplateGenerator {
|
||||||
const fspath = utils.joinPath(destinationFolder, name);
|
const fspath = utils.joinPath(destinationFolder, name);
|
||||||
this.checkPathExists(fspath);
|
this.checkPathExists(fspath);
|
||||||
const self = this;
|
const self = this;
|
||||||
let {url, filePath} = this.getExternalProject(uri);
|
let ext;
|
||||||
|
try {
|
||||||
|
ext = this.getExternalProject(uri);
|
||||||
|
} catch (e) {
|
||||||
|
console.error(utils.errorMessage(e).red);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
let {url, filePath, browse} = ext;
|
||||||
let tmpFilePath = fs.tmpDir(filePath);
|
let tmpFilePath = fs.tmpDir(filePath);
|
||||||
console.log(__('Installing Template from ' + uri + '....').green);
|
console.log(__('Installing template from ' + browse).green);
|
||||||
|
console.log(__('Downloading template...').green);
|
||||||
|
|
||||||
fs.mkdirpSync(utils.dirname(tmpFilePath));
|
fs.mkdirpSync(utils.dirname(tmpFilePath));
|
||||||
utils.downloadFile(url, tmpFilePath, (err) => {
|
utils.downloadFile(url, tmpFilePath, (err) => {
|
||||||
|
@ -45,12 +54,15 @@ class TemplateGenerator {
|
||||||
generate(destinationFolder, name) {
|
generate(destinationFolder, name) {
|
||||||
const fspath = utils.joinPath(destinationFolder, name);
|
const fspath = utils.joinPath(destinationFolder, name);
|
||||||
this.checkPathExists(fspath);
|
this.checkPathExists(fspath);
|
||||||
console.log(__('Initializing Embark Template...').green);
|
console.log(__('Initializing Embark template...').green);
|
||||||
|
|
||||||
let templatePath = fs.embarkPath(utils.joinPath('templates', this.templateName));
|
let templatePath = fs.embarkPath(utils.joinPath('templates', this.templateName));
|
||||||
fs.copySync(templatePath, fspath);
|
fs.copySync(templatePath, fspath);
|
||||||
|
|
||||||
this.installTemplate(fspath, name, (name === 'embark_demo'));
|
this.installTemplate(
|
||||||
|
fspath,
|
||||||
|
name,
|
||||||
|
(this.templateName === 'boilerplate' || this.templateName === 'demo')
|
||||||
|
);
|
||||||
|
|
||||||
if (name === 'embark_demo') {
|
if (name === 'embark_demo') {
|
||||||
console.log('-------------------'.yellow);
|
console.log('-------------------'.yellow);
|
||||||
|
@ -74,79 +86,32 @@ class TemplateGenerator {
|
||||||
|
|
||||||
if (installPackages) {
|
if (installPackages) {
|
||||||
console.log(__('Installing packages...').green);
|
console.log(__('Installing packages...').green);
|
||||||
utils.runCmd('npm install');
|
utils.runCmd('npm install', null, (err) => {
|
||||||
|
if (err) {
|
||||||
|
return console.error(err.red);
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log(__('Init complete').green);
|
console.log(__('Init complete').green);
|
||||||
console.log('\n' + __('App ready at ').green + templatePath);
|
console.log('\n' + __('App ready at ').green + templatePath);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
extractGithubUrlAndFolder(uri){
|
|
||||||
|
|
||||||
/* first matching group is the url, second the repoPart and third the branch with a hash in the beginning (if existing)
|
|
||||||
e.g. (["git@github.com/status-im/dappcon-workshop-dapp#master", "status-im/dappcon-workshop-dapp", "#master" ])
|
|
||||||
should work with all formats of the following:
|
|
||||||
* git@github.com/status-im/dappcon-workshop-dapp#start-here
|
|
||||||
* git@github.com/status-im/dappcon-workshop-dapp
|
|
||||||
* http://www.github.com/status-im/dappcon-workshop-dapp
|
|
||||||
* https://www.github.com/status-im/dappcon-workshop-dapp
|
|
||||||
* github.com/status-im/dappcon-workshop-dapp#start-here
|
|
||||||
|
|
||||||
sadly it doesn't extract from http(s)://github.com/status-im/dappcon-workshop-dapp/tree/start-here
|
|
||||||
thats why we have a special case later
|
|
||||||
*/
|
|
||||||
const match = uri.match(/github\.com+\/(.+?)(#.*)?$/);
|
|
||||||
const githubPart = "https://github.com/";
|
|
||||||
let repoPart = match !== null? match[1] : null;
|
|
||||||
let branchName = match !== null? match[2] : null;
|
|
||||||
|
|
||||||
if (branchName && branchName !== '#'){
|
|
||||||
branchName = branchName.substring(1);
|
|
||||||
} else {
|
|
||||||
branchName = "master";
|
|
||||||
}
|
|
||||||
|
|
||||||
let url, folder;
|
|
||||||
if (uri.includes("/tree")){
|
|
||||||
//e.g http(s)://github.com/status-im/dappcon-workshop-dapp/tree/start-here
|
|
||||||
let repoPartAndBranch = repoPart.split("/tree/");
|
|
||||||
repoPart = repoPartAndBranch[0];
|
|
||||||
branchName = repoPartAndBranch[1];
|
|
||||||
url = "https://github.com/" + repoPart + "/archive/"+ branchName +".zip";
|
|
||||||
folder = repoPart + "/" + branchName;
|
|
||||||
} else if (repoPart !== undefined) {
|
|
||||||
url = githubPart + repoPart + "/archive/" + branchName + ".zip";
|
|
||||||
folder = repoPart + "/" + branchName;
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
'url': url,
|
|
||||||
'folder': folder
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
getExternalProject(uri) {
|
getExternalProject(uri) {
|
||||||
let url, folder;
|
let url, folder, hgi;
|
||||||
const regex = /^((git@)?(www\.)?github\.com\/)|(https?:\/\/)/;
|
hgi = hostedGitInfo.fromUrl(uri);
|
||||||
if (!uri.match(regex) && uri.split('/').length >= 2) {
|
if (!hgi || hgi.user.includes('#')) {
|
||||||
//e.g embark-framework/embark, embark-framework/embark#branch, embark-framework/embark#features/branch
|
let templateAndBranch = uri.split('#');
|
||||||
let repoPartAndBranch = uri.split('#');
|
templateAndBranch[0] = `embark-framework/embark-${templateAndBranch[0]}-template`;
|
||||||
let repoPart = repoPartAndBranch[0];
|
hgi = hostedGitInfo.fromUrl(templateAndBranch.join('#'));
|
||||||
let branchName = (repoPartAndBranch.length === 2)? repoPartAndBranch[1] : "master";
|
|
||||||
url = "https://github.com/" + repoPart + "/archive/"+ branchName + ".zip";
|
|
||||||
folder = repoPart + "/" + branchName;
|
|
||||||
} else if (uri.indexOf('/') === -1) {
|
|
||||||
url = "https://github.com/embark-framework/embark-" + uri + "-template/archive/master.zip";
|
|
||||||
folder = "embark-framework/embark-" + uri + "-template";
|
|
||||||
} else {
|
|
||||||
let urlAndFolder = this.extractGithubUrlAndFolder(uri);
|
|
||||||
url = urlAndFolder.url;
|
|
||||||
folder = urlAndFolder.folder;
|
|
||||||
}
|
}
|
||||||
|
if(!hgi) { throw new Error('Unsupported template name or git host URL'); }
|
||||||
|
url = hgi.tarball();
|
||||||
|
folder = `${hgi.user}/${hgi.project}/${hgi.committish || 'master'}`;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
url,
|
url,
|
||||||
filePath: utils.joinPath(".embark/templates/", folder, "archive.zip")
|
filePath: utils.joinPath(".embark/templates/", folder, "archive.zip"),
|
||||||
|
browse: decodeURIComponent(hgi.browse())
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -56,6 +56,7 @@
|
||||||
"glob": "7.1.3",
|
"glob": "7.1.3",
|
||||||
"globule": "1.2.1",
|
"globule": "1.2.1",
|
||||||
"hard-source-webpack-plugin": "0.11.2",
|
"hard-source-webpack-plugin": "0.11.2",
|
||||||
|
"hosted-git-info": "2.7.1",
|
||||||
"http-proxy": "1.17.0",
|
"http-proxy": "1.17.0",
|
||||||
"http-shutdown": "1.2.0",
|
"http-shutdown": "1.2.0",
|
||||||
"i18n": "0.8.3",
|
"i18n": "0.8.3",
|
||||||
|
|
130
test/template.js
130
test/template.js
|
@ -10,63 +10,117 @@ describe('TemplateGenerator', function () {
|
||||||
templateGenerator = new TemplateGenerator();
|
templateGenerator = new TemplateGenerator();
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('with github link', function () {
|
describe('with named template', function () {
|
||||||
|
|
||||||
it('return correct zip filename for https link', function () {
|
it('returns correct info for named template', function () {
|
||||||
let result = templateGenerator.getExternalProject("https://github.com/embark-framework/embark");
|
let result = templateGenerator.getExternalProject("typescript");
|
||||||
assert.strictEqual(result.url, "https://github.com/embark-framework/embark/archive/master.zip");
|
assert.strictEqual(result.url, "https://codeload.github.com/embark-framework/embark-typescript-template/tar.gz/master");
|
||||||
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/master/archive.zip");
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark-typescript-template/master/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://github.com/embark-framework/embark-typescript-template");
|
||||||
|
|
||||||
|
result = templateGenerator.getExternalProject("typescript#features/branch");
|
||||||
|
assert.strictEqual(result.url, "https://codeload.github.com/embark-framework/embark-typescript-template/tar.gz/features%2Fbranch");
|
||||||
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark-typescript-template/features/branch/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://github.com/embark-framework/embark-typescript-template/tree/features/branch");
|
||||||
});
|
});
|
||||||
|
|
||||||
it('return correct zip filename for https link with branch specified', function () {
|
|
||||||
let result = templateGenerator.getExternalProject("https://github.com/embark-framework/embark/tree/develop");
|
|
||||||
assert.strictEqual(result.url, "https://github.com/embark-framework/embark/archive/develop.zip");
|
|
||||||
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/develop/archive.zip");
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('return correct zip filename for http link', function () {
|
describe('with git host URL', function () {
|
||||||
|
|
||||||
|
it('returns correct info for GitHub URL', function () {
|
||||||
let result = templateGenerator.getExternalProject("http://github.com/embark-framework/embark");
|
let result = templateGenerator.getExternalProject("http://github.com/embark-framework/embark");
|
||||||
assert.strictEqual(result.url, "https://github.com/embark-framework/embark/archive/master.zip");
|
assert.strictEqual(result.url, "https://codeload.github.com/embark-framework/embark/tar.gz/master");
|
||||||
|
|
||||||
|
result = templateGenerator.getExternalProject("https://github.com/embark-framework/embark");
|
||||||
|
assert.strictEqual(result.url, "https://codeload.github.com/embark-framework/embark/tar.gz/master");
|
||||||
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/master/archive.zip");
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/master/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://github.com/embark-framework/embark");
|
||||||
|
|
||||||
|
result = templateGenerator.getExternalProject("https://github.com/embark-framework/embark#features/branch");
|
||||||
|
assert.strictEqual(result.url, "https://codeload.github.com/embark-framework/embark/tar.gz/features%2Fbranch");
|
||||||
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/features/branch/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://github.com/embark-framework/embark/tree/features/branch");
|
||||||
});
|
});
|
||||||
|
|
||||||
it('return correct zip filename for http link with branch specified', function () {
|
it('returns correct info for Bitbucket URL', function () {
|
||||||
let result = templateGenerator.getExternalProject("http://github.com/embark-framework/embark/tree/develop");
|
let result = templateGenerator.getExternalProject("https://bitbucket.org/embark-framework/embark");
|
||||||
assert.strictEqual(result.url, "https://github.com/embark-framework/embark/archive/develop.zip");
|
assert.strictEqual(result.url, "https://bitbucket.org/embark-framework/embark/get/master.tar.gz");
|
||||||
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/develop/archive.zip");
|
|
||||||
});
|
|
||||||
|
|
||||||
it('return correct zip filename without protocol specified ', function () {
|
|
||||||
let result = templateGenerator.getExternalProject("github.com/embark-framework/embark");
|
|
||||||
assert.strictEqual(result.url, "https://github.com/embark-framework/embark/archive/master.zip");
|
|
||||||
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/master/archive.zip");
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/master/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://bitbucket.org/embark-framework/embark");
|
||||||
|
|
||||||
|
result = templateGenerator.getExternalProject("https://bitbucket.org/embark-framework/embark#features/branch");
|
||||||
|
assert.strictEqual(result.url, "https://bitbucket.org/embark-framework/embark/get/features%2Fbranch.tar.gz");
|
||||||
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/features/branch/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://bitbucket.org/embark-framework/embark/src/features/branch");
|
||||||
});
|
});
|
||||||
|
|
||||||
it('return correct zip filename without protocol with branch specified', function () {
|
it('returns correct info for GitLab URL', function () {
|
||||||
let result = templateGenerator.getExternalProject("github.com/embark-framework/embark#develop");
|
let result = templateGenerator.getExternalProject("https://gitlab.com/embark-framework/embark");
|
||||||
assert.strictEqual(result.url, "https://github.com/embark-framework/embark/archive/develop.zip");
|
assert.strictEqual(result.url, "https://gitlab.com/embark-framework/embark/repository/archive.tar.gz?ref=master");
|
||||||
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/develop/archive.zip");
|
|
||||||
});
|
|
||||||
|
|
||||||
it('return correct zip filename with just username/repo specified', function () {
|
|
||||||
let result = templateGenerator.getExternalProject("embark-framework/embark");
|
|
||||||
assert.strictEqual(result.url, "https://github.com/embark-framework/embark/archive/master.zip");
|
|
||||||
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/master/archive.zip");
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/master/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://gitlab.com/embark-framework/embark");
|
||||||
|
|
||||||
|
result = templateGenerator.getExternalProject("https://gitlab.com/embark-framework/embark#features/branch");
|
||||||
|
assert.strictEqual(result.url, "https://gitlab.com/embark-framework/embark/repository/archive.tar.gz?ref=features%2Fbranch");
|
||||||
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/features/branch/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://gitlab.com/embark-framework/embark/tree/features/branch");
|
||||||
});
|
});
|
||||||
|
|
||||||
it('return correct zip filename with just username/repo and branch specified', function () {
|
|
||||||
let result = templateGenerator.getExternalProject("embark-framework/embark#develop");
|
|
||||||
assert.strictEqual(result.url, "https://github.com/embark-framework/embark/archive/develop.zip");
|
|
||||||
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/develop/archive.zip");
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('return correct zip filename with just embark template specified', function () {
|
describe('with git host shortcut', function () {
|
||||||
let result = templateGenerator.getExternalProject("react");
|
|
||||||
assert.strictEqual(result.url, "https://github.com/embark-framework/embark-react-template/archive/master.zip");
|
it('returns correct info for GitHub shortcut', function () {
|
||||||
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark-react-template/archive.zip");
|
let result = templateGenerator.getExternalProject("github:embark-framework/embark");
|
||||||
|
assert.strictEqual(result.url, "https://codeload.github.com/embark-framework/embark/tar.gz/master");
|
||||||
|
|
||||||
|
result = templateGenerator.getExternalProject("embark-framework/embark");
|
||||||
|
assert.strictEqual(result.url, "https://codeload.github.com/embark-framework/embark/tar.gz/master");
|
||||||
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/master/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://github.com/embark-framework/embark");
|
||||||
|
|
||||||
|
result = templateGenerator.getExternalProject("embark-framework/embark#features/branch");
|
||||||
|
assert.strictEqual(result.url, "https://codeload.github.com/embark-framework/embark/tar.gz/features%2Fbranch");
|
||||||
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/features/branch/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://github.com/embark-framework/embark/tree/features/branch");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('returns correct info for Bitbucket shortcut', function () {
|
||||||
|
let result = templateGenerator.getExternalProject("bitbucket:embark-framework/embark");
|
||||||
|
assert.strictEqual(result.url, "https://bitbucket.org/embark-framework/embark/get/master.tar.gz");
|
||||||
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/master/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://bitbucket.org/embark-framework/embark");
|
||||||
|
|
||||||
|
result = templateGenerator.getExternalProject("bitbucket:embark-framework/embark#features/branch");
|
||||||
|
assert.strictEqual(result.url, "https://bitbucket.org/embark-framework/embark/get/features%2Fbranch.tar.gz");
|
||||||
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/features/branch/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://bitbucket.org/embark-framework/embark/src/features/branch");
|
||||||
|
});
|
||||||
|
|
||||||
|
it('returns correct info for GitLab shortcut', function () {
|
||||||
|
let result = templateGenerator.getExternalProject("gitlab:embark-framework/embark");
|
||||||
|
assert.strictEqual(result.url, "https://gitlab.com/embark-framework/embark/repository/archive.tar.gz?ref=master");
|
||||||
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/master/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://gitlab.com/embark-framework/embark");
|
||||||
|
|
||||||
|
result = templateGenerator.getExternalProject("gitlab:embark-framework/embark#features/branch");
|
||||||
|
assert.strictEqual(result.url, "https://gitlab.com/embark-framework/embark/repository/archive.tar.gz?ref=features%2Fbranch");
|
||||||
|
assert.strictEqual(result.filePath.replace(/\\/g,'/'), ".embark/templates/embark-framework/embark/features/branch/archive.zip");
|
||||||
|
assert.strictEqual(result.browse, "https://gitlab.com/embark-framework/embark/tree/features/branch");
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('with unsupported template specifier', function () {
|
||||||
|
|
||||||
|
it('raises an exception', function () {
|
||||||
|
assert.throws(() => templateGenerator.getExternalProject("bad://format"), /Unsupported/);
|
||||||
|
assert.throws(() => templateGenerator.getExternalProject("bad://format#/also/bad"), /Unsupported/);
|
||||||
|
assert.throws(() => templateGenerator.getExternalProject(/force an error/), Error);
|
||||||
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue