Files
scylladb/redis/reply.hh
Etienne Adam 63a1a4cbb9 redis: add hgetall and hdel commands
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>
2020-09-08 11:59:52 +03:00

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");
}
};
}