Commit c0aed241 authored by zhujiashun's avatar zhujiashun

redis_server_protocol: refine code

parent 01da505f
...@@ -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());
......
...@@ -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;
} }
......
...@@ -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"
......
...@@ -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
...@@ -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 SetBasicString(str, REDIS_REPLY_STATUS); return SetStringImpl(str, REDIS_REPLY_STATUS);
} }
inline bool RedisReply::SetError(const std::string& str) { inline void RedisReply::SetError(const std::string& str) {
return SetBasicString(str, REDIS_REPLY_ERROR); return SetStringImpl(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 SetBasicString(str, REDIS_REPLY_STRING); return SetStringImpl(str, REDIS_REPLY_STRING);
} }
inline const char* RedisReply::c_str() const { inline const char* RedisReply::c_str() const {
......
This diff is collapsed.
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment