From 784e5a56c7cf200d957cd1f06480265a33114434 Mon Sep 17 00:00:00 2001 From: Kevin Krauss Date: Tue, 6 Mar 2018 14:39:24 -0800 Subject: [PATCH 1/3] =?UTF-8?q?Don=E2=80=99t=20try=20to=20connect=20to=20b?= =?UTF-8?q?lacklisted=20resources?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- lib/makara/connection_wrapper.rb | 1 + 1 file changed, 1 insertion(+) diff --git a/lib/makara/connection_wrapper.rb b/lib/makara/connection_wrapper.rb index e63b2eb6..07438fc6 100644 --- a/lib/makara/connection_wrapper.rb +++ b/lib/makara/connection_wrapper.rb @@ -64,6 +64,7 @@ def _makara_connected? end def _makara_connection + return if _makara_blacklisted? current = @connection if current From 6eb7192415b3b49c7a612edb44b459c63a00b600 Mon Sep 17 00:00:00 2001 From: Kevin Krauss Date: Wed, 7 Mar 2018 11:36:12 -0800 Subject: [PATCH 2/3] Tests passing when not connecting to blacklisted hosts --- lib/makara/pool.rb | 8 +------- .../connection_adapters/makara_postgresql_adapter_spec.rb | 8 ++++---- 2 files changed, 5 insertions(+), 11 deletions(-) diff --git a/lib/makara/pool.rb b/lib/makara/pool.rb index a9df9d1d..0a2350f9 100644 --- a/lib/makara/pool.rb +++ b/lib/makara/pool.rb @@ -106,13 +106,10 @@ def provide value - # if we've made any connections within this pool, we should report the blackout. - elsif connection_made? + else err = Makara::Errors::AllConnectionsBlacklisted.new(self, @blacklist_errors) @blacklist_errors = [] raise err - else - raise Makara::Errors::NoConnectionsAvailable.new(@role) unless @disabled end # when a connection causes a blacklist error within the provided block, we blacklist it then retry @@ -122,11 +119,8 @@ def provide retry end - - protected - # have we connected to any of the underlying connections. def connection_made? @connections.any?(&:_makara_connected?) diff --git a/spec/active_record/connection_adapters/makara_postgresql_adapter_spec.rb b/spec/active_record/connection_adapters/makara_postgresql_adapter_spec.rb index 6702c465..45f981d0 100644 --- a/spec/active_record/connection_adapters/makara_postgresql_adapter_spec.rb +++ b/spec/active_record/connection_adapters/makara_postgresql_adapter_spec.rb @@ -77,7 +77,7 @@ allow_any_instance_of(Makara::Strategies::RoundRobin).to receive(:single_one?){ true } Test::User.exists? # flush other (schema) things that need to happen - + con = connection.slave_pool.connections.first if (ActiveRecord::VERSION::MAJOR == 4 && ActiveRecord::VERSION::MINOR >= 2) || (ActiveRecord::VERSION::MAJOR == 5 && ActiveRecord::VERSION::MINOR <= 0) @@ -100,8 +100,8 @@ allow(ActiveRecord::Base).to receive(:postgresql_connection).and_raise(StandardError.new('could not connect to server: Connection refused')) ActiveRecord::Base.establish_connection(config) - expect { connection.execute('SELECT * FROM users') }.to raise_error(Makara::Errors::NoConnectionsAvailable) - expect { connection.execute('INSERT INTO users (name) VALUES (\'John\')') }.to raise_error(Makara::Errors::NoConnectionsAvailable) + expect { connection.execute('SELECT * FROM users') }.to raise_error(Makara::Errors::AllConnectionsBlacklisted) + expect { connection.execute('INSERT INTO users (name) VALUES (\'John\')') }.to raise_error(Makara::Errors::AllConnectionsBlacklisted) end end @@ -130,7 +130,7 @@ establish_connection(custom_config) connection.execute('SELECT * FROM users') - expect { connection.execute('INSERT INTO users (name) VALUES (\'John\')') }.to raise_error(Makara::Errors::NoConnectionsAvailable) + expect { connection.execute('INSERT INTO users (name) VALUES (\'John\')') }.to raise_error(Makara::Errors::AllConnectionsBlacklisted) end end From fc2650da14d572e9fcdcb3c2a68eb8e00abcbc86 Mon Sep 17 00:00:00 2001 From: Kevin Krauss Date: Wed, 7 Mar 2018 13:19:29 -0800 Subject: [PATCH 3/3] refactor this so that NoConnectionsAvailable error is still valid All the connections are blacklisted because there is only one --- lib/makara/connection_wrapper.rb | 1 - lib/makara/pool.rb | 16 ++++++++++------ .../makara_mysql2_adapter_spec.rb | 2 +- .../makara_postgis_adapter_spec.rb | 6 +++--- 4 files changed, 14 insertions(+), 11 deletions(-) diff --git a/lib/makara/connection_wrapper.rb b/lib/makara/connection_wrapper.rb index 07438fc6..e63b2eb6 100644 --- a/lib/makara/connection_wrapper.rb +++ b/lib/makara/connection_wrapper.rb @@ -64,7 +64,6 @@ def _makara_connected? end def _makara_connection - return if _makara_blacklisted? current = @connection if current diff --git a/lib/makara/pool.rb b/lib/makara/pool.rb index 0a2350f9..145bfdc8 100644 --- a/lib/makara/pool.rb +++ b/lib/makara/pool.rb @@ -93,9 +93,13 @@ def send_to_all(method, *args, &block) # Provide a connection that is not blacklisted and connected. Handle any errors # that may occur within the block. def provide - provided_connection = self.next + if all_connections_blacklisted? + err = Makara::Errors::AllConnectionsBlacklisted.new(self, @blacklist_errors) + @blacklist_errors = [] + raise err + end - # nil implies that it's blacklisted + provided_connection = self.next if provided_connection value = @proxy.error_handler.handle(provided_connection) do @@ -105,11 +109,8 @@ def provide @blacklist_errors = [] value - else - err = Makara::Errors::AllConnectionsBlacklisted.new(self, @blacklist_errors) - @blacklist_errors = [] - raise err + raise Makara::Errors::NoConnectionsAvailable.new(@role) unless @disabled end # when a connection causes a blacklist error within the provided block, we blacklist it then retry @@ -126,6 +127,9 @@ def connection_made? @connections.any?(&:_makara_connected?) end + def all_connections_blacklisted? + @connections.all?(&:_makara_blacklisted?) + end # Get the next non-blacklisted connection. If the proxy is setup # to be sticky, provide back the current connection assuming it is diff --git a/spec/active_record/connection_adapters/makara_mysql2_adapter_spec.rb b/spec/active_record/connection_adapters/makara_mysql2_adapter_spec.rb index 27731e8e..80be8379 100644 --- a/spec/active_record/connection_adapters/makara_mysql2_adapter_spec.rb +++ b/spec/active_record/connection_adapters/makara_mysql2_adapter_spec.rb @@ -163,7 +163,7 @@ allow_any_instance_of(Makara::Strategies::RoundRobin).to receive(:single_one?){ true } Test::User.exists? # flush other (schema) things that need to happen - + con = connection.slave_pool.connections.first if (ActiveRecord::VERSION::MAJOR == 5 && ActiveRecord::VERSION::MINOR <= 0) expect(con).to receive(:execute).with(/SELECT\s+1\s*(AS one)?\s+FROM .?users.?\s+LIMIT\s+.?1/, any_args).once.and_call_original diff --git a/spec/active_record/connection_adapters/makara_postgis_adapter_spec.rb b/spec/active_record/connection_adapters/makara_postgis_adapter_spec.rb index b701bd86..30baa20b 100644 --- a/spec/active_record/connection_adapters/makara_postgis_adapter_spec.rb +++ b/spec/active_record/connection_adapters/makara_postgis_adapter_spec.rb @@ -118,8 +118,8 @@ allow(ActiveRecord::Base).to receive(:postgis_connection).and_raise(StandardError.new('could not connect to server: Connection refused')) ActiveRecord::Base.establish_connection(config) - expect { connection.execute('SELECT * FROM users') }.to raise_error(Makara::Errors::NoConnectionsAvailable) - expect { connection.execute('INSERT INTO users (name) VALUES (\'John\')') }.to raise_error(Makara::Errors::NoConnectionsAvailable) + expect { connection.execute('SELECT * FROM users') }.to raise_error(Makara::Errors::AllConnectionsBlacklisted) + expect { connection.execute('INSERT INTO users (name) VALUES (\'John\')') }.to raise_error(Makara::Errors::AllConnectionsBlacklisted) end end @@ -148,7 +148,7 @@ ActiveRecord::Base.establish_connection(custom_config) connection.execute('SELECT * FROM users') - expect { connection.execute('INSERT INTO users (name) VALUES (\'John\')') }.to raise_error(Makara::Errors::NoConnectionsAvailable) + expect { connection.execute('INSERT INTO users (name) VALUES (\'John\')') }.to raise_error(Makara::Errors::AllConnectionsBlacklisted) end end end