// Copyright © 2017, 2019, Oracle and/or its affiliates. All rights reserved. // // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License, version 2.0, as // published by the Free Software Foundation. // // This program is also distributed with certain software (including // but not limited to OpenSSL) that is licensed under separate terms, // as designated in a particular file or component or in included license // documentation. The authors of MySQL hereby grant you an // additional permission to link the program and your derivative works // with the separately licensed software that they have included with // MySQL. // // Without limiting anything contained in the foregoing, this file, // which is part of MySQL Connector/NET, is also subject to the // Universal FOSS Exception, version 1.0, a copy of which can be found at // http://oss.oracle.com/licenses/universal-foss-exception. // // 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, version 2.0, for more details. // // You should have received a copy of the GNU General Public License // along with this program; if not, write to the Free Software Foundation, Inc., // 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA using MySql.Data.MySqlClient; using System.Collections.Generic; using Sog.Properties; namespace MySql.Data.Failover { /// /// Manages the hosts available for client side failover using the Sequential Failover method. /// The Sequential Failover method attempts to connect to the hosts specified in the list one after another until the initial host is reached. /// internal class SequentialFailoverGroup : FailoverGroup { /// /// The initial host taken from the list. /// private FailoverServer _initialHost; /// /// The index of the current host. /// private int _hostIndex; /// /// The host for the current connection attempt. /// private FailoverServer _currentHost; public SequentialFailoverGroup(List hosts) : base(hosts) { this._hostIndex = 0; } /// /// Sets the initial active host. /// protected internal override void SetInitialActiveServer() { if (this.Hosts == null || this.Hosts.Count == 0) { throw new MySqlException(Resources.Replication_NoAvailableServer); } this._initialHost = this.Hosts[0]; this.Hosts[0].IsActive = true; this._activeHost = this.Hosts[0]; this._currentHost = this._activeHost; } /// /// Determines the next host. /// /// A object that represents the next available host. protected internal override FailoverServer GetNextHost() { if (this.Hosts == null || this.Hosts?.Count == 0) { throw new MySqlException(Resources.Replication_NoAvailableServer); } var currentServer = this.Hosts.Find(h => h.Host == this._currentHost.Host && h.Port == this._currentHost.Port); currentServer.IsActive = false; this._hostIndex = this.Hosts.IndexOf(currentServer); if (this.Hosts.Count > 1) { this._activeHost = this._hostIndex == this.Hosts.Count - 1 ? this.Hosts[0] : this.Hosts[this._hostIndex + 1]; this._activeHost.IsActive = true; this._currentHost = this._activeHost; this._hostIndex++; } else { this._activeHost = this._initialHost; } return this._activeHost; } } }