diff options
Diffstat (limited to 'Sparkles/Git/Git.Command.cs')
-rw-r--r-- | Sparkles/Git/Git.Command.cs | 231 |
1 files changed, 231 insertions, 0 deletions
diff --git a/Sparkles/Git/Git.Command.cs b/Sparkles/Git/Git.Command.cs new file mode 100644 index 0000000..ba314c7 --- /dev/null +++ b/Sparkles/Git/Git.Command.cs @@ -0,0 +1,231 @@ +// 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 + } + } +} |