aboutsummaryrefslogtreecommitdiffstats
path: root/src/plugins/cvs/cvsutils.cpp
diff options
context:
space:
mode:
authorFriedemann Kleint <[email protected]>2009-07-15 12:28:40 +0200
committerFriedemann Kleint <[email protected]>2009-07-15 12:28:40 +0200
commit18f9375501926ab637ccf68ca01bc9ea615ffced (patch)
tree0369a9ec55802780d0af0fc332edb6265abbb4c4 /src/plugins/cvs/cvsutils.cpp
parentd53129d336408aa162207bb501235f8378b7a131 (diff)
Add a CVS plugin for use with UNIX cvs or Tortoise CVS.
Diffstat (limited to 'src/plugins/cvs/cvsutils.cpp')
-rw-r--r--src/plugins/cvs/cvsutils.cpp238
1 files changed, 238 insertions, 0 deletions
diff --git a/src/plugins/cvs/cvsutils.cpp b/src/plugins/cvs/cvsutils.cpp
new file mode 100644
index 00000000000..fcd193dd3f4
--- /dev/null
+++ b/src/plugins/cvs/cvsutils.cpp
@@ -0,0 +1,238 @@
+/**************************************************************************
+**
+** This file is part of Qt Creator
+**
+** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
+**
+** Contact: Nokia Corporation ([email protected])
+**
+** Commercial Usage
+**
+** Licensees holding valid Qt Commercial licenses may use this file in
+** accordance with the Qt Commercial License Agreement provided with the
+** Software or, alternatively, in accordance with the terms contained in
+** a written agreement between you and Nokia.
+**
+** 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.
+**
+** If you are unsure which license is appropriate for your use, please
+** contact the sales department at http://www.qtsoftware.com/contact.
+**
+**************************************************************************/
+
+#include "cvsutils.h"
+
+#include <QtCore/QDebug>
+#include <QtCore/QRegExp>
+#include <QtCore/QStringList>
+
+namespace CVS {
+namespace Internal {
+
+CVS_Revision::CVS_Revision(const QString &rev) :
+ revision(rev)
+{
+}
+
+CVS_LogEntry::CVS_LogEntry(const QString &f) :
+ file(f)
+{
+}
+
+QDebug operator<<(QDebug d, const CVS_LogEntry &e)
+{
+ QDebug nospace = d.nospace();
+ nospace << "File: " << e.file << e.revisions.size() << '\n';
+ foreach(const CVS_Revision &r, e.revisions)
+ nospace << " " << r.revision << ' ' << r.date << ' ' << r.commitId << '\n';
+ return d;
+}
+
+/* Parse:
+\code
+RCS file: /repo/foo.h
+Working file: foo.h
+head: 1.2
+...
+----------------------------
+revision 1.2
+date: 2009-07-14 13:30:25 +0200; author: <author>; state: dead; lines: +0 -0; commitid: <id>;
+<message>
+----------------------------
+revision 1.1
+...
+=============================================================================
+\endcode */
+
+QList<CVS_LogEntry> parseLogEntries(const QString &o,
+ const QString &directory,
+ const QString filterCommitId)
+{
+ enum ParseState { FileState, RevisionState, StatusLineState };
+
+ QList<CVS_LogEntry> rc;
+ const QStringList lines = o.split(QString(QLatin1Char('\n')), QString::SkipEmptyParts);
+ ParseState state = FileState;
+
+ const QString workingFilePrefix = QLatin1String("Working file: ");
+ const QString revisionPrefix = QLatin1String("revision ");
+ const QString statusPrefix = QLatin1String("date: ");
+ const QString commitId = QLatin1String("commitid: ");
+ const QRegExp statusPattern = QRegExp(QLatin1String("^date: ([\\d\\-]+) .*commitid: ([^;]+);$"));
+ const QRegExp revisionPattern = QRegExp(QLatin1String("^revision ([\\d\\.]+)$"));
+ const QChar slash = QLatin1Char('/');
+ Q_ASSERT(statusPattern.isValid() && revisionPattern.isValid());
+ const QString fileSeparator = QLatin1String("=============================================================================");
+
+ // Parse using a state enumeration and regular expressions as not to fall for weird
+ // commit messages in state 'RevisionState'
+ foreach(const QString &line, lines) {
+ switch (state) {
+ case FileState:
+ if (line.startsWith(workingFilePrefix)) {
+ QString file = directory;
+ if (!file.isEmpty())
+ file += slash;
+ file += line.mid(workingFilePrefix.size()).trimmed();
+ rc.push_back(CVS_LogEntry(file));
+ state = RevisionState;
+ }
+ break;
+ case RevisionState:
+ if (revisionPattern.exactMatch(line)) {
+ rc.back().revisions.push_back(CVS_Revision(revisionPattern.cap(1)));
+ state = StatusLineState;
+ } else {
+ if (line == fileSeparator)
+ state = FileState;
+ }
+ break;
+ case StatusLineState:
+ if (statusPattern.exactMatch(line)) {
+ const QString commitId = statusPattern.cap(2);
+ if (filterCommitId.isEmpty() || filterCommitId == commitId) {
+ rc.back().revisions.back().date = statusPattern.cap(1);
+ rc.back().revisions.back().commitId = commitId;
+ } else {
+ rc.back().revisions.pop_back();
+ }
+ state = RevisionState;
+ }
+ }
+ }
+ // Purge out files with no matching commits
+ if (!filterCommitId.isEmpty()) {
+ for (QList<CVS_LogEntry>::iterator it = rc.begin(); it != rc.end(); ) {
+ if (it->revisions.empty()) {
+ it = rc.erase(it);
+ } else {
+ ++it;
+ }
+ }
+ }
+ return rc;
+}
+
+QString fixDiffOutput(QString d)
+{
+ if (d.isEmpty())
+ return d;
+ // Kill all lines starting with '?'
+ const QChar questionMark = QLatin1Char('?');
+ const QChar newLine = QLatin1Char('\n');
+ for (int pos = 0; pos < d.size(); ) {
+ const int endOfLinePos = d.indexOf(newLine, pos);
+ if (endOfLinePos == -1)
+ break;
+ const int nextLinePos = endOfLinePos + 1;
+ if (d.at(pos) == questionMark) {
+ d.remove(pos, nextLinePos - pos);
+ } else {
+ pos = nextLinePos;
+ }
+ }
+ return d;
+}
+
+// Parse "cvs status" output for added/modified/deleted files
+// "File: <foo> Status: Up-to-date"
+// "File: <foo> Status: Locally Modified"
+// "File: no file <foo> Status: Locally Removed"
+// "File: hup Status: Locally Added"
+// Not handled for commit purposes: "Needs Patch/Needs Merge"
+// In between, we might encounter "cvs status: Examining subdir"...
+// As we run the status command from the repository directory,
+// we need to add the full path, again.
+// stdout/stderr need to be merged to catch directories.
+
+// Parse out status keywords, return state enum or -1
+inline int stateFromKeyword(const QString &s)
+{
+ if (s == QLatin1String("Up-to-date"))
+ return -1;
+ if (s == QLatin1String("Locally Modified"))
+ return CVSSubmitEditor::LocallyModified;
+ if (s == QLatin1String("Locally Added"))
+ return CVSSubmitEditor::LocallyAdded;
+ if (s == QLatin1String("Locally Removed"))
+ return CVSSubmitEditor::LocallyRemoved;
+ return -1;
+}
+
+StateList parseStatusOutput(const QString &directory, const QString &output)
+{
+ StateList changeSet;
+ const QString fileKeyword = QLatin1String("File: ");
+ const QString statusKeyword = QLatin1String("Status: ");
+ const QString noFileKeyword = QLatin1String("no file ");
+ const QString directoryKeyword = QLatin1String("cvs status: Examining ");
+ const QString dotDir = QString(QLatin1Char('.'));
+ const QChar slash = QLatin1Char('/');
+
+ const QStringList list = output.split(QLatin1Char('\n'), QString::SkipEmptyParts);
+
+ QString path = directory;
+ if (!path.isEmpty())
+ path += slash;
+ foreach (const QString &l, list) {
+ // Status line containing file
+ if (l.startsWith(fileKeyword)) {
+ // Parse state
+ const int statusPos = l.indexOf(statusKeyword);
+ if (statusPos == -1)
+ continue;
+ const int state = stateFromKeyword(l.mid(statusPos + statusKeyword.size()).trimmed());
+ if (state == -1)
+ continue;
+ // Concatenate file name, Correct "no file <foo>"
+ QString fileName = l.mid(fileKeyword.size(), statusPos - fileKeyword.size()).trimmed();
+ if (state == CVSSubmitEditor::LocallyRemoved && fileName.startsWith(noFileKeyword))
+ fileName.remove(0, noFileKeyword.size());
+ changeSet.push_back(CVSSubmitEditor::StateFilePair(static_cast<CVSSubmitEditor::State>(state), path + fileName));
+ continue;
+ }
+ // Examining a new subdirectory
+ if (l.startsWith(directoryKeyword)) {
+ path = directory;
+ if (!path.isEmpty())
+ path += slash;
+ const QString newSubDir = l.mid(directoryKeyword.size()).trimmed();
+ if (newSubDir != dotDir) { // Skip Examining '.'
+ path += newSubDir;
+ path += slash;
+ }
+ continue;
+ }
+ }
+ return changeSet;
+}
+
+} // namespace Internal
+} // namespace CVS