/**************************************************************************** ** ** Copyright (C) 2017 Orgad Shaneh . ** Contact: https://www.qt.io/licensing/ ** ** 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 The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ****************************************************************************/ #include "authenticationdialog.h" #include "gerritparameters.h" #include "gerritserver.h" #include "../gitplugin.h" #include "../gitclient.h" #include #include #include #include #include using namespace Utils; using namespace Git::Internal; namespace Gerrit { namespace Internal { static const char defaultHostC[] = "codereview.qt-project.org"; static const char accountUrlC[] = "/accounts/self"; bool GerritUser::isSameAs(const GerritUser &other) const { if (!userName.isEmpty() && !other.userName.isEmpty()) return userName == other.userName; if (!fullName.isEmpty() && !other.fullName.isEmpty()) return fullName == other.fullName; return false; } GerritServer::GerritServer() : host(defaultHost()) , port(defaultPort) { } GerritServer::GerritServer(const QString &host, unsigned short port, const QString &userName, HostType type) : host(host) , port(port) , type(type) { user.userName = userName; } bool GerritServer::operator==(const GerritServer &other) const { if (port && other.port && port != other.port) return false; return host == other.host && user.isSameAs(other.user) && type == other.type; } QString GerritServer::defaultHost() { return QLatin1String(defaultHostC); } QString GerritServer::hostArgument() const { if (!authenticated || user.userName.isEmpty()) return host; return user.userName + '@' + host; } QString GerritServer::url(UrlType urlType) const { QString protocol; switch (type) { case Ssh: protocol = "ssh"; break; case Http: protocol = "http"; break; case Https: protocol = "https"; break; } QString res = protocol + "://"; if (type == Ssh || urlType != DefaultUrl) res += hostArgument(); else res += host; if (port) res += ':' + QString::number(port); if (type != Ssh) { res += rootPath; if (authenticated && urlType == RestUrl) res += "/a"; } return res; } bool GerritServer::fillFromRemote(const QString &remote, const GerritParameters ¶meters) { const GitRemote r(remote); if (!r.isValid) return false; if (r.protocol == "https") type = GerritServer::Https; else if (r.protocol == "http") type = GerritServer::Http; else if (r.protocol.isEmpty() || r.protocol == "ssh") type = GerritServer::Ssh; else return false; if (r.host.contains("github.com")) // Clearly not gerrit return false; host = r.host; port = r.port; user.userName = r.userName.isEmpty() ? parameters.server.user.userName : r.userName; if (type == GerritServer::Ssh) return true; curlBinary = parameters.curl; if (curlBinary.isEmpty() || !QFile::exists(curlBinary)) return false; rootPath = r.path; // Strip the last part of the path, which is always the repo name // The rest of the path needs to be inspected to find the root path // (can be http://example.net/review) ascendPath(); return resolveRoot(); } QStringList GerritServer::curlArguments() { // -k - insecure - do not validate certificate // -f - fail silently on server error // -n - use credentials from ~/.netrc (or ~/_netrc on Windows) // -sS - silent, except server error (no progress) // --basic, --digest - try both authentication types return {"-kfnsS", "--basic", "--digest"}; } int GerritServer::testConnection() { static GitClient *const client = GitPlugin::client(); const QStringList arguments = curlArguments() << (url(RestUrl) + accountUrlC); const SynchronousProcessResponse resp = client->vcsFullySynchronousExec( QString(), FileName::fromString(curlBinary), arguments, Core::ShellCommand::NoOutput); if (resp.result == SynchronousProcessResponse::Finished) { QString output = resp.stdOut(); output.remove(0, output.indexOf('\n')); // Strip first line QJsonDocument doc = QJsonDocument::fromJson(output.toUtf8()); if (!doc.isNull()) user.fullName = doc.object().value("name").toString(); return 200; } const QRegularExpression errorRegexp("returned error: (\\d+)"); QRegularExpressionMatch match = errorRegexp.match(resp.stdErr()); if (match.hasMatch()) return match.captured(1).toInt(); return 400; } bool GerritServer::setupAuthentication() { AuthenticationDialog dialog(this); if (!dialog.exec()) return false; authenticated = dialog.isAuthenticated(); return true; } bool GerritServer::ascendPath() { const int lastSlash = rootPath.lastIndexOf('/'); if (lastSlash == -1) return false; rootPath = rootPath.left(lastSlash); return true; } bool GerritServer::resolveRoot() { for (;;) { switch (testConnection()) { case 200: return true; case 401: return setupAuthentication(); case 404: if (!ascendPath()) return false; break; default: // unknown error - fail return false; } } return false; } } // namespace Internal } // namespace Gerrit