diff options
Diffstat (limited to 'examples/redis-unstable/tests/support/cluster.tcl')
| -rw-r--r-- | examples/redis-unstable/tests/support/cluster.tcl | 372 |
1 files changed, 0 insertions, 372 deletions
diff --git a/examples/redis-unstable/tests/support/cluster.tcl b/examples/redis-unstable/tests/support/cluster.tcl deleted file mode 100644 index 3a66684..0000000 --- a/examples/redis-unstable/tests/support/cluster.tcl +++ /dev/null @@ -1,372 +0,0 @@ -# Tcl redis cluster client as a wrapper of redis.rb. -# -# Copyright (C) 2014-Present, Redis Ltd. -# All Rights reserved. -# -# Licensed under your choice of (a) the Redis Source Available License 2.0 -# (RSALv2); or (b) the Server Side Public License v1 (SSPLv1); or (c) the -# GNU Affero General Public License v3 (AGPLv3). -# -# Example usage: -# -# set c [redis_cluster {127.0.0.1:6379 127.0.0.1:6380}] -# $c set foo -# $c get foo -# $c close - -package require Tcl 8.5 -package provide redis_cluster 0.1 - -namespace eval redis_cluster {} -set ::redis_cluster::internal_id 0 -set ::redis_cluster::id 0 -array set ::redis_cluster::startup_nodes {} -array set ::redis_cluster::nodes {} -array set ::redis_cluster::slots {} -array set ::redis_cluster::tls {} - -# List of "plain" commands, which are commands where the sole key is always -# the first argument. -set ::redis_cluster::plain_commands { - get set setnx setex psetex append strlen exists setbit getbit - setrange getrange substr incr decr rpush lpush rpushx lpushx - linsert rpop lpop brpop llen lindex lset lrange ltrim lrem - sadd srem sismember smismember scard spop srandmember smembers sscan zadd - zincrby zrem zremrangebyscore zremrangebyrank zremrangebylex zrange - zrangebyscore zrevrangebyscore zrangebylex zrevrangebylex zcount - zlexcount zrevrange zcard zscore zmscore zrank zrevrank zscan hset hsetnx - hget hmset hmget hincrby hincrbyfloat hdel hlen hkeys hvals - hgetall hexists hscan incrby decrby incrbyfloat getset move - expire expireat pexpire pexpireat type ttl pttl persist restore - dump bitcount bitpos pfadd pfcount cluster ssubscribe spublish - sunsubscribe -} - -# Create a cluster client. The nodes are given as a list of host:port. The TLS -# parameter (1 or 0) is optional and defaults to the global $::tls. -proc redis_cluster {nodes {tls -1}} { - set id [incr ::redis_cluster::id] - set ::redis_cluster::startup_nodes($id) $nodes - set ::redis_cluster::nodes($id) {} - set ::redis_cluster::slots($id) {} - set ::redis_cluster::tls($id) [expr $tls == -1 ? $::tls : $tls] - set handle [interp alias {} ::redis_cluster::instance$id {} ::redis_cluster::__dispatch__ $id] - $handle refresh_nodes_map - return $handle -} - -# Totally reset the slots / nodes state for the client, calls -# CLUSTER NODES in the first startup node available, populates the -# list of nodes ::redis_cluster::nodes($id) with an hash mapping node -# ip:port to a representation of the node (another hash), and finally -# maps ::redis_cluster::slots($id) with an hash mapping slot numbers -# to node IDs. -# -# This function is called when a new Redis Cluster client is initialized -# and every time we get a -MOVED redirection error. -proc ::redis_cluster::__method__refresh_nodes_map {id} { - # Contact the first responding startup node. - set idx 0; # Index of the node that will respond. - set errmsg {} - foreach start_node $::redis_cluster::startup_nodes($id) { - set ip_port [lindex [split $start_node @] 0] - lassign [split $ip_port :] start_host start_port - set tls $::redis_cluster::tls($id) - if {[catch { - set r {} - set r [redis $start_host $start_port 0 $tls] - set nodes_descr [$r cluster nodes] - $r close - } e]} { - if {$r ne {}} {catch {$r close}} - incr idx - if {[string length $errmsg] < 200} { - append errmsg " $ip_port: $e" - } - continue ; # Try next. - } else { - break; # Good node found. - } - } - - if {$idx == [llength $::redis_cluster::startup_nodes($id)]} { - error "No good startup node found. $errmsg" - } - - # Put the node that responded as first in the list if it is not - # already the first. - if {$idx != 0} { - set l $::redis_cluster::startup_nodes($id) - set left [lrange $l 0 [expr {$idx-1}]] - set right [lrange $l [expr {$idx+1}] end] - set l [concat [lindex $l $idx] $left $right] - set ::redis_cluster::startup_nodes($id) $l - } - - # Parse CLUSTER NODES output to populate the nodes description. - set nodes {} ; # addr -> node description hash. - foreach line [split $nodes_descr "\n"] { - set line [string trim $line] - if {$line eq {}} continue - set args [split $line " "] - lassign $args nodeid addr flags slaveof pingsent pongrecv configepoch linkstate - set slots [lrange $args 8 end] - set addr [lindex [split $addr @] 0] - if {$addr eq {:0}} { - set addr $start_host:$start_port - } - lassign [split $addr :] host port - - # Connect to the node - set link {} - set tls $::redis_cluster::tls($id) - catch {set link [redis $host $port 0 $tls]} - - # Build this node description as an hash. - set node [dict create \ - id $nodeid \ - internal_id $id \ - addr $addr \ - host $host \ - port $port \ - flags $flags \ - slaveof $slaveof \ - slots $slots \ - link $link \ - ] - dict set nodes $addr $node - lappend ::redis_cluster::startup_nodes($id) $addr - } - - # Close all the existing links in the old nodes map, and set the new - # map as current. - foreach n $::redis_cluster::nodes($id) { - catch { - [dict get $n link] close - } - } - set ::redis_cluster::nodes($id) $nodes - - # Populates the slots -> nodes map. - dict for {addr node} $nodes { - foreach slotrange [dict get $node slots] { - lassign [split $slotrange -] start end - if {$end == {}} {set end $start} - for {set j $start} {$j <= $end} {incr j} { - dict set ::redis_cluster::slots($id) $j $addr - } - } - } - - # Only retain unique entries in the startup nodes list - set ::redis_cluster::startup_nodes($id) [lsort -unique $::redis_cluster::startup_nodes($id)] -} - -# Free a redis_cluster handle. -proc ::redis_cluster::__method__close {id} { - catch { - set nodes $::redis_cluster::nodes($id) - dict for {addr node} $nodes { - catch { - [dict get $node link] close - } - } - } - catch {unset ::redis_cluster::startup_nodes($id)} - catch {unset ::redis_cluster::nodes($id)} - catch {unset ::redis_cluster::slots($id)} - catch {unset ::redis_cluster::tls($id)} - catch {interp alias {} ::redis_cluster::instance$id {}} -} - -proc ::redis_cluster::__method__masternode_for_slot {id slot} { - # Get the node mapped to this slot. - set node_addr [dict get $::redis_cluster::slots($id) $slot] - if {$node_addr eq {}} { - error "No mapped node for slot $slot." - } - return [dict get $::redis_cluster::nodes($id) $node_addr] -} - -proc ::redis_cluster::__method__masternode_notfor_slot {id slot} { - # Get a node that is not mapped to this slot. - set node_addr [dict get $::redis_cluster::slots($id) $slot] - set addrs [dict keys $::redis_cluster::nodes($id)] - foreach addr [lshuffle $addrs] { - set node [dict get $::redis_cluster::nodes($id) $addr] - if {$node_addr ne $addr && [dict get $node slaveof] eq "-"} { - return $node - } - } - error "Slot $slot is everywhere" -} - -proc ::redis_cluster::__dispatch__ {id method args} { - if {[info command ::redis_cluster::__method__$method] eq {}} { - # Get the keys from the command. - set keys [::redis_cluster::get_keys_from_command $method $args] - if {$keys eq {}} { - error "Redis command '$method' is not supported by redis_cluster." - } - - # Resolve the keys in the corresponding hash slot they hash to. - set slot [::redis_cluster::get_slot_from_keys $keys] - if {$slot eq {}} { - error "Invalid command: multiple keys not hashing to the same slot." - } - - # Get the node mapped to this slot. - set node_addr [dict get $::redis_cluster::slots($id) $slot] - if {$node_addr eq {}} { - error "No mapped node for slot $slot." - } - - # Execute the command in the node we think is the slot owner. - set retry 100 - set asking 0 - while {[incr retry -1]} { - if {$retry < 5} {after 100} - set node [dict get $::redis_cluster::nodes($id) $node_addr] - set link [dict get $node link] - if {$asking} { - $link ASKING - set asking 0 - } - if {[catch {$link $method {*}$args} e]} { - if {$link eq {} || \ - [string range $e 0 4] eq {MOVED} || \ - [string range $e 0 2] eq {I/O} \ - } { - # MOVED redirection. - ::redis_cluster::__method__refresh_nodes_map $id - set node_addr [dict get $::redis_cluster::slots($id) $slot] - continue - } elseif {[string range $e 0 2] eq {ASK}} { - # ASK redirection. - set node_addr [lindex $e 2] - set asking 1 - continue - } else { - # Non redirecting error. - error $e $::errorInfo $::errorCode - } - } else { - # OK query went fine - return $e - } - } - error "Too many redirections or failures contacting Redis Cluster." - } else { - uplevel 1 [list ::redis_cluster::__method__$method $id] $args - } -} - -proc ::redis_cluster::get_keys_from_command {cmd argv} { - set cmd [string tolower $cmd] - # Most Redis commands get just one key as first argument. - if {[lsearch -exact $::redis_cluster::plain_commands $cmd] != -1} { - return [list [lindex $argv 0]] - } - - # Special handling for other commands - switch -exact $cmd { - mget {return $argv} - eval {return [lrange $argv 2 1+[lindex $argv 1]]} - evalsha {return [lrange $argv 2 1+[lindex $argv 1]]} - spublish {return [list [lindex $argv 1]]} - } - - # All the remaining commands are not handled. - return {} -} - -# Returns the CRC16 of the specified string. -# The CRC parameters are described in the Redis Cluster specification. -set ::redis_cluster::XMODEMCRC16Lookup { - 0x0000 0x1021 0x2042 0x3063 0x4084 0x50a5 0x60c6 0x70e7 - 0x8108 0x9129 0xa14a 0xb16b 0xc18c 0xd1ad 0xe1ce 0xf1ef - 0x1231 0x0210 0x3273 0x2252 0x52b5 0x4294 0x72f7 0x62d6 - 0x9339 0x8318 0xb37b 0xa35a 0xd3bd 0xc39c 0xf3ff 0xe3de - 0x2462 0x3443 0x0420 0x1401 0x64e6 0x74c7 0x44a4 0x5485 - 0xa56a 0xb54b 0x8528 0x9509 0xe5ee 0xf5cf 0xc5ac 0xd58d - 0x3653 0x2672 0x1611 0x0630 0x76d7 0x66f6 0x5695 0x46b4 - 0xb75b 0xa77a 0x9719 0x8738 0xf7df 0xe7fe 0xd79d 0xc7bc - 0x48c4 0x58e5 0x6886 0x78a7 0x0840 0x1861 0x2802 0x3823 - 0xc9cc 0xd9ed 0xe98e 0xf9af 0x8948 0x9969 0xa90a 0xb92b - 0x5af5 0x4ad4 0x7ab7 0x6a96 0x1a71 0x0a50 0x3a33 0x2a12 - 0xdbfd 0xcbdc 0xfbbf 0xeb9e 0x9b79 0x8b58 0xbb3b 0xab1a - 0x6ca6 0x7c87 0x4ce4 0x5cc5 0x2c22 0x3c03 0x0c60 0x1c41 - 0xedae 0xfd8f 0xcdec 0xddcd 0xad2a 0xbd0b 0x8d68 0x9d49 - 0x7e97 0x6eb6 0x5ed5 0x4ef4 0x3e13 0x2e32 0x1e51 0x0e70 - 0xff9f 0xefbe 0xdfdd 0xcffc 0xbf1b 0xaf3a 0x9f59 0x8f78 - 0x9188 0x81a9 0xb1ca 0xa1eb 0xd10c 0xc12d 0xf14e 0xe16f - 0x1080 0x00a1 0x30c2 0x20e3 0x5004 0x4025 0x7046 0x6067 - 0x83b9 0x9398 0xa3fb 0xb3da 0xc33d 0xd31c 0xe37f 0xf35e - 0x02b1 0x1290 0x22f3 0x32d2 0x4235 0x5214 0x6277 0x7256 - 0xb5ea 0xa5cb 0x95a8 0x8589 0xf56e 0xe54f 0xd52c 0xc50d - 0x34e2 0x24c3 0x14a0 0x0481 0x7466 0x6447 0x5424 0x4405 - 0xa7db 0xb7fa 0x8799 0x97b8 0xe75f 0xf77e 0xc71d 0xd73c - 0x26d3 0x36f2 0x0691 0x16b0 0x6657 0x7676 0x4615 0x5634 - 0xd94c 0xc96d 0xf90e 0xe92f 0x99c8 0x89e9 0xb98a 0xa9ab - 0x5844 0x4865 0x7806 0x6827 0x18c0 0x08e1 0x3882 0x28a3 - 0xcb7d 0xdb5c 0xeb3f 0xfb1e 0x8bf9 0x9bd8 0xabbb 0xbb9a - 0x4a75 0x5a54 0x6a37 0x7a16 0x0af1 0x1ad0 0x2ab3 0x3a92 - 0xfd2e 0xed0f 0xdd6c 0xcd4d 0xbdaa 0xad8b 0x9de8 0x8dc9 - 0x7c26 0x6c07 0x5c64 0x4c45 0x3ca2 0x2c83 0x1ce0 0x0cc1 - 0xef1f 0xff3e 0xcf5d 0xdf7c 0xaf9b 0xbfba 0x8fd9 0x9ff8 - 0x6e17 0x7e36 0x4e55 0x5e74 0x2e93 0x3eb2 0x0ed1 0x1ef0 -} - -proc ::redis_cluster::crc16 {s} { - set s [encoding convertto ascii $s] - set crc 0 - foreach char [split $s {}] { - scan $char %c byte - set crc [expr {(($crc<<8)&0xffff) ^ [lindex $::redis_cluster::XMODEMCRC16Lookup [expr {(($crc>>8)^$byte) & 0xff}]]}] - } - return $crc -} - -# Hash a single key returning the slot it belongs to, Implemented hash -# tags as described in the Redis Cluster specification. -proc ::redis_cluster::hash {key} { - set keylen [string length $key] - set s {} - set e {} - for {set s 0} {$s < $keylen} {incr s} { - if {[string index $key $s] eq "\{"} break - } - - if {[expr {$s == $keylen}]} { - set res [expr {[crc16 $key] & 16383}] - return $res - } - - for {set e [expr {$s+1}]} {$e < $keylen} {incr e} { - if {[string index $key $e] == "\}"} break - } - - if {$e == $keylen || $e == [expr {$s+1}]} { - set res [expr {[crc16 $key] & 16383}] - return $res - } - - set key_sub [string range $key [expr {$s+1}] [expr {$e-1}]] - return [expr {[crc16 $key_sub] & 16383}] -} - -# Return the slot the specified keys hash to. -# If the keys hash to multiple slots, an empty string is returned to -# signal that the command can't be run in Redis Cluster. -proc ::redis_cluster::get_slot_from_keys {keys} { - set slot {} - foreach k $keys { - set s [::redis_cluster::hash $k] - if {$slot eq {}} { - set slot $s - } elseif {$slot != $s} { - return {} ; # Error - } - } - return $slot -} |
