/* Copyright (c) Cloud Software Group, Inc. * * Redistribution and use in source and binary forms, * with or without modification, are permitted provided * that the following conditions are met: * * * Redistributions of source code must retain the above * copyright notice, this list of conditions and the * following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the * following disclaimer in the documentation and/or other * materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ using System; using System.Collections.Generic; using System.Globalization; using System.Net; using System.Reflection; using System.Threading; using XenAdmin.Actions; using XenAdmin.Core; using XenAPI; using XenCenterLib; using System.Diagnostics; using System.IO; using System.Linq; using System.Xml; using System.Xml.Serialization; namespace XenAdmin.Network { [DebuggerDisplay("IXenConnection :{HostnameWithPort}")] public class XenConnection : IXenConnection, IXmlSerializable { private static readonly log4net.ILog log = log4net.LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); public string Hostname { get; set; } public int Port { get; set; } public string Username { get; set; } public string Password { get; set; } public string FriendlyName { get; set; } /// /// The last known name for the pool, in the form "'Pool friendly name' (hostname or IP)". /// public string LastConnectionFullName = ""; /// /// Whether this connection is saved in a disconnected state (i.e. don't reconnect on session restore). /// public bool SaveDisconnected { get; set; } /// /// Indicates whether this connection was created in the Add Server dialog. /// public bool fromDialog = false; private volatile bool _expectedDisruption; private volatile bool _suppressErrors; private volatile bool _preventResettingPasswordPrompt; private volatile bool _coordinatorMayChange; /// /// Whether we're expecting network disruption, say because we're reconfiguring the network at this time. In that case, we ignore /// keepalive failures, and expect task polling to be disrupted. /// public bool ExpectDisruption { get => _expectedDisruption; set => _expectedDisruption = value; } /// /// If we are 'expecting' this connection's Password property to contain the correct password /// (i.e. false if the user has just entered the password, true if it was restored from the saved session). /// public bool ExpectPasswordIsCorrect { get; set; } /// /// Used by the patch wizard, suppress any errors coming from reconnect attempts /// public bool SuppressErrors { get => _suppressErrors; set => _suppressErrors = value; } /// /// The password prompting function () is set to null when the connection is closed. /// Set this value to true in order to prevent that from happening. /// n.b.: remember to set the value back to false once it's not needed anymore /// public bool PreventResettingPasswordPrompt { get => _preventResettingPasswordPrompt; set => _preventResettingPasswordPrompt = value; } /// /// Indicates whether we are expecting the pool coordinator to change soon (e.g. when explicitly designating a new coordinator). /// public bool CoordinatorMayChange { get => _coordinatorMayChange; set => _coordinatorMayChange = value; } /// /// Set when we detect that Event.next() has become blocked and we need to reset the connection. See CA-33145. /// private bool EventNextBlocked = false; /// /// A cache of the pool's opaque_ref, last time this connection was connected. This will be /// Helper.NullOpaqueRef if it's never been connected. /// private string PoolOpaqueRef = Helper.NullOpaqueRef; /// /// Set after a successful connection attempt, before the cache is populated. /// private string CoordinatorIPAddress = ""; /// /// The lock that must be taken around connectTask and heartbeat. /// private readonly object connectTaskLock = new object(); private ConnectTask connectTask = null; /// /// This is the metrics monitor. Has to be accessed within the connectTaskLock. /// private Heartbeat heartbeat; /// /// Whether we are trying to automatically connect to the new coordinator. Set in HandleConnectionLost. /// Note: I think we are not using this correctly -- see CA-37864 for details -- but I'm not going /// to fix it unless it gives rise to a reported bug, because I can't test the fix. /// private volatile bool FindingNewCoordinator = false; /// /// The time at which we started looking for the new coordinator. /// private DateTime FindingNewCoordinatorStartedAt = DateTime.MinValue; /// /// Timeout before we consider that Event.next() has got blocked: see CA-33145 /// private const int EVENT_NEXT_TIMEOUT = 120 * 1000; // 2 minutes private string LastCoordinatorHostname = ""; public readonly object PoolMembersLock = new object(); private List _poolMembers = new List(); public List PoolMembers { get { return _poolMembers; } set { _poolMembers = value; } } private int PoolMemberIndex = 0; private System.Threading.Timer ReconnectionTimer = null; private ActionBase ConnectAction; private DateTime m_startTime = DateTime.MinValue; private int m_lastDebug; private TimeSpan _serverTimeOffset = TimeSpan.Zero; private object ServerTimeOffsetLock = new object(); /// /// The offset between the clock at the client and the clock at the server. server time + ServerTimeOffset = client time. /// This does not take the local timezone into account -- all calculations should be in UTC. /// For Orlando and greater, this value is set by XenMetricsMonitor, calling Host.get_servertime on a heartbeat. /// public TimeSpan ServerTimeOffset { get { lock (ServerTimeOffsetLock) { return _serverTimeOffset; } } set { lock (ServerTimeOffsetLock) { TimeSpan diff = _serverTimeOffset - value; if (diff.TotalSeconds < -1 || 1 < diff.TotalSeconds) { var now = DateTime.UtcNow; var debugMsg = string.Format("Time offset for {0} is now {1}. It's now {2} UTC here, and {3} UTC on the server.", Hostname, value, now.ToString("o", CultureInfo.InvariantCulture), (now.Subtract(value)).ToString("o", CultureInfo.InvariantCulture)); if (m_startTime.Ticks == 0)//log it the first time it is detected { m_startTime = now; log.Info(debugMsg); } //then log every 5mins int currDebug = (int)((now - m_startTime).TotalSeconds) / 300; if (currDebug > m_lastDebug) { m_lastDebug = currDebug; log.InfoFormat(debugMsg); } } _serverTimeOffset = value; } } } public string HostnameWithPort { get { return Port == ConnectionsManager.DEFAULT_XEN_PORT ? Hostname : Hostname + ':' + Port.ToString(); } } public string UriScheme { get { return Port == 8080 || Port == 80 ? Uri.UriSchemeHttp : Uri.UriSchemeHttps; } } public string Name { get { string result = Helpers.GetName(Helpers.GetPoolOfOne(this)); return !string.IsNullOrEmpty(result) ? result : !string.IsNullOrEmpty(FriendlyName) ? FriendlyName : Hostname; } } /// /// The cache of XenAPI objects for this connection. /// public ICache Cache { get; } = new Cache(); private readonly LockFreeQueue eventQueue = new LockFreeQueue(); private readonly System.Threading.Timer cacheUpdateTimer; /// /// Whether the cache for this connection has been populated. /// public bool CacheIsPopulated { get => _cacheIsPopulated; private set { _cacheIsPopulated = value; if (_cacheIsPopulated) CachePopulated?.Invoke(this); } } private bool _cacheIsPopulated; private bool cacheUpdaterRunning = false; private bool updatesWaiting = false; /// /// Initializes a new instance of the class. /// public XenConnection() { Port = ConnectionsManager.DEFAULT_XEN_PORT; Username = "root"; SaveDisconnected = false; ExpectPasswordIsCorrect = true; cacheUpdateTimer = new Timer(cacheUpdater); } #region Events /// /// Fired just before the cache is cleared (i.e. the cache is still populated). /// public event Action ClearingCache; public event Action CachePopulated; public event EventHandler ConnectionResult; public event Action ConnectionStateChanged; public event Action ConnectionLost; public event Action ConnectionClosed; public event Action ConnectionReconnecting; public event Action BeforeConnectionEnd; public event Action ConnectionMessageChanged; public event Action BeforeMajorChange; public event Action AfterMajorChange; /// /// Fired on the UI thread, once per batch of events in CacheUpdater. /// public event EventHandler XenObjectsUpdated; #endregion public void LoadFromDatabaseFile(string databaseFile) { IsSimulatedConnection = true; var foundConn = ConnectionsManager.XenConnections.Cast().FirstOrDefault(c => c.Hostname == databaseFile && c.CoordinatorIPAddress == CoordinatorIPAddress); if (foundConn != null && foundConn.IsConnected) throw new ConnectionExists(this); var document = new XmlDocument(); using (var sr = new StreamReader(databaseFile)) document.LoadXml(sr.ReadToEnd()); var db = new Db(document); var events = db.Tables.SelectMany(t => t.Rows).Select(r => r.ObjectChange).ToList(); var session = new Session(this, Path.GetFileName(Hostname), Port); connectTask = new ConnectTask(Hostname, Port) { Connected = true, Session = session }; OnBeforeMajorChange(false); Cache.Clear(); CacheIsPopulated = false; OnAfterMajorChange(false); OnBeforeMajorChange(false); Cache.UpdateFrom(this, events); OnAfterMajorChange(false); CacheIsPopulated = true; var pool = Cache.Pools[0]; CoordinatorIPAddress = Cache.Hosts.FirstOrDefault(h => h.opaque_ref == pool.master)?.address; HandleSuccessfulConnection(connectTask, pool); MarkConnectActionComplete(); OnXenObjectsUpdated(); } public NetworkCredential NetworkCredential { get; set; } public bool IsSimulatedConnection { get; private set; } public bool IsConnected { get { // sneaky, but avoids a lock ConnectTask t = connectTask; return t != null && t.Connected; } } public bool InProgress { get { return connectTask != null; } } /// /// Gets the Session passed to ConnectWorkerThread and used to update the cache in XenObjectDownloader. /// May return null. /// public Session Session { get { ConnectTask t = connectTask; return t == null ? null : t.Session; } } /// /// Create a duplicate of the Session that this connection is currently using. /// This will use a separate TCP stream, but the same authentication credentials. /// /// public Session DuplicateSession() { return DuplicateSession(Session.STANDARD_TIMEOUT); } public Session DuplicateSession(int timeout) { Session s = Session; if (s == null) throw new DisconnectionException(); return new Session(s, this) { Timeout = timeout }; } /// /// For retrieving an extra session using different credentials to those stored in the connection. Used for the sudo /// function in actions. Does not prompt for new credentials if the authentication fails. /// Does not change connection's Username and Password. /// /// /// /// public Session ElevatedSession(string username, string password) { return GetNewSession(Hostname, Port, username, password, true); } /// /// For retrieving a new session. /// /// /// /// /// /// /// null if the server password has changed, the user has been prompted for the new password, /// but the user has clicked cancel on the dialog. private Session GetNewSession(string hostname, int port, string username, string password, bool isElevated) { const int DELAY = 250; // unit = ms int attempt = 0; while (true) { attempt++; string uname = isElevated ? username : Username; string pwd = isElevated ? password : Password; // Keep the password that we're using for this iteration, as it may // be changed by another thread handling an authentication failure. // For elevated session we use the elevated username and password passed into this function, // as the connection's Username and Password are not updated. Session session = new Session(this, hostname, port); if (isElevated) session.IsElevatedSession = true; try { session.login_with_password(uname, pwd, Helper.APIVersionString(API_Version.LATEST), Session.UserAgent); NetworkCredential = new NetworkCredential(uname, pwd); return session; } catch (Failure f) { if (connectTask == null || connectTask.Cancelled) // the user has clicked cancel on the connection to server dialog { attempt = DEFAULT_MAX_SESSION_LOGIN_ATTEMPTS; // make sure we throw rather than try again throw new CancelledException(); // make the dialog pop up again } else if (f.ErrorDescription.Count > 0) { switch (f.ErrorDescription[0]) { case Failure.SESSION_AUTHENTICATION_FAILED: if (isElevated) throw; if (PromptForNewPassword(pwd)) attempt = 0; else { //user cannot provide correct credentials, we d/c now to save the confusion of having the server available //but unusable. EndConnect(); throw new CancelledException(); } break; case Failure.HOST_IS_SLAVE: // we know it is a supporter so there there is no need to try and connect again, we need to connect to the coordinator case Failure.RBAC_PERMISSION_DENIED: // No point retrying this, the user needs the read only role at least to log in case Failure.HOST_UNKNOWN_TO_MASTER: // Will never succeed, CA-74718 throw; default: if (isElevated) { if (attempt >= DEFAULT_MAX_SESSION_LOGIN_ATTEMPTS) throw; else break; } else break; } } else { if (attempt >= DEFAULT_MAX_SESSION_LOGIN_ATTEMPTS) throw; } } catch (WebException e) { if (e.Status == WebExceptionStatus.TrustFailure) throw new CancelledException(); if (e.Status == WebExceptionStatus.NameResolutionFailure || e.Status == WebExceptionStatus.ProtocolError || attempt >= DEFAULT_MAX_SESSION_LOGIN_ATTEMPTS) throw; } catch (UriFormatException) { // No point trying to connect more than once to a duff URI throw; } catch (Exception) { if (attempt >= DEFAULT_MAX_SESSION_LOGIN_ATTEMPTS) throw; } Thread.Sleep(DELAY); } } // This CompareTo() mimics the sort order we get from XenSearch, with // pools first, then hosts, then disconnected connections. It is used // in some dialogs where we don't do a proper search: see CA-57131 & // CA-60517 for examples. public int CompareTo(IXenConnection other) { if (this == other) return 0; if (other == null) return -1; Pool thisPool = Helpers.GetPool(this); Pool otherPool = Helpers.GetPool(other); int thisClass = (this.IsConnected ? (thisPool == null ? 2 : 1) : 3); int otherClass = (other.IsConnected ? (otherPool == null ? 2 : 1) : 3); if (thisClass != otherClass) return thisClass - otherClass; int result = StringUtility.NaturalCompare(Name, other.Name); if (result != 0) return result; Pool p1 = Helpers.GetPoolOfOne(this); Pool p2 = Helpers.GetPoolOfOne(other); if (p1 == null || p2 == null) return 0; // shouldn't happen once connected, but let's be safe return p1.opaque_ref.CompareTo(p2.opaque_ref); } private void SetPoolAndHostInAction(ActionBase action, Pool pool, string poolopaqueref) { if (pool != null && !string.IsNullOrEmpty(poolopaqueref)) { Pool p = new Pool(); p.Connection = this; p.opaque_ref = poolopaqueref; p.UpdateFrom(pool); Host h = new Host(); h.Connection = this; h.opaque_ref = pool.master.opaque_ref; h.name_label = Hostname; action.Pool = p; action.Host = h; } else { // Match the hack in XenSearch/GroupAlg.cs. We are creating fake Host objects to // represent the disconnected server, with the opaque_ref set to the connection's HostnameWithPort. // We need to do the same here, so that Actions for disconnected hosts (like failed connections) // are attached to the disconnected server correctly. Host host = new Host(); host.Connection = this; host.opaque_ref = HostnameWithPort; action.Host = host; } } private Func _promptForNewPassword; /// /// /// /// If true, if connection to the coordinator fails we will start trying to connect to /// each remembered supporter in turn. /// A function that prompts the user for the changed password for a server. public void BeginConnect(bool initiateCoordinatorSearch, Func promptForNewPassword) { _promptForNewPassword = promptForNewPassword; //InvokeHelper.Synchronizer is used for synchronizing the cache update. Must not be null at this point. It can be initialized through InvokeHelper.Initialize() Trace.Assert(InvokeHelper.Synchronizer != null); InvokeHelper.AssertOnEventThread(); if (initiateCoordinatorSearch) { FindingNewCoordinator = true; FindingNewCoordinatorStartedAt = DateTime.Now; } CoordinatorMayChange = false; if (!HandlePromptForNewPassword()) return; lock (connectTaskLock) { //if connectTask != null a connection is already in progress if (connectTask == null) { ClearEventQueue(); OnBeforeMajorChange(false); Cache.Clear(); OnAfterMajorChange(false); connectTask = new ConnectTask(Hostname, Port); heartbeat?.Stop(); heartbeat = null; heartbeat = new Heartbeat(this, XenAdminConfigManager.Provider.ConnectionTimeout); Thread t = new Thread(ConnectWorkerThread); t.Name = "Connection to " + Hostname; t.IsBackground = true; t.Start(connectTask); } } } private static object WaitForMonitor = new object(); private static int WaitForEventRegistered = 0; private static object WaitForEventRegisteredLock = new object(); public void WaitFor(Func predicate, Func cancelling) { lock (WaitForEventRegisteredLock) { if (WaitForEventRegistered == 0) XenObjectsUpdated += WakeWaitFor; WaitForEventRegistered++; } try { for (int i = 0; i < 120; i++) { lock (WaitForMonitor) { if (predicate() || (cancelling != null && cancelling())) return; System.Threading.Monitor.Wait(WaitForMonitor, 500); } } } finally { lock (WaitForEventRegisteredLock) { WaitForEventRegistered--; if (WaitForEventRegistered == 0) XenObjectsUpdated -= WakeWaitFor; } } } private void WakeWaitFor(object sender, EventArgs e) { lock (WaitForMonitor) { System.Threading.Monitor.PulseAll(WaitForMonitor); } } /// /// Equivalent to WaitForCache(xenref, null). /// /// /// /// public T WaitForCache(XenRef xenref) where T : XenObject { return WaitForCache(xenref, null); } /// /// Wait for the given object to arrive in our cache. Returns the XenObject, or null if it does /// not appear within the timeout (1 minute). /// /// This blocks, so must only be used on a background thread. /// /// /// A delegate to check whether to cancel. May be null, in which case it's ignored public T WaitForCache(XenRef xenref, Func cancelling) where T : XenObject { lock (WaitForEventRegisteredLock) { if (WaitForEventRegistered == 0) XenObjectsUpdated += WakeWaitFor; WaitForEventRegistered++; } try { for (int i = 0; i < 120; i++) { lock (WaitForMonitor) { T result = Resolve(xenref); if (result != null || (cancelling != null && cancelling())) return result; System.Threading.Monitor.Wait(WaitForMonitor, 500); } } return null; } finally { lock (WaitForEventRegisteredLock) { WaitForEventRegistered--; if (WaitForEventRegistered == 0) XenObjectsUpdated -= WakeWaitFor; } } } /// Whether the cache should be cleared (requires invoking onto the GUI thread) /// public void EndConnect(bool clearCache = true, bool exiting = false) { ConnectTask t = connectTask; connectTask = null; EndConnect(clearCache, t, exiting); } /// /// Closes the connecting dialog, stops the XenMetricsMonitor thread, marks this.task as Cancelled and /// logs out of the task's Session on a background thread. /// /// Whether the cache should be cleared (requires invoking onto the GUI thread) /// /// private void EndConnect(bool clearCache, ConnectTask task, bool exiting) { OnBeforeConnectionEnd(); lock (connectTaskLock) { heartbeat?.Stop(); heartbeat = null; if (task != null) { task.Cancelled = true; Session session = task.Session; task.Session = null; if (session != null) { Logout(session, exiting); } } } MarkConnectActionComplete(); log.Info($"Connection to {Hostname} is ended."); // Save list of addresses of current hosts in pool List members = new List(); foreach (Host host in Cache.Hosts) { members.Add(host.address); } lock (PoolMembersLock) { PoolMembers = members; } // Clear the XenAPI object cache if (clearCache) { ClearCache(); } if (!PreventResettingPasswordPrompt) { // CA-371356: Preventing the reset of the prompt allows // for it to be shown when attempting to reconnect to a host // whose password has changed since last login _promptForNewPassword = null; } OnConnectionClosed(); } /// /// Try to logout the given session. This will cause any threads blocking on Event.next() to get /// a XenAPI.Failure (which is better than them freezing around forever). /// Do on a background thread - otherwise, if the coordinator has died, then this will block /// until the timeout is reached (default 20s). /// However, in the case of exiting, the thread need to be set as foreground. /// Otherwise the logging out operation can be terminated when other foreground threads finish. /// /// May be null, in which case nothing happens. /// public void Logout(Session session, bool exiting = false) { if (session == null || session.opaque_ref == null) return; Thread t = new Thread(Logout_); t.Name = string.Format("Logging out session {0}", session.opaque_ref); if (exiting) { t.IsBackground = false; t.Priority = ThreadPriority.AboveNormal; } else { t.IsBackground = true; t.Priority = ThreadPriority.Lowest; } t.Start(session); } public void Logout() { Logout(Session); } private static void Logout_(object o) { Session session = (Session)o; try { log.Debug("Trying Session.logout() on background thread"); session.logout(); log.Debug("Session.logout() succeeded"); } catch (Exception e) { log.Debug("Session.logout() failed", e); } } public void Interrupt() { ConnectTask t = connectTask; ICache coll = Cache; connectTask = null; if (t != null && t.Connected) { string poolopaqueref = null; Pool pool = coll == null ? null : getAPool(coll, out poolopaqueref); t.Cancelled = true; HandleConnectionLost(t, pool, poolopaqueref); } } private void ClearCache() { OnClearingCache(); ClearEventQueue(); // This call to Clear needs to occur on the background thread, otherwise the event firing in response to all the changes // block in one big lump rather than the smaller pieces that you get when invoking onto the event thread on a finer // granularity. If you do all this on the event thread, then the app tends to go (Not Responding) when you lose a connection. // It doesn't actually occur on the background thread all the time. There's a path from AddServerDialog.ConnectToServer. Cache.Clear(); } private string GetReason(Exception error) { if (error is ArgumentException) { // This happens if the server API is incompatible with our bindings. This should // never happen in production, but will happen during development if a field // changes type, for example. return string.Format(Messages.SERVER_API_INCOMPATIBLE, BrandManager.BrandConsole); } else if (error is WebException) { WebException w = error as WebException; if (w.Status == WebExceptionStatus.NameResolutionFailure) { return string.Format(Messages.CONNECT_RESOLUTION_FAILURE, this.Hostname); } else if (w.Status == WebExceptionStatus.ConnectFailure) { return string.Format(Messages.CONNCET_CONNECTION_FAILURE, this.Hostname); } else if (w.Status == WebExceptionStatus.ReceiveFailure) { return string.Format(Messages.ERROR_NO_XENSERVER, this.Hostname); } else if (w.Status == WebExceptionStatus.SecureChannelFailure) { return string.Format(Messages.ERROR_SECURE_CHANNEL_FAILURE, this.Hostname); } else { return w.Message; } } else if (error is Failure && error != null && !string.IsNullOrEmpty(error.Message)) { Failure f = error as Failure; if (f.ErrorDescription[0] == Failure.RBAC_PERMISSION_DENIED) { // we use a different error message here from the standard one in friendly names return Messages.ERROR_NO_PERMISSION; } return error.Message; } else if (error is NullReferenceException && error.Source.StartsWith("CookComputing")) { return string.Format(Messages.CONNCET_CONNECTION_FAILURE, this.Hostname); } else if (error != null && !string.IsNullOrEmpty(error.Message)) { return error.Message; } else { return null; } } private void HandleSuccessfulConnection(ConnectTask task, Pool pool) { // Remove any other (disconnected) entries for this server from the tree var existingConnections = ConnectionsManager.XenConnections.Where(c => c.Hostname.Equals(task.Hostname) && !c.IsConnected).ToList(); foreach (var conn in existingConnections) ConnectionsManager.ClearCacheAndRemoveConnection(conn); // add server name to history (if it's not already there) XenAdminConfigManager.Provider.UpdateServerHistory(HostnameWithPort); if (!ConnectionsManager.XenConnectionsContains(this)) { lock (ConnectionsManager.ConnectionsLock) { ConnectionsManager.XenConnections.Add(this); } InvokeHelper.Invoke(XenAdminConfigManager.Provider.SaveSettingsIfRequired); } string name = string.IsNullOrEmpty(FriendlyName) || FriendlyName == task.Hostname ? task.Hostname : string.Format("{0} ({1})", FriendlyName, task.Hostname); string title = string.Format(Messages.CONNECTING_NOTICE_TITLE, name); string msg = string.Format(Messages.CONNECTING_NOTICE_TEXT, name); ConnectAction = new DummyAction(title, msg); SetPoolAndHostInAction(ConnectAction, pool, PoolOpaqueRef); ExpectPasswordIsCorrect = true; OnConnectionResult(true, null, null); log.InfoFormat("Completed connection phase for pool {0} ({1}:{2}, {3}).", FriendlyName, task.Hostname, task.Port, PoolOpaqueRef); } /// /// Check the password isn't null, which happens when the session is restored without remembering passwords. /// /// Whether to continue. private bool HandlePromptForNewPassword() { if (Password == null && !PromptForNewPassword(Password)) { // if false the user has cancelled, set the password back to null and return Password = null; return false; } return true; } private readonly object PromptLock = new object(); /// /// Prompts the user for the changed password for a server. /// /// /// private bool PromptForNewPassword(string old_password) { // Serialise prompting for new passwords, so that we don't get multiple dialogs pop up. lock (PromptLock) { if (Password != old_password) { // Some other thread has changed the password already. Retry using that one. return true; } bool result = (_promptForNewPassword != null) ? _promptForNewPassword(this, old_password) : false; return result; } } private void ClearEventQueue() { while (eventQueue.NotEmpty) { eventQueue.Dequeue(); } // Suspend the cache update timer cacheUpdateTimer.Change(Timeout.Infinite, Timeout.Infinite); } private void EventsPending() { lock (cacheUpdateTimer) { if (cacheUpdaterRunning) updatesWaiting = true; else cacheUpdateTimer.Change(50, -1); } } private void cacheUpdater(object state) { lock (cacheUpdateTimer) { if (cacheUpdaterRunning) { // there is a race-condition here which can be observed under high load: It is possible for the timer to fire even when // cacheUpdaterRunning = true. The check ensures we don't get multiple threads calling cacheUpdater // on the same connection. updatesWaiting = true; return; } cacheUpdaterRunning = true; updatesWaiting = false; } try { cacheUpdater_(); } finally { bool waiting; lock (cacheUpdateTimer) { waiting = updatesWaiting; cacheUpdaterRunning = false; } if (waiting) cacheUpdater(null); } } private void cacheUpdater_() { // Copy events off the event queue // as we don't want events delivered while we're // on the GUI thread to be included in this set of // updates, otherwise we might hose the gui thread // during an event storm (ie deleting 1000 vms) List events = new List(); while (eventQueue.NotEmpty) events.Add(eventQueue.Dequeue()); if (events.Count > 0) { InvokeHelper.Invoke(() => { OnBeforeMajorChange(false); bool fire = Cache.UpdateFrom(this, events); OnAfterMajorChange(false); if (fire) OnXenObjectsUpdated(); }); if (!CacheIsPopulated) { lock (connectTaskLock) heartbeat?.Start(); CacheIsPopulated = true; MarkConnectActionComplete(); log.Info($"Connection to {Hostname} successful."); } } } /// /// if not called the action will never finish and the gui will never close /// private void MarkConnectActionComplete() { if (ConnectAction != null && !ConnectAction.IsCompleted) { string title = string.Format(Messages.CONNECTION_OK_NOTICE_TITLE, Hostname); string msg = string.Format(Messages.CONNECTION_OK_NOTICE_TEXT, Hostname); ConnectAction.Title = title; ConnectAction.Description = msg; Pool pool = Helpers.GetPoolOfOne(this); if (pool != null) SetPoolAndHostInAction(ConnectAction, pool, PoolOpaqueRef); // mark the connect action as completed ConnectAction.Finished = DateTime.Now; ConnectAction.PercentComplete = 100; ConnectAction.IsCompleted = true; } } private const int DEFAULT_MAX_SESSION_LOGIN_ATTEMPTS = 3; private readonly string eventNextConnectionGroupName = Guid.NewGuid().ToString(); /// /// The main method for a connection to a XenServer. This method runs until the connection is lost, and /// contains a loop that after doing an initial cache fill processes events off the wire when they arrive. /// /// private void ConnectWorkerThread(object o) { ConnectTask task = (ConnectTask)o; Exception error = null; Pool pool = null; try { log.DebugFormat("IXenConnection: trying to connect to {0}", HostnameWithPort); Session session = GetNewSession(task.Hostname, task.Port, Username, Password, false); // Save the session so we can log it out later task.Session = session; if (session.APIVersion < API_Version.API_2_6) throw new ServerNotSupported(); // Event.next uses a different session with a shorter timeout: see CA-33145. Session eventNextSession = DuplicateSession(EVENT_NEXT_TIMEOUT); eventNextSession.ConnectionGroupName = eventNextConnectionGroupName; // this will force the eventNextSession onto its own set of TCP streams (see CA-108676) CacheIsPopulated = false; session.CacheWarming = true; string token = ""; bool eventsExceptionLogged = false; while (true) { if (task.Cancelled) break; EventNextBlocked = false; if (session.CacheWarming) { if (!task.Connected) { // We've started cache sync: update the dialog text OnConnectionMessageChanged(string.Format(Messages.LABEL_SYNC, this.Hostname)); } log.Debug("Cache is warming. Starting XenObjectDownloader.GetAllObjects"); XenObjectDownloader.GetAllObjects(session, eventQueue, task.GetCancelled, ref token); log.Debug("Cache is warming. XenObjectDownloader.GetAllObjects finished successfully"); session.CacheWarming = false; } else { try { log.Debug("Starting XenObjectDownloader.GetEvents"); XenObjectDownloader.GetEvents(eventNextSession, eventQueue, task.GetCancelled, ref token); log.Debug("Starting XenObjectDownloader.GetEvents finished successfully"); eventsExceptionLogged = false; } catch (Exception exn) { if (!ExpectDisruption) throw; log.DebugFormat("Exception (disruption is expected) in XenObjectDownloader.GetEvents: {0}", exn.GetType().Name); // ignoring some exceptions when disruption is expected if (exn is IOException || (exn is WebException webEx && (webEx.Status == WebExceptionStatus.KeepAliveFailure || webEx.Status == WebExceptionStatus.ConnectFailure))) { if (!eventsExceptionLogged) { log.Debug("Ignoring keepalive/connect failure, because disruption is expected"); eventsExceptionLogged = true; } } else { throw; } } } // check if requested to cancel if (task.Cancelled) break; if (!task.Connected) { lock (ConnectionsManager.ConnectionsLock) { pool = ObjectChange.GetPool(eventQueue, out PoolOpaqueRef); Host coordinator = ObjectChange.GetCoordinator(eventQueue); CoordinatorIPAddress = coordinator.address; foreach (IXenConnection iconn in ConnectionsManager.XenConnections) { XenConnection connection = iconn as XenConnection; Trace.Assert(connection != null); if (!connection.IsConnected) continue; bool sameRef = PoolOpaqueRef == connection.PoolOpaqueRef; if (!sameRef) continue; bool sameCoordinator = CoordinatorIPAddress == connection.CoordinatorIPAddress; if (sameRef && sameCoordinator) throw new ConnectionExists(connection); // CA-15633: XenCenter does not allow connection to host on which backup is restored. throw new BadRestoreDetected(connection); } task.Connected = true; string poolName = pool.Name(); FriendlyName = !string.IsNullOrEmpty(poolName) ? poolName : !string.IsNullOrEmpty(coordinator.Name()) ? coordinator.Name() : task.Hostname; } // ConnectionsLock log.DebugFormat("Getting server time for pool {0} ({1})...", FriendlyName, PoolOpaqueRef); SetServerTimeOffset(session, pool.master.opaque_ref); HandleSuccessfulConnection(task, pool); } EventsPending(); } } catch (Failure e) { if (task.Cancelled && e.ErrorDescription.Count > 0 && e.ErrorDescription[0] == Failure.SESSION_INVALID) { // Do nothing: this is probably a result of the user disconnecting, and us calling session.logout() } else { error = e; log.Warn(e); } } catch (WebException e) { error = e; log.Debug(e.Message); } catch (TargetInvocationException e) { error = e.InnerException; log.Error("TargetInvocationException", e); } catch (UriFormatException e) { // This can happen when the user types gobbledy-gook into the host-name field // of the add server dialog... error = e; log.Debug(e.Message); } catch (CancelledException) { task.Cancelled = true; } catch (DisconnectionException e) { error = e; log.Debug(e.Message); } catch (EventFromBlockedException e) { EventNextBlocked = true; error = e; log.Error(e, e); } catch (Exception e) { error = e; log.Error(e, e); } finally { HandleConnectionResult(task, error, pool); } } private void HandleConnectionResult(ConnectTask task, Exception error, Pool pool) { if (task != connectTask) { // We've been superseded by a newer ConnectTask. Exit silently without firing events. // Can happen when user disconnects while sync is taking place, then reconnects // (creating a new _connectTask) before the sync is complete. } else { ClearEventQueue(); connectTask = null; // clean up action so we don't stay open forever if (ConnectAction != null) ConnectAction.IsCompleted = true; if (error is ServerNotSupported) { EndConnect(true, task, false); log.Info(error.Message); OnConnectionResult(false, error.Message, error); } else if (task.Cancelled) { task.Connected = false; log.InfoFormat("IXenConnection: closing connection to {0}", this.HostnameWithPort); EndConnect(true, task, false); OnConnectionClosed(); } else if (task.Connected) { HandleConnectionLost(task, pool, PoolOpaqueRef); } else { // We never connected string reason = GetReason(error); log.WarnFormat("IXenConnection: failed to connect to {0}: {1}", HostnameWithPort, reason); Failure f = error as Failure; if (f != null && f.ErrorDescription[0] == Failure.HOST_IS_SLAVE) { //do not log an event in this case } else if (error is ConnectionExists) { //do not log an event in this case } else { // Create a new log message to say the connection attempt failed string title = string.Format(Messages.CONNECTION_FAILED_TITLE, HostnameWithPort); var action = new DummyAction(title, reason, reason); SetPoolAndHostInAction(action, pool, PoolOpaqueRef); action.Run(); } // We only want to continue the coordinator search in certain circumstances if (FindingNewCoordinator && (error is WebException || (f != null && f.ErrorDescription[0] != Failure.RBAC_PERMISSION_DENIED))) { if (f != null) { if (f.ErrorDescription[0] == XenAPI.Failure.HOST_IS_SLAVE) { log.DebugFormat("Found a member of {0} at {1}; redirecting to the coordinator at {2}", LastCoordinatorHostname, Hostname, f.ErrorDescription[1]); Hostname = f.ErrorDescription[1]; OnConnectionMessageChanged(string.Format(Messages.CONNECTION_REDIRECTING, LastCoordinatorHostname, Hostname)); ReconnectCoordinator(); } else if (f.ErrorDescription[0] == XenAPI.Failure.HOST_STILL_BOOTING) { log.DebugFormat("Found a member of {0} at {1}, but it's still booting; trying the next pool member", LastCoordinatorHostname, Hostname); MaybeStartNextPoolMemberTimer(reason, error); } else { log.DebugFormat("Found a member of {0} at {1}, but got a failure; trying the next pool member", LastCoordinatorHostname, Hostname); MaybeStartNextPoolMemberTimer(reason, error); } } else if (PoolMemberRemaining()) { log.DebugFormat("Connection to {0} failed; trying the next pool member", Hostname); MaybeStartNextPoolMemberTimer(reason, error); } else { if (ExpectDisruption || DateTime.Now - FindingNewCoordinatorStartedAt < SEARCH_NEW_COORDINATOR_STOP_AFTER) { log.DebugFormat("While trying to find a connection for {0}, tried to connect to every remembered host. Will now loop back through pool members again.", this.HostnameWithPort); lock (PoolMembersLock) { PoolMemberIndex = 0; } MaybeStartNextPoolMemberTimer(reason, error); } else if (LastCoordinatorHostname != "") { log.DebugFormat("Stopping search for new coordinator for {0}: timeout reached without success. Trying the old coordinator one last time", LastConnectionFullName); FindingNewCoordinator = false; Hostname = LastCoordinatorHostname; ReconnectCoordinator(); } else { OnConnectionResult(false, reason, error); } } } else { OnConnectionResult(false, reason, error); } } } } private void SetServerTimeOffset(Session session, string coordinator_opaqueref) { DateTime t = Host.get_servertime(session, coordinator_opaqueref); ServerTimeOffset = DateTime.UtcNow - t; } /// /// Called when a connection that had been made successfully is then lost. /// /// /// /// private void HandleConnectionLost(ConnectTask task, Pool pool, string poolopaqueref) { task.Connected = false; log.WarnFormat("Lost connection to {0}", this.HostnameWithPort); // Cancel all current actions to do with this connection if (!ExpectDisruption) { InvokeHelper.Invoke(() => ConnectionsManager.CancelAllActions(this)); } // Save list of addresses of current hosts in pool List members = new List(); foreach (Host host in Cache.Hosts) { if (!string.IsNullOrEmpty(host.address)) members.Add(host.address); } // Save coordinator's address so we don't try to reconnect to it first Host coordinator = Helpers.GetCoordinator(this); // Save ha_enabled status before we clear the cache bool ha_enabled = IsHAEnabled(); // NB line below clears the cache EndConnect(true, task, false); string description; LastCoordinatorHostname = Hostname; string poolName = pool.Name(); if (string.IsNullOrEmpty(poolName)) { LastConnectionFullName = HostnameWithPort; } else { LastConnectionFullName = string.Format("'{0}' ({1})", poolName, HostnameWithPort); } if (!EventNextBlocked && (CoordinatorMayChange || ha_enabled) && members.Count > 1) { log.DebugFormat("Will now try to connect to another pool member"); lock (PoolMembersLock) { PoolMembers.Clear(); PoolMembers.AddRange(members); PoolMemberIndex = 0; // Don't reconnect to the coordinator straight away, try a supporter first if (coordinator != null && PoolMembers[0] == coordinator.address && PoolMembers.Count > 1) { PoolMemberIndex = 1; } } FindingNewCoordinator = true; // Record the time at which we started the new coordinator search. FindingNewCoordinatorStartedAt = DateTime.Now; StartReconnectCoordinatorTimer(); description = string.Format(Messages.CONNECTION_LOST_NOTICE_COORDINATOR_IN_X_SECONDS, LastConnectionFullName, XenConnection.SEARCH_NEW_COORDINATOR_TIMEOUT_MS / 1000); log.DebugFormat("Beginning search for new coordinator; will give up after {0} seconds", SEARCH_NEW_COORDINATOR_STOP_AFTER.TotalSeconds); } else { log.DebugFormat("Will retry connection to {0} in {1} ms.", LastConnectionFullName, ReconnectHostTimeoutMs); StartReconnectSingleHostTimer(); description = string.Format(Messages.CONNECTION_LOST_RECONNECT_IN_X_SECONDS, LastConnectionFullName, ReconnectHostTimeoutMs / 1000); } string title = string.Format(Messages.CONNECTION_LOST_NOTICE_TITLE, LastConnectionFullName); var action = new DummyAction(title, description, description); SetPoolAndHostInAction(action, pool, poolopaqueref); action.Run(); OnConnectionLost(); } private bool PoolMemberRemaining() { lock (PoolMembersLock) { return PoolMemberIndex < PoolMembers.Count; } } private bool IsHAEnabled() { Pool pool = Helpers.GetPoolOfOne(this); return pool != null && pool.ha_enabled; } /// /// When we lose connection to a non-HA host, the timeout before we try reconnecting. /// private const int RECONNECT_HOST_TIMEOUT_MS = 120 * 1000; private const int RECONNECT_SHORT_TIMEOUT_MS = 5 * 1000; private int ReconnectHostTimeoutMs { get { if (EventNextBlocked || IsSimulatedConnection) return RECONNECT_SHORT_TIMEOUT_MS; return RECONNECT_HOST_TIMEOUT_MS; } } /// /// When HA is enabled, the timeout after losing connection to the coordinator before we start searching for a new coordinator. /// i.e. This should be the time it takes coordinator failover to be sorted out on the server, plus a margin. /// NB we already have an additional built-in delay - it takes time for us to decide that the host is not responding, /// and stop the connection to the dead host, before starting the search. /// private const int SEARCH_NEW_COORDINATOR_TIMEOUT_MS = 60 * 1000; /// /// When HA is enabled, and going through each of the supporters to try and find the new coordinator, the time between failing /// to connect to one supporter and trying to connect to the next in the list. /// private const int SEARCH_NEXT_SUPPORTER_TIMEOUT_MS = 15 * 1000; /// /// When going through each of the remembered members of the pool looking for the new coordinator, don't start another pass /// through connecting to each of the hosts if we've already been looking for this long. /// private static readonly TimeSpan SEARCH_NEW_COORDINATOR_STOP_AFTER = TimeSpan.FromMinutes(6); private void StartReconnectSingleHostTimer() { ReconnectionTimer = new System.Threading.Timer((TimerCallback)ReconnectSingleHostTimer, null, ReconnectHostTimeoutMs, ReconnectHostTimeoutMs); } private void StartReconnectCoordinatorTimer() { StartReconnectCoordinatorTimer(SEARCH_NEW_COORDINATOR_TIMEOUT_MS); } private void MaybeStartNextPoolMemberTimer(string reason, Exception error) { if (PoolMemberRemaining()) StartReconnectCoordinatorTimer(SEARCH_NEXT_SUPPORTER_TIMEOUT_MS); else OnConnectionResult(false, reason, error); } private void StartReconnectCoordinatorTimer(int timeout) { OnConnectionMessageChanged(string.Format(Messages.CONNECTION_WILL_RETRY_SUPPORTER, LastConnectionFullName.Ellipsise(25), timeout / 1000)); ReconnectionTimer = new System.Threading.Timer((TimerCallback)ReconnectCoordinatorTimer, null, timeout, Timeout.Infinite); } private void ReconnectSingleHostTimer(object state) { if (IsConnected || !ConnectionsManager.XenConnectionsContains(this)) { log.DebugFormat("Host {0} already reconnected", Hostname); if (ReconnectionTimer != null) { ReconnectionTimer.Dispose(); ReconnectionTimer = null; } return; } if (!ExpectDisruption) // only try once unless expect disruption { if (ReconnectionTimer != null) { ReconnectionTimer.Dispose(); ReconnectionTimer = null; } } log.DebugFormat("Reconnecting to server {0}...", Hostname); if (!XenAdminConfigManager.Provider.Exiting) { InvokeHelper.Invoke(delegate () { /*ConnectionResult += new EventHandler(XenConnection_ConnectionResult); CachePopulated += new EventHandler(XenConnection_CachePopulated);*/ BeginConnect(false, _promptForNewPassword); }); OnConnectionReconnecting(); } } private void ReconnectCoordinatorTimer(object state) { if (IsConnected || !ConnectionsManager.XenConnectionsContains(this)) { log.DebugFormat("Coordinator has been found for {0} at {1}", LastCoordinatorHostname, Hostname); return; } lock (PoolMembersLock) { if (PoolMemberIndex < PoolMembers.Count) { Hostname = PoolMembers[PoolMemberIndex]; PoolMemberIndex++; } } OnConnectionMessageChanged(string.Format(Messages.CONNECTION_RETRYING_SUPPORTER, LastConnectionFullName.Ellipsise(25), Hostname)); ReconnectCoordinator(); } private void ReconnectCoordinator() { // Add an informational entry to the log string title = string.Format(Messages.CONNECTION_FINDING_COORDINATOR_TITLE, LastConnectionFullName); string descr = string.Format(Messages.CONNECTION_FINDING_COORDINATOR_DESCRIPTION, LastConnectionFullName, Hostname); var action = new DummyAction(title, descr); SetPoolAndHostInAction(action, null, PoolOpaqueRef); action.Run(); log.DebugFormat("Looking for coordinator for {0} on {1}...", LastConnectionFullName, Hostname); if (!XenAdminConfigManager.Provider.Exiting) { InvokeHelper.Invoke(() => BeginConnect(false, _promptForNewPassword)); OnConnectionReconnecting(); } } private Pool getAPool(ICache objects, out string opaqueref) { var pools = objects.Pools; if (pools.Length > 0) { var pool = pools.First(); opaqueref = pool.opaque_ref; return pool; } Trace.Assert(false); opaqueref = null; return null; } private void OnClearingCache() { ClearingCache?.Invoke(this); } private void OnConnectionResult(bool connected, string reason, Exception error) { if (ConnectionResult != null) ConnectionResult(this, new ConnectionResultEventArgs(connected, reason, error)); OnConnectionStateChanged(); } private void OnConnectionClosed() { ConnectionClosed?.Invoke(this); OnConnectionStateChanged(); } private void OnConnectionLost() { ConnectionLost?.Invoke(this); OnConnectionStateChanged(); } private void OnConnectionReconnecting() { ConnectionReconnecting?.Invoke(this); OnConnectionStateChanged(); } private void OnBeforeConnectionEnd() { BeforeConnectionEnd?.Invoke(this); } private void OnConnectionStateChanged() { ConnectionStateChanged?.Invoke(this); } private void OnConnectionMessageChanged(string message) { ConnectionMessageChanged?.Invoke(this, message); } public void OnBeforeMajorChange(bool background) { BeforeMajorChange?.Invoke(this, background); } public void OnAfterMajorChange(bool background) { AfterMajorChange?.Invoke(this, background); } private void OnXenObjectsUpdated() { // Using BeginInvoke here means that the XenObjectsUpdated event gets fired after any // CollectionChanged events fired by ChangeableDictionary during Cache.UpdateFrom. InvokeHelper.BeginInvoke(delegate () { if (XenObjectsUpdated != null) XenObjectsUpdated(this, null); }); } public T TryResolveWithTimeout(XenRef t) where T : XenObject { log.DebugFormat("Resolving {0} {1}", t, t.opaque_ref); int timeout = 120; // two minutes; while (timeout > 0) { T obj = Resolve(t); if (obj != null) return obj; Thread.Sleep(1000); timeout--; } if (typeof(T) == typeof(Host)) throw new Failure(Failure.HOST_OFFLINE); throw new Failure(Failure.HANDLE_INVALID, typeof(T).Name, t.opaque_ref); } /// /// Stub to Cache.Resolve /// /// /// May be null, in which case null is returned. /// public virtual T Resolve(XenRef xenRef) where T : XenObject { return Cache.Resolve(xenRef); } /// /// Resolve every object in the given list. Skip any references that don't resolve. /// /// /// May be null, in which case the empty list is returned. /// public List ResolveAll(IEnumerable> xenRefs) where T : XenObject { List result = new List(); if (xenRefs != null) { foreach (XenRef xenRef in xenRefs) { T o = Resolve(xenRef); if (o != null) result.Add(o); } } return result; } public List ResolveAllShownXenModelObjects(List> xenRefs, bool showHiddenObjects) { List result = ResolveAll(xenRefs); result.RemoveAll(vdi => !vdi.Show(showHiddenObjects)); return result; } public static T FindByUUIDXenObject(string uuid) where T : XenObject { foreach (IXenConnection c in ConnectionsManager.XenConnectionsCopy) { T o = c.Cache.Find_By_Uuid(uuid); if (o != null) return o; } return null; } /// /// Find a XenObject corresponding to the given XenRef, or null if no such object is found. /// public static T FindByRef(XenRef needle) where T : XenObject { foreach (IXenConnection c in ConnectionsManager.XenConnectionsCopy) { T o = c.Resolve(needle); if (o != null) return o; } return null; } public static string ConnectedElsewhere(string hostname) { lock (ConnectionsManager.ConnectionsLock) { foreach (IXenConnection connection in ConnectionsManager.XenConnections) { Pool pool = Helpers.GetPoolOfOne(connection); if (pool == null) continue; Host coordinator = connection.Resolve(pool.master); if (coordinator == null) continue; if (coordinator.address == hostname) { // we have tried to connect to a supporter that is a member of a pool we are already connected to. return pool.Name(); } } } return null; } #region IXmlSerializable Members public System.Xml.Schema.XmlSchema GetSchema() { return null; } public void ReadXml(System.Xml.XmlReader reader) { Hostname = reader["Hostname"]; } public void WriteXml(System.Xml.XmlWriter writer) { writer.WriteElementString("Hostname", Hostname); } #endregion #region IDisposable Members /// /// Disposing this class will make it unusable - make sure you want to do this /// public void Dispose() { Dispose(true); GC.SuppressFinalize(this); } private bool disposed; protected virtual void Dispose(bool disposing) { if (!disposed) { if (disposing) { ClearingCache = null; CachePopulated = null; ConnectionResult = null; ConnectionStateChanged = null; ConnectionLost = null; ConnectionClosed = null; ConnectionReconnecting = null; BeforeConnectionEnd = null; ConnectionMessageChanged = null; BeforeMajorChange = null; AfterMajorChange = null; XenObjectsUpdated = null; if (ReconnectionTimer != null) ReconnectionTimer.Dispose(); if (cacheUpdateTimer != null) cacheUpdateTimer.Dispose(); } disposed = true; } } #endregion } public class ServerNotSupported : DisconnectionException { public override string Message => string.Format(Messages.SERVER_TOO_OLD, BrandManager.BrandConsole, BrandManager.ProductVersion712, BrandManager.ProductVersion80); } public class ConnectionExists : DisconnectionException { protected readonly IXenConnection Connection; public ConnectionExists(IXenConnection connection) { Connection = connection; } public override string Message { get { if (Connection == null) return Messages.CONNECTION_EXISTS_NULL; return string.Format(Messages.CONNECTION_EXISTS, Connection.Hostname); } } public virtual string GetDialogMessage() { if (Connection == null) return Messages.CONNECTION_EXISTS_NULL; Pool p = Helpers.GetPool(Connection); if (p == null) return string.Format(Messages.ALREADY_CONNECTED, Connection.Hostname); return string.Format(Messages.SUPPORTER_ALREADY_CONNECTED, Connection.Hostname, p.Name()); } } internal class BadRestoreDetected : ConnectionExists { public BadRestoreDetected(IXenConnection xc) : base(xc) { } public override string Message { get { if (Connection == null) return Messages.CONNECTION_EXISTS_NULL; return string.Format(Messages.BAD_RESTORE_DETECTED, Connection.Name); } } public override string GetDialogMessage() { return Message; } } }