231 lines
8.2 KiB
C#
231 lines
8.2 KiB
C#
// SparkleShare, a collaboration and sharing tool.
|
|
// Copyright (C) 2010 Hylke Bons <hi@planetpeanut.uk>
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Lesser General Public License as
|
|
// published by the Free Software Foundation, either version 3 of the
|
|
// License, or (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
using System;
|
|
using System.Globalization;
|
|
using System.Text.RegularExpressions;
|
|
|
|
namespace Sparkles.Git {
|
|
|
|
public class GitCommand : SSHCommand {
|
|
|
|
public static string ExecPath;
|
|
|
|
|
|
static string git_path;
|
|
|
|
public static string GitPath {
|
|
get {
|
|
if (git_path == null)
|
|
git_path = LocateCommand ("git");
|
|
|
|
return git_path;
|
|
}
|
|
|
|
set {
|
|
git_path = value;
|
|
}
|
|
}
|
|
|
|
|
|
public static string GitVersion {
|
|
get {
|
|
if (GitPath == null)
|
|
GitPath = LocateCommand ("git");
|
|
|
|
var git_version = new Command (GitPath, "--version", false);
|
|
|
|
if (ExecPath != null)
|
|
git_version.SetEnvironmentVariable ("GIT_EXEC_PATH", ExecPath);
|
|
|
|
string version = git_version.StartAndReadStandardOutput ();
|
|
return version.Replace ("git version ", "");
|
|
}
|
|
}
|
|
|
|
|
|
public static string GitLFSVersion {
|
|
get {
|
|
if (GitPath == null)
|
|
GitPath = LocateCommand ("git");
|
|
|
|
var git_lfs_version = new Command (GitPath, "lfs version", false);
|
|
|
|
if (ExecPath != null)
|
|
git_lfs_version.SetEnvironmentVariable ("GIT_EXEC_PATH", ExecPath);
|
|
|
|
string version = git_lfs_version.StartAndReadStandardOutput ();
|
|
return version.Replace ("git-lfs/", "").Split (' ') [0];
|
|
}
|
|
}
|
|
|
|
|
|
public GitCommand (string working_dir, string args) : this (working_dir, args, null)
|
|
{
|
|
}
|
|
|
|
|
|
public GitCommand (string working_dir, string args, SSHAuthenticationInfo auth_info) : base (GitPath, args)
|
|
{
|
|
StartInfo.WorkingDirectory = working_dir;
|
|
|
|
string GIT_SSH_COMMAND = SSHCommand.SSHCommandPath;
|
|
|
|
if (auth_info != null)
|
|
GIT_SSH_COMMAND = FormatGitSSHCommand (auth_info);
|
|
|
|
if (ExecPath != null)
|
|
SetEnvironmentVariable ("GIT_EXEC_PATH", ExecPath);
|
|
|
|
SetEnvironmentVariable ("GIT_SSH_COMMAND", GIT_SSH_COMMAND);
|
|
SetEnvironmentVariable ("GIT_TERMINAL_PROMPT", "0");
|
|
|
|
// Don't let Git try to read the config options in PREFIX/etc or ~
|
|
SetEnvironmentVariable ("GIT_CONFIG_NOSYSTEM", "1");
|
|
SetEnvironmentVariable ("PREFIX", "");
|
|
SetEnvironmentVariable ("HOME", "");
|
|
|
|
SetEnvironmentVariable ("LANG", "en_US");
|
|
}
|
|
|
|
|
|
static Regex progress_regex = new Regex (@"([0-9]+)%", RegexOptions.Compiled);
|
|
static Regex progress_regex_lfs = new Regex (@".*\(([0-9]+) of ([0-9]+) files\).*", RegexOptions.Compiled);
|
|
static Regex progress_regex_lfs_skipped = new Regex (@".*\(([0-9]+) of ([0-9]+) files, ([0-9]+) skipped\).*", RegexOptions.Compiled);
|
|
static Regex speed_regex = new Regex (@"([0-9\.]+) ([KM])iB/s", RegexOptions.Compiled);
|
|
|
|
public static ErrorStatus ParseProgress (string line, out double percentage, out double speed, out string information)
|
|
{
|
|
percentage = 0;
|
|
speed = 0;
|
|
information = "";
|
|
|
|
Match match;
|
|
|
|
if (line.StartsWith ("Git LFS:")) {
|
|
match = progress_regex_lfs_skipped.Match (line);
|
|
|
|
int current_file = 0;
|
|
int total_file_count = 0;
|
|
int skipped_file_count = 0;
|
|
|
|
if (match.Success) {
|
|
// "skipped" files are objects that have already been transferred
|
|
skipped_file_count = int.Parse (match.Groups [3].Value);
|
|
|
|
} else {
|
|
|
|
match = progress_regex_lfs.Match (line);
|
|
|
|
if (!match.Success)
|
|
return ErrorStatus.None;
|
|
}
|
|
|
|
current_file = int.Parse (match.Groups [1].Value);
|
|
|
|
if (current_file == 0)
|
|
return ErrorStatus.None;
|
|
|
|
total_file_count = int.Parse (match.Groups [2].Value) - skipped_file_count;
|
|
|
|
percentage = Math.Round ((double) current_file / total_file_count * 100, 0);
|
|
information = string.Format ("{0} of {1} files", current_file, total_file_count);
|
|
|
|
return ErrorStatus.None;
|
|
}
|
|
|
|
match = progress_regex.Match (line);
|
|
|
|
if (!match.Success || string.IsNullOrWhiteSpace (line)) {
|
|
if (!string.IsNullOrWhiteSpace (line))
|
|
Logger.LogInfo ("Git", line);
|
|
|
|
return FindError (line);
|
|
}
|
|
|
|
int number = int.Parse (match.Groups [1].Value);
|
|
|
|
// The transfer process consists of two stages: the "Compressing
|
|
// objects" stage which we count as 20% of the total progress, and
|
|
// the "Writing objects" stage which we count as the last 80%
|
|
if (line.Contains ("Compressing objects")) {
|
|
// "Compressing objects" stage
|
|
percentage = (number / 100 * 20);
|
|
|
|
} else if (line.Contains ("Writing objects")) {
|
|
percentage = (number / 100 * 80 + 20);
|
|
Match speed_match = speed_regex.Match (line);
|
|
|
|
if (speed_match.Success) {
|
|
speed = double.Parse (speed_match.Groups [1].Value, new CultureInfo ("en-US")) * 1024;
|
|
|
|
if (speed_match.Groups [2].Value.Equals ("M"))
|
|
speed = speed * 1024;
|
|
|
|
information = speed.ToSize ();
|
|
}
|
|
}
|
|
|
|
return ErrorStatus.None;
|
|
}
|
|
|
|
|
|
static ErrorStatus FindError (string line)
|
|
{
|
|
ErrorStatus error = ErrorStatus.None;
|
|
|
|
if (line.Contains ("WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED!") ||
|
|
line.Contains ("WARNING: POSSIBLE DNS SPOOFING DETECTED!")) {
|
|
|
|
error = ErrorStatus.HostIdentityChanged;
|
|
}
|
|
|
|
if (line.StartsWith ("Permission denied") ||
|
|
line.StartsWith ("ssh_exchange_identification: Connection closed by remote host") ||
|
|
line.StartsWith ("The authenticity of host")) {
|
|
|
|
error = ErrorStatus.AuthenticationFailed;
|
|
}
|
|
|
|
if (line.EndsWith ("does not appear to be a git repository"))
|
|
error = ErrorStatus.NotFound;
|
|
|
|
if (line.EndsWith ("expected old/new/ref, got 'shallow"))
|
|
error = ErrorStatus.IncompatibleClientServer;
|
|
|
|
if (line.StartsWith ("error: Disk space exceeded") ||
|
|
line.EndsWith ("No space left on device") ||
|
|
line.EndsWith ("file write error (Disk quota exceeded)")) {
|
|
|
|
error = ErrorStatus.DiskSpaceExceeded;
|
|
}
|
|
|
|
return error;
|
|
}
|
|
|
|
|
|
public static string FormatGitSSHCommand (SSHAuthenticationInfo auth_info)
|
|
{
|
|
return SSHCommandPath + " " +
|
|
"-i " + auth_info.PrivateKeyFilePath.Replace ("\\", "/").Replace (" ", "\\ ") + " " +
|
|
"-o UserKnownHostsFile=" + auth_info.KnownHostsFilePath.Replace ("\\", "/").Replace (" ", "\\ ") + " " +
|
|
"-o IdentitiesOnly=yes" + " " + // Don't fall back to other keys on the system
|
|
"-o PasswordAuthentication=no" + " " + // Don't hang on possible password prompts
|
|
"-F /dev/null"; // Ignore the system's SSH config file
|
|
}
|
|
}
|
|
}
|