mirror of
https://codeberg.org/redict/redict.git
synced 2025-01-22 08:08:53 -05:00
50ee0f5be8
Based on feedback from interested parties
447 lines
14 KiB
Tcl
447 lines
14 KiB
Tcl
# Copyright (C) 2009-2014 Salvatore Sanfilippo
|
|
# 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
|
|
|
|
package require Tcl 8.5
|
|
package provide redict 0.1
|
|
|
|
source [file join [file dirname [info script]] "response_transformers.tcl"]
|
|
|
|
namespace eval redict {}
|
|
set ::redict::id 0
|
|
array set ::redict::fd {}
|
|
array set ::redict::addr {}
|
|
array set ::redict::blocking {}
|
|
array set ::redict::deferred {}
|
|
array set ::redict::readraw {}
|
|
array set ::redict::attributes {} ;# Holds the RESP3 attributes from the last call
|
|
array set ::redict::reconnect {}
|
|
array set ::redict::tls {}
|
|
array set ::redict::callback {}
|
|
array set ::redict::state {} ;# State in non-blocking reply reading
|
|
array set ::redict::statestack {} ;# Stack of states, for nested mbulks
|
|
array set ::redict::curr_argv {} ;# Remember the current argv, to be used in response_transformers.tcl
|
|
array set ::redict::testing_resp3 {} ;# Indicating if the current client is using RESP3 (only if the test is trying to test RESP3 specific behavior. It won't be on in case of force_resp3)
|
|
|
|
set ::force_resp3 0
|
|
set ::log_req_res 0
|
|
|
|
proc redict {{server 127.0.0.1} {port 6379} {defer 0} {tls 0} {tlsoptions {}} {readraw 0}} {
|
|
if {$tls} {
|
|
package require tls
|
|
::tls::init \
|
|
-cafile "$::tlsdir/ca.crt" \
|
|
-certfile "$::tlsdir/client.crt" \
|
|
-keyfile "$::tlsdir/client.key" \
|
|
{*}$tlsoptions
|
|
set fd [::tls::socket $server $port]
|
|
} else {
|
|
set fd [socket $server $port]
|
|
}
|
|
fconfigure $fd -translation binary
|
|
set id [incr ::redict::id]
|
|
set ::redict::fd($id) $fd
|
|
set ::redict::addr($id) [list $server $port]
|
|
set ::redict::blocking($id) 1
|
|
set ::redict::deferred($id) $defer
|
|
set ::redict::readraw($id) $readraw
|
|
set ::redict::reconnect($id) 0
|
|
set ::redict::curr_argv($id) 0
|
|
set ::redict::testing_resp3($id) 0
|
|
set ::redict::tls($id) $tls
|
|
::redict::redict_reset_state $id
|
|
interp alias {} ::redict::redictHandle$id {} ::redict::__dispatch__ $id
|
|
}
|
|
|
|
# On recent versions of tcl-tls/OpenSSL, reading from a dropped connection
|
|
# results with an error we need to catch and mimic the old behavior.
|
|
proc ::redict::redict_safe_read {fd len} {
|
|
if {$len == -1} {
|
|
set err [catch {set val [read $fd]} msg]
|
|
} else {
|
|
set err [catch {set val [read $fd $len]} msg]
|
|
}
|
|
if {!$err} {
|
|
return $val
|
|
}
|
|
if {[string match "*connection abort*" $msg]} {
|
|
return {}
|
|
}
|
|
error $msg
|
|
}
|
|
|
|
proc ::redict::redict_safe_gets {fd} {
|
|
if {[catch {set val [gets $fd]} msg]} {
|
|
if {[string match "*connection abort*" $msg]} {
|
|
return {}
|
|
}
|
|
error $msg
|
|
}
|
|
return $val
|
|
}
|
|
|
|
# This is a wrapper to the actual dispatching procedure that handles
|
|
# reconnection if needed.
|
|
proc ::redict::__dispatch__ {id method args} {
|
|
set errorcode [catch {::redict::__dispatch__raw__ $id $method $args} retval]
|
|
if {$errorcode && $::redict::reconnect($id) && $::redict::fd($id) eq {}} {
|
|
# Try again if the connection was lost.
|
|
# FIXME: we don't re-select the previously selected DB, nor we check
|
|
# if we are inside a transaction that needs to be re-issued from
|
|
# scratch.
|
|
set errorcode [catch {::redict::__dispatch__raw__ $id $method $args} retval]
|
|
}
|
|
return -code $errorcode $retval
|
|
}
|
|
|
|
proc ::redict::__dispatch__raw__ {id method argv} {
|
|
set fd $::redict::fd($id)
|
|
|
|
# Reconnect the link if needed.
|
|
if {$fd eq {} && $method ne {close}} {
|
|
lassign $::redict::addr($id) host port
|
|
if {$::redict::tls($id)} {
|
|
set ::redict::fd($id) [::tls::socket $host $port]
|
|
} else {
|
|
set ::redict::fd($id) [socket $host $port]
|
|
}
|
|
fconfigure $::redict::fd($id) -translation binary
|
|
set fd $::redict::fd($id)
|
|
}
|
|
|
|
# Transform HELLO 2 to HELLO 3 if force_resp3
|
|
# All set the connection var testing_resp3 in case of HELLO 3
|
|
if {[llength $argv] > 0 && [string compare -nocase $method "HELLO"] == 0} {
|
|
if {[lindex $argv 0] == 3} {
|
|
set ::redict::testing_resp3($id) 1
|
|
} else {
|
|
set ::redict::testing_resp3($id) 0
|
|
if {$::force_resp3} {
|
|
# If we are in force_resp3 we run HELLO 3 instead of HELLO 2
|
|
lset argv 0 3
|
|
}
|
|
}
|
|
}
|
|
|
|
set blocking $::redict::blocking($id)
|
|
set deferred $::redict::deferred($id)
|
|
if {$blocking == 0} {
|
|
if {[llength $argv] == 0} {
|
|
error "Please provide a callback in non-blocking mode"
|
|
}
|
|
set callback [lindex $argv end]
|
|
set argv [lrange $argv 0 end-1]
|
|
}
|
|
if {[info command ::redict::__method__$method] eq {}} {
|
|
catch {unset ::redict::attributes($id)}
|
|
set cmd "*[expr {[llength $argv]+1}]\r\n"
|
|
append cmd "$[string length $method]\r\n$method\r\n"
|
|
foreach a $argv {
|
|
append cmd "$[string length $a]\r\n$a\r\n"
|
|
}
|
|
::redict::redict_write $fd $cmd
|
|
if {[catch {flush $fd}]} {
|
|
catch {close $fd}
|
|
set ::redict::fd($id) {}
|
|
return -code error "I/O error reading reply"
|
|
}
|
|
|
|
set ::redict::curr_argv($id) [concat $method $argv]
|
|
if {!$deferred} {
|
|
if {$blocking} {
|
|
::redict::redict_read_reply $id $fd
|
|
} else {
|
|
# Every well formed reply read will pop an element from this
|
|
# list and use it as a callback. So pipelining is supported
|
|
# in non blocking mode.
|
|
lappend ::redict::callback($id) $callback
|
|
fileevent $fd readable [list ::redict::redict_readable $fd $id]
|
|
}
|
|
}
|
|
} else {
|
|
uplevel 1 [list ::redict::__method__$method $id $fd] $argv
|
|
}
|
|
}
|
|
|
|
proc ::redict::__method__blocking {id fd val} {
|
|
set ::redict::blocking($id) $val
|
|
fconfigure $fd -blocking $val
|
|
}
|
|
|
|
proc ::redict::__method__reconnect {id fd val} {
|
|
set ::redict::reconnect($id) $val
|
|
}
|
|
|
|
proc ::redict::__method__read {id fd} {
|
|
::redict::redict_read_reply $id $fd
|
|
}
|
|
|
|
proc ::redict::__method__rawread {id fd {len -1}} {
|
|
return [redict_safe_read $fd $len]
|
|
}
|
|
|
|
proc ::redict::__method__write {id fd buf} {
|
|
::redict::redict_write $fd $buf
|
|
}
|
|
|
|
proc ::redict::__method__flush {id fd} {
|
|
flush $fd
|
|
}
|
|
|
|
proc ::redict::__method__close {id fd} {
|
|
catch {close $fd}
|
|
catch {unset ::redict::fd($id)}
|
|
catch {unset ::redict::addr($id)}
|
|
catch {unset ::redict::blocking($id)}
|
|
catch {unset ::redict::deferred($id)}
|
|
catch {unset ::redict::readraw($id)}
|
|
catch {unset ::redict::attributes($id)}
|
|
catch {unset ::redict::reconnect($id)}
|
|
catch {unset ::redict::tls($id)}
|
|
catch {unset ::redict::state($id)}
|
|
catch {unset ::redict::statestack($id)}
|
|
catch {unset ::redict::callback($id)}
|
|
catch {unset ::redict::curr_argv($id)}
|
|
catch {unset ::redict::testing_resp3($id)}
|
|
catch {interp alias {} ::redict::redictHandle$id {}}
|
|
}
|
|
|
|
proc ::redict::__method__channel {id fd} {
|
|
return $fd
|
|
}
|
|
|
|
proc ::redict::__method__deferred {id fd val} {
|
|
set ::redict::deferred($id) $val
|
|
}
|
|
|
|
proc ::redict::__method__readraw {id fd val} {
|
|
set ::redict::readraw($id) $val
|
|
}
|
|
|
|
proc ::redict::__method__readingraw {id fd} {
|
|
return $::redict::readraw($id)
|
|
}
|
|
|
|
proc ::redict::__method__attributes {id fd} {
|
|
set _ $::redict::attributes($id)
|
|
}
|
|
|
|
proc ::redict::redict_write {fd buf} {
|
|
puts -nonewline $fd $buf
|
|
}
|
|
|
|
proc ::redict::redict_writenl {fd buf} {
|
|
redict_write $fd $buf
|
|
redict_write $fd "\r\n"
|
|
flush $fd
|
|
}
|
|
|
|
proc ::redict::redict_readnl {fd len} {
|
|
set buf [redict_safe_read $fd $len]
|
|
redict_safe_read $fd 2 ; # discard CR LF
|
|
return $buf
|
|
}
|
|
|
|
proc ::redict::redict_bulk_read {fd} {
|
|
set count [redict_read_line $fd]
|
|
if {$count == -1} return {}
|
|
set buf [redict_readnl $fd $count]
|
|
return $buf
|
|
}
|
|
|
|
proc ::redict::redict_multi_bulk_read {id fd} {
|
|
set count [redict_read_line $fd]
|
|
if {$count == -1} return {}
|
|
set l {}
|
|
set err {}
|
|
for {set i 0} {$i < $count} {incr i} {
|
|
if {[catch {
|
|
lappend l [redict_read_reply_logic $id $fd]
|
|
} e] && $err eq {}} {
|
|
set err $e
|
|
}
|
|
}
|
|
if {$err ne {}} {return -code error $err}
|
|
return $l
|
|
}
|
|
|
|
proc ::redict::redict_read_map {id fd} {
|
|
set count [redict_read_line $fd]
|
|
if {$count == -1} return {}
|
|
set d {}
|
|
set err {}
|
|
for {set i 0} {$i < $count} {incr i} {
|
|
if {[catch {
|
|
set k [redict_read_reply_logic $id $fd] ; # key
|
|
set v [redict_read_reply_logic $id $fd] ; # value
|
|
dict set d $k $v
|
|
} e] && $err eq {}} {
|
|
set err $e
|
|
}
|
|
}
|
|
if {$err ne {}} {return -code error $err}
|
|
return $d
|
|
}
|
|
|
|
proc ::redict::redict_read_line fd {
|
|
string trim [redict_safe_gets $fd]
|
|
}
|
|
|
|
proc ::redict::redict_read_null fd {
|
|
redict_safe_gets $fd
|
|
return {}
|
|
}
|
|
|
|
proc ::redict::redict_read_bool fd {
|
|
set v [redict_read_line $fd]
|
|
if {$v == "t"} {return 1}
|
|
if {$v == "f"} {return 0}
|
|
return -code error "Bad protocol, '$v' as bool type"
|
|
}
|
|
|
|
proc ::redict::redict_read_double {id fd} {
|
|
set v [redict_read_line $fd]
|
|
# unlike many other DTs, there is a textual difference between double and a string with the same value,
|
|
# so we need to transform to double if we are testing RESP3 (i.e. some tests check that a
|
|
# double reply is "1.0" and not "1")
|
|
if {[should_transform_to_resp2 $id]} {
|
|
return $v
|
|
} else {
|
|
return [expr {double($v)}]
|
|
}
|
|
}
|
|
|
|
proc ::redict::redict_read_verbatim_str fd {
|
|
set v [redict_bulk_read $fd]
|
|
# strip the first 4 chars ("txt:")
|
|
return [string range $v 4 end]
|
|
}
|
|
|
|
proc ::redict::redict_read_reply_logic {id fd} {
|
|
if {$::redict::readraw($id)} {
|
|
return [redict_read_line $fd]
|
|
}
|
|
|
|
while {1} {
|
|
set type [redict_safe_read $fd 1]
|
|
switch -exact -- $type {
|
|
_ {return [redict_read_null $fd]}
|
|
: -
|
|
( -
|
|
+ {return [redict_read_line $fd]}
|
|
, {return [redict_read_double $id $fd]}
|
|
# {return [redict_read_bool $fd]}
|
|
= {return [redict_read_verbatim_str $fd]}
|
|
- {return -code error [redict_read_line $fd]}
|
|
$ {return [redict_bulk_read $fd]}
|
|
> -
|
|
~ -
|
|
* {return [redict_multi_bulk_read $id $fd]}
|
|
% {return [redict_read_map $id $fd]}
|
|
| {
|
|
set attrib [redict_read_map $id $fd]
|
|
set ::redict::attributes($id) $attrib
|
|
continue
|
|
}
|
|
default {
|
|
if {$type eq {}} {
|
|
catch {close $fd}
|
|
set ::redict::fd($id) {}
|
|
return -code error "I/O error reading reply"
|
|
}
|
|
return -code error "Bad protocol, '$type' as reply type byte"
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
proc ::redict::redict_read_reply {id fd} {
|
|
set response [redict_read_reply_logic $id $fd]
|
|
::response_transformers::transform_response_if_needed $id $::redict::curr_argv($id) $response
|
|
}
|
|
|
|
proc ::redict::redict_reset_state id {
|
|
set ::redict::state($id) [dict create buf {} mbulk -1 bulk -1 reply {}]
|
|
set ::redict::statestack($id) {}
|
|
}
|
|
|
|
proc ::redict::redis.call_callback {id type reply} {
|
|
set cb [lindex $::redict::callback($id) 0]
|
|
set ::redict::callback($id) [lrange $::redict::callback($id) 1 end]
|
|
uplevel #0 $cb [list ::redict::redictHandle$id $type $reply]
|
|
::redict::redict_reset_state $id
|
|
}
|
|
|
|
# Read a reply in non-blocking mode.
|
|
proc ::redict::redict_readable {fd id} {
|
|
if {[eof $fd]} {
|
|
redis.call_callback $id eof {}
|
|
::redict::__method__close $id $fd
|
|
return
|
|
}
|
|
if {[dict get $::redict::state($id) bulk] == -1} {
|
|
set line [gets $fd]
|
|
if {$line eq {}} return ;# No complete line available, return
|
|
switch -exact -- [string index $line 0] {
|
|
: -
|
|
+ {redis.call_callback $id reply [string range $line 1 end-1]}
|
|
- {redis.call_callback $id err [string range $line 1 end-1]}
|
|
( {redis.call_callback $id reply [string range $line 1 end-1]}
|
|
$ {
|
|
dict set ::redict::state($id) bulk \
|
|
[expr [string range $line 1 end-1]+2]
|
|
if {[dict get $::redict::state($id) bulk] == 1} {
|
|
# We got a $-1, hack the state to play well with this.
|
|
dict set ::redict::state($id) bulk 2
|
|
dict set ::redict::state($id) buf "\r\n"
|
|
::redict::redict_readable $fd $id
|
|
}
|
|
}
|
|
* {
|
|
dict set ::redict::state($id) mbulk [string range $line 1 end-1]
|
|
# Handle *-1
|
|
if {[dict get $::redict::state($id) mbulk] == -1} {
|
|
redis.call_callback $id reply {}
|
|
}
|
|
}
|
|
default {
|
|
redis.call_callback $id err \
|
|
"Bad protocol, $type as reply type byte"
|
|
}
|
|
}
|
|
} else {
|
|
set totlen [dict get $::redict::state($id) bulk]
|
|
set buflen [string length [dict get $::redict::state($id) buf]]
|
|
set toread [expr {$totlen-$buflen}]
|
|
set data [read $fd $toread]
|
|
set nread [string length $data]
|
|
dict append ::redict::state($id) buf $data
|
|
# Check if we read a complete bulk reply
|
|
if {[string length [dict get $::redict::state($id) buf]] ==
|
|
[dict get $::redict::state($id) bulk]} {
|
|
if {[dict get $::redict::state($id) mbulk] == -1} {
|
|
redis.call_callback $id reply \
|
|
[string range [dict get $::redict::state($id) buf] 0 end-2]
|
|
} else {
|
|
dict with ::redict::state($id) {
|
|
lappend reply [string range $buf 0 end-2]
|
|
incr mbulk -1
|
|
set bulk -1
|
|
}
|
|
if {[dict get $::redict::state($id) mbulk] == 0} {
|
|
redis.call_callback $id reply \
|
|
[dict get $::redict::state($id) reply]
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# when forcing resp3 some tests that rely on resp2 can fail, so we have to translate the resp3 response to resp2
|
|
proc ::redict::should_transform_to_resp2 {id} {
|
|
return [expr {$::force_resp3 && !$::redict::testing_resp3($id)}]
|
|
}
|