Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in / Register
Toggle navigation
B
brpc
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Packages
Packages
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
submodule
brpc
Commits
684f4ed1
Commit
684f4ed1
authored
Dec 06, 2019
by
zhujiashun
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
redis_server_protocol: refine code, add comment and more ut
parent
ba7f3d5f
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
160 additions
and
95 deletions
+160
-95
redis_server.cpp
example/redis_c++/redis_server.cpp
+20
-14
redis_protocol.cpp
src/brpc/policy/redis_protocol.cpp
+43
-28
redis.cpp
src/brpc/redis.cpp
+8
-7
redis.h
src/brpc/redis.h
+2
-8
redis_reply.h
src/brpc/redis_reply.h
+44
-13
brpc_redis_unittest.cpp
test/brpc_redis_unittest.cpp
+43
-25
No files found.
example/redis_c++/redis_server.cpp
View file @
684f4ed1
...
@@ -64,23 +64,26 @@ public:
...
@@ -64,23 +64,26 @@ public:
brpc
::
RedisReply
*
output
)
{
brpc
::
RedisReply
*
output
)
{
std
::
string
key
;
std
::
string
key
;
bool
parse_command
=
false
;
bool
parse_command
=
false
;
butil
::
StringSplitter
sp
(
args
,
' '
);
const
char
*
start
=
args
;
for
(;
sp
;
++
sp
)
{
const
char
*
end
=
NULL
;
while
((
end
=
strchr
(
start
,
' '
))
!=
NULL
)
{
if
(
!
parse_command
)
{
if
(
!
parse_command
)
{
parse_command
=
true
;
parse_command
=
true
;
}
else
if
(
key
.
empty
())
{
}
else
if
(
key
.
empty
())
{
key
.
assign
(
sp
.
field
(),
sp
.
length
());
key
.
assign
(
start
,
end
);
}
else
{
LOG
(
WARNING
)
<<
"unknown args: "
<<
sp
;
}
}
start
=
end
+
1
;
}
}
if
(
key
.
empty
()
)
{
if
(
!
parse_command
)
{
output
->
SetError
(
"ERR wrong number of arguments for 'get' command"
);
output
->
SetError
(
"ERR wrong number of arguments for 'get' command"
);
return
brpc
::
RedisCommandHandler
::
OK
;
return
brpc
::
RedisCommandHandler
::
OK
;
}
}
if
(
key
.
empty
())
{
key
.
assign
(
start
);
}
std
::
string
value
;
std
::
string
value
;
if
(
_rsimpl
->
Get
(
key
,
&
value
))
{
if
(
_rsimpl
->
Get
(
key
,
&
value
))
{
output
->
Set
Bulk
String
(
value
);
output
->
SetString
(
value
);
}
else
{
}
else
{
output
->
SetNilString
();
output
->
SetNilString
();
}
}
...
@@ -102,22 +105,25 @@ public:
...
@@ -102,22 +105,25 @@ public:
std
::
string
key
;
std
::
string
key
;
std
::
string
value
;
std
::
string
value
;
bool
parse_command
=
false
;
bool
parse_command
=
false
;
butil
::
StringSplitter
sp
(
args
,
' '
);
const
char
*
start
=
args
;
for
(;
sp
;
++
sp
)
{
const
char
*
end
=
NULL
;
while
((
end
=
strchr
(
start
,
' '
))
!=
NULL
)
{
if
(
!
parse_command
)
{
if
(
!
parse_command
)
{
parse_command
=
true
;
parse_command
=
true
;
}
else
if
(
key
.
empty
())
{
}
else
if
(
key
.
empty
())
{
key
.
assign
(
s
p
.
field
(),
sp
.
length
()
);
key
.
assign
(
s
tart
,
end
);
}
else
if
(
value
.
empty
())
{
}
else
if
(
value
.
empty
())
{
value
.
assign
(
sp
.
field
(),
sp
.
length
());
value
.
assign
(
start
,
end
);
}
else
{
LOG
(
WARNING
)
<<
"unknown args: "
<<
sp
;
}
}
start
=
end
+
1
;
}
}
if
(
key
.
empty
()
||
value
.
empty
())
{
if
(
!
parse_command
||
key
.
empty
())
{
output
->
SetError
(
"ERR wrong number of arguments for 'set' command"
);
output
->
SetError
(
"ERR wrong number of arguments for 'set' command"
);
return
brpc
::
RedisCommandHandler
::
OK
;
return
brpc
::
RedisCommandHandler
::
OK
;
}
}
if
(
value
.
empty
())
{
value
.
assign
(
start
);
}
_rsimpl
->
Set
(
key
,
value
);
_rsimpl
->
Set
(
key
,
value
);
output
->
SetStatus
(
"OK"
);
output
->
SetStatus
(
"OK"
);
return
brpc
::
RedisCommandHandler
::
OK
;
return
brpc
::
RedisCommandHandler
::
OK
;
...
...
src/brpc/policy/redis_protocol.cpp
View file @
684f4ed1
...
@@ -44,8 +44,9 @@ namespace policy {
...
@@ -44,8 +44,9 @@ namespace policy {
DEFINE_bool
(
redis_verbose
,
false
,
DEFINE_bool
(
redis_verbose
,
false
,
"[DEBUG] Print EVERY redis request/response"
);
"[DEBUG] Print EVERY redis request/response"
);
DEFINE_int32
(
redis_batch_flush_max_size
,
4096
,
"beyond which the server response"
DEFINE_int32
(
redis_batch_flush_data_size
,
4096
,
"If the total data size of buffered "
" are forced to write to socket"
);
"responses is beyond this value, then data is forced to write to socket"
"to avoid latency of the front responses being too big"
);
struct
InputResponse
:
public
InputMessageBase
{
struct
InputResponse
:
public
InputMessageBase
{
bthread_id_t
id_wait
;
bthread_id_t
id_wait
;
...
@@ -57,21 +58,34 @@ struct InputResponse : public InputMessageBase {
...
@@ -57,21 +58,34 @@ struct InputResponse : public InputMessageBase {
}
}
};
};
static
bool
ParseArgs
(
const
RedisReply
&
message
,
std
::
ostringstream
&
o
s
)
{
static
bool
ParseArgs
(
const
RedisReply
&
message
,
std
::
unique_ptr
<
char
[]
>*
arg
s
)
{
if
(
!
message
.
is_array
()
||
message
.
size
()
==
0
)
{
if
(
!
message
.
is_array
()
||
message
.
size
()
==
0
)
{
LOG
(
WARNING
)
<<
"request message is not array or size equals to zero"
;
LOG
(
WARNING
)
<<
"request message is not array or size equals to zero"
;
return
false
;
return
false
;
}
}
int
total_size
=
0
;
for
(
size_t
i
=
0
;
i
<
message
.
size
();
++
i
)
{
for
(
size_t
i
=
0
;
i
<
message
.
size
();
++
i
)
{
if
(
!
message
[
i
].
is_string
())
{
if
(
!
message
[
i
].
is_string
())
{
LOG
(
WARNING
)
<<
"request message["
<<
i
<<
"] is not array"
;
LOG
(
WARNING
)
<<
"request message["
<<
i
<<
"] is not array"
;
return
false
;
return
false
;
}
}
if
(
i
!=
0
)
{
if
(
i
!=
0
)
{
os
<<
" "
;
total_size
++
;
// add one byte for ' '
}
}
os
<<
message
[
i
].
c_str
();
total_size
+=
message
[
i
].
size
();
}
}
args
->
reset
(
new
char
[
total_size
+
1
/* NULL */
]);
int
len
=
0
;
for
(
size_t
i
=
0
;
i
<
message
.
size
();
++
i
)
{
if
(
i
!=
0
)
{
(
*
args
)[
len
++
]
=
' '
;
}
memcpy
(
args
->
get
()
+
len
,
message
[
i
].
c_str
(),
message
[
i
].
size
());
len
+=
message
[
i
].
size
();
}
(
*
args
)[
len
]
=
'\0'
;
CHECK
(
len
==
total_size
)
<<
"implementation of ParseArgs is buggy, len="
<<
len
<<
" expected="
<<
total_size
;
return
true
;
return
true
;
}
}
...
@@ -81,19 +95,19 @@ struct RedisTask {
...
@@ -81,19 +95,19 @@ struct RedisTask {
};
};
// This class is as parsing_context in socket.
// This class is as parsing_context in socket.
class
RedisConnContext
:
public
SharedObject
class
RedisConnContext
:
public
Destroyable
{
,
public
Destroyable
{
public
:
public
:
RedisConnContext
()
RedisConnContext
()
:
handler_continue
(
NULL
)
{}
:
redis_service
(
NULL
)
,
handler_continue
(
NULL
)
{}
~
RedisConnContext
();
~
RedisConnContext
();
// @Destroyable
// @Destroyable
void
Destroy
();
void
Destroy
()
override
;
int
Init
();
int
Init
();
SocketId
socket_id
;
SocketId
socket_id
;
RedisService
::
CommandMap
command_map
;
RedisService
*
redis_service
;
// If user starts a transaction, handler_continue indicates the
// If user starts a transaction, handler_continue indicates the
// first handler pointer that triggers the transaction.
// first handler pointer that triggers the transaction.
RedisCommandHandler
*
handler_continue
;
RedisCommandHandler
*
handler_continue
;
...
@@ -106,33 +120,31 @@ public:
...
@@ -106,33 +120,31 @@ public:
int
ConsumeTask
(
RedisConnContext
*
ctx
,
RedisTask
*
task
,
butil
::
IOBuf
*
sendbuf
)
{
int
ConsumeTask
(
RedisConnContext
*
ctx
,
RedisTask
*
task
,
butil
::
IOBuf
*
sendbuf
)
{
RedisReply
output
(
&
task
->
arena
);
RedisReply
output
(
&
task
->
arena
);
std
::
ostringstream
o
s
;
std
::
unique_ptr
<
char
[]
>
arg
s
;
if
(
!
ParseArgs
(
task
->
input_message
,
o
s
))
{
if
(
!
ParseArgs
(
task
->
input_message
,
&
arg
s
))
{
LOG
(
ERROR
)
<<
"ERR command not string"
;
LOG
(
ERROR
)
<<
"ERR command not string"
;
output
.
SetError
(
"ERR command not string"
);
output
.
SetError
(
"ERR command not string"
);
return
-
1
;
return
-
1
;
}
}
if
(
ctx
->
handler_continue
)
{
if
(
ctx
->
handler_continue
)
{
RedisCommandHandler
::
Result
result
=
RedisCommandHandler
::
Result
result
=
ctx
->
handler_continue
->
Run
(
os
.
str
().
c_str
(),
&
output
);
ctx
->
handler_continue
->
Run
(
args
.
get
(),
&
output
);
if
(
result
==
RedisCommandHandler
::
OK
)
{
if
(
result
==
RedisCommandHandler
::
OK
)
{
ctx
->
handler_continue
=
NULL
;
ctx
->
handler_continue
=
NULL
;
}
}
}
else
{
}
else
{
std
::
string
comm
;
std
::
string
comm
=
task
->
input_message
[
0
].
c_str
();
comm
.
reserve
(
8
);
std
::
transform
(
comm
.
begin
(),
comm
.
end
(),
comm
.
begin
(),
for
(
const
char
*
c
=
task
->
input_message
[
0
].
c_str
();
*
c
;
++
c
)
{
[](
unsigned
char
c
){
return
std
::
tolower
(
c
);
});
comm
.
push_back
(
std
::
tolower
(
*
c
));
RedisCommandHandler
*
ch
=
ctx
->
redis_service
->
FindCommandHandler
(
comm
);
}
if
(
!
ch
)
{
auto
it
=
ctx
->
command_map
.
find
(
comm
);
if
(
it
==
ctx
->
command_map
.
end
())
{
char
buf
[
64
];
char
buf
[
64
];
snprintf
(
buf
,
sizeof
(
buf
),
"ERR unknown command `%s`"
,
comm
.
c_str
());
snprintf
(
buf
,
sizeof
(
buf
),
"ERR unknown command `%s`"
,
comm
.
c_str
());
output
.
SetError
(
buf
);
output
.
SetError
(
buf
);
}
else
{
}
else
{
RedisCommandHandler
::
Result
result
=
it
->
second
->
Run
(
os
.
str
().
c_str
(),
&
output
);
RedisCommandHandler
::
Result
result
=
ch
->
Run
(
args
.
get
(),
&
output
);
if
(
result
==
RedisCommandHandler
::
CONTINUE
)
{
if
(
result
==
RedisCommandHandler
::
CONTINUE
)
{
ctx
->
handler_continue
=
it
->
second
.
get
()
;
ctx
->
handler_continue
=
ch
;
}
}
}
}
}
}
...
@@ -143,7 +155,7 @@ int ConsumeTask(RedisConnContext* ctx, RedisTask* task, butil::IOBuf* sendbuf) {
...
@@ -143,7 +155,7 @@ int ConsumeTask(RedisConnContext* ctx, RedisTask* task, butil::IOBuf* sendbuf) {
int
Consume
(
void
*
ctx
,
bthread
::
TaskIterator
<
RedisTask
*>&
iter
)
{
int
Consume
(
void
*
ctx
,
bthread
::
TaskIterator
<
RedisTask
*>&
iter
)
{
RedisConnContext
*
qctx
=
static_cast
<
RedisConnContext
*>
(
ctx
);
RedisConnContext
*
qctx
=
static_cast
<
RedisConnContext
*>
(
ctx
);
if
(
iter
.
is_queue_stopped
())
{
if
(
iter
.
is_queue_stopped
())
{
qctx
->
RemoveRefManually
()
;
delete
qctx
;
return
0
;
return
0
;
}
}
SocketUniquePtr
s
;
SocketUniquePtr
s
;
...
@@ -164,7 +176,12 @@ int Consume(void* ctx, bthread::TaskIterator<RedisTask*>& iter) {
...
@@ -164,7 +176,12 @@ int Consume(void* ctx, bthread::TaskIterator<RedisTask*>& iter) {
has_err
=
true
;
has_err
=
true
;
continue
;
continue
;
}
}
if
((
int
)
sendbuf
.
size
()
>=
FLAGS_redis_batch_flush_max_size
)
{
// If there are too many tasks to execute, latency of the front
// responses will be increased by waiting the following tasks to
// be completed. To prevent this, if the current buf size is greater
// than FLAGS_redis_batch_flush_max_size, we just write the current
// buf first.
if
((
int
)
sendbuf
.
size
()
>=
FLAGS_redis_batch_flush_data_size
)
{
LOG_IF
(
WARNING
,
s
->
Write
(
&
sendbuf
,
&
wopt
)
!=
0
)
LOG_IF
(
WARNING
,
s
->
Write
(
&
sendbuf
,
&
wopt
)
!=
0
)
<<
"Fail to send redis reply"
;
<<
"Fail to send redis reply"
;
}
}
...
@@ -211,12 +228,10 @@ ParseResult ParseRedisMessage(butil::IOBuf* source, Socket* socket,
...
@@ -211,12 +228,10 @@ ParseResult ParseRedisMessage(butil::IOBuf* source, Socket* socket,
RedisConnContext
*
ctx
=
static_cast
<
RedisConnContext
*>
(
socket
->
parsing_context
());
RedisConnContext
*
ctx
=
static_cast
<
RedisConnContext
*>
(
socket
->
parsing_context
());
if
(
ctx
==
NULL
)
{
if
(
ctx
==
NULL
)
{
ctx
=
new
RedisConnContext
;
ctx
=
new
RedisConnContext
;
// add ref that removed in Consume()
ctx
->
AddRefManually
();
ctx
->
socket_id
=
socket
->
id
();
ctx
->
socket_id
=
socket
->
id
();
rs
->
CloneCommandMap
(
&
ctx
->
command_map
)
;
ctx
->
redis_service
=
rs
;
if
(
ctx
->
Init
()
!=
0
)
{
if
(
ctx
->
Init
()
!=
0
)
{
ctx
->
RemoveRefManually
()
;
delete
ctx
;
LOG
(
ERROR
)
<<
"Fail to init redis RedisConnContext"
;
LOG
(
ERROR
)
<<
"Fail to init redis RedisConnContext"
;
return
MakeParseError
(
PARSE_ERROR_NO_RESOURCE
);
return
MakeParseError
(
PARSE_ERROR_NO_RESOURCE
);
}
}
...
...
src/brpc/redis.cpp
View file @
684f4ed1
...
@@ -438,10 +438,9 @@ std::ostream& operator<<(std::ostream& os, const RedisResponse& response) {
...
@@ -438,10 +438,9 @@ std::ostream& operator<<(std::ostream& os, const RedisResponse& response) {
bool
RedisService
::
AddCommandHandler
(
const
std
::
string
&
name
,
RedisCommandHandler
*
handler
)
{
bool
RedisService
::
AddCommandHandler
(
const
std
::
string
&
name
,
RedisCommandHandler
*
handler
)
{
std
::
string
lcname
;
std
::
string
lcname
;
lcname
.
reserve
(
name
.
size
());
lcname
.
resize
(
name
.
size
());
for
(
auto
c
:
name
)
{
std
::
transform
(
name
.
begin
(),
name
.
end
(),
lcname
.
begin
(),
lcname
.
push_back
(
std
::
tolower
(
c
));
[](
unsigned
char
c
){
return
std
::
tolower
(
c
);
});
}
if
(
_command_map
.
count
(
lcname
))
{
if
(
_command_map
.
count
(
lcname
))
{
LOG
(
ERROR
)
<<
"redis command name="
<<
name
<<
" exist"
;
LOG
(
ERROR
)
<<
"redis command name="
<<
name
<<
" exist"
;
return
false
;
return
false
;
...
@@ -450,10 +449,12 @@ bool RedisService::AddCommandHandler(const std::string& name, RedisCommandHandle
...
@@ -450,10 +449,12 @@ bool RedisService::AddCommandHandler(const std::string& name, RedisCommandHandle
return
true
;
return
true
;
}
}
void
RedisService
::
CloneCommandMap
(
CommandMap
*
map
)
{
RedisCommandHandler
*
RedisService
::
FindCommandHandler
(
const
std
::
string
&
name
)
{
for
(
auto
it
:
_command_map
)
{
auto
it
=
_command_map
.
find
(
name
);
(
*
map
)[
it
.
first
].
reset
(
it
.
second
->
New
());
if
(
it
!=
_command_map
.
end
())
{
return
it
->
second
.
get
();
}
}
return
NULL
;
}
}
}
// namespace brpc
}
// namespace brpc
src/brpc/redis.h
View file @
684f4ed1
...
@@ -226,12 +226,12 @@ public:
...
@@ -226,12 +226,12 @@ public:
bool
AddCommandHandler
(
const
std
::
string
&
name
,
RedisCommandHandler
*
handler
);
bool
AddCommandHandler
(
const
std
::
string
&
name
,
RedisCommandHandler
*
handler
);
// This function should be touched by user and used by brpc deverloper only.
// This function should be touched by user and used by brpc deverloper only.
void
CloneCommandMap
(
CommandMap
*
map
);
RedisCommandHandler
*
FindCommandHandler
(
const
std
::
string
&
name
);
private
:
private
:
CommandMap
_command_map
;
CommandMap
_command_map
;
};
};
// The Command handler for a redis request. User should impletement Run()
and New()
.
// The Command handler for a redis request. User should impletement Run().
class
RedisCommandHandler
{
class
RedisCommandHandler
{
public
:
public
:
enum
Result
{
enum
Result
{
...
@@ -259,12 +259,6 @@ public:
...
@@ -259,12 +259,6 @@ public:
// all once the ending marker is received.
// all once the ending marker is received.
virtual
RedisCommandHandler
::
Result
Run
(
const
char
*
args
,
virtual
RedisCommandHandler
::
Result
Run
(
const
char
*
args
,
RedisReply
*
output
)
=
0
;
RedisReply
*
output
)
=
0
;
// Whenever a tcp connection is established, a bunch of new handlers would be created
// using New() of the corresponding handler and brpc makes sure that all requests from
// one connection with the same command name would be redirected to the same New()-ed
// command handler.
virtual
RedisCommandHandler
*
New
()
=
0
;
};
};
}
// namespace brpc
}
// namespace brpc
...
...
src/brpc/redis_reply.h
View file @
684f4ed1
...
@@ -59,12 +59,32 @@ public:
...
@@ -59,12 +59,32 @@ public:
bool
is_string
()
const
;
// True if the reply is a string.
bool
is_string
()
const
;
// True if the reply is a string.
bool
is_array
()
const
;
// True if the reply is an array.
bool
is_array
()
const
;
// True if the reply is an array.
bool
SetNilString
();
// "$-1\r\n"
// Set the reply to the nil string. Return True if it is set
bool
SetArray
(
int
size
);
// size == -1 means nil array("*-1\r\n")
// successfully. If the reply has already been set, return false.
bool
SetNilString
();
// Set the reply to the array with `size' elements. If `size'
// is -1, then it is a nil array. After call SetArray, use
// operator[] to visit sub replies and set their value. Return
// True if it is set successfully. If the reply has already
// been set, return false.
bool
SetArray
(
int
size
);
// Set the reply to status message `str'. Return True if it is set
// successfully. If the reply has already been set, return false.
bool
SetStatus
(
const
std
::
string
&
str
);
bool
SetStatus
(
const
std
::
string
&
str
);
// Set the reply to error message `str'. Return True if it is set
// successfully. If the reply has already been set, return false.
bool
SetError
(
const
std
::
string
&
str
);
bool
SetError
(
const
std
::
string
&
str
);
// Set the reply to integer `value'. Return True if it is set
// successfully. If the reply has already been set, return false.
bool
SetInteger
(
int64_t
value
);
bool
SetInteger
(
int64_t
value
);
bool
SetBulkString
(
const
std
::
string
&
str
);
// Set the reply to string `str'. Return True if it is set
// successfully. If the reply has already been set, return false.
bool
SetString
(
const
std
::
string
&
str
);
// Convert the reply into a signed 64-bit integer(according to
// Convert the reply into a signed 64-bit integer(according to
// http://redis.io/topics/protocol). If the reply is not an integer,
// http://redis.io/topics/protocol). If the reply is not an integer,
...
@@ -84,8 +104,9 @@ public:
...
@@ -84,8 +104,9 @@ public:
// If you need a std::string, call .data().as_string() (which allocates mem)
// If you need a std::string, call .data().as_string() (which allocates mem)
butil
::
StringPiece
data
()
const
;
butil
::
StringPiece
data
()
const
;
// Return number of sub replies in the array. If this reply is not an array,
// Return number of sub replies in the array if this reply is an array, or
// 0 is returned (call stacks are not logged).
// return the length of string if this reply is a string, otherwise 0 is
// returned (call stacks are not logged).
size_t
size
()
const
;
size_t
size
()
const
;
// Get the index-th sub reply. If this reply is not an array, a nil reply
// Get the index-th sub reply. If this reply is not an array, a nil reply
// is returned (call stacks are not logged)
// is returned (call stacks are not logged)
...
@@ -147,6 +168,7 @@ private:
...
@@ -147,6 +168,7 @@ private:
uint64_t
padding
[
2
];
// For swapping, must cover all bytes.
uint64_t
padding
[
2
];
// For swapping, must cover all bytes.
}
_data
;
}
_data
;
butil
::
Arena
*
_arena
;
butil
::
Arena
*
_arena
;
bool
_has_set
;
};
};
// =========== inline impl. ==============
// =========== inline impl. ==============
...
@@ -164,7 +186,8 @@ inline RedisReply::RedisReply(butil::Arena* arena)
...
@@ -164,7 +186,8 @@ inline RedisReply::RedisReply(butil::Arena* arena)
inline
RedisReply
::
RedisReply
()
inline
RedisReply
::
RedisReply
()
:
_type
(
REDIS_REPLY_NIL
)
:
_type
(
REDIS_REPLY_NIL
)
,
_length
(
0
)
,
_length
(
0
)
,
_arena
(
NULL
)
{
,
_arena
(
NULL
)
,
_has_set
(
false
)
{
_data
.
array
.
last_index
=
-
1
;
_data
.
array
.
last_index
=
-
1
;
_data
.
array
.
replies
=
NULL
;
_data
.
array
.
replies
=
NULL
;
}
}
...
@@ -190,14 +213,15 @@ inline int64_t RedisReply::integer() const {
...
@@ -190,14 +213,15 @@ inline int64_t RedisReply::integer() const {
}
}
inline
bool
RedisReply
::
SetNilString
()
{
inline
bool
RedisReply
::
SetNilString
()
{
if
(
!
_arena
)
return
false
;
if
(
!
_arena
||
_has_set
)
return
false
;
_type
=
REDIS_REPLY_STRING
;
_type
=
REDIS_REPLY_STRING
;
_length
=
npos
;
_length
=
npos
;
_has_set
=
true
;
return
true
;
return
true
;
}
}
inline
bool
RedisReply
::
SetArray
(
int
size
)
{
inline
bool
RedisReply
::
SetArray
(
int
size
)
{
if
(
!
_arena
)
{
if
(
!
_arena
||
_has_set
)
{
return
false
;
return
false
;
}
}
_type
=
REDIS_REPLY_ARRAY
;
_type
=
REDIS_REPLY_ARRAY
;
...
@@ -218,19 +242,20 @@ inline bool RedisReply::SetArray(int size) {
...
@@ -218,19 +242,20 @@ inline bool RedisReply::SetArray(int size) {
}
}
_length
=
size
;
_length
=
size
;
_data
.
array
.
replies
=
subs
;
_data
.
array
.
replies
=
subs
;
_has_set
=
true
;
return
true
;
return
true
;
}
}
inline
bool
RedisReply
::
SetBasicString
(
const
std
::
string
&
str
,
RedisReplyType
type
)
{
inline
bool
RedisReply
::
SetBasicString
(
const
std
::
string
&
str
,
RedisReplyType
type
)
{
if
(
!
_arena
)
{
if
(
!
_arena
||
_has_set
)
{
return
false
;
return
false
;
}
}
size_t
size
=
str
.
size
();
const
size_t
size
=
str
.
size
();
if
(
size
<
sizeof
(
_data
.
short_str
))
{
if
(
size
<
sizeof
(
_data
.
short_str
))
{
memcpy
(
_data
.
short_str
,
str
.
c_str
(),
size
);
memcpy
(
_data
.
short_str
,
str
.
c_str
(),
size
);
_data
.
short_str
[
size
]
=
'\0'
;
_data
.
short_str
[
size
]
=
'\0'
;
}
else
{
}
else
{
char
*
d
=
(
char
*
)
_arena
->
allocate
((
_length
/
8
+
1
)
*
8
);
char
*
d
=
(
char
*
)
_arena
->
allocate
((
size
/
8
+
1
)
*
8
);
if
(
!
d
)
{
if
(
!
d
)
{
LOG
(
FATAL
)
<<
"Fail to allocate string["
<<
size
<<
"]"
;
LOG
(
FATAL
)
<<
"Fail to allocate string["
<<
size
<<
"]"
;
return
false
;
return
false
;
...
@@ -241,6 +266,7 @@ inline bool RedisReply::SetBasicString(const std::string& str, RedisReplyType ty
...
@@ -241,6 +266,7 @@ inline bool RedisReply::SetBasicString(const std::string& str, RedisReplyType ty
}
}
_type
=
type
;
_type
=
type
;
_length
=
size
;
_length
=
size
;
_has_set
=
true
;
return
true
;
return
true
;
}
}
...
@@ -253,13 +279,17 @@ inline bool RedisReply::SetError(const std::string& str) {
...
@@ -253,13 +279,17 @@ inline bool RedisReply::SetError(const std::string& str) {
}
}
inline
bool
RedisReply
::
SetInteger
(
int64_t
value
)
{
inline
bool
RedisReply
::
SetInteger
(
int64_t
value
)
{
if
(
!
_arena
||
_has_set
)
{
return
false
;
}
_type
=
REDIS_REPLY_INTEGER
;
_type
=
REDIS_REPLY_INTEGER
;
_length
=
0
;
_length
=
0
;
_data
.
integer
=
value
;
_data
.
integer
=
value
;
_has_set
=
true
;
return
true
;
return
true
;
}
}
inline
bool
RedisReply
::
Set
Bulk
String
(
const
std
::
string
&
str
)
{
inline
bool
RedisReply
::
SetString
(
const
std
::
string
&
str
)
{
return
SetBasicString
(
str
,
REDIS_REPLY_STRING
);
return
SetBasicString
(
str
,
REDIS_REPLY_STRING
);
}
}
...
@@ -303,7 +333,7 @@ inline const char* RedisReply::error_message() const {
...
@@ -303,7 +333,7 @@ inline const char* RedisReply::error_message() const {
}
}
inline
size_t
RedisReply
::
size
()
const
{
inline
size_t
RedisReply
::
size
()
const
{
return
(
is_array
(
)
?
_length
:
0
);
return
(
(
is_array
()
||
is_string
()
)
?
_length
:
0
);
}
}
inline
RedisReply
&
RedisReply
::
operator
[](
size_t
index
)
{
inline
RedisReply
&
RedisReply
::
operator
[](
size_t
index
)
{
...
@@ -331,6 +361,7 @@ inline void RedisReply::Clear() {
...
@@ -331,6 +361,7 @@ inline void RedisReply::Clear() {
_length
=
0
;
_length
=
0
;
_data
.
array
.
last_index
=
-
1
;
_data
.
array
.
last_index
=
-
1
;
_data
.
array
.
replies
=
NULL
;
_data
.
array
.
replies
=
NULL
;
_has_set
=
false
;
}
}
inline
void
RedisReply
::
CopyFromSameArena
(
const
RedisReply
&
other
)
{
inline
void
RedisReply
::
CopyFromSameArena
(
const
RedisReply
&
other
)
{
...
...
test/brpc_redis_unittest.cpp
View file @
684f4ed1
...
@@ -549,7 +549,7 @@ TEST_F(RedisTest, quote_and_escape) {
...
@@ -549,7 +549,7 @@ TEST_F(RedisTest, quote_and_escape) {
request
.
Clear
();
request
.
Clear
();
}
}
TEST_F
(
RedisTest
,
codec
)
{
TEST_F
(
RedisTest
,
redis_reply_
codec
)
{
butil
::
Arena
arena
;
butil
::
Arena
arena
;
// status
// status
{
{
...
@@ -591,7 +591,7 @@ TEST_F(RedisTest, codec) {
...
@@ -591,7 +591,7 @@ TEST_F(RedisTest, codec) {
ASSERT_TRUE
(
r
.
is_nil
());
ASSERT_TRUE
(
r
.
is_nil
());
r
.
Clear
();
r
.
Clear
();
ASSERT_TRUE
(
r
.
Set
Bulk
String
(
"abcde'hello world"
));
ASSERT_TRUE
(
r
.
SetString
(
"abcde'hello world"
));
ASSERT_TRUE
(
r
.
SerializeToIOBuf
(
&
buf
));
ASSERT_TRUE
(
r
.
SerializeToIOBuf
(
&
buf
));
ASSERT_STREQ
(
buf
.
to_string
().
c_str
(),
"$17
\r\n
abcde'hello world
\r\n
"
);
ASSERT_STREQ
(
buf
.
to_string
().
c_str
(),
"$17
\r\n
abcde'hello world
\r\n
"
);
ASSERT_STREQ
(
r
.
c_str
(),
"abcde'hello world"
);
ASSERT_STREQ
(
r
.
c_str
(),
"abcde'hello world"
);
...
@@ -627,9 +627,9 @@ TEST_F(RedisTest, codec) {
...
@@ -627,9 +627,9 @@ TEST_F(RedisTest, codec) {
ASSERT_TRUE
(
r
.
SetArray
(
3
));
ASSERT_TRUE
(
r
.
SetArray
(
3
));
brpc
::
RedisReply
&
sub_reply
=
r
[
0
];
brpc
::
RedisReply
&
sub_reply
=
r
[
0
];
sub_reply
.
SetArray
(
2
);
sub_reply
.
SetArray
(
2
);
sub_reply
[
0
].
Set
Bulk
String
(
"hello, it's me"
);
sub_reply
[
0
].
SetString
(
"hello, it's me"
);
sub_reply
[
1
].
SetInteger
(
422
);
sub_reply
[
1
].
SetInteger
(
422
);
r
[
1
].
Set
Bulk
String
(
"To go over everything"
);
r
[
1
].
SetString
(
"To go over everything"
);
r
[
2
].
SetInteger
(
1
);
r
[
2
].
SetInteger
(
1
);
ASSERT_TRUE
(
r
[
3
].
is_nil
());
ASSERT_TRUE
(
r
[
3
].
is_nil
());
ASSERT_TRUE
(
r
.
SerializeToIOBuf
(
&
buf
));
ASSERT_TRUE
(
r
.
SerializeToIOBuf
(
&
buf
));
...
@@ -659,6 +659,44 @@ TEST_F(RedisTest, codec) {
...
@@ -659,6 +659,44 @@ TEST_F(RedisTest, codec) {
ASSERT_EQ
(
r
.
ConsumePartialIOBuf
(
buf
,
&
arena
),
brpc
::
PARSE_OK
);
ASSERT_EQ
(
r
.
ConsumePartialIOBuf
(
buf
,
&
arena
),
brpc
::
PARSE_OK
);
ASSERT_TRUE
(
r
.
is_nil
());
ASSERT_TRUE
(
r
.
is_nil
());
}
}
// CopyFromDifferentArena
{
brpc
::
RedisReply
r
(
&
arena
);
ASSERT_TRUE
(
r
.
SetArray
(
1
));
brpc
::
RedisReply
&
sub_reply
=
r
[
0
];
sub_reply
.
SetArray
(
2
);
sub_reply
[
0
].
SetString
(
"hello, it's me"
);
sub_reply
[
1
].
SetInteger
(
422
);
brpc
::
RedisReply
r2
(
NULL
);
r2
.
CopyFromDifferentArena
(
r
,
&
arena
);
ASSERT_TRUE
(
r2
.
is_array
());
ASSERT_EQ
((
int
)
r2
[
0
].
size
(),
2
);
ASSERT_STREQ
(
r2
[
0
][
0
].
c_str
(),
sub_reply
[
0
].
c_str
());
ASSERT_EQ
(
r2
[
0
][
1
].
integer
(),
sub_reply
[
1
].
integer
());
}
// SetXXX can only be called once
{
brpc
::
RedisReply
r
(
&
arena
);
ASSERT_TRUE
(
r
.
SetStatus
(
"OK"
));
ASSERT_FALSE
(
r
.
SetStatus
(
"OK"
));
ASSERT_FALSE
(
r
.
SetNilString
());
ASSERT_FALSE
(
r
.
SetArray
(
2
));
ASSERT_FALSE
(
r
.
SetString
(
"OK"
));
ASSERT_FALSE
(
r
.
SetError
(
"OK"
));
ASSERT_FALSE
(
r
.
SetInteger
(
42
));
}
{
brpc
::
RedisReply
r
(
&
arena
);
ASSERT_TRUE
(
r
.
SetInteger
(
42
));
ASSERT_FALSE
(
r
.
SetStatus
(
"OK"
));
ASSERT_FALSE
(
r
.
SetNilString
());
ASSERT_FALSE
(
r
.
SetArray
(
2
));
ASSERT_FALSE
(
r
.
SetString
(
"OK"
));
ASSERT_FALSE
(
r
.
SetError
(
"OK"
));
ASSERT_FALSE
(
r
.
SetStatus
(
"OK"
));
}
}
}
butil
::
Mutex
s_mutex
;
butil
::
Mutex
s_mutex
;
...
@@ -694,11 +732,6 @@ public:
...
@@ -694,11 +732,6 @@ public:
output
->
SetStatus
(
"OK"
);
output
->
SetStatus
(
"OK"
);
return
brpc
::
RedisCommandHandler
::
OK
;
return
brpc
::
RedisCommandHandler
::
OK
;
}
}
RedisCommandHandler
*
New
()
{
_new_count
++
;
return
new
SetCommandHandler
();
}
int
new_count
()
{
return
_new_count
;
}
private
:
int
_new_count
=
0
;
};
};
class
GetCommandHandler
:
public
brpc
::
RedisCommandHandler
{
class
GetCommandHandler
:
public
brpc
::
RedisCommandHandler
{
...
@@ -725,17 +758,12 @@ public:
...
@@ -725,17 +758,12 @@ public:
}
}
auto
it
=
m
.
find
(
key
);
auto
it
=
m
.
find
(
key
);
if
(
it
!=
m
.
end
())
{
if
(
it
!=
m
.
end
())
{
output
->
Set
Bulk
String
(
it
->
second
);
output
->
SetString
(
it
->
second
);
}
else
{
}
else
{
output
->
SetNilString
();
output
->
SetNilString
();
}
}
return
brpc
::
RedisCommandHandler
::
OK
;
return
brpc
::
RedisCommandHandler
::
OK
;
}
}
RedisCommandHandler
*
New
()
{
_new_count
++
;
return
new
GetCommandHandler
();
}
int
new_count
()
{
return
_new_count
;
}
private
:
int
_new_count
=
0
;
};
};
class
IncrCommandHandler
:
public
brpc
::
RedisCommandHandler
{
class
IncrCommandHandler
:
public
brpc
::
RedisCommandHandler
{
...
@@ -767,11 +795,6 @@ public:
...
@@ -767,11 +795,6 @@ public:
output
->
SetInteger
(
value
);
output
->
SetInteger
(
value
);
return
brpc
::
RedisCommandHandler
::
OK
;
return
brpc
::
RedisCommandHandler
::
OK
;
}
}
RedisCommandHandler
*
New
()
{
_new_count
++
;
return
new
IncrCommandHandler
();
}
int
new_count
()
{
return
_new_count
;
}
private
:
int
_new_count
=
0
;
};
};
class
RedisServiceImpl
:
public
brpc
::
RedisService
{
};
class
RedisServiceImpl
:
public
brpc
::
RedisService
{
};
...
@@ -820,10 +843,6 @@ TEST_F(RedisTest, server_sanity) {
...
@@ -820,10 +843,6 @@ TEST_F(RedisTest, server_sanity) {
ASSERT_STREQ
(
"value2"
,
response
.
reply
(
5
).
c_str
());
ASSERT_STREQ
(
"value2"
,
response
.
reply
(
5
).
c_str
());
ASSERT_EQ
(
brpc
::
REDIS_REPLY_ERROR
,
response
.
reply
(
6
).
type
());
ASSERT_EQ
(
brpc
::
REDIS_REPLY_ERROR
,
response
.
reply
(
6
).
type
());
ASSERT_TRUE
(
butil
::
StringPiece
(
response
.
reply
(
6
).
error_message
()).
starts_with
(
"ERR unknown command"
));
ASSERT_TRUE
(
butil
::
StringPiece
(
response
.
reply
(
6
).
error_message
()).
starts_with
(
"ERR unknown command"
));
ASSERT_EQ
(
gh
->
new_count
(),
1
);
ASSERT_EQ
(
sh
->
new_count
(),
1
);
ASSERT_EQ
(
ih
->
new_count
(),
1
);
}
}
void
*
incr_thread
(
void
*
arg
)
{
void
*
incr_thread
(
void
*
arg
)
{
...
@@ -871,7 +890,6 @@ TEST_F(RedisTest, server_concurrency) {
...
@@ -871,7 +890,6 @@ TEST_F(RedisTest, server_concurrency) {
delete
channels
[
i
];
delete
channels
[
i
];
}
}
ASSERT_EQ
(
int_map
[
"count"
],
10
*
5000LL
);
ASSERT_EQ
(
int_map
[
"count"
],
10
*
5000LL
);
ASSERT_EQ
(
ih
->
new_count
(),
N
);
}
}
class
MultiCommandHandler
:
public
brpc
::
RedisCommandHandler
{
class
MultiCommandHandler
:
public
brpc
::
RedisCommandHandler
{
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment