forked from qt-creator/qt-creator
Conflicts: src/plugins/git/changeselectiondialog.cpp src/plugins/git/changeselectiondialog.h src/plugins/git/gerrit/gerritplugin.cpp src/plugins/git/gitclient.cpp src/plugins/git/gitclient.h src/plugins/git/gitsettings.cpp src/plugins/git/gitsettings.h src/plugins/git/mergetool.cpp Change-Id: Icd1b2741da96395ed1b41903f453049a303e4791
448 lines
21 KiB
C++
448 lines
21 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
|
|
** Contact: http://www.qt-project.org/legal
|
|
**
|
|
** This file is part of Qt Creator.
|
|
**
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and Digia. For licensing terms and
|
|
** conditions see http://qt.digia.com/licensing. For further information
|
|
** use the contact form at http://qt.digia.com/contact-us.
|
|
**
|
|
** GNU Lesser General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
** General Public License version 2.1 as published by the Free Software
|
|
** Foundation and appearing in the file LICENSE.LGPL included in the
|
|
** packaging of this file. Please review the following information to
|
|
** ensure the GNU Lesser General Public License version 2.1 requirements
|
|
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
**
|
|
** In addition, as a special exception, Digia gives you certain additional
|
|
** rights. These rights are described in the Digia Qt LGPL Exception
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
**
|
|
****************************************************************************/
|
|
|
|
#ifndef GITCLIENT_H
|
|
#define GITCLIENT_H
|
|
|
|
#include "gitsettings.h"
|
|
#include "commitdata.h"
|
|
|
|
#include <coreplugin/editormanager/ieditor.h>
|
|
|
|
#include <utils/fileutils.h>
|
|
|
|
#include <QFutureSynchronizer>
|
|
#include <QObject>
|
|
#include <QString>
|
|
#include <QStringList>
|
|
#include <QWidget>
|
|
|
|
QT_BEGIN_NAMESPACE
|
|
class QCheckBox;
|
|
class QSignalMapper;
|
|
class QDebug;
|
|
class QProcessEnvironment;
|
|
class QMenu;
|
|
QT_END_NAMESPACE
|
|
|
|
namespace Core { class ICore; }
|
|
|
|
namespace VcsBase {
|
|
class Command;
|
|
class SubmitFileModel;
|
|
class VcsBaseEditorWidget;
|
|
class VcsBaseEditorParameterWidget;
|
|
}
|
|
|
|
namespace Utils { struct SynchronousProcessResponse; }
|
|
|
|
namespace DiffEditor {
|
|
class DiffEditorDocument;
|
|
class DiffEditorController;
|
|
}
|
|
|
|
namespace Git {
|
|
namespace Internal {
|
|
|
|
class CommitData;
|
|
struct GitSubmitEditorPanelData;
|
|
class Stash;
|
|
|
|
enum StatusMode
|
|
{
|
|
ShowAll = 0,
|
|
NoUntracked = 1,
|
|
NoSubmodules = 2
|
|
};
|
|
|
|
enum StashFlag {
|
|
Default = 0x00, /* Prompt and do not allow unstashed */
|
|
AllowUnstashed = 0x01,
|
|
NoPrompt = 0x02
|
|
};
|
|
|
|
class SubmoduleData
|
|
{
|
|
public:
|
|
QString dir;
|
|
QString url;
|
|
QString ignore;
|
|
};
|
|
|
|
typedef QMap<QString, SubmoduleData> SubmoduleDataMap;
|
|
|
|
class GitClient : public QObject
|
|
{
|
|
Q_OBJECT
|
|
|
|
public:
|
|
enum CommandInProgress { NoCommand, Revert, CherryPick,
|
|
Rebase, Merge, RebaseMerge };
|
|
|
|
class StashInfo
|
|
{
|
|
public:
|
|
StashInfo();
|
|
enum StashResult { StashUnchanged, StashCanceled, StashFailed,
|
|
Stashed, NotStashed /* User did not want it */ };
|
|
|
|
bool init(const QString &workingDirectory, const QString &command,
|
|
StashFlag flag = Default, PushAction pushAction = NoPush);
|
|
bool stashingFailed() const;
|
|
void end();
|
|
StashResult result() const { return m_stashResult; }
|
|
QString stashMessage() const { return m_message; }
|
|
|
|
private:
|
|
void stashPrompt(const QString &command, const QString &statusOutput, QString *errorMessage);
|
|
void executeStash(const QString &command, QString *errorMessage);
|
|
|
|
StashResult m_stashResult;
|
|
QString m_message;
|
|
QString m_workingDir;
|
|
GitClient *m_client;
|
|
StashFlag m_flags;
|
|
PushAction m_pushAction;
|
|
};
|
|
|
|
static const char *stashNamePrefix;
|
|
|
|
explicit GitClient(GitSettings *settings);
|
|
~GitClient();
|
|
|
|
Utils::FileName gitExecutable(bool *ok = 0, QString *errorMessage = 0) const;
|
|
unsigned gitVersion(QString *errorMessage = 0) const;
|
|
|
|
QString findRepositoryForDirectory(const QString &dir);
|
|
QString findGitDirForRepository(const QString &repositoryDir) const;
|
|
bool managesFile(const QString &workingDirectory, const QString &fileName) const;
|
|
|
|
void diff(const QString &workingDirectory, const QString &fileName);
|
|
void diff(const QString &workingDirectory,
|
|
const QStringList &unstagedFileNames,
|
|
const QStringList &stagedFileNames = QStringList());
|
|
void diffBranch(const QString &workingDirectory,
|
|
const QString &branchName);
|
|
void merge(const QString &workingDirectory, const QStringList &unmergedFileNames = QStringList());
|
|
|
|
void status(const QString &workingDirectory);
|
|
void log(const QString &workingDirectory, const QString &fileName = QString(),
|
|
bool enableAnnotationContextMenu = false, const QStringList &args = QStringList());
|
|
void reflog(const QString &workingDirectory);
|
|
void blame(const QString &workingDirectory, const QStringList &args, const QString &fileName,
|
|
const QString &revision = QString(), int lineNumber = -1);
|
|
void reset(const QString &workingDirectory, const QString &argument, const QString &commit = QString());
|
|
void addFile(const QString &workingDirectory, const QString &fileName);
|
|
bool synchronousLog(const QString &workingDirectory,
|
|
const QStringList &arguments,
|
|
QString *output,
|
|
QString *errorMessage = 0,
|
|
unsigned flags = 0);
|
|
bool synchronousAdd(const QString &workingDirectory, const QStringList &files);
|
|
bool synchronousDelete(const QString &workingDirectory,
|
|
bool force,
|
|
const QStringList &files);
|
|
bool synchronousMove(const QString &workingDirectory,
|
|
const QString &from,
|
|
const QString &to);
|
|
bool synchronousReset(const QString &workingDirectory,
|
|
const QStringList &files = QStringList(),
|
|
QString *errorMessage = 0);
|
|
bool synchronousCleanList(const QString &workingDirectory, QStringList *files, QStringList *ignoredFiles, QString *errorMessage);
|
|
bool synchronousApplyPatch(const QString &workingDirectory, const QString &file, QString *errorMessage, const QStringList &arguments = QStringList());
|
|
bool synchronousInit(const QString &workingDirectory);
|
|
bool synchronousCheckoutFiles(const QString &workingDirectory,
|
|
QStringList files = QStringList(),
|
|
QString revision = QString(), QString *errorMessage = 0,
|
|
bool revertStaging = true);
|
|
// Checkout ref
|
|
bool stashAndCheckout(const QString &workingDirectory, const QString &ref);
|
|
bool synchronousCheckout(const QString &workingDirectory, const QString &ref,
|
|
QString *errorMessage = 0);
|
|
|
|
QStringList setupCheckoutArguments(const QString &workingDirectory, const QString &ref);
|
|
void updateSubmodulesIfNeeded(const QString &workingDirectory, bool prompt);
|
|
|
|
// Do a stash and return identier.
|
|
enum { StashPromptDescription = 0x1, StashImmediateRestore = 0x2, StashIgnoreUnchanged = 0x4 };
|
|
QString synchronousStash(const QString &workingDirectory,
|
|
const QString &messageKeyword = QString(),
|
|
unsigned flags = 0, bool *unchanged = 0);
|
|
|
|
bool executeSynchronousStash(const QString &workingDirectory,
|
|
const QString &message = QString(),
|
|
QString *errorMessage = 0);
|
|
bool synchronousStashRestore(const QString &workingDirectory,
|
|
const QString &stash,
|
|
bool pop = false,
|
|
const QString &branch = QString());
|
|
bool synchronousStashRemove(const QString &workingDirectory,
|
|
const QString &stash = QString(),
|
|
QString *errorMessage = 0);
|
|
bool synchronousBranchCmd(const QString &workingDirectory, QStringList branchArgs,
|
|
QString *output, QString *errorMessage);
|
|
bool synchronousTagCmd(const QString &workingDirectory, QStringList tagArgs,
|
|
QString *output, QString *errorMessage);
|
|
bool synchronousForEachRefCmd(const QString &workingDirectory, QStringList args,
|
|
QString *output, QString *errorMessage = 0);
|
|
bool synchronousRemoteCmd(const QString &workingDirectory, QStringList remoteArgs,
|
|
QString *output, QString *errorMessage, bool silent = false);
|
|
|
|
QMap<QString,QString> synchronousRemotesList(const QString &workingDirectory,
|
|
QString *errorMessage = 0);
|
|
QStringList synchronousSubmoduleStatus(const QString &workingDirectory,
|
|
QString *errorMessage = 0);
|
|
SubmoduleDataMap submoduleList(const QString &workingDirectory);
|
|
bool synchronousShow(const QString &workingDirectory, const QString &id,
|
|
QString *output, QString *errorMessage);
|
|
|
|
bool synchronousRevListCmd(const QString &workingDirectory, const QStringList &arguments,
|
|
QString *output, QString *errorMessage = 0);
|
|
|
|
bool synchronousParentRevisions(const QString &workingDirectory,
|
|
const QStringList &files /* = QStringList() */,
|
|
const QString &revision,
|
|
QStringList *parents,
|
|
QString *errorMessage);
|
|
QString synchronousShortDescription(const QString &workingDirectory, const QString &revision);
|
|
QString synchronousShortDescription(const QString &workingDirectory, const QString &revision,
|
|
const QString &format);
|
|
|
|
QString synchronousCurrentLocalBranch(const QString &workingDirectory);
|
|
|
|
bool synchronousHeadRefs(const QString &workingDirectory, QStringList *output,
|
|
QString *errorMessage = 0);
|
|
QString synchronousTopic(const QString &workingDirectory);
|
|
bool synchronousRevParseCmd(const QString &workingDirectory, const QString &ref,
|
|
QString *output, QString *errorMessage = 0) const;
|
|
QString synchronousTopRevision(const QString &workingDirectory, QString *errorMessage = 0);
|
|
void synchronousTagsForCommit(const QString &workingDirectory, const QString &revision,
|
|
QString &precedes, QString &follows);
|
|
bool isRemoteCommit(const QString &workingDirectory, const QString &commit);
|
|
bool isFastForwardMerge(const QString &workingDirectory, const QString &branch);
|
|
|
|
bool cloneRepository(const QString &directory, const QByteArray &url);
|
|
QString vcsGetRepositoryURL(const QString &directory);
|
|
void fetch(const QString &workingDirectory, const QString &remote);
|
|
bool synchronousPull(const QString &workingDirectory, bool rebase);
|
|
void push(const QString &workingDirectory, const QStringList &pushArgs = QStringList());
|
|
bool synchronousMerge(const QString &workingDirectory, const QString &branch,
|
|
bool allowFastForward = true);
|
|
bool canRebase(const QString &workingDirectory) const;
|
|
void rebase(const QString &workingDirectory, const QString &argument);
|
|
void cherryPick(const QString &workingDirectory, const QString &argument);
|
|
void revert(const QString &workingDirectory, const QString &argument);
|
|
void asyncCommand(const QString &workingDirectory, const QStringList &arguments,
|
|
bool hasProgress = false);
|
|
bool synchronousRevert(const QString &workingDirectory, const QString &commit);
|
|
bool synchronousCherryPick(const QString &workingDirectory, const QString &commit);
|
|
void interactiveRebase(const QString &workingDirectory, const QString &commit, bool fixup);
|
|
void synchronousAbortCommand(const QString &workingDir, const QString &abortCommand);
|
|
QString synchronousTrackingBranch(const QString &workingDirectory,
|
|
const QString &branch = QString());
|
|
bool synchronousSetTrackingBranch(const QString &workingDirectory,
|
|
const QString &branch,
|
|
const QString &tracking);
|
|
|
|
// git svn support (asynchronous).
|
|
void synchronousSubversionFetch(const QString &workingDirectory);
|
|
void subversionLog(const QString &workingDirectory);
|
|
|
|
void stashPop(const QString &workingDirectory, const QString &stash);
|
|
void stashPop(const QString &workingDirectory);
|
|
void revert(const QStringList &files, bool revertStaging);
|
|
bool synchronousStashList(const QString &workingDirectory,
|
|
QList<Stash> *stashes,
|
|
QString *errorMessage = 0);
|
|
// Resolve a stash name from message (for IVersionControl's names).
|
|
bool stashNameFromMessage(const QString &workingDirectory,
|
|
const QString &messge, QString *name,
|
|
QString *errorMessage = 0);
|
|
|
|
QString readConfigValue(const QString &workingDirectory, const QString &configVar) const;
|
|
|
|
QTextCodec *encoding(const QString &workingDirectory, const QByteArray &configVar) const;
|
|
bool readDataFromCommit(const QString &repoDirectory, const QString &commit,
|
|
CommitData &commitData, QString *errorMessage = 0,
|
|
QString *commitTemplate = 0);
|
|
bool getCommitData(const QString &workingDirectory, QString *commitTemplate,
|
|
CommitData &commitData, QString *errorMessage);
|
|
|
|
bool addAndCommit(const QString &workingDirectory,
|
|
const GitSubmitEditorPanelData &data,
|
|
CommitType commitType,
|
|
const QString &amendSHA1,
|
|
const QString &messageFile,
|
|
VcsBase::SubmitFileModel *model);
|
|
|
|
enum StatusResult { StatusChanged, StatusUnchanged, StatusFailed };
|
|
StatusResult gitStatus(const QString &workingDirectory,
|
|
StatusMode mode,
|
|
QString *output = 0,
|
|
QString *errorMessage = 0);
|
|
|
|
CommandInProgress checkCommandInProgress(const QString &workingDirectory);
|
|
QString commandInProgressDescription(const QString &workingDirectory);
|
|
|
|
void continueCommandIfNeeded(const QString &workingDirectory, bool allowContinue = true);
|
|
|
|
QString extendedShowDescription(const QString &workingDirectory, const QString &text);
|
|
|
|
void launchGitK(const QString &workingDirectory, const QString &fileName);
|
|
void launchGitK(const QString &workingDirectory) { launchGitK(workingDirectory, QString()); }
|
|
bool launchGitGui(const QString &workingDirectory);
|
|
Utils::FileName gitBinDirectory();
|
|
|
|
void launchRepositoryBrowser(const QString &workingDirectory);
|
|
|
|
QStringList synchronousRepositoryBranches(const QString &repositoryURL);
|
|
|
|
GitSettings *settings() const;
|
|
|
|
QProcessEnvironment processEnvironment() const;
|
|
|
|
bool beginStashScope(const QString &workingDirectory, const QString &command,
|
|
StashFlag flag = Default, PushAction pushAction = NoPush);
|
|
StashInfo &stashInfo(const QString &workingDirectory);
|
|
void endStashScope(const QString &workingDirectory);
|
|
bool isValidRevision(const QString &revision) const;
|
|
void handleMergeConflicts(const QString &workingDir, const QString &commit, const QStringList &files, const QString &abortCommand);
|
|
void addFuture(const QFuture<void> &future);
|
|
|
|
static QString msgNoChangedFiles();
|
|
static QString msgNoCommits(bool includeRemote);
|
|
|
|
public slots:
|
|
void show(const QString &source,
|
|
const QString &id,
|
|
const QString &name = QString());
|
|
void saveSettings();
|
|
|
|
private slots:
|
|
void slotBlameRevisionRequested(const QString &workingDirectory, const QString &file,
|
|
QString change, int lineNumber);
|
|
void finishSubmoduleUpdate();
|
|
void fetchFinished(const QVariant &cookie);
|
|
void slotChunkActionsRequested(QMenu *menu, int diffFileIndex, int chunkIndex);
|
|
void slotStageChunk();
|
|
void slotUnstageChunk();
|
|
void branchesForCommit(const QString &revision);
|
|
|
|
private:
|
|
QString makePatch(int diffFileIndex, int chunkIndex, bool revert) const;
|
|
void stage(const QString &patch, bool revert);
|
|
QByteArray readConfigBytes(const QString &workingDirectory, const QString &configVar) const;
|
|
QTextCodec *getSourceCodec(const QString &file) const;
|
|
VcsBase::VcsBaseEditorWidget *findExistingVCSEditor(const char *registerDynamicProperty,
|
|
const QString &dynamicPropertyValue) const;
|
|
|
|
enum CodecType { CodecSource, CodecLogOutput, CodecNone };
|
|
VcsBase::VcsBaseEditorWidget *createVcsEditor(Core::Id kind,
|
|
QString title,
|
|
const QString &source,
|
|
CodecType codecType,
|
|
const char *registerDynamicProperty,
|
|
const QString &dynamicPropertyValue,
|
|
VcsBase::VcsBaseEditorParameterWidget *configWidget) const;
|
|
DiffEditor::DiffEditorDocument *createDiffEditor(const QString &documentId,
|
|
const QString &source,
|
|
const QString &title) const;
|
|
|
|
VcsBase::Command *createCommand(const QString &workingDirectory,
|
|
VcsBase::VcsBaseEditorWidget* editor = 0,
|
|
bool useOutputToWindow = false,
|
|
int editorLineNumber = -1);
|
|
|
|
VcsBase::Command *executeGit(const QString &workingDirectory,
|
|
const QStringList &arguments,
|
|
VcsBase::VcsBaseEditorWidget* editor = 0,
|
|
bool useOutputToWindow = false,
|
|
unsigned additionalFlags = 0,
|
|
int editorLineNumber = -1);
|
|
|
|
// Fully synchronous git execution (QProcess-based).
|
|
bool fullySynchronousGit(const QString &workingDirectory,
|
|
const QStringList &arguments,
|
|
QByteArray* outputText,
|
|
QByteArray* errorText = 0,
|
|
unsigned flags = 0) const;
|
|
|
|
// Synchronous git execution using Utils::SynchronousProcess, with
|
|
// log windows updating (using VcsBasePlugin::runVcs with flags).
|
|
inline Utils::SynchronousProcessResponse
|
|
synchronousGit(const QString &workingDirectory, const QStringList &arguments,
|
|
unsigned flags = 0, QTextCodec *outputCodec = 0);
|
|
|
|
// determine version as '(major << 16) + (minor << 8) + patch' or 0.
|
|
unsigned synchronousGitVersion(QString *errorMessage = 0) const;
|
|
|
|
enum RevertResult { RevertOk, RevertUnchanged, RevertCanceled, RevertFailed };
|
|
RevertResult revertI(QStringList files,
|
|
bool *isDirectory,
|
|
QString *errorMessage,
|
|
bool revertStaging);
|
|
void connectRepositoryChanged(const QString & repository, VcsBase::Command *cmd);
|
|
bool executeAndHandleConflicts(const QString &workingDirectory, const QStringList &arguments,
|
|
const QString &abortCommand = QString());
|
|
bool tryLauchingGitK(const QProcessEnvironment &env,
|
|
const QString &workingDirectory,
|
|
const QString &fileName,
|
|
const QString &gitBinDirectory);
|
|
bool cleanList(const QString &workingDirectory, const QString &flag, QStringList *files, QString *errorMessage);
|
|
|
|
enum ContinueCommandMode {
|
|
ContinueOnly,
|
|
SkipOnly,
|
|
SkipIfNoChanges
|
|
};
|
|
|
|
void continuePreviousGitCommand(const QString &workingDirectory, const QString &msgBoxTitle,
|
|
QString msgBoxText, const QString &buttonName,
|
|
const QString &gitCommand, ContinueCommandMode continueMode);
|
|
|
|
mutable Utils::FileName m_gitVersionForBinary;
|
|
mutable unsigned m_cachedGitVersion;
|
|
|
|
const QString m_msgWait;
|
|
GitSettings *m_settings;
|
|
QString m_gitQtcEditor;
|
|
QMap<QString, StashInfo> m_stashInfo;
|
|
QStringList m_updatedSubmodules;
|
|
bool m_disableEditor;
|
|
int m_contextDiffFileIndex;
|
|
int m_contextChunkIndex;
|
|
QPointer<DiffEditor::DiffEditorController> m_contextController;
|
|
QFutureSynchronizer<void> m_synchronizer; // for commit updates
|
|
};
|
|
|
|
} // namespace Internal
|
|
} // namespace Git
|
|
|
|
#endif // GITCLIENT_H
|