mirror of
https://github.com/smallmain/cocos-enhance-kit.git
synced 2025-01-14 23:11:06 +00:00
393 lines
12 KiB
JavaScript
393 lines
12 KiB
JavaScript
'use strict';
|
||
|
||
var Path = require('path');
|
||
var Chalk = require('chalk');
|
||
var Spawn = require('child_process').spawn;
|
||
var treekill = require('tree-kill');
|
||
var Async = require('async');
|
||
|
||
function exec(cmdArgs, path, cb, options) {
|
||
var timeout = (options && options.timeout) || 600000;
|
||
var autoRetry = options && options.autoRetry;
|
||
var autoKill = !options || (options.autoKill !== false);
|
||
|
||
console.log(Chalk.yellow('git ' + cmdArgs.join(' ')), 'in', Chalk.magenta(path));
|
||
|
||
var child = Spawn('git', cmdArgs, {
|
||
cwd: path,
|
||
stdio: [0, 'pipe', 'pipe']
|
||
});
|
||
|
||
var offbranch = false;
|
||
var aborted = false;
|
||
var timerId = -1;
|
||
|
||
function retry () {
|
||
console.log(Chalk.yellow(`restart "${cmdArgs[0]}": ${Path.basename(path)}`));
|
||
exec(cmdArgs, path, cb, options); // Object.assign({}, options, { autoRetry: false })
|
||
}
|
||
|
||
function onConnectionError () {
|
||
aborted = true;
|
||
clearTimeout(timerId);
|
||
console.log(Chalk.yellow(`connection timeout/error: ${Path.basename(path)}`));
|
||
treekill(child.pid);
|
||
if (autoRetry && !offbranch) {
|
||
retry();
|
||
}
|
||
else {
|
||
// console.log('+++send callback from connection timeout: ' + Path.basename(path));
|
||
cb(null, { offbranch });
|
||
}
|
||
}
|
||
|
||
timerId = setTimeout(onConnectionError, timeout);
|
||
|
||
child.stdout.on('data', function (data) {
|
||
if (aborted) return;
|
||
|
||
var text = path + ' ' + data.toString().trim();
|
||
|
||
// git stash pop
|
||
if (text.indexOf('CONFLICT (content): Merge conflict in') !== -1) {
|
||
console.error(Chalk.red(text));
|
||
process.exit(1);
|
||
return;
|
||
}
|
||
});
|
||
child.stderr.on('data', function(data) {
|
||
if (aborted) return;
|
||
|
||
var text = path + ' ' + data.toString().trim();
|
||
|
||
// git checkout ("overwritten by checkout")
|
||
// git pull ("overwritten by merge")
|
||
if (text.includes('Your local changes to the following files would be overwritten by')) {
|
||
if (!autoKill) {
|
||
console.log(Chalk.yellow(text));
|
||
clearTimeout(timerId);
|
||
aborted = true;
|
||
return cb(new Error(text));
|
||
}
|
||
}
|
||
|
||
// git pull ("error: cannot lock ref '...': ... (unable to update local ref)")
|
||
if (text.includes('error: cannot lock ref')) {
|
||
console.log(Chalk.yellow(text));
|
||
aborted = true;
|
||
setTimeout(retry, 500);
|
||
return;
|
||
}
|
||
|
||
if (text.includes('Aborting') || text.includes('fatal')) {
|
||
if (
|
||
text.indexOf('Invalid refspec') === -1 &&
|
||
text.indexOf('Couldn\'t find remote ref') === -1 &&
|
||
text.indexOf('remote fireball already exists') === -1
|
||
) {
|
||
if (text.includes('Could not read from remote repository') ||
|
||
text.includes('The remote end hung up unexpectedly')
|
||
) {
|
||
console.log(Chalk.yellow(text));
|
||
onConnectionError();
|
||
}
|
||
else {
|
||
console.error(Chalk.red(text));
|
||
process.exit(1);
|
||
}
|
||
return;
|
||
}
|
||
|
||
offbranch = true;
|
||
}
|
||
|
||
// normal message, not error
|
||
console.log(text);
|
||
});
|
||
if (cb) {
|
||
child.on('close', function (code, signal) {
|
||
if (aborted) return;
|
||
// console.log(`====closing process: ${Path.basename(path)}, code: ${code}, signal: ${signal}`);
|
||
clearTimeout(timerId);
|
||
// console.log('+++send callback from close event: ' + Path.basename(path));
|
||
cb (null, { offbranch });
|
||
});
|
||
}
|
||
}
|
||
|
||
function commit( repo, message, cb ) {
|
||
exec(['commit', '-m', message], repo, cb);
|
||
}
|
||
|
||
function checkout( repo, branch, fetch, cb ) {
|
||
var Async = require('async');
|
||
Async.series([
|
||
function ( next ) {
|
||
if (!fetch) return next();
|
||
exec(['fetch', '--all'], repo, next );
|
||
},
|
||
|
||
function ( next ) {
|
||
exec(['checkout', branch], repo, next);
|
||
},
|
||
], function ( err ) {
|
||
if ( err ) {
|
||
console.error(Chalk.red('Failed to checkout ' + repo + '. Message: ' + err.message ));
|
||
if (cb) cb (err);
|
||
return;
|
||
}
|
||
if (cb) cb();
|
||
});
|
||
}
|
||
|
||
function fetch( branch, repo, callback) {
|
||
exec(['fetch', 'fireball', branch], repo, callback, { autoRetry: true, timeout: 30000 } );
|
||
}
|
||
|
||
function checkoutTrack( branch, repo, callback) {
|
||
exec(['checkout', '-B', branch, '--track', 'fireball/' + branch], repo, callback, { autoRetry: true, timeout: 30000, autoKill: false } );
|
||
}
|
||
|
||
// 检查 git 的错误信息,看看是否可以通过调用指定的重置方法进行文件重置操作
|
||
function resetFiles (repo, log, resetFunctions) {
|
||
var fileListRE = /overwritten by \w+:\s*\n((?:\s+.*\n)*\s+.*)/;
|
||
var matches = fileListRE.exec(log);
|
||
if (matches) {
|
||
// pending changes detected
|
||
var filesToReset = matches[1].split(/\r\n|\r|\n/).map(x => x.trimLeft()).filter(x => resetFunctions[x]);
|
||
if (filesToReset.length > 0) {
|
||
// auto reset
|
||
var postProcesses = [];
|
||
for (var i = 0; i < filesToReset.length; ++i) {
|
||
var filename = filesToReset[i];
|
||
var postProcess = resetFunctions[filename](repo, Path.join(repo, filename));
|
||
if (postProcess) {
|
||
postProcesses.push(postProcess);
|
||
}
|
||
}
|
||
|
||
var postProcessAll = postProcesses.length > 0 ? function () {
|
||
for (var i = postProcesses.length - 1; i >= 0; --i) {
|
||
postProcesses[i]();
|
||
}
|
||
} : (function () {});
|
||
|
||
return { changed: true, postProcess: postProcessAll };
|
||
}
|
||
}
|
||
|
||
return { changed: false };
|
||
}
|
||
|
||
// autoResetFiles - 可选字典,用于指定要自动重置的文件。key 是文件名,例如 "package.json"
|
||
// value 是重置用的回调函数。回调函数调用时会传入要重置的文件路径以及重置后的回调函数。
|
||
function checkoutRemoteBranch (repo, branch, autoResetFiles, cb) {
|
||
if (typeof autoResetFiles === 'function') {
|
||
cb = autoResetFiles;
|
||
autoResetFiles = null;
|
||
}
|
||
Async.series([
|
||
function ( next ) {
|
||
fetch( branch, repo, next);
|
||
},
|
||
|
||
function ( next ) {
|
||
checkoutTrack(branch, repo, autoResetFiles ? function (err) {
|
||
if (!err) {
|
||
return next();
|
||
}
|
||
var res = resetFiles(repo, err.message, autoResetFiles);
|
||
if (res.changed) {
|
||
// retry after reset
|
||
checkoutTrack(branch, repo, (err) => {
|
||
res.postProcess(err);
|
||
next(err);
|
||
});
|
||
}
|
||
else {
|
||
next(err);
|
||
}
|
||
} : next);
|
||
}
|
||
], function ( err ) {
|
||
if ( err ) {
|
||
console.error(Chalk.red('Failed to checkout ' + repo + '. Message: ' + err.message ));
|
||
if (cb) cb (err);
|
||
return;
|
||
}
|
||
if (cb) cb();
|
||
});
|
||
}
|
||
|
||
function pull (repo, remote, branch, autoResetFiles, cb) {
|
||
if (!cb) {
|
||
cb = autoResetFiles;
|
||
autoResetFiles = null;
|
||
}
|
||
exec(['pull', remote, branch], repo, function (err, result) {
|
||
if (result && result.offbranch) {
|
||
console.log(Chalk.red(`Skip repos that has custom local branch checkout: "${repo}"`));
|
||
}
|
||
else if (err && autoResetFiles) {
|
||
var res = resetFiles(repo, err.message, autoResetFiles);
|
||
if (res.changed) {
|
||
// retry after reset
|
||
pull(repo, remote, branch, (err, result) => {
|
||
res.postProcess(err, result);
|
||
cb(err, result);
|
||
});
|
||
return;
|
||
}
|
||
}
|
||
cb(err);
|
||
}, { autoRetry: true, timeout: 50000, autoKill: !autoResetFiles });
|
||
}
|
||
|
||
function push( repo, remote, branch, cb ) {
|
||
var Async = require('async');
|
||
Async.series([
|
||
function ( next ) {
|
||
exec(['push', remote, branch], repo, next );
|
||
},
|
||
|
||
function ( next ) {
|
||
exec(['push', remote, '--tags'], repo, next );
|
||
},
|
||
], function ( err ) {
|
||
if ( err ) {
|
||
console.error(Chalk.red('Failed to push ' + repo + '. Message: ' + err.message ));
|
||
if (cb) cb (err);
|
||
return;
|
||
}
|
||
if (cb) cb ();
|
||
});
|
||
}
|
||
|
||
function getCurrentBranch( path ) {
|
||
var spawnSync = require('child_process').spawnSync;
|
||
var output = spawnSync('git', ['symbolic-ref', '--short', '-q', 'HEAD'], {
|
||
cwd: path,
|
||
});
|
||
// console.log(output);
|
||
return output.stdout.toString().trim().replace(/^heads\//, '');
|
||
}
|
||
|
||
function getCurrentCommit( path, cb ) {
|
||
var child = Spawn('git', ['rev-parse', 'HEAD'], {
|
||
cwd: path,
|
||
});
|
||
var commit, err;
|
||
child.stdout.on('data', function(data) {
|
||
commit = data.toString().trim();
|
||
});
|
||
child.stderr.on('data', function(data) {
|
||
err = data.toString();
|
||
});
|
||
child.on('close', function() {
|
||
cb(err, commit);
|
||
});
|
||
// console.log(output);
|
||
// return output.stdout.toString().trim();
|
||
}
|
||
|
||
function reportStatus( path, cb ) {
|
||
var output = '';
|
||
var Async = require('async');
|
||
Async.series([
|
||
function( next ) {
|
||
var statusOut = Spawn('git', ['status', '-s'], {
|
||
cwd: path,
|
||
});
|
||
statusOut.stdout.on('data', function(data) {
|
||
output += Chalk.yellow(data.toString());
|
||
});
|
||
statusOut.on('close', function() {
|
||
next();
|
||
});
|
||
},
|
||
// function( next ) {
|
||
// var cherryOut = Spawn('git', ['cherry', '-v'], {
|
||
// cwd: path,
|
||
// });
|
||
// cherryOut.stdout.on('data', function(data) {
|
||
// output += data.toString();
|
||
// });
|
||
// cherryOut.on('close', function() {
|
||
// next();
|
||
// });
|
||
// }
|
||
], function ( err ) {
|
||
if ( err ) {
|
||
console.error(Chalk.red('Failed to report status in ' + path + '. Message: ' + err.message ));
|
||
if (cb) cb (err);
|
||
return;
|
||
}
|
||
if (cb) cb (null, output);
|
||
});
|
||
}
|
||
|
||
function parseRepo( category, name ) {
|
||
var orgName = '';
|
||
var branch = '';
|
||
var localFolder = '';
|
||
|
||
switch(category) {
|
||
case 'fireball':
|
||
return {
|
||
name: 'fireball',
|
||
branch: '',
|
||
localPath: '.',
|
||
url: ''
|
||
};
|
||
case 'builtin':
|
||
orgName = 'cocos-creator-packages';
|
||
localFolder = 'builtin';
|
||
break;
|
||
case 'hosts':
|
||
orgName = 'cocos-creator';
|
||
break;
|
||
}
|
||
|
||
var nameList = name.split('/');
|
||
if (nameList.length === 2) {
|
||
orgName = nameList[0];
|
||
name = nameList[1];
|
||
}
|
||
|
||
var branchList = name.split('#');
|
||
if (branchList.length === 2) {
|
||
name = branchList[0];
|
||
branch = branchList[1];
|
||
}
|
||
|
||
var url = 'git@github.com:' + orgName + '/' + name;
|
||
var localPath = Path.join(localFolder, name);
|
||
if (!branch) branch = 'master';
|
||
var repoInfo = {
|
||
name: name,
|
||
branch: branch,
|
||
url: url,
|
||
localPath: localPath
|
||
};
|
||
return repoInfo;
|
||
}
|
||
|
||
function updateOriginUrl(path, newRemote, cb) {
|
||
var newUrl = 'git@github.com:' + newRemote + '/' + Path.basename(path);
|
||
exec(['remote', 'set-url', 'origin', newUrl], path, cb);
|
||
}
|
||
|
||
module.exports = {
|
||
exec: exec,
|
||
checkout: checkout,
|
||
checkoutRemoteBranch: checkoutRemoteBranch,
|
||
commit: commit,
|
||
pull: pull,
|
||
push: push,
|
||
getCurrentBranch: getCurrentBranch,
|
||
getCurrentCommit: getCurrentCommit,
|
||
reportStatus: reportStatus,
|
||
parseRepo: parseRepo,
|
||
updateOriginUrl: updateOriginUrl
|
||
};
|