From 7aff371495f28e892f1cb024bb0fae98066b824e Mon Sep 17 00:00:00 2001 From: Mark Paluch Date: Tue, 19 Dec 2017 15:30:53 +0100 Subject: [PATCH] Consider Sentinel slave changes to refresh topology #669 Lettuce now considers Sentinel messages on channels +sdown, -sdown and +slave as signals to refresh topology. This change allows to add new nodes during runtime and close connections to nodes that are not available (connectable). Temporary failures to connect a slave result in closing the client connection until the node is reachable again. --- .../masterslave/SentinelTopologyRefresh.java | 16 +++----- .../SentinelTopologyRefreshTest.java | 41 ++++++++++++++++++- 2 files changed, 46 insertions(+), 11 deletions(-) diff --git a/src/main/java/io/lettuce/core/masterslave/SentinelTopologyRefresh.java b/src/main/java/io/lettuce/core/masterslave/SentinelTopologyRefresh.java index 60698a8948..70f82c432f 100644 --- a/src/main/java/io/lettuce/core/masterslave/SentinelTopologyRefresh.java +++ b/src/main/java/io/lettuce/core/masterslave/SentinelTopologyRefresh.java @@ -249,6 +249,8 @@ interface MessagePredicate extends BiPredicate { private static class TopologyRefreshMessagePredicate implements MessagePredicate { private final String masterId; + private Set TOPOLOGY_CHANGE_CHANNELS = new HashSet<>(Arrays.asList("+slave", "+sdown", "-sdown", + "fix-slave-config", "+convert-to-slave", "+role-change")); TopologyRefreshMessagePredicate(String masterId) { this.masterId = masterId; @@ -258,26 +260,20 @@ private static class TopologyRefreshMessagePredicate implements MessagePredicate public boolean test(String channel, String message) { // trailing spaces after the master name are not bugs - if (channel.equals("+elected-leader")) { + if (channel.equals("+elected-leader") || channel.equals("+reset-master")) { if (message.startsWith(String.format("master %s ", masterId))) { return true; } } - if (channel.equals("+switch-master")) { - if (message.startsWith(String.format("%s ", masterId))) { - return true; - } - } - - if (channel.equals("fix-slave-config")) { + if (TOPOLOGY_CHANGE_CHANNELS.contains(channel)) { if (message.contains(String.format("@ %s ", masterId))) { return true; } } - if (channel.equals("+convert-to-slave") || channel.equals("+role-change")) { - if (message.contains(String.format("@ %s ", masterId))) { + if (channel.equals("+switch-master")) { + if (message.startsWith(String.format("%s ", masterId))) { return true; } } diff --git a/src/test/java/io/lettuce/core/masterslave/SentinelTopologyRefreshTest.java b/src/test/java/io/lettuce/core/masterslave/SentinelTopologyRefreshTest.java index c0bfb270d1..0294e80286 100644 --- a/src/test/java/io/lettuce/core/masterslave/SentinelTopologyRefreshTest.java +++ b/src/test/java/io/lettuce/core/masterslave/SentinelTopologyRefreshTest.java @@ -198,13 +198,52 @@ public void shouldNotProcessOtherEvents() { RedisPubSubAdapter adapter = getAdapter(); sut.bind(refreshRunnable); - adapter.message("*", "*", "irreleval"); + adapter.message("*", "*", "irrelevant"); verify(redisClient, times(2)).getResources(); verify(redisClient).connectPubSub(any(), any()); verifyNoMoreInteractions(redisClient); } + @Test + public void shouldProcessSlaveDown() { + + RedisPubSubAdapter adapter = getAdapter(); + sut.bind(refreshRunnable); + + adapter.message("*", "+sdown", "slave 127.0.0.1:6483 127.0.0.1 6483 @ mymaster 127.0.0.1 6482"); + + verify(eventExecutors, times(1)).schedule(captor.capture(), anyLong(), any()); + captor.getValue().run(); + verify(refreshRunnable, times(1)).run(); + } + + @Test + public void shouldProcessSlaveAdded() { + + RedisPubSubAdapter adapter = getAdapter(); + sut.bind(refreshRunnable); + + adapter.message("*", "+slave", "slave 127.0.0.1:8483 127.0.0.1 8483 @ mymaster 127.0.0.1 6482"); + + verify(eventExecutors, times(1)).schedule(captor.capture(), anyLong(), any()); + captor.getValue().run(); + verify(refreshRunnable, times(1)).run(); + } + + @Test + public void shouldProcessSlaveBackUp() { + + RedisPubSubAdapter adapter = getAdapter(); + sut.bind(refreshRunnable); + + adapter.message("*", "-sdown", "slave 127.0.0.1:6483 127.0.0.1 6483 @ mymaster 127.0.0.1 6482"); + + verify(eventExecutors, times(1)).schedule(captor.capture(), anyLong(), any()); + captor.getValue().run(); + verify(refreshRunnable, times(1)).run(); + } + @Test public void shouldProcessElectedLeader() {