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
c0aed241
Commit
c0aed241
authored
Dec 17, 2019
by
zhujiashun
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
redis_server_protocol: refine code
parent
01da505f
Expand all
Show whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
91 additions
and
91 deletions
+91
-91
redis_protocol.cpp
src/brpc/policy/redis_protocol.cpp
+28
-28
redis.cpp
src/brpc/redis.cpp
+6
-6
redis_command.h
src/brpc/redis_command.h
+1
-0
redis_reply.cpp
src/brpc/redis_reply.cpp
+16
-12
redis_reply.h
src/brpc/redis_reply.h
+40
-45
brpc_redis_unittest.cpp
test/brpc_redis_unittest.cpp
+0
-0
No files found.
src/brpc/policy/redis_protocol.cpp
View file @
c0aed241
...
@@ -67,59 +67,52 @@ public:
...
@@ -67,59 +67,52 @@ public:
SocketId
socket_id
;
SocketId
socket_id
;
RedisService
*
redis_service
;
RedisService
*
redis_service
;
// If user starts a transaction,
handler_continue
indicates the
// If user starts a transaction,
transaction_handler
indicates the
// handler pointer that runs the transaction command.
// handler pointer that runs the transaction command.
std
::
unique_ptr
<
RedisCommandHandler
>
handler_continue
;
std
::
unique_ptr
<
RedisCommandHandler
>
transaction_handler
;
// >0 if command handler is run in batched mode.
// >0 if command handler is run in batched mode.
int
batched_size
;
int
batched_size
;
RedisCommandParser
parser
;
RedisCommandParser
parser
;
};
};
static
std
::
string
ToLowercase
(
const
std
::
string
&
command
)
{
std
::
string
res
;
res
.
resize
(
command
.
size
());
std
::
transform
(
command
.
begin
(),
command
.
end
(),
res
.
begin
(),
[](
unsigned
char
c
){
return
std
::
tolower
(
c
);
});
return
res
;
}
int
ConsumeCommand
(
RedisConnContext
*
ctx
,
int
ConsumeCommand
(
RedisConnContext
*
ctx
,
const
std
::
unique_ptr
<
const
char
*
[]
>&
commands
,
const
std
::
unique_ptr
<
const
char
*
[]
>&
commands
,
int
len
,
butil
::
Arena
*
arena
,
int
command_
len
,
butil
::
Arena
*
arena
,
bool
is_last
,
bool
is_last
,
butil
::
IOBuf
*
sendbuf
)
{
butil
::
IOBuf
*
sendbuf
)
{
RedisReply
output
(
arena
);
RedisReply
output
(
arena
);
RedisCommandHandler
::
Result
result
=
RedisCommandHandler
::
OK
;
RedisCommandHandler
::
Result
result
=
RedisCommandHandler
::
OK
;
if
(
ctx
->
handler_continue
)
{
if
(
ctx
->
transaction_handler
)
{
result
=
ctx
->
handler_continue
->
Run
(
len
,
commands
.
get
(),
&
output
,
is_last
);
result
=
ctx
->
transaction_handler
->
Run
(
command_len
,
commands
.
get
(),
&
output
,
is_last
);
if
(
result
==
RedisCommandHandler
::
OK
)
{
if
(
result
==
RedisCommandHandler
::
OK
)
{
ctx
->
handler_continue
.
reset
(
NULL
);
ctx
->
transaction_handler
.
reset
(
NULL
);
}
else
if
(
result
==
RedisCommandHandler
::
BATCHED
)
{
}
else
if
(
result
==
RedisCommandHandler
::
BATCHED
)
{
LOG
(
ERROR
)
<<
"BATCHED should not be returned
in redis transaction process
."
;
LOG
(
ERROR
)
<<
"BATCHED should not be returned
by a transaction handler
."
;
return
-
1
;
return
-
1
;
}
}
}
else
{
}
else
{
std
::
string
lcname
=
ToLowercase
(
commands
[
0
]);
RedisCommandHandler
*
ch
=
ctx
->
redis_service
->
FindCommandHandler
(
commands
[
0
]);
RedisCommandHandler
*
ch
=
ctx
->
redis_service
->
FindCommandHandler
(
lcname
);
if
(
!
ch
)
{
if
(
!
ch
)
{
char
buf
[
64
];
char
buf
[
64
];
snprintf
(
buf
,
sizeof
(
buf
),
"ERR unknown command `%s`"
,
lcname
.
c_str
()
);
snprintf
(
buf
,
sizeof
(
buf
),
"ERR unknown command `%s`"
,
commands
[
0
]
);
output
.
SetError
(
buf
);
output
.
SetError
(
buf
);
}
else
{
}
else
{
result
=
ch
->
Run
(
len
,
commands
.
get
(),
&
output
,
is_last
);
result
=
ch
->
Run
(
command_
len
,
commands
.
get
(),
&
output
,
is_last
);
if
(
result
==
RedisCommandHandler
::
CONTINUE
)
{
if
(
result
==
RedisCommandHandler
::
CONTINUE
)
{
if
(
ctx
->
batched_size
)
{
if
(
ctx
->
batched_size
)
{
LOG
(
ERROR
)
<<
"CONTINUE should not be returned in redis batched process."
;
LOG
(
ERROR
)
<<
"CONTINUE should not be returned in redis batched process."
;
return
-
1
;
return
-
1
;
}
}
ctx
->
handler_continue
.
reset
(
ch
->
NewTransactionHandler
());
ctx
->
transaction_handler
.
reset
(
ch
->
NewTransactionHandler
());
}
else
if
(
result
==
RedisCommandHandler
::
BATCHED
)
{
}
else
if
(
result
==
RedisCommandHandler
::
BATCHED
)
{
ctx
->
batched_size
++
;
ctx
->
batched_size
++
;
}
}
}
}
}
}
if
(
result
==
RedisCommandHandler
::
OK
&&
ctx
->
batched_size
)
{
if
(
result
==
RedisCommandHandler
::
OK
)
{
if
(
ctx
->
batched_size
)
{
if
((
int
)
output
.
size
()
!=
(
ctx
->
batched_size
+
1
))
{
if
((
int
)
output
.
size
()
!=
(
ctx
->
batched_size
+
1
))
{
LOG
(
ERROR
)
<<
"reply array size can't be matched with batched size, "
LOG
(
ERROR
)
<<
"reply array size can't be matched with batched size, "
<<
" expected="
<<
ctx
->
batched_size
+
1
<<
" actual="
<<
output
.
size
();
<<
" expected="
<<
ctx
->
batched_size
+
1
<<
" actual="
<<
output
.
size
();
...
@@ -129,9 +122,17 @@ int ConsumeCommand(RedisConnContext* ctx,
...
@@ -129,9 +122,17 @@ int ConsumeCommand(RedisConnContext* ctx,
output
[
i
].
SerializeTo
(
sendbuf
);
output
[
i
].
SerializeTo
(
sendbuf
);
}
}
ctx
->
batched_size
=
0
;
ctx
->
batched_size
=
0
;
}
else
if
(
result
!=
RedisCommandHandler
::
BATCHED
)
{
}
else
{
output
.
SerializeTo
(
sendbuf
);
output
.
SerializeTo
(
sendbuf
);
}
// else result == RedisCommandHandler::BATCHED, do not serialize to buf
}
}
else
if
(
result
==
RedisCommandHandler
::
CONTINUE
)
{
output
.
SerializeTo
(
sendbuf
);
}
else
if
(
result
==
RedisCommandHandler
::
BATCHED
)
{
// just do nothing and wait handler to return OK.
}
else
{
LOG
(
ERROR
)
<<
"unknown status="
<<
result
;
return
-
1
;
}
return
0
;
return
0
;
}
}
...
@@ -180,16 +181,15 @@ ParseResult ParseRedisMessage(butil::IOBuf* source, Socket* socket,
...
@@ -180,16 +181,15 @@ ParseResult ParseRedisMessage(butil::IOBuf* source, Socket* socket,
if
(
err
!=
PARSE_OK
)
{
if
(
err
!=
PARSE_OK
)
{
break
;
break
;
}
}
// safe to read first element.
if
(
ConsumeCommand
(
ctx
,
current_commands
,
current_len
,
&
arena
,
// current_commands and next_commands both have at least one element(NULL).
false
,
&
sendbuf
)
!=
0
)
{
bool
is_last
=
(
strcasecmp
(
current_commands
[
0
],
next_commands
[
0
])
!=
0
);
if
(
ConsumeCommand
(
ctx
,
current_commands
,
current_len
,
&
arena
,
is_last
,
&
sendbuf
)
!=
0
)
{
return
MakeParseError
(
PARSE_ERROR_ABSOLUTELY_WRONG
);
return
MakeParseError
(
PARSE_ERROR_ABSOLUTELY_WRONG
);
}
}
current_commands
.
swap
(
next_commands
);
current_commands
.
swap
(
next_commands
);
current_len
=
next_len
;
current_len
=
next_len
;
}
}
if
(
ConsumeCommand
(
ctx
,
current_commands
,
current_len
,
&
arena
,
true
,
&
sendbuf
)
!=
0
)
{
if
(
ConsumeCommand
(
ctx
,
current_commands
,
current_len
,
&
arena
,
true
/* must be last message */
,
&
sendbuf
)
!=
0
)
{
return
MakeParseError
(
PARSE_ERROR_ABSOLUTELY_WRONG
);
return
MakeParseError
(
PARSE_ERROR_ABSOLUTELY_WRONG
);
}
}
CHECK
(
!
sendbuf
.
empty
());
CHECK
(
!
sendbuf
.
empty
());
...
...
src/brpc/redis.cpp
View file @
c0aed241
...
@@ -20,6 +20,7 @@
...
@@ -20,6 +20,7 @@
#include <google/protobuf/reflection_ops.h> // ReflectionOps::Merge
#include <google/protobuf/reflection_ops.h> // ReflectionOps::Merge
#include <gflags/gflags.h>
#include <gflags/gflags.h>
#include "butil/status.h"
#include "butil/status.h"
#include "butil/strings/string_util.h" // StringToLowerASCII
#include "brpc/redis.h"
#include "brpc/redis.h"
#include "brpc/redis_command.h"
#include "brpc/redis_command.h"
...
@@ -437,11 +438,9 @@ std::ostream& operator<<(std::ostream& os, const RedisResponse& response) {
...
@@ -437,11 +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
=
StringToLowerASCII
(
name
);
lcname
.
resize
(
name
.
size
());
auto
it
=
_command_map
.
find
(
lcname
);
std
::
transform
(
name
.
begin
(),
name
.
end
(),
lcname
.
begin
(),
if
(
it
!=
_command_map
.
end
())
{
[](
unsigned
char
c
){
return
std
::
tolower
(
c
);
});
if
(
_command_map
.
count
(
lcname
))
{
LOG
(
ERROR
)
<<
"redis command name="
<<
name
<<
" exist"
;
LOG
(
ERROR
)
<<
"redis command name="
<<
name
<<
" exist"
;
return
false
;
return
false
;
}
}
...
@@ -450,7 +449,8 @@ bool RedisService::AddCommandHandler(const std::string& name, RedisCommandHandle
...
@@ -450,7 +449,8 @@ bool RedisService::AddCommandHandler(const std::string& name, RedisCommandHandle
}
}
RedisCommandHandler
*
RedisService
::
FindCommandHandler
(
const
std
::
string
&
name
)
{
RedisCommandHandler
*
RedisService
::
FindCommandHandler
(
const
std
::
string
&
name
)
{
auto
it
=
_command_map
.
find
(
name
);
std
::
string
lcname
=
StringToLowerASCII
(
name
);
auto
it
=
_command_map
.
find
(
lcname
);
if
(
it
!=
_command_map
.
end
())
{
if
(
it
!=
_command_map
.
end
())
{
return
it
->
second
;
return
it
->
second
;
}
}
...
...
src/brpc/redis_command.h
View file @
c0aed241
...
@@ -20,6 +20,7 @@
...
@@ -20,6 +20,7 @@
#ifndef BRPC_REDIS_COMMAND_H
#ifndef BRPC_REDIS_COMMAND_H
#define BRPC_REDIS_COMMAND_H
#define BRPC_REDIS_COMMAND_H
#include <memory> // std::unique_ptr
#include <vector>
#include <vector>
#include "butil/iobuf.h"
#include "butil/iobuf.h"
#include "butil/status.h"
#include "butil/status.h"
...
...
src/brpc/redis_reply.cpp
View file @
c0aed241
...
@@ -413,34 +413,39 @@ void RedisReply::CopyFromDifferentArena(const RedisReply& other,
...
@@ -413,34 +413,39 @@ void RedisReply::CopyFromDifferentArena(const RedisReply& other,
}
}
}
}
bool
RedisReply
::
SetArray
(
int
size
)
{
void
RedisReply
::
SetArray
(
int
size
)
{
if
(
!
_arena
||
_type
!=
REDIS_REPLY_NIL
)
{
if
(
!
_arena
)
{
return
false
;
return
;
}
if
(
_type
!=
REDIS_REPLY_NIL
)
{
Reset
();
}
}
_type
=
REDIS_REPLY_ARRAY
;
_type
=
REDIS_REPLY_ARRAY
;
if
(
size
<
0
)
{
if
(
size
<
0
)
{
LOG
(
ERROR
)
<<
"negative size="
<<
size
<<
" when calling SetArray"
;
LOG
(
ERROR
)
<<
"negative size="
<<
size
<<
" when calling SetArray"
;
return
false
;
return
;
}
else
if
(
size
==
0
)
{
}
else
if
(
size
==
0
)
{
_length
=
0
;
_length
=
0
;
return
true
;
return
;
}
}
RedisReply
*
subs
=
(
RedisReply
*
)
_arena
->
allocate
(
sizeof
(
RedisReply
)
*
size
);
RedisReply
*
subs
=
(
RedisReply
*
)
_arena
->
allocate
(
sizeof
(
RedisReply
)
*
size
);
if
(
!
subs
)
{
if
(
!
subs
)
{
LOG
(
FATAL
)
<<
"Fail to allocate RedisReply["
<<
size
<<
"]"
;
LOG
(
FATAL
)
<<
"Fail to allocate RedisReply["
<<
size
<<
"]"
;
return
false
;
return
;
}
}
for
(
int
i
=
0
;
i
<
size
;
++
i
)
{
for
(
int
i
=
0
;
i
<
size
;
++
i
)
{
new
(
&
subs
[
i
])
RedisReply
(
_arena
);
new
(
&
subs
[
i
])
RedisReply
(
_arena
);
}
}
_length
=
size
;
_length
=
size
;
_data
.
array
.
replies
=
subs
;
_data
.
array
.
replies
=
subs
;
return
true
;
}
}
bool
RedisReply
::
SetBasicString
(
const
std
::
string
&
str
,
RedisReplyType
type
)
{
void
RedisReply
::
SetStringImpl
(
const
std
::
string
&
str
,
RedisReplyType
type
)
{
if
(
!
_arena
||
_type
!=
REDIS_REPLY_NIL
)
{
if
(
!
_arena
)
{
return
false
;
return
;
}
if
(
_type
!=
REDIS_REPLY_NIL
)
{
Reset
();
}
}
const
size_t
size
=
str
.
size
();
const
size_t
size
=
str
.
size
();
if
(
size
<
sizeof
(
_data
.
short_str
))
{
if
(
size
<
sizeof
(
_data
.
short_str
))
{
...
@@ -450,7 +455,7 @@ bool RedisReply::SetBasicString(const std::string& str, RedisReplyType type) {
...
@@ -450,7 +455,7 @@ bool RedisReply::SetBasicString(const std::string& str, RedisReplyType type) {
char
*
d
=
(
char
*
)
_arena
->
allocate
((
size
/
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
;
}
}
memcpy
(
d
,
str
.
c_str
(),
size
);
memcpy
(
d
,
str
.
c_str
(),
size
);
d
[
size
]
=
'\0'
;
d
[
size
]
=
'\0'
;
...
@@ -458,7 +463,6 @@ bool RedisReply::SetBasicString(const std::string& str, RedisReplyType type) {
...
@@ -458,7 +463,6 @@ bool RedisReply::SetBasicString(const std::string& str, RedisReplyType type) {
}
}
_type
=
type
;
_type
=
type
;
_length
=
size
;
_length
=
size
;
return
true
;
}
}
}
// namespace brpc
}
// namespace brpc
src/brpc/redis_reply.h
View file @
c0aed241
...
@@ -59,35 +59,28 @@ public:
...
@@ -59,35 +59,28 @@ 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.
// Set the reply to the null string. Return True if it is set
// Set the reply to the null string.
// successfully. If the reply has already been set, return false.
void
SetNullString
();
bool
SetNullString
();
// Set the reply to the null array. Return True if it is set
// Set the reply to the null array.
// successfully. If the reply has already been set, return false.
void
SetNullArray
();
bool
SetNullArray
();
// Set the reply to the array with `size' elements. After calling
// Set the reply to the array with `size' elements. After calling
// SetArray, use operator[] to visit sub replies and set their
// SetArray, use operator[] to visit sub replies and set their
// value. Return True if it is set successfully. If the reply has
// value.
// already been set, return false.
void
SetArray
(
int
size
);
bool
SetArray
(
int
size
);
// Set the reply to status message `str'. Return True if it is set
// Set the reply to status message `str'.
// successfully. If the reply has already been set, return false.
void
SetStatus
(
const
std
::
string
&
str
);
bool
SetStatus
(
const
std
::
string
&
str
);
// Set the reply to error message `str'. Return True if it is set
// Set the reply to error message `str'.
// successfully. If the reply has already been set, return false.
void
SetError
(
const
std
::
string
&
str
);
bool
SetError
(
const
std
::
string
&
str
);
// Set the reply to integer `value'. Return True if it is set
// Set the reply to integer `value'.
// successfully. If the reply has already been set, return false.
void
SetInteger
(
int64_t
value
);
bool
SetInteger
(
int64_t
value
);
// Set the reply to string `str'. Return True if it is set
// Set the reply to string `str'.
// successfully. If the reply has already been set, return false.
void
SetString
(
const
std
::
string
&
str
);
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,
...
@@ -156,7 +149,8 @@ private:
...
@@ -156,7 +149,8 @@ private:
// by calling CopyFrom[Different|Same]Arena.
// by calling CopyFrom[Different|Same]Arena.
DISALLOW_COPY_AND_ASSIGN
(
RedisReply
);
DISALLOW_COPY_AND_ASSIGN
(
RedisReply
);
bool
SetBasicString
(
const
std
::
string
&
str
,
RedisReplyType
type
);
void
SetStringImpl
(
const
std
::
string
&
str
,
RedisReplyType
type
);
void
Reset
();
RedisReplyType
_type
;
RedisReplyType
_type
;
uint32_t
_length
;
// length of short_str/long_str, count of replies
uint32_t
_length
;
// length of short_str/long_str, count of replies
...
@@ -180,17 +174,21 @@ inline std::ostream& operator<<(std::ostream& os, const RedisReply& r) {
...
@@ -180,17 +174,21 @@ inline std::ostream& operator<<(std::ostream& os, const RedisReply& r) {
return
os
;
return
os
;
}
}
inline
RedisReply
::
RedisReply
(
butil
::
Arena
*
arena
)
inline
void
RedisReply
::
Reset
()
{
:
RedisReply
()
{
_type
=
REDIS_REPLY_NIL
;
_length
=
0
;
_data
.
array
.
last_index
=
-
1
;
_data
.
array
.
replies
=
NULL
;
}
inline
RedisReply
::
RedisReply
(
butil
::
Arena
*
arena
)
{
Reset
();
_arena
=
arena
;
_arena
=
arena
;
}
}
inline
RedisReply
::
RedisReply
()
inline
RedisReply
::
RedisReply
()
{
:
_type
(
REDIS_REPLY_NIL
)
Reset
();
,
_length
(
0
)
_arena
=
NULL
;
,
_arena
(
NULL
)
{
_data
.
array
.
last_index
=
-
1
;
_data
.
array
.
replies
=
NULL
;
}
}
inline
bool
RedisReply
::
is_nil
()
const
{
inline
bool
RedisReply
::
is_nil
()
const
{
...
@@ -213,44 +211,41 @@ inline int64_t RedisReply::integer() const {
...
@@ -213,44 +211,41 @@ inline int64_t RedisReply::integer() const {
return
0
;
return
0
;
}
}
inline
bool
RedisReply
::
SetNullArray
()
{
inline
void
RedisReply
::
SetNullArray
()
{
if
(
_type
!=
REDIS_REPLY_NIL
)
{
if
(
_type
!=
REDIS_REPLY_NIL
)
{
return
false
;
Reset
()
;
}
}
_type
=
REDIS_REPLY_ARRAY
;
_type
=
REDIS_REPLY_ARRAY
;
_length
=
npos
;
_length
=
npos
;
return
true
;
}
}
inline
bool
RedisReply
::
SetNullString
()
{
inline
void
RedisReply
::
SetNullString
()
{
if
(
_type
!=
REDIS_REPLY_NIL
)
{
if
(
_type
!=
REDIS_REPLY_NIL
)
{
return
false
;
Reset
()
;
}
}
_type
=
REDIS_REPLY_STRING
;
_type
=
REDIS_REPLY_STRING
;
_length
=
npos
;
_length
=
npos
;
return
true
;
}
}
inline
bool
RedisReply
::
SetStatus
(
const
std
::
string
&
str
)
{
inline
void
RedisReply
::
SetStatus
(
const
std
::
string
&
str
)
{
return
Set
BasicString
(
str
,
REDIS_REPLY_STATUS
);
return
Set
StringImpl
(
str
,
REDIS_REPLY_STATUS
);
}
}
inline
bool
RedisReply
::
SetError
(
const
std
::
string
&
str
)
{
inline
void
RedisReply
::
SetError
(
const
std
::
string
&
str
)
{
return
Set
BasicString
(
str
,
REDIS_REPLY_ERROR
);
return
Set
StringImpl
(
str
,
REDIS_REPLY_ERROR
);
}
}
inline
bool
RedisReply
::
SetInteger
(
int64_t
value
)
{
inline
void
RedisReply
::
SetInteger
(
int64_t
value
)
{
if
(
_type
!=
REDIS_REPLY_NIL
)
{
if
(
_type
!=
REDIS_REPLY_NIL
)
{
return
false
;
Reset
()
;
}
}
_type
=
REDIS_REPLY_INTEGER
;
_type
=
REDIS_REPLY_INTEGER
;
_length
=
0
;
_length
=
0
;
_data
.
integer
=
value
;
_data
.
integer
=
value
;
return
true
;
}
}
inline
bool
RedisReply
::
SetString
(
const
std
::
string
&
str
)
{
inline
void
RedisReply
::
SetString
(
const
std
::
string
&
str
)
{
return
Set
BasicString
(
str
,
REDIS_REPLY_STRING
);
return
Set
StringImpl
(
str
,
REDIS_REPLY_STRING
);
}
}
inline
const
char
*
RedisReply
::
c_str
()
const
{
inline
const
char
*
RedisReply
::
c_str
()
const
{
...
...
test/brpc_redis_unittest.cpp
View file @
c0aed241
This diff is collapsed.
Click to expand it.
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