summaryrefslogtreecommitdiff
path: root/Sparkles/Git/Git.Command.cs
diff options
context:
space:
mode:
Diffstat (limited to 'Sparkles/Git/Git.Command.cs')
-rw-r--r--Sparkles/Git/Git.Command.cs231
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
+ }
+ }
+}