This patch adds support for 2 hash commands HDEL and HGETALL. Internally it introduces the hashes_result_builder class to read hashes and stored them in a std::map. Other changes: - one exception return string was fixed - tests now use pytest.raises Signed-off-by: Etienne Adam <etienne.adam@gmail.com> Message-Id: <20200907202528.4985-1-etienne.adam@gmail.com>
127 lines
4.7 KiB
C++
127 lines
4.7 KiB
C++
/*
|
|
* Copyright (C) 2019 pengjian.uestc @ gmail.com
|
|
*/
|
|
|
|
/*
|
|
* This file is part of Scylla.
|
|
*
|
|
* Scylla is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* Scylla is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with Scylla. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include "bytes.hh"
|
|
#include "seastar/core/sharded.hh"
|
|
#include "seastar/core/shared_ptr.hh"
|
|
#include <seastar/core/print.hh>
|
|
#include "seastar/core/scattered_message.hh"
|
|
#include "redis/exceptions.hh"
|
|
|
|
using namespace seastar;
|
|
|
|
namespace redis {
|
|
|
|
class redis_message final {
|
|
seastar::lw_shared_ptr<scattered_message<char>> _message;
|
|
public:
|
|
redis_message() = delete;
|
|
redis_message(const redis_message&) = delete;
|
|
redis_message& operator=(const redis_message&) = delete;
|
|
redis_message(redis_message&& o) noexcept : _message(std::move(o._message)) {}
|
|
redis_message(lw_shared_ptr<scattered_message<char>> m) noexcept : _message(m) {}
|
|
static future<redis_message> ok() {
|
|
auto m = make_lw_shared<scattered_message<char>> ();
|
|
m->append_static("+OK\r\n");
|
|
return make_ready_future<redis_message>(m);
|
|
}
|
|
static future<redis_message> pong() {
|
|
auto m = make_lw_shared<scattered_message<char>> ();
|
|
m->append_static("+PONG\r\n");
|
|
return make_ready_future<redis_message>(m);
|
|
}
|
|
static future<redis_message> zero() {
|
|
auto m = make_lw_shared<scattered_message<char>> ();
|
|
m->append_static(":0\r\n");
|
|
return make_ready_future<redis_message>(m);
|
|
}
|
|
static future<redis_message> one() {
|
|
auto m = make_lw_shared<scattered_message<char>> ();
|
|
m->append_static(":1\r\n");
|
|
return make_ready_future<redis_message>(m);
|
|
}
|
|
static future<redis_message> nil() {
|
|
auto m = make_lw_shared<scattered_message<char>> ();
|
|
m->append_static("$-1\r\n");
|
|
return make_ready_future<redis_message>(m);
|
|
}
|
|
static future<redis_message> err() {
|
|
return zero();
|
|
}
|
|
static future<redis_message> number(size_t n) {
|
|
auto m = make_lw_shared<scattered_message<char>> ();
|
|
m->append(sprint(":%zu\r\n", n));
|
|
return make_ready_future<redis_message>(m);
|
|
}
|
|
static future<redis_message> make_list_result(std::map<bytes, bytes>& list_result) {
|
|
auto m = make_lw_shared<scattered_message<char>> ();
|
|
m->append(sprint("*%u\r\n", list_result.size() * 2));
|
|
for (auto r : list_result) {
|
|
write_bytes(m, (bytes&)r.first);
|
|
write_bytes(m, r.second);
|
|
}
|
|
return make_ready_future<redis_message>(m);
|
|
}
|
|
static future<redis_message> make_strings_result(bytes result) {
|
|
auto m = make_lw_shared<scattered_message<char>> ();
|
|
write_bytes(m, result);
|
|
return make_ready_future<redis_message>(m);
|
|
}
|
|
static future<redis_message> unknown(const bytes& name) {
|
|
return from_exception(make_message("-ERR unknown command '%s'\r\n", to_sstring(name)));
|
|
}
|
|
static future<redis_message> exception(const sstring& em) {
|
|
auto m = make_lw_shared<scattered_message<char>> ();
|
|
m->append(make_message("-ERR %s\r\n", em));
|
|
return make_ready_future<redis_message>(m);
|
|
}
|
|
static future<redis_message> exception(const redis_exception& e) {
|
|
return exception(e.what_message());
|
|
}
|
|
inline lw_shared_ptr<scattered_message<char>> message() { return _message; }
|
|
private:
|
|
static future<redis_message> from_exception(sstring data) {
|
|
auto m = make_lw_shared<scattered_message<char>> ();
|
|
m->append(data);
|
|
return make_ready_future<redis_message>(m);
|
|
}
|
|
template<typename... Args>
|
|
static inline sstring make_message(const char* fmt, Args&&... args) noexcept {
|
|
try {
|
|
return sprint(fmt, std::forward<Args>(args)...);
|
|
} catch (...) {
|
|
return sstring();
|
|
}
|
|
}
|
|
static sstring to_sstring(const bytes& b) {
|
|
return sstring(reinterpret_cast<const char*>(b.data()), b.size());
|
|
}
|
|
static void write_bytes(lw_shared_ptr<scattered_message<char>> m, bytes& b) {
|
|
m->append(sprint("$%d\r\n", b.size()));
|
|
m->append(std::string_view(reinterpret_cast<const char*>(b.data()), b.size()));
|
|
m->append_static("\r\n");
|
|
}
|
|
};
|
|
|
|
}
|