2022-06-25 00:23:03 +08:00

393 lines
12 KiB
JavaScript
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

'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
};