// Copyright © 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; using System.Collections.Generic; using Sog.Properties; namespace MySql.Data.Failover { /// /// Manages the hosts available for client side failover using the Random Failover method. /// The Random Failover method attempts to connect to the hosts specified in the list randomly until all the hosts have been attempted. /// internal class RandomFailoverGroup : FailoverGroup { /// /// The initial host taken from the list. /// private FailoverServer _initialHost; /// /// The host for the current connection attempt. /// private FailoverServer _currentHost; /// /// Random object to get the next host. /// private readonly Random rnd = new Random(); public RandomFailoverGroup(List hosts) : base(hosts) { } /// /// Sets the initial active host. /// protected internal override void SetInitialActiveServer() { if (this.Hosts == null || this.Hosts.Count == 0) { throw new MySqlException(Resources.Replication_NoAvailableServer); } var initialIndex = this.rnd.Next(this.Hosts.Count); this._initialHost = this.Hosts[initialIndex]; this._initialHost.IsActive = true; this._initialHost.Attempted = true; this._activeHost = this._initialHost; 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); } this.Hosts.Find(h => h.Host == this._currentHost.Host && h.Port == this._currentHost.Port).IsActive = false; var notAttemptedHosts = this.Hosts.FindAll(h => h.Attempted == false); if (notAttemptedHosts.Count > 0) { this._activeHost = notAttemptedHosts[this.rnd.Next(notAttemptedHosts.Count)]; this._activeHost.IsActive = true; this._activeHost.Attempted = true; this._currentHost = this._activeHost; } else { this._activeHost = this._initialHost; } return this._activeHost; } } }