var _ = require('underscore'); var intl = require('../intl'); var Errors = require('../util/errors'); var CommandProcessError = Errors.CommandProcessError; var GitError = Errors.GitError; var Warning = Errors.Warning; var CommandResult = Errors.CommandResult; var crappyUnescape = function(str) { return str.replace(/'/g, "'").replace(///g, "/"); }; var assertOriginSpecified = function(generalArgs) { if (generalArgs[0] !== 'origin') { throw new GitError({ msg: intl.todo( generalArgs[0] + ' is not a remote in your repository! try origin' ) }); } }; var assertBranchIsRemoteTracking = function(engine, branchName) { branchName = crappyUnescape(branchName); if (!engine.refs[branchName]) { throw new GitError({ msg: intl.todo(branchName + ' is not a branch!') }); } var branch = engine.resolveID(branchName); if (branch.get('type') !== 'branch') { throw new GitError({ msg: intl.todo(branchName + ' is not a branch!') }); } var tracking = branch.getRemoteTrackingBranchID(); if (!tracking) { throw new GitError({ msg: intl.todo(branchName + ' is not a remote tracking branch!') }); } return tracking; }; var commandConfig = { commit: { sc: /^(gc|git ci)($|\s)/, regex: /^git +commit($|\s)/, options: [ '--amend', '-a', '-am', '-m' ], execute: function(engine, command) { var commandOptions = command.getOptionsMap(); command.acceptNoGeneralArgs(); if (commandOptions['-am'] && ( commandOptions['-a'] || commandOptions['-m'])) { throw new GitError({ msg: intl.str('git-error-options') }); } var msg = null; var args = null; if (commandOptions['-a']) { command.addWarning(intl.str('git-warning-add')); } if (commandOptions['-am']) { args = commandOptions['-am']; command.validateArgBounds(args, 1, 1, '-am'); msg = args[0]; } if (commandOptions['-m']) { args = commandOptions['-m']; command.validateArgBounds(args, 1, 1, '-m'); msg = args[0]; } var newCommit = engine.commit({ isAmend: commandOptions['--amend'] }); if (msg) { msg = msg .replace(/"/g, '"') .replace(/^"/g, '') .replace(/"$/g, ''); newCommit.set('commitMessage', msg); } var promise = engine.animationFactory.playCommitBirthPromiseAnimation( newCommit, engine.gitVisuals ); engine.animationQueue.thenFinish(promise); } }, cherrypick: { displayName: 'cherry-pick', regex: /^git +cherry-pick($|\s)/, execute: function(engine, command) { var commandOptions = command.getOptionsMap(); var generalArgs = command.getGeneralArgs(); command.validateArgBounds(generalArgs, 1, Number.MAX_VALUE); var set = engine.getUpstreamSet('HEAD'); // first resolve all the refs (as an error check) var toCherrypick = _.map(generalArgs, function(arg) { var commit = engine.getCommitFromRef(arg); // and check that its not upstream if (set[commit.get('id')]) { throw new GitError({ msg: intl.str( 'git-error-already-exists', { commit: commit.get('id') } ) }); } return commit; }, this); engine.setupCherrypickChain(toCherrypick); } }, pull: { regex: /^git +pull($|\s)/, options: [ '--rebase' ], execute: function(engine, command) { if (!engine.hasOrigin()) { throw new GitError({ msg: intl.str('git-error-origin-required') }); } // here is the deal -- git pull is pretty complex with // the arguments it wants. Either you can: // A) specify the remote branch you want to // merge & fetch, in which case it completely // ignores the properties of branch you are on, or // // B) specify no args, in which case it figures out // the branch to fetch from the remote tracking // and merges those in. // so lets switch on A/B here var commandOptions = command.getOptionsMap(); var generalArgs = command.getGeneralArgs(); command.twoArgsImpliedOrigin(generalArgs); assertOriginSpecified(generalArgs); var tracking; if (generalArgs[1]) { tracking = assertBranchIsRemoteTracking(engine, generalArgs[1]); } else { // cant be detached if (engine.getDetachedHead()) { throw new GitError({ msg: intl.todo('Git pull can not be executed in detached HEAD mode if no remote branch specified!') }); } var oneBefore = engine.getOneBeforeCommit('HEAD'); tracking = assertBranchIsRemoteTracking(engine, oneBefore.get('id')); } engine.pull({ source: tracking, isRebase: commandOptions['--rebase'] }); } }, fakeTeamwork: { regex: /^git +fakeTeamwork($|\s)/, execute: function(engine, command) { var generalArgs = command.getGeneralArgs(); if (!engine.hasOrigin()) { throw new GitError({ msg: intl.str('git-error-origin-required') }); } command.validateArgBounds(generalArgs, 0, 2); // allow formats of: git Faketeamwork 2 or git Faketeamwork side 3 var branch = (engine.origin.refs[generalArgs[0]]) ? generalArgs[0] : 'master'; var numToMake = parseInt(generalArgs[0], 10) || generalArgs[1] || 1; // make sure its a branch and exists var destBranch = engine.origin.resolveID(branch); if (destBranch.get('type') !== 'branch') { throw new GitError({ msg: intl.str('git-error-options') }); } engine.fakeTeamwork(numToMake, branch); } }, clone: { regex: /^git +clone *?$/, execute: function(engine, command) { command.acceptNoGeneralArgs(); engine.makeOrigin(engine.printTree()); } }, remote: { regex: /^git +remote($|\s)/, options: [ '-v' ], execute: function(engine, command) { command.acceptNoGeneralArgs(); if (!engine.hasOrigin()) { throw new CommandResult({ msg: '' }); } engine.printRemotes({ verbose: !!command.getOptionsMap()['-v'] }); } }, fetch: { regex: /^git +fetch($|\s)/, execute: function(engine, command) { var options = {}; if (!engine.hasOrigin()) { throw new GitError({ msg: intl.str('git-error-origin-required') }); } var generalArgs = command.getGeneralArgs(); command.twoArgsImpliedOrigin(generalArgs); assertOriginSpecified(generalArgs); if (generalArgs[1]) { var tracking = assertBranchIsRemoteTracking(engine, generalArgs[1]); options.branches = [engine.refs[tracking]]; } engine.fetch(options); } }, branch: { sc: /^(gb|git br)($|\s)/, regex: /^git +branch($|\s)/, options: [ '-d', '-D', '-f', '-a', '-r', '--contains' ], execute: function(engine, command) { var commandOptions = command.getOptionsMap(); var generalArgs = command.getGeneralArgs(); var args = null; // handle deletion first if (commandOptions['-d'] || commandOptions['-D']) { var names = commandOptions['-d'] || commandOptions['-D']; command.validateArgBounds(names, 1, Number.MAX_VALUE, '-d'); _.each(names, function(name) { engine.deleteBranch(name); }); return; } if (commandOptions['--contains']) { args = commandOptions['--contains']; command.validateArgBounds(args, 1, 1, '--contains'); engine.printBranchesWithout(args[0]); return; } if (commandOptions['-f']) { args = commandOptions['-f']; command.twoArgsImpliedHead(args, '-f'); // we want to force a branch somewhere engine.forceBranch(args[0], args[1]); return; } if (generalArgs.length === 0) { var branches; if (commandOptions['-a']) { branches = engine.getBranches(); } else if (commandOptions['-r']) { branches = engine.getRemoteBranches(); } else { branches = engine.getLocalBranches(); } engine.printBranches(branches); return; } command.twoArgsImpliedHead(generalArgs); engine.branch(generalArgs[0], generalArgs[1]); } }, add: { dontCountForGolf: true, sc: /^ga($|\s)/, regex: /^git +add($|\s)/, execute: function() { throw new CommandResult({ msg: intl.str('git-error-staging') }); } }, reset: { regex: /^git +reset($|\s)/, options: [ '--hard', '--soft' ], execute: function(engine, command) { var commandOptions = command.getOptionsMap(); var generalArgs = command.getGeneralArgs(); if (commandOptions['--soft']) { throw new GitError({ msg: intl.str('git-error-staging') }); } if (commandOptions['--hard']) { command.addWarning( intl.str('git-warning-hard') ); // dont absorb the arg off of --hard generalArgs = generalArgs.concat(commandOptions['--hard']); } command.validateArgBounds(generalArgs, 1, 1); if (engine.getDetachedHead()) { throw new GitError({ msg: intl.str('git-error-reset-detached') }); } engine.reset(generalArgs[0]); } }, revert: { regex: /^git +revert($|\s)/, execute: function(engine, command) { var generalArgs = command.getGeneralArgs(); command.validateArgBounds(generalArgs, 1, Number.MAX_VALUE); engine.revert(generalArgs); } }, merge: { regex: /^git +merge($|\s)/, options: [ '--no-ff' ], execute: function(engine, command) { var commandOptions = command.getOptionsMap(); var generalArgs = command.getGeneralArgs(); command.validateArgBounds(generalArgs, 1, 1); var newCommit = engine.merge( generalArgs[0], { noFF: !!commandOptions['--no-ff'] } ); if (newCommit === undefined) { // its just a fast forwrard engine.animationFactory.refreshTree( engine.animationQueue, engine.gitVisuals ); return; } engine.animationFactory.genCommitBirthAnimation( engine.animationQueue, newCommit, engine.gitVisuals ); } }, log: { dontCountForGolf: true, regex: /^git +log($|\s)/, execute: function(engine, command) { var generalArgs = command.getGeneralArgs(); if (generalArgs.length == 2) { // do fancy git log branchA ^branchB if (generalArgs[1][0] == '^') { engine.logWithout(generalArgs[0], generalArgs[1]); } else { throw new GitError({ msg: intl.str('git-error-options') }); } } command.oneArgImpliedHead(generalArgs); engine.log(generalArgs[0]); } }, show: { dontCountForGolf: true, regex: /^git +show($|\s)/, execute: function(engine, command) { var generalArgs = command.getGeneralArgs(); command.oneArgImpliedHead(generalArgs); engine.show(generalArgs[0]); } }, rebase: { sc: /^gr($|\s)/, options: [ '-i', '--aboveAll', '-p', '--preserve-merges' ], regex: /^git +rebase($|\s)/, execute: function(engine, command) { var commandOptions = command.getOptionsMap(); var generalArgs = command.getGeneralArgs(); if (commandOptions['-i']) { var args = commandOptions['-i']; command.twoArgsImpliedHead(args, ' -i'); engine.rebaseInteractive( args[0], args[1], { aboveAll: !!commandOptions['--aboveAll'] } ); return; } command.twoArgsImpliedHead(generalArgs); engine.rebase(generalArgs[0], generalArgs[1], { preserveMerges: commandOptions['-p'] || commandOptions['--preserve-merges'] }); } }, status: { dontCountForGolf: true, sc: /^(gst|gs|git st)($|\s)/, regex: /^git +status($|\s)/, execute: function(engine) { // no parsing at all engine.status(); } }, checkout: { sc: /^(go|git co)($|\s)/, regex: /^git +checkout($|\s)/, options: [ '-b', '-B', '-' ], execute: function(engine, command) { var commandOptions = command.getOptionsMap(); var generalArgs = command.getGeneralArgs(); var args = null; if (commandOptions['-b']) { if (generalArgs.length) { throw new GitError({ msg: intl.str('git-error-options') }); } // the user is really trying to just make a branch and then switch to it. so first: args = commandOptions['-b']; command.twoArgsImpliedHead(args, '-b'); var validId = engine.validateBranchName(args[0]); engine.branch(validId, args[1]); engine.checkout(validId); return; } if (commandOptions['-']) { // get the heads last location var lastPlace = engine.HEAD.get('lastLastTarget'); if (!lastPlace) { throw new GitError({ msg: intl.str('git-result-nothing') }); } engine.HEAD.set('target', lastPlace); return; } if (commandOptions['-B']) { args = commandOptions['-B']; command.twoArgsImpliedHead(args, '-B'); engine.forceBranch(args[0], args[1]); engine.checkout(args[0]); return; } command.validateArgBounds(generalArgs, 1, 1); engine.checkout(engine.crappyUnescape(generalArgs[0])); } }, push: { regex: /^git +push($|\s)/, execute: function(engine, command) { if (!engine.hasOrigin()) { throw new GitError({ msg: intl.str('git-error-origin-required') }); } var options = {}; // git push is pretty complex in terms of // the arguments it wants as well -- see // git pull for a more detailed description. var generalArgs = command.getGeneralArgs(); command.twoArgsImpliedOrigin(generalArgs); assertOriginSpecified(generalArgs); var tracking; if (generalArgs[1]) { tracking = assertBranchIsRemoteTracking(engine, generalArgs[1]); } else { var oneBefore = engine.getOneBeforeCommit('HEAD'); tracking = assertBranchIsRemoteTracking(engine, oneBefore.get('id')); } engine.push({ destination: tracking }); } } }; var instantCommands = [ [/^(git help($|\s)|git$)/, function() { var lines = [ intl.str('git-version'), '
', intl.str('git-usage'), _.escape(intl.str('git-usage-command')), '
', intl.str('git-supported-commands'), '
' ]; var commands = require('../commands').commands.getOptionMap()['git']; // build up a nice display of what we support _.each(commands, function(commandOptions, command) { lines.push('git ' + command); _.each(commandOptions, function(vals, optionName) { lines.push('\t ' + optionName); }, this); }, this); // format and throw var msg = lines.join('\n'); msg = msg.replace(/\t/g, '   '); throw new CommandResult({ msg: msg }); }] ]; exports.commandConfig = commandConfig; exports.instantCommands = instantCommands;