using Duplicati.Library.Common; using Duplicati.Library.Common.IO; using System; using System.Collections.Generic; using System.Linq; namespace Duplicati.Library.Snapshots { public class MSSQLDB : IEquatable { public string Name { get; } public string ID { get; } public List DataPaths { get; } public MSSQLDB(string Name, string ID, List DataPaths) { this.Name = Name; this.ID = ID; this.DataPaths = DataPaths; } bool IEquatable.Equals(MSSQLDB other) { return ID.Equals(other.ID); } public override int GetHashCode() { return ID.GetHashCode(); } public override bool Equals(object obj) { MSSQLDB db = obj as MSSQLDB; if (db != null) { return Equals(db); } else { return false; } } public static bool operator ==(MSSQLDB db1, MSSQLDB db2) { if (object.ReferenceEquals(db1, db2)) return true; if (object.ReferenceEquals(db1, null)) return false; if (object.ReferenceEquals(db2, null)) return false; return db1.Equals(db2); } public static bool operator !=(MSSQLDB db1, MSSQLDB db2) { if (object.ReferenceEquals(db1, db2)) return false; if (object.ReferenceEquals(db1, null)) return true; if (object.ReferenceEquals(db2, null)) return true; return !db1.Equals(db2); } } public class MSSQLUtility { /// /// The tag used for logging /// private static readonly string LOGTAG = Logging.Log.LogTagFromType(); /// /// The MS SQL VSS Writer Guid /// public static readonly Guid MSSQLWriterGuid = new Guid("a65faa63-5ea8-4ebc-9dbd-a0c4db26912a"); /// /// MS SQL is supported only on Windows platform /// public bool IsMSSQLInstalled { get; } /// /// Enumerated MS SQL DBs /// public List DBs { get { return m_DBs; } } private readonly List m_DBs; public MSSQLUtility() { m_DBs = new List(); if (!Platform.IsClientWindows) { IsMSSQLInstalled = false; return; } string[] arrInstalledInstances = null; var installed = Microsoft.Win32.Registry.GetValue(@"HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Microsoft SQL Server", "InstalledInstances", ""); if (installed is string s) { if (!string.IsNullOrWhiteSpace(s)) arrInstalledInstances = new string[] { s }; } else if (installed is string[] strings) arrInstalledInstances = strings; else if (installed != null) try { arrInstalledInstances = (string[])installed; } catch { } if(Environment.Is64BitOperatingSystem && arrInstalledInstances == null) { var installed32on64 = Microsoft.Win32.Registry.GetValue(@"HKEY_LOCAL_MACHINE\SOFTWARE\Wow6432Node\Microsoft\Microsoft SQL Server", "InstalledInstances", ""); if (installed32on64 is string on64) { if (!string.IsNullOrWhiteSpace(on64)) arrInstalledInstances = new string[] { on64 }; } else if (installed32on64 is string[] strings) arrInstalledInstances = strings; else if (installed32on64 != null) try { arrInstalledInstances = (string[])installed32on64; } catch { } } IsMSSQLInstalled = arrInstalledInstances != null && arrInstalledInstances.Length > 0; if (!IsMSSQLInstalled) Logging.Log.WriteInformationMessage(LOGTAG, "NoMSSQLInstance", "Cannot find any MS SQL Server instance. MS SQL Server is probably not installed."); } /// /// For all MS SQL databases it enumerate all associated paths using VSS data /// /// A collection of DBs and paths public void QueryDBsInfo() { if (!IsMSSQLInstalled) return; m_DBs.Clear(); using (var vssBackupComponents = new VssBackupComponents()) { var writerGUIDS = new [] { MSSQLWriterGuid }; try { vssBackupComponents.SetupWriters(writerGUIDS, null); } catch (Exception) { throw new Interface.UserInformationException("Microsoft SQL Server VSS Writer not found - cannot backup SQL databases.", "NoMsSqlVssWriter"); } foreach (var o in vssBackupComponents.ParseWriterMetaData(writerGUIDS)) { m_DBs.Add(new MSSQLDB(o.Name, o.LogicalPath + "\\" + o.Name, o.Paths.ConvertAll(m => m[0].ToString().ToUpperInvariant() + m.Substring(1)) .Distinct(Utility.Utility.ClientFilenameStringComparer) .OrderBy(a => a).ToList())); } } } } }