// SparkleShare, a collaboration and sharing tool. // Copyright (C) 2010 Hylke Bons // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU 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 . using System; using System.Collections.Generic; using System.Diagnostics; using System.IO; using System.Security.Cryptography; using System.Text; using System.Text.RegularExpressions; using System.Threading; namespace SparkleLib { // Sets up a fetcher that can get remote folders public abstract class SparkleFetcherBase { public delegate void StartedEventHandler (); public delegate void FinishedEventHandler (string [] warnings); public delegate void FailedEventHandler (); public delegate void ProgressChangedEventHandler (double percentage); public event StartedEventHandler Started; public event FinishedEventHandler Finished; public event FailedEventHandler Failed; public event ProgressChangedEventHandler ProgressChanged; public abstract bool Fetch (); public abstract void Stop (); public string TargetFolder; public Uri RemoteUrl; public string [] ExcludeRules; public List Warnings = new List (); public bool IsActive { get; private set; } public string RequiredFingerprint; public bool FetchPriorHistory = false; private Thread thread; public SparkleFetcherBase (string server, string required_fingerprint, string remote_path, string target_folder, bool fetch_prior_history) { RequiredFingerprint = required_fingerprint; FetchPriorHistory = fetch_prior_history; remote_path = remote_path.Trim ("/".ToCharArray ()); if (server.EndsWith ("/")) server = server.Substring (0, server.Length - 1); if (!remote_path.StartsWith ("/")) remote_path = "/" + remote_path; if (!server.Contains ("://")) server = "ssh://" + server; TargetFolder = target_folder; RemoteUrl = new Uri (server + remote_path); IsActive = false; ExcludeRules = new string [] { // gedit and emacs "*~", // LibreOffice ".~lock.*", // Firefox and Chromium temporary download files "*.part", "*.crdownload", // vi(m) ".*.sw[a-z]", "*.un~", "*.swp", "*.swo", // KDE ".directory", // Mac OS X ".DS_Store", "Icon?", "._*", ".Spotlight-V100", ".Trashes", // Omnigraffle "*(Autosaved).graffle", // Windows "Thumbs.db", "Desktop.ini", // MS Office "~*.tmp", "~*.TMP", "*~*.tmp", "*~*.TMP", "~*.ppt", "~*.PPT", "~*.pptx", "~*.PPTX", "~*.xls", "~*.XLS", "~*.xlsx", "~*.XLSX", "~*.doc", "~*.DOC", "~*.docx", "~*.DOCX", // CVS "*/CVS/*", ".cvsignore", "*/.cvsignore", // Subversion "/.svn/*", "*/.svn/*", // Mercurial "/.hg/*", "*/.hg/*", "*/.hgignore", // Bazaar "/.bzr/*", "*/.bzr/*", "*/.bzrignore" }; } public void Start () { IsActive = true; if (Started != null) Started (); SparkleHelpers.DebugInfo ("Fetcher", "[" + TargetFolder + "] Fetching folder: " + RemoteUrl); if (Directory.Exists (TargetFolder)) Directory.Delete (TargetFolder, true); string host = RemoteUrl.Host; if (string.IsNullOrEmpty (host)) { if (Failed != null) Failed (); return; } string host_key = GetHostKey (); if (host_key == null) { if (Failed != null) Failed (); return; } bool warn = true; if (RequiredFingerprint != null) { string host_fingerprint = GetFingerprint (host_key); if (!RequiredFingerprint.Equals (host_fingerprint)) { SparkleHelpers.DebugInfo ("Auth", "Fingerprint doesn't match"); if (Failed != null) Failed (); return; } warn = false; SparkleHelpers.DebugInfo ("Auth", "Fingerprint matches"); } else { SparkleHelpers.DebugInfo ("Auth", "Skipping fingerprint check"); } AcceptHostKey (host_key, warn); this.thread = new Thread (new ThreadStart (delegate { if (Fetch ()) { Thread.Sleep (500); SparkleHelpers.DebugInfo ("Fetcher", "Finished"); IsActive = false; if (Finished != null) Finished (Warnings.ToArray ()); } else { Thread.Sleep (500); SparkleHelpers.DebugInfo ("Fetcher", "Failed"); IsActive = false; if (Failed != null) Failed (); } })); this.thread.Start (); } public void Dispose () { if (this.thread != null) { this.thread.Abort (); this.thread.Join (); } } protected void OnProgressChanged (double percentage) { if (ProgressChanged != null) ProgressChanged (percentage); } private string GetHostKey () { string host = RemoteUrl.Host; SparkleHelpers.DebugInfo ("Auth", "Fetching host key for " + host); Process process = new Process () { EnableRaisingEvents = true }; process.StartInfo.WorkingDirectory = SparkleConfig.DefaultConfig.TmpPath; process.StartInfo.UseShellExecute = false; process.StartInfo.RedirectStandardOutput = true; process.StartInfo.CreateNoWindow = true; process.StartInfo.FileName = "ssh-keyscan"; process.StartInfo.Arguments = "-t rsa " + host; process.Start (); // Reading the standard output HAS to go before // WaitForExit, or it will hang forever on output > 4096 bytes string host_key = process.StandardOutput.ReadToEnd ().Trim (); process.WaitForExit (); if (process.ExitCode == 0) return host_key; else return null; } // FIXME: Calculate fingerprint natively: decode base64 -> md5 private string GetFingerprint (string public_key) { string tmp_file_path = Path.Combine (SparkleConfig.DefaultConfig.TmpPath, "hostkey.tmp"); File.WriteAllText (tmp_file_path, public_key + Environment.NewLine); Process process = new Process () { EnableRaisingEvents = true }; process.StartInfo.WorkingDirectory = SparkleConfig.DefaultConfig.TmpPath; process.StartInfo.UseShellExecute = false; process.StartInfo.RedirectStandardOutput = true; process.StartInfo.CreateNoWindow = true; process.StartInfo.FileName = "ssh-keygen"; process.StartInfo.Arguments = "-lf " + tmp_file_path; process.Start (); // Reading the standard output HAS to go before // WaitForExit, or it will hang forever on output > 4096 bytes string fingerprint = process.StandardOutput.ReadToEnd ().Trim (); process.WaitForExit (); fingerprint = fingerprint.Substring (fingerprint.IndexOf (" ") + 1, 47); File.Delete (tmp_file_path); return fingerprint; } private void AcceptHostKey (string host_key, bool warn) { string ssh_config_path = Path.Combine (SparkleConfig.DefaultConfig.HomePath, ".ssh"); string known_hosts_file_path = Path.Combine (ssh_config_path, "known_hosts"); if (!File.Exists (known_hosts_file_path)) { if (!Directory.Exists (ssh_config_path)) Directory.CreateDirectory (ssh_config_path); File.Create (known_hosts_file_path).Close (); } string host = RemoteUrl.Host; string known_hosts = File.ReadAllText (known_hosts_file_path); string [] known_hosts_lines = File.ReadAllLines (known_hosts_file_path); foreach (string line in known_hosts_lines) { if (line.StartsWith (host + " ")) return; } if (known_hosts.EndsWith ("\n")) File.AppendAllText (known_hosts_file_path, host_key + "\n"); else File.AppendAllText (known_hosts_file_path, "\n" + host_key + "\n"); SparkleHelpers.DebugInfo ("Auth", "Accepted host key for " + host); if (warn) { Warnings.Add ("The accepted fingerprint is:\n" + GetFingerprint (host_key) + "\n\nIf the above key doesn't match due to a possible attack, delete this project folder immediately."); } } } }