/************************************************************************** ** ** This file is part of Qt Creator ** ** Copyright (c) 2010 Hugues Delorme ** ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** No Commercial Usage ** ** This file contains pre-release code and may not be distributed. ** You may use this file in accordance with the terms and conditions ** contained in the Technology Preview License Agreement accompanying ** this package. ** ** 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, Nokia gives you certain additional ** rights. These rights are described in the Nokia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** If you have questions regarding the use of this file, please contact ** Nokia at qt-info@nokia.com. ** **************************************************************************/ #include "bazaarclient.h" #include "constants.h" #include #include #include #include #include namespace { void addBoolArgument(const QVariant &optionValue, const QString &optionName, QStringList *arguments) { if (arguments == 0) return; Q_ASSERT(optionValue.canConvert(QVariant::Bool)); if (optionValue.toBool()) arguments->append(optionName); } void addRevisionArgument(const QVariant &optionValue, QStringList *arguments) { if (arguments == 0) return; Q_ASSERT(optionValue.canConvert(QVariant::String)); const QString revision = optionValue.toString(); if (!revision.isEmpty()) (*arguments) << QLatin1String("-r") << revision; } } // Anonymous namespace namespace Bazaar { namespace Internal { BazaarClient::BazaarClient(const VCSBase::VCSBaseClientSettings &settings) : VCSBase::VCSBaseClient(settings) { } BranchInfo BazaarClient::synchronousBranchQuery(const QString &repositoryRoot) const { QFile branchConfFile(repositoryRoot + QDir::separator() + QLatin1String(Constants::BAZAARREPO) + QLatin1String("/branch/branch.conf")); if (!branchConfFile.open(QIODevice::ReadOnly)) return BranchInfo(QString(), false); QTextStream ts(&branchConfFile); QString branchLocation; QString isBranchBound; const QRegExp branchLocationRx("bound_location\\s*=\\s*(.+)$"); const QRegExp isBranchBoundRx("bound\\s*=\\s*(.+)$"); while (!ts.atEnd() && (branchLocation.isEmpty() || isBranchBound.isEmpty())) { const QString line = ts.readLine(); if (branchLocationRx.indexIn(line) != -1) branchLocation = branchLocationRx.cap(1); else if (isBranchBoundRx.indexIn(line) != -1) isBranchBound = isBranchBoundRx.cap(1); } if (isBranchBound.simplified().toLower() == QLatin1String("true")) return BranchInfo(branchLocation, true); return BranchInfo(repositoryRoot, false); } QString BazaarClient::findTopLevelForFile(const QFileInfo &file) const { const QString repositoryCheckFile = QLatin1String(Constants::BAZAARREPO) + QLatin1String("/branch-format"); return file.isDir() ? VCSBase::VCSBasePlugin::findRepositoryForDirectory(file.absoluteFilePath(), repositoryCheckFile) : VCSBase::VCSBasePlugin::findRepositoryForDirectory(file.absolutePath(), repositoryCheckFile); } QString BazaarClient::vcsEditorKind(VCSCommand cmd) const { switch(cmd) { case AnnotateCommand : return QLatin1String(Constants::ANNOTATELOG); case DiffCommand : return QLatin1String(Constants::DIFFLOG); case LogCommand : return QLatin1String(Constants::FILELOG); default : return QLatin1String(""); } return QLatin1String(""); } QStringList BazaarClient::cloneArguments(const QString &srcLocation, const QString &dstLocation, const ExtraCommandOptions &extraOptions) const { QStringList args; // Fetch extra options foreach (int iOption, extraOptions.keys()) { const QVariant iOptValue = extraOptions[iOption]; switch (iOption) { case UseExistingDirCloneOptionId : ::addBoolArgument(iOptValue, QLatin1String("--use-existing-dir"), &args); break; case StackedCloneOptionId : ::addBoolArgument(iOptValue, QLatin1String("--stacked"), &args); break; case StandAloneCloneOptionId : ::addBoolArgument(iOptValue, QLatin1String("--standalone"), &args); break; case BindCloneOptionId : ::addBoolArgument(iOptValue, QLatin1String("--bind"), &args); break; case SwitchCloneOptionId : ::addBoolArgument(iOptValue, QLatin1String("--switch"), &args); break; case HardLinkCloneOptionId : ::addBoolArgument(iOptValue, QLatin1String("--hardlink"), &args); break; case NoTreeCloneOptionId : ::addBoolArgument(iOptValue, QLatin1String("--no-tree"), &args); break; case RevisionCloneOptionId : ::addRevisionArgument(iOptValue, &args); break; default : Q_ASSERT(false); // Invalid option ! } } // end foreach () // Add arguments for common options args << srcLocation; if (!dstLocation.isEmpty()) args << dstLocation; return args; } QStringList BazaarClient::pullArguments(const QString &srcLocation, const ExtraCommandOptions &extraOptions) const { // Fetch extra options QStringList args(commonPullOrPushArguments(extraOptions)); foreach (int iOption, extraOptions.keys()) { const QVariant iOptValue = extraOptions[iOption]; switch (iOption) { case RememberPullOrPushOptionId : break; case OverwritePullOrPushOptionId : break; case RevisionPullOrPushOptionId : break; case LocalPullOptionId : ::addBoolArgument(iOptValue, QLatin1String("--local"), &args); break; default : Q_ASSERT(false); // Invalid option ! } } // end foreach () // Add arguments for common options if (!srcLocation.isEmpty()) args << srcLocation; return args; } QStringList BazaarClient::pushArguments(const QString &dstLocation, const ExtraCommandOptions &extraOptions) const { // Fetch extra options QStringList args(commonPullOrPushArguments(extraOptions)); foreach (int iOption, extraOptions.keys()) { const QVariant iOptValue = extraOptions[iOption]; switch (iOption) { case RememberPullOrPushOptionId : break; case OverwritePullOrPushOptionId : break; case RevisionPullOrPushOptionId : break; case UseExistingDirPushOptionId : ::addBoolArgument(iOptValue, QLatin1String("--use-existing-dir"), &args); break; case CreatePrefixPushOptionId : ::addBoolArgument(iOptValue, QLatin1String("--create-prefix"), &args); break; default : Q_ASSERT(false); // Invalid option ! } } // end foreach () // Add arguments for common options if (!dstLocation.isEmpty()) args << dstLocation; return args; } QStringList BazaarClient::commitArguments(const QStringList &files, const QString &commitMessageFile, const ExtraCommandOptions &extraOptions) const { QStringList args; // Fetch extra options foreach (int iOption, extraOptions.keys()) { const QVariant iOptValue = extraOptions[iOption]; switch (iOption) { case AuthorCommitOptionId : { Q_ASSERT(iOptValue.canConvert(QVariant::String)); const QString committerInfo = iOptValue.toString(); if (!committerInfo.isEmpty()) args << QString("--author=%1").arg(committerInfo); break; } case FixesCommitOptionId : { Q_ASSERT(iOptValue.canConvert(QVariant::StringList)); foreach (const QString& iFix, iOptValue.toStringList()) { if (!iFix.isEmpty()) args << QLatin1String("--fixes") << iFix; } break; } case LocalCommitOptionId : ::addBoolArgument(iOptValue, QLatin1String("--local"), &args); break; default : Q_ASSERT(false); // Invalid option ! } } // end foreach () // Add arguments for common options args << QLatin1String("-F") << commitMessageFile; args << files; return args; } QStringList BazaarClient::importArguments(const QStringList &files) const { QStringList args; if (!files.isEmpty()) args.append(files); return args; } QStringList BazaarClient::updateArguments(const QString &revision) const { QStringList args; if (!revision.isEmpty()) args << QLatin1String("-r") << revision; return args; } QStringList BazaarClient::revertArguments(const QString &file, const QString &revision) const { QStringList args; if (!revision.isEmpty()) args << QLatin1String("-r") << revision; if (!file.isEmpty()) args << file; return args; } QStringList BazaarClient::revertAllArguments(const QString &revision) const { QStringList args; if (!revision.isEmpty()) args << QLatin1String("-r") << revision; return args; } QStringList BazaarClient::annotateArguments(const QString &file, const QString &revision, int lineNumber) const { Q_UNUSED(lineNumber); QStringList args(QLatin1String("--long")); if (!revision.isEmpty()) args << QLatin1String("-r") << revision; return args << file; } QStringList BazaarClient::diffArguments(const QStringList &files) const { QStringList args; if (!files.isEmpty()) args.append(files); return args; } QStringList BazaarClient::logArguments(const QStringList &files) const { return diffArguments(files); } QStringList BazaarClient::statusArguments(const QString &file) const { QStringList args; args.append(QLatin1String("--short")); if (!file.isEmpty()) args.append(file); return args; } QStringList BazaarClient::viewArguments(const QString &revision) const { QStringList args(QLatin1String("log")); args << QLatin1String("-p") << QLatin1String("--gnu-changelog") << QLatin1String("-r") << revision; return args; } QPair BazaarClient::parseStatusLine(const QString &line) const { QPair status; if (!line.isEmpty()) { const QChar flagVersion = line[0]; if (flagVersion == QLatin1Char('+')) status.first = QLatin1String("Versioned"); else if (flagVersion == QLatin1Char('-')) status.first = QLatin1String("Unversioned"); else if (flagVersion == QLatin1Char('R')) status.first = QLatin1String("Renamed"); else if (flagVersion == QLatin1Char('?')) status.first = QLatin1String("Unknown"); else if (flagVersion == QLatin1Char('X')) status.first = QLatin1String("Nonexistent"); else if (flagVersion == QLatin1Char('C')) status.first = QLatin1String("Conflict"); else if (flagVersion == QLatin1Char('P')) status.first = QLatin1String("PendingMerge"); const int lineLength = line.length(); if (lineLength >= 2) { const QChar flagContents = line[1]; if (flagContents == QLatin1Char('N')) status.first = QLatin1String("Created"); else if (flagContents == QLatin1Char('D')) status.first = QLatin1String("Deleted"); else if (flagContents == QLatin1Char('K')) status.first = QLatin1String("KindChanged"); else if (flagContents == QLatin1Char('M')) status.first = QLatin1String("Modified"); } if (lineLength >= 3) { const QChar flagExec = line[2]; if (flagExec == QLatin1Char('*')) status.first = QLatin1String("ExecuteBitChanged"); } // The status string should be similar to "xxx file_with_changes" // so just should take the file name part and store it status.second = line.mid(4); } return status; } QStringList BazaarClient::commonPullOrPushArguments(const ExtraCommandOptions &extraOptions) const { QStringList args; foreach (int iOption, extraOptions.keys()) { const QVariant iOptValue = extraOptions[iOption]; switch (iOption) { case RememberPullOrPushOptionId : ::addBoolArgument(iOptValue, QLatin1String("--remember"), &args); break; case OverwritePullOrPushOptionId : ::addBoolArgument(iOptValue, QLatin1String("--overwrite"), &args); break; case RevisionPullOrPushOptionId : ::addRevisionArgument(iOptValue, &args); break; default : break; // Unknown option, do nothing } } // end foreach () return args; } } //namespace Internal } // namespace Bazaar