]> git.saurik.com Git - redis.git/commitdiff
Merge branch 'unstable' of github.com:antirez/redis into unstable
authorantirez <antirez@gmail.com>
Wed, 13 Apr 2011 08:39:06 +0000 (10:39 +0200)
committerantirez <antirez@gmail.com>
Wed, 13 Apr 2011 08:39:06 +0000 (10:39 +0200)
1  2 
src/redis-trib.rb

diff --combined src/redis-trib.rb
index f2e14010abd7caa69c9660857bee43c7f6cfdfa1,5a0fe363acadcf40d3fe43d9484f9ab6c3f1f645..cd35587fe7779e41d6d38d2313f63539dd448b1c
@@@ -3,6 -3,8 +3,8 @@@
  require 'rubygems'
  require 'redis'
  
+ ClusterHashSlots = 4096
  def xputs(s)
      printf s
      STDOUT.flush
@@@ -17,6 -19,8 +19,8 @@@ class ClusterNod
          end
          @host = s[0]
          @port = s[1]
+         @slots = {}
+         @dirty = false
      end
  
      def to_s
          end
      end
  
+     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
+     def info
+         slots = @slots.map{|k,v| k}.reduce{|a,b|
+             a = [(a..a)] if !a.is_a?(Array)
+             if b == (a[-1].last)+1
+                 a[-1] = (a[-1].first)..b
+                 a
+             else
+                 a << (b..b)
+             end
+         }.map{|x|
+             (x.first == x.last) ? x.first.to_s : "#{x.first}-#{x.last}"
+         }.join(",")
+         "#{self.to_s.ljust(25)} slots:#{slots}"
+     end
+     
+     def is_dirty?
+         @dirty
+     end
      def r
          @r
      end
  end
  
  class RedisTrib
+     def initialize
+         @nodes = []
+     end
      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))
              node.connect
              node.assert_cluster
              node.assert_empty
+             @nodes << node
          }
+         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"
+         puts "Sending CLUSTER MEET messages to join the cluster"
+         join_cluster
+     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|
+             puts n.info
+         }
+     end
+     def join_cluster
+     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
      end
  end
  
  COMMANDS={
 -    "create-cluster" => ["create_cluster", -2]
 +    "create-cluster" => ["create_cluster", -2, "node1_addr node2_addr ..."]
  }
  
  # Sanity check
  if ARGV.length == 0
      puts "Usage: redis-trib <command> <arguments ...>"
 +    puts
 +    COMMANDS.each{|k,v|
 +        puts "  #{k.ljust(20)} #{v[2]}"
 +    }
 +    puts
      exit 1
  end