// Native const path = require('path'); const url = require('url'); const childProcess = require('child_process'); // Packages const fs = require('fs-promise'); const download = require('download'); const tmp = require('tmp-promise'); const isURL = require('is-url'); const cloneRepo = (parts, tmpDir) => new Promise((resolve, reject) => { let host; switch (parts.type) { case 'GitLab': host = `gitlab.com`; break; case 'Bitbucket': host = `bitbucket.org`; break; default: host = `github.com`; } const url = `https://${host}/${parts.main}`; const ref = parts.ref || (parts.type === 'Bitbucket' ? 'default' : 'master'); const cmd = `git clone ${url} --single-branch ${ref}`; childProcess.exec(cmd, { cwd: tmpDir.path }, (err, stdout) => { if (err) { reject(err); } resolve(stdout); }); }); const renameRepoDir = async (pathParts, tmpDir) => { const tmpContents = await fs.readdir(tmpDir.path); const oldTemp = path.join(tmpDir.path, tmpContents[0]); const newTemp = path.join(tmpDir.path, pathParts.main.replace('/', '-')); await fs.rename(oldTemp, newTemp); tmpDir.path = newTemp; return tmpDir; }; const capitalizePlatform = name => { const names = { github: 'GitHub', gitlab: 'GitLab', bitbucket: 'Bitbucket' }; return names[name]; }; const splittedURL = fullURL => { const parsedURL = url.parse(fullURL); const pathParts = parsedURL.path.split('/'); pathParts.shift(); // Set path to repo... const main = pathParts[0] + '/' + pathParts[1]; // ...and then remove it from the parts pathParts.splice(0, 2); // Assign Git reference let ref = pathParts.length >= 2 ? pathParts[1] : ''; // Firstly be sure that we haven know the ref type if (pathParts[0]) { // Then shorten the SHA of the commit if (pathParts[0] === 'commit' || pathParts[0] === 'commits') { ref = ref.substring(0, 7); } } // We're deploying master by default, // so there's no need to indicate it explicitly if (ref === 'master') { ref = ''; } return { main, ref, type: capitalizePlatform(parsedURL.host.split('.')[0]) }; }; const gitPathParts = main => { let ref = ''; if (isURL(main)) { return splittedURL(main); } if (main.split('/')[1].includes('#')) { const parts = main.split('#'); ref = parts[1]; main = parts[0]; } return { main, ref, type: capitalizePlatform('github') }; }; const downloadRepo = async repoPath => { const pathParts = gitPathParts(repoPath); const tmpDir = await tmp.dir({ // We'll remove it manually once deployment is done keep: true, // Recursively remove directory when calling respective method unsafeCleanup: true }); let gitInstalled = true; try { await cloneRepo(pathParts, tmpDir); } catch (err) { gitInstalled = false; } if (gitInstalled) { const renaming = await renameRepoDir(pathParts, tmpDir); return renaming; } let url; switch (pathParts.type) { case 'GitLab': { const ref = pathParts.ref ? `?ref=${pathParts.ref}` : ''; url = `https://gitlab.com/${pathParts.main}/repository/archive.tar` + ref; break; } case 'Bitbucket': url = `https://bitbucket.org/${pathParts.main}/get/${pathParts.ref || 'default'}.zip`; break; default: url = `https://api.github.com/repos/${pathParts.main}/tarball/${pathParts.ref}`; } try { await download(url, tmpDir.path, { extract: true }); } catch (err) { tmpDir.cleanup(); return false; } const renaming = await renameRepoDir(pathParts, tmpDir); return renaming; }; const isRepoPath = path => { if (!path) { return false; } const allowedHosts = ['github.com', 'gitlab.com', 'bitbucket.org']; if (isURL(path)) { const urlParts = url.parse(path); const slashSplitted = urlParts.path.split('/').filter(n => n); const notBare = slashSplitted.length >= 2; if (allowedHosts.includes(urlParts.host) && notBare) { return true; } return 'no-valid-url'; } return /[^\s\\]\/[^\s\\]/g.test(path); }; const fromGit = async (path, debug) => { let tmpDir = false; try { tmpDir = await downloadRepo(path); } catch (err) { if (debug) { console.log(`Could not download "${path}" repo from GitHub`); } } return tmpDir; }; module.exports = { gitPathParts, isRepoPath, fromGit };