redis-trib.rb 6.3 KB
Newer Older
1 2 3 4 5
#!/usr/bin/env ruby

require 'rubygems'
require 'redis'

6 7
ClusterHashSlots = 4096

8 9 10 11
def xputs(s)
    printf s
    STDOUT.flush
end
12

13 14 15
class ClusterNode
    def initialize(addr)
        s = addr.split(":")
16
        if s.length != 2
A
antirez 已提交
17
            puts "Invalid node name #{addr}"
18 19
            exit 1
        end
20
        @r = nil
21 22
        @host = s[0]
        @port = s[1]
23 24
        @slots = {}
        @dirty = false
25 26
    end

27 28 29 30
    def to_s
        "#{@host}:#{@port}"
    end

31
    def connect(o={})
32
        xputs "Connecting to node #{self}: "
33
        begin
34 35
            @r = Redis.new(:host => @ost, :port => @port)
            @r.ping
36 37
        rescue
            puts "ERROR"
38
            puts "Sorry, can't connect to node #{self}"
39 40
            exit 1 if o[:abort]
            @r = nil
41 42 43 44
        end
        puts "OK"
    end

45 46 47 48 49 50 51 52
    def assert_cluster
        info = @r.info
        if !info["cluster_enabled"] || info["cluster_enabled"].to_i == 0
            puts "Error: Node #{self} is not configured as a cluster node."
            exit 1
        end
    end

53 54 55 56 57 58 59 60
    def assert_empty
        if !(@r.cluster("info").split("\r\n").index("cluster_known_nodes:1")) ||
            (@r.info['db0'])
            puts "Error: Node #{self} is not empty. Either the node already knows other nodes (check with nodes-info) or contains some key in database 0."
            exit 1
        end
    end

61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80
    def add_slots(slots)
        slots.each{|s|
            @slots[s] = :new
        }
        @dirty = true
    end

    def flush_node_config
        return if !@dirty
        new = []
        @slots.each{|s,val|
            if val == :new
                new << s
                @slots[s] = true
            end
        }
        @r.cluster("addslots",*new)
        @dirty = false
    end

81
    def info_string
82
        # We want to display the hash slots assigned to this node
A
antirez 已提交
83
        # as ranges, like in: "1-5,8-9,20-25,30"
84 85 86 87 88 89 90 91 92 93 94
        #
        # Note: this could be easily written without side effects,
        # we use 'slots' just to split the computation into steps.
        
        # First step: we want an increasing array of integers
        # for instance: [1,2,3,4,5,8,9,20,21,22,23,24,25,30]
        slots = @slots.keys.sort

        # As we want to aggregate adiacent slots we convert all the
        # slot integers into ranges (with just one element)
        # So we have something like [1..1,2..2, ... and so forth.
A
antirez 已提交
95
        slots.map!{|x| x..x}
96 97 98 99 100

        # Finally we group ranges with adiacent elements.
        slots = slots.reduce([]) {|a,b|
            if !a.empty? && b.first == (a[-1].last)+1
                a[0..-2] + [(a[-1].first)..(b.last)]
101
            else
102
                a + [b]
103
            end
104 105 106 107 108 109 110
        }

        # Now our task is easy, we just convert ranges with just one
        # element into a number, and a real range into a start-end format.
        # Finally we join the array using the comma as separator.
        slots = slots.map{|x|
            x.count == 1 ? x.first.to_s : "#{x.first}-#{x.last}"
111
        }.join(",")
112

113 114
        "#{self.to_s.ljust(25)} slots:#{slots}"
    end
115 116 117 118 119 120 121 122 123

    def info
        {
            :host => @host,
            :port => @port,
            :slots => @slots,
            :dirty => @dirty
        }
    end
124 125 126 127 128
    
    def is_dirty?
        @dirty
    end

129 130 131 132 133 134
    def r
        @r
    end
end

class RedisTrib
135 136 137 138
    def initialize
        @nodes = []
    end

139 140 141 142 143 144 145 146
    def check_arity(req_args, num_args)
        if ((req_args > 0 and num_args != req_args) ||
           (req_args < 0 and num_args < req_args.abs))
           puts "Wrong number of arguments for specified sub command"
           exit 1
        end
    end

147 148 149 150
    def add_node(node)
        @nodes << node
    end

151 152
    def create_cluster
        puts "Creating cluster"
153 154
        ARGV[1..-1].each{|n|
            node = ClusterNode.new(n)
155
            node.connect(:abort => true)
156
            node.assert_cluster
157
            node.assert_empty
158
            add_node(node)
159
        }
160 161 162 163 164 165
        puts "Performing hash slots allocation on #{@nodes.length} nodes..."
        alloc_slots
        show_nodes
        yes_or_die "Can I set the above configuration?"
        flush_nodes_config
        puts "** Nodes configuration updated"
166
        puts "** Sending CLUSTER MEET messages to join the cluster"
167
        join_cluster
168 169 170 171
        check_cluster
    end

    def check_cluster
172 173 174 175 176 177 178 179 180
        puts "Performing Cluster Check (node #{ARGV[1]})"
        node = ClusterNode.new(ARGV[1])
        node.connect(:abort => true)
        node.assert_cluster
        node.add_slots(10..15)
        node.add_slots(30..30)
        node.add_slots(5..5)
        add_node(node)
        show_nodes
181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202
    end

    def alloc_slots
        slots_per_node = ClusterHashSlots/@nodes.length
        i = 0
        @nodes.each{|n|
            first = i*slots_per_node
            last = first+slots_per_node-1
            last = ClusterHashSlots-1 if i == @nodes.length-1
            n.add_slots first..last
            i += 1
        }
    end

    def flush_nodes_config
        @nodes.each{|n|
            n.flush_node_config
        }
    end

    def show_nodes
        @nodes.each{|n|
203
            puts n.info_string
204 205 206 207
        }
    end

    def join_cluster
208 209 210 211 212 213 214 215 216 217
        # We use a brute force approach to make sure the node will meet
        # each other, that is, sending CLUSTER MEET messages to all the nodes
        # about the very same node.
        # Thanks to gossip this information should propagate across all the
        # cluster in a matter of seconds.
        first = false
        @nodes.each{|n|
            if !first then first = n.info; next; end # Skip the first node
            n.r.cluster("meet",first[:host],first[:port])
        }
218 219 220 221 222 223 224 225 226
    end

    def yes_or_die(msg)
        print "#{msg} (type 'yes' to accept): "
        STDOUT.flush
        if !(STDIN.gets.chomp.downcase == "yes")
            puts "Aborting..."
            exit 1
        end
227 228 229 230
    end
end

COMMANDS={
231
    "create" => ["create_cluster", -2, "host1:port host2:port ... hostN:port"],
232
    "check" =>  ["check_cluster", 2, "host:port"]
233 234 235 236 237
}

# Sanity check
if ARGV.length == 0
    puts "Usage: redis-trib <command> <arguments ...>"
A
antirez 已提交
238 239 240 241 242
    puts
    COMMANDS.each{|k,v|
        puts "  #{k.ljust(20)} #{v[2]}"
    }
    puts
243 244 245 246 247 248 249 250 251 252 253 254 255
    exit 1
end

rt = RedisTrib.new
cmd_spec = COMMANDS[ARGV[0].downcase]
if !cmd_spec
    puts "Unknown redis-trib subcommand '#{ARGV[0]}'"
    exit 1
end
rt.check_arity(cmd_spec[1],ARGV.length)

# Dispatch
rt.send(cmd_spec[0])