mirror of
https://codeberg.org/redict/redict.git
synced 2025-01-22 08:08:53 -05:00
50ee0f5be8
Based on feedback from interested parties
270 lines
9.2 KiB
Tcl
270 lines
9.2 KiB
Tcl
# SPDX-FileCopyrightText: 2024 Redict Contributors
|
|
# SPDX-FileCopyrightText: 2024 Salvatore Sanfilippo <antirez at gmail dot com>
|
|
#
|
|
# SPDX-License-Identifier: BSD-3-Clause
|
|
# SPDX-License-Identifier: LGPL-3.0-only
|
|
|
|
start_server {tags {"bitops"}} {
|
|
test {BITFIELD signed SET and GET basics} {
|
|
r del bits
|
|
set results {}
|
|
lappend results [r bitfield bits set i8 0 -100]
|
|
lappend results [r bitfield bits set i8 0 101]
|
|
lappend results [r bitfield bits get i8 0]
|
|
set results
|
|
} {0 -100 101}
|
|
|
|
test {BITFIELD unsigned SET and GET basics} {
|
|
r del bits
|
|
set results {}
|
|
lappend results [r bitfield bits set u8 0 255]
|
|
lappend results [r bitfield bits set u8 0 100]
|
|
lappend results [r bitfield bits get u8 0]
|
|
set results
|
|
} {0 255 100}
|
|
|
|
test {BITFIELD signed SET and GET together} {
|
|
r del bits
|
|
set results [r bitfield bits set i8 0 255 set i8 0 100 get i8 0]
|
|
} {0 -1 100}
|
|
|
|
test {BITFIELD unsigned with SET, GET and INCRBY arguments} {
|
|
r del bits
|
|
set results [r bitfield bits set u8 0 255 incrby u8 0 100 get u8 0]
|
|
} {0 99 99}
|
|
|
|
test {BITFIELD with only key as argument} {
|
|
r del bits
|
|
set result [r bitfield bits]
|
|
assert {$result eq {}}
|
|
}
|
|
|
|
test {BITFIELD #<idx> form} {
|
|
r del bits
|
|
set results {}
|
|
r bitfield bits set u8 #0 65
|
|
r bitfield bits set u8 #1 66
|
|
r bitfield bits set u8 #2 67
|
|
r get bits
|
|
} {ABC}
|
|
|
|
test {BITFIELD basic INCRBY form} {
|
|
r del bits
|
|
set results {}
|
|
r bitfield bits set u8 #0 10
|
|
lappend results [r bitfield bits incrby u8 #0 100]
|
|
lappend results [r bitfield bits incrby u8 #0 100]
|
|
set results
|
|
} {110 210}
|
|
|
|
test {BITFIELD chaining of multiple commands} {
|
|
r del bits
|
|
set results {}
|
|
r bitfield bits set u8 #0 10
|
|
lappend results [r bitfield bits incrby u8 #0 100 incrby u8 #0 100]
|
|
set results
|
|
} {{110 210}}
|
|
|
|
test {BITFIELD unsigned overflow wrap} {
|
|
r del bits
|
|
set results {}
|
|
r bitfield bits set u8 #0 100
|
|
lappend results [r bitfield bits overflow wrap incrby u8 #0 257]
|
|
lappend results [r bitfield bits get u8 #0]
|
|
lappend results [r bitfield bits overflow wrap incrby u8 #0 255]
|
|
lappend results [r bitfield bits get u8 #0]
|
|
} {101 101 100 100}
|
|
|
|
test {BITFIELD unsigned overflow sat} {
|
|
r del bits
|
|
set results {}
|
|
r bitfield bits set u8 #0 100
|
|
lappend results [r bitfield bits overflow sat incrby u8 #0 257]
|
|
lappend results [r bitfield bits get u8 #0]
|
|
lappend results [r bitfield bits overflow sat incrby u8 #0 -255]
|
|
lappend results [r bitfield bits get u8 #0]
|
|
} {255 255 0 0}
|
|
|
|
test {BITFIELD signed overflow wrap} {
|
|
r del bits
|
|
set results {}
|
|
r bitfield bits set i8 #0 100
|
|
lappend results [r bitfield bits overflow wrap incrby i8 #0 257]
|
|
lappend results [r bitfield bits get i8 #0]
|
|
lappend results [r bitfield bits overflow wrap incrby i8 #0 255]
|
|
lappend results [r bitfield bits get i8 #0]
|
|
} {101 101 100 100}
|
|
|
|
test {BITFIELD signed overflow sat} {
|
|
r del bits
|
|
set results {}
|
|
r bitfield bits set u8 #0 100
|
|
lappend results [r bitfield bits overflow sat incrby i8 #0 257]
|
|
lappend results [r bitfield bits get i8 #0]
|
|
lappend results [r bitfield bits overflow sat incrby i8 #0 -255]
|
|
lappend results [r bitfield bits get i8 #0]
|
|
} {127 127 -128 -128}
|
|
|
|
test {BITFIELD overflow detection fuzzing} {
|
|
for {set j 0} {$j < 1000} {incr j} {
|
|
set bits [expr {[randomInt 64]+1}]
|
|
set sign [randomInt 2]
|
|
set range [expr {2**$bits}]
|
|
if {$bits == 64} {set sign 1} ; # u64 is not supported by BITFIELD.
|
|
if {$sign} {
|
|
set min [expr {-($range/2)}]
|
|
set type "i$bits"
|
|
} else {
|
|
set min 0
|
|
set type "u$bits"
|
|
}
|
|
set max [expr {$min+$range-1}]
|
|
|
|
# Compare Tcl vs Redict
|
|
set range2 [expr {$range*2}]
|
|
set value [expr {($min*2)+[randomInt $range2]}]
|
|
set increment [expr {($min*2)+[randomInt $range2]}]
|
|
if {$value > 9223372036854775807} {
|
|
set value 9223372036854775807
|
|
}
|
|
if {$value < -9223372036854775808} {
|
|
set value -9223372036854775808
|
|
}
|
|
if {$increment > 9223372036854775807} {
|
|
set increment 9223372036854775807
|
|
}
|
|
if {$increment < -9223372036854775808} {
|
|
set increment -9223372036854775808
|
|
}
|
|
|
|
set overflow 0
|
|
if {$value > $max || $value < $min} {set overflow 1}
|
|
if {($value + $increment) > $max} {set overflow 1}
|
|
if {($value + $increment) < $min} {set overflow 1}
|
|
|
|
r del bits
|
|
set res1 [r bitfield bits overflow fail set $type 0 $value]
|
|
set res2 [r bitfield bits overflow fail incrby $type 0 $increment]
|
|
|
|
if {$overflow && [lindex $res1 0] ne {} &&
|
|
[lindex $res2 0] ne {}} {
|
|
fail "OW not detected where needed: $type $value+$increment"
|
|
}
|
|
if {!$overflow && ([lindex $res1 0] eq {} ||
|
|
[lindex $res2 0] eq {})} {
|
|
fail "OW detected where NOT needed: $type $value+$increment"
|
|
}
|
|
}
|
|
}
|
|
|
|
test {BITFIELD overflow wrap fuzzing} {
|
|
for {set j 0} {$j < 1000} {incr j} {
|
|
set bits [expr {[randomInt 64]+1}]
|
|
set sign [randomInt 2]
|
|
set range [expr {2**$bits}]
|
|
if {$bits == 64} {set sign 1} ; # u64 is not supported by BITFIELD.
|
|
if {$sign} {
|
|
set min [expr {-($range/2)}]
|
|
set type "i$bits"
|
|
} else {
|
|
set min 0
|
|
set type "u$bits"
|
|
}
|
|
set max [expr {$min+$range-1}]
|
|
|
|
# Compare Tcl vs Redict
|
|
set range2 [expr {$range*2}]
|
|
set value [expr {($min*2)+[randomInt $range2]}]
|
|
set increment [expr {($min*2)+[randomInt $range2]}]
|
|
if {$value > 9223372036854775807} {
|
|
set value 9223372036854775807
|
|
}
|
|
if {$value < -9223372036854775808} {
|
|
set value -9223372036854775808
|
|
}
|
|
if {$increment > 9223372036854775807} {
|
|
set increment 9223372036854775807
|
|
}
|
|
if {$increment < -9223372036854775808} {
|
|
set increment -9223372036854775808
|
|
}
|
|
|
|
r del bits
|
|
r bitfield bits overflow wrap set $type 0 $value
|
|
r bitfield bits overflow wrap incrby $type 0 $increment
|
|
set res [lindex [r bitfield bits get $type 0] 0]
|
|
|
|
set expected 0
|
|
if {$sign} {incr expected [expr {$max+1}]}
|
|
incr expected $value
|
|
incr expected $increment
|
|
set expected [expr {$expected % $range}]
|
|
if {$sign} {incr expected $min}
|
|
|
|
if {$res != $expected} {
|
|
fail "WRAP error: $type $value+$increment = $res, should be $expected"
|
|
}
|
|
}
|
|
}
|
|
|
|
test {BITFIELD regression for #3221} {
|
|
r set bits 1
|
|
r bitfield bits get u1 0
|
|
} {0}
|
|
|
|
test {BITFIELD regression for #3564} {
|
|
for {set j 0} {$j < 10} {incr j} {
|
|
r del mystring
|
|
set res [r BITFIELD mystring SET i8 0 10 SET i8 64 10 INCRBY i8 10 99900]
|
|
assert {$res eq {0 0 60}}
|
|
}
|
|
r del mystring
|
|
}
|
|
|
|
test {BITFIELD_RO with only key as argument} {
|
|
set res [r bitfield_ro bits]
|
|
assert {$res eq {}}
|
|
}
|
|
|
|
test {BITFIELD_RO fails when write option is used} {
|
|
catch {r bitfield_ro bits set u8 0 100 get u8 0} err
|
|
assert_match {*ERR BITFIELD_RO only supports the GET subcommand*} $err
|
|
}
|
|
}
|
|
|
|
start_server {tags {"repl external:skip"}} {
|
|
start_server {} {
|
|
set master [srv -1 client]
|
|
set master_host [srv -1 host]
|
|
set master_port [srv -1 port]
|
|
set slave [srv 0 client]
|
|
|
|
test {BITFIELD: setup slave} {
|
|
$slave slaveof $master_host $master_port
|
|
wait_for_condition 50 100 {
|
|
[s 0 master_link_status] eq {up}
|
|
} else {
|
|
fail "Replication not started."
|
|
}
|
|
}
|
|
|
|
test {BITFIELD: write on master, read on slave} {
|
|
$master del bits
|
|
assert_equal 0 [$master bitfield bits set u8 0 255]
|
|
assert_equal 255 [$master bitfield bits set u8 0 100]
|
|
wait_for_ofs_sync $master $slave
|
|
assert_equal 100 [$slave bitfield_ro bits get u8 0]
|
|
}
|
|
|
|
test {BITFIELD_RO with only key as argument on read-only replica} {
|
|
set res [$slave bitfield_ro bits]
|
|
assert {$res eq {}}
|
|
}
|
|
|
|
test {BITFIELD_RO fails when write option is used on read-only replica} {
|
|
catch {$slave bitfield_ro bits set u8 0 100 get u8 0} err
|
|
assert_match {*ERR BITFIELD_RO only supports the GET subcommand*} $err
|
|
}
|
|
}
|
|
}
|