test_helper.tcl 6.0 KB
Newer Older
1 2 3 4 5
# Redis test suite. Copyright (C) 2009 Salvatore Sanfilippo antirez@gmail.com
# This softare is released under the BSD License. See the COPYING file for
# more information.

set tcl_precision 17
6 7 8 9 10
source tests/support/redis.tcl
source tests/support/server.tcl
source tests/support/tmpfile.tcl
source tests/support/test.tcl
source tests/support/util.tcl
11 12

set ::host 127.0.0.1
13
set ::port 16379
14
set ::traceleaks 0
15
set ::valgrind 0
A
antirez 已提交
16
set ::verbose 0
P
Pieter Noordhuis 已提交
17 18
set ::denytags {}
set ::allowtags {}
19
set ::external 0; # If "1" this means, we are running against external instance
20
set ::file ""; # If set, runs only the tests in this comma separated list
21
set ::curfile ""; # Hold the filename of the current suite
22
set ::diskstore 0; # Don't touch this by hand. The test itself will toggle it.
23 24

proc execute_tests name {
25 26 27
    set path "tests/$name.tcl"
    set ::curfile $path
    source $path
28 29
}

30 31 32 33
# Setup a list to hold a stack of server configs. When calls to start_server
# are nested, use "srv 0 pid" to get the pid of the inner server. To access
# outer servers, use "srv -1 pid" etcetera.
set ::servers {}
34 35 36 37 38 39 40 41
proc srv {args} {
    set level 0
    if {[string is integer [lindex $args 0]]} {
        set level [lindex $args 0]
        set property [lindex $args 1]
    } else {
        set property [lindex $args 0]
    }
42 43 44 45 46 47 48
    set srv [lindex $::servers end+$level]
    dict get $srv $property
}

# Provide easy access to the client for the inner server. It's possible to
# prepend the argument list with a negative level to access clients for
# servers running in outer blocks.
49
proc r {args} {
50 51 52 53 54 55 56 57
    set level 0
    if {[string is integer [lindex $args 0]]} {
        set level [lindex $args 0]
        set args [lrange $args 1 end]
    }
    [srv $level "client"] {*}$args
}

P
Pieter Noordhuis 已提交
58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79
proc reconnect {args} {
    set level [lindex $args 0]
    if {[string length $level] == 0 || ![string is integer $level]} {
        set level 0
    }

    set srv [lindex $::servers end+$level]
    set host [dict get $srv "host"]
    set port [dict get $srv "port"]
    set config [dict get $srv "config"]
    set client [redis $host $port]
    dict set srv "client" $client

    # select the right db when we don't have to authenticate
    if {![dict exists $config "requirepass"]} {
        $client select 9
    }

    # re-set $srv in the servers list
    set ::servers [lreplace $::servers end+$level 1 $srv]
}

80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95
proc redis_deferring_client {args} {
    set level 0
    if {[llength $args] > 0 && [string is integer [lindex $args 0]]} {
        set level [lindex $args 0]
        set args [lrange $args 1 end]
    }

    # create client that defers reading reply
    set client [redis [srv $level "host"] [srv $level "port"] 1]

    # select the right db and read the response (OK)
    $client select 9
    $client read
    return $client
}

96 97 98 99 100 101 102 103
# Provide easy access to INFO properties. Same semantic as "proc r".
proc s {args} {
    set level 0
    if {[string is integer [lindex $args 0]]} {
        set level [lindex $args 0]
        set args [lrange $args 1 end]
    }
    status [srv $level "client"] [lindex $args 0]
104 105
}

106
proc cleanup {} {
107
    puts "Cleanup: warning may take some time..."
108 109
    catch {exec rm -rf {*}[glob tests/tmp/redis.conf.*]}
    catch {exec rm -rf {*}[glob tests/tmp/server.*]}
110 111
}

112
proc execute_everything {} {
113
    if 0 {
A
antirez 已提交
114 115 116 117 118 119
        # Use this when hacking on new tests.
        set ::verbose 1
        execute_tests "unit/first"
        return
    }

120
    execute_tests "unit/printver"
121 122 123 124 125 126 127 128 129 130
    execute_tests "unit/auth"
    execute_tests "unit/protocol"
    execute_tests "unit/basic"
    execute_tests "unit/type/list"
    execute_tests "unit/type/set"
    execute_tests "unit/type/zset"
    execute_tests "unit/type/hash"
    execute_tests "unit/sort"
    execute_tests "unit/expire"
    execute_tests "unit/other"
A
antirez 已提交
131
    execute_tests "unit/cas"
P
Pieter Noordhuis 已提交
132
    execute_tests "unit/quit"
133
    execute_tests "integration/replication"
134
    execute_tests "integration/aof"
P
Pieter Noordhuis 已提交
135
    execute_tests "unit/pubsub"
A
antirez 已提交
136
    execute_tests "unit/slowlog"
137 138 139 140
}

proc main {} {
    cleanup
A
antirez 已提交
141
    set exit_with_error 0
142 143 144 145 146 147 148 149

    if {[string length $::file] > 0} {
        foreach {file} [split $::file ,] {
            execute_tests $file
        }
    } else {
        execute_everything
    }
150 151

    cleanup
152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172
    puts "\n[expr $::num_tests] tests, $::num_passed passed, $::num_failed failed\n"
    if {$::num_failed > 0} {
        set curheader ""
        puts "Failures:"
        foreach {test} $::tests_failed {
            set header [lindex $test 0]
            append header " ("
            append header [join [lindex $test 1] ","]
            append header ")"

            if {$curheader ne $header} {
                set curheader $header
                puts "\n$curheader:"
            }

            set name [lindex $test 2]
            set msg [lindex $test 3]
            puts "- $name: $msg"
        }

        puts ""
A
antirez 已提交
173
        incr exit_with_error
174
    }
A
antirez 已提交
175 176 177 178 179 180

    if {[string length $::valgrind_errors]} {
        puts "Valgrind errors:\n$::valgrind_errors"
        incr exit_with_error
    }
    if {$exit_with_error} {exit 1}
181 182
}

183 184 185 186 187 188 189 190 191 192 193 194 195
# parse arguments
for {set j 0} {$j < [llength $argv]} {incr j} {
    set opt [lindex $argv $j]
    set arg [lindex $argv [expr $j+1]]
    if {$opt eq {--tags}} {
        foreach tag $arg {
            if {[string index $tag 0] eq "-"} {
                lappend ::denytags [string range $tag 1 end]
            } else {
                lappend ::allowtags $tag
            }
        }
        incr j
196 197
    } elseif {$opt eq {--valgrind}} {
        set ::valgrind 1
198 199 200
    } elseif {$opt eq {--file}} {
        set ::file $arg
        incr j
201 202 203 204 205 206 207
    } elseif {$opt eq {--host}} {
        set ::external 1
        set ::host $arg
        incr j
    } elseif {$opt eq {--port}} {
        set ::port $arg
        incr j
208 209
    } elseif {$opt eq {--verbose}} {
        set ::verbose 1
210 211 212 213 214 215
    } else {
        puts "Wrong argument: $opt"
        exit 1
    }
}

216 217 218 219
if {[catch { main } err]} {
    if {[string length $err] > 0} {
        # only display error when not generated by the test suite
        if {$err ne "exception"} {
220
            puts $::errorInfo
221 222 223 224
        }
        exit 1
    }
}