Files
scylladb/redis/server.cc

169 lines
6.4 KiB
C++

/*
* Copyright (C) 2019 pengjian.uestc @ gmail.com
*/
/*
* SPDX-License-Identifier: LicenseRef-ScyllaDB-Source-Available-1.0
*/
#include "redis/server.hh"
#include "generic_server.hh"
#include "redis/request.hh"
#include "redis/reply.hh"
#include "db/consistency_level_type.hh"
#include "db/config.hh"
#include <seastar/core/semaphore.hh>
#include <seastar/core/shared_ptr.hh>
#include <seastar/core/future-util.hh>
#include <seastar/core/seastar.hh>
#include <seastar/net/byteorder.hh>
#include <seastar/core/execution_stage.hh>
#include <seastar/util/defer.hh>
#include <cassert>
#include <unordered_map>
namespace redis_transport {
static logging::logger logging("redis_server");
redis_server::redis_server(seastar::sharded<redis::query_processor>& qp, auth::service& auth_service, const db::config& cfg, redis_server_config config)
: server("Redis", logging, generic_server::config{cfg.uninitialized_connections_semaphore_cpu_concurrency})
, _query_processor(qp)
, _config(std::move(config))
, _max_request_size(_config._max_request_size)
, _memory_available(_max_request_size)
, _auth_service(auth_service)
, _total_redis_db_count(_config._total_redis_db_count)
{
}
shared_ptr<generic_server::connection>
redis_server::make_connection(socket_address server_addr, connected_socket&& fd, socket_address addr, named_semaphore& sem, semaphore_units<named_semaphore_exception_factory> initial_sem_units) {
return make_shared<connection>(*this, server_addr, std::move(fd), std::move(addr), sem, std::move(initial_sem_units));
}
future<redis_server::result> redis_server::connection::process_request_one(redis::request&& request, redis::redis_options& opts, service_permit permit) {
return futurize_invoke([this, request = std::move(request), &opts, permit] () mutable {
return _server._query_processor.local().process(std::move(request), seastar::ref(opts), permit).then([] (auto&& message) {
return make_ready_future<redis_server::result> (std::move(message));
});
});
}
redis_server::connection::connection(redis_server& server, socket_address server_addr, connected_socket&& fd, socket_address addr, named_semaphore& sem, semaphore_units<named_semaphore_exception_factory> initial_sem_units)
: generic_server::connection(server, std::move(fd), sem, std::move(initial_sem_units))
, _server(server)
, _server_addr(server_addr)
, _options(server._config._read_consistency_level, server._config._write_consistency_level, server._config._timeout_config, server._auth_service, addr, server._total_redis_db_count)
{
++_server._stats._connects;
++_server._stats._connections;
}
redis_server::connection::~connection() {
--_server._stats._connections;
}
thread_local redis_server::connection::execution_stage_type redis_server::connection::_process_request_stage {"redis_transport", &connection::process_request_one};
future<redis_server::result> redis_server::connection::process_request_internal() {
return _process_request_stage(this, _parser.get_request(), seastar::ref(_options), empty_service_permit());
}
void redis_server::connection::write_reply(const redis_exception& e)
{
_ready_to_respond = _ready_to_respond.then([this, exception_message = e.what_message()] () mutable {
return redis_message::exception(exception_message).then([this] (auto&& result) {
auto m = result.message();
return _write_buf.write(std::move(*m)).then([this] {
return _write_buf.flush();
});
});
});
}
void redis_server::connection::write_reply(redis_server::result result)
{
_ready_to_respond = _ready_to_respond.then([this, result = std::move(result)] () mutable {
auto m = result.make_message();
return _write_buf.write(std::move(*m)).then([this] {
return _write_buf.flush();
});
});
}
future<> redis_server::connection::process_request() {
_parser.init();
return _read_buf.consume(_parser).then([this] {
if (_parser.eof()) {
return make_ready_future<>();
}
++_server._stats._requests_serving;
_pending_requests_gate.enter();
utils::latency_counter lc;
lc.start();
on_connection_ready();
auto leave = defer([this] () noexcept { _pending_requests_gate.leave(); });
return process_request_internal().then([this, leave = std::move(leave), lc = std::move(lc)] (auto&& result) mutable {
--_server._stats._requests_serving;
try {
if (_parser.failed()) {
logging.error("request parse failed");
const auto e = redis_exception("unknown command ''");
write_reply(std::move(e));
}else{
write_reply(std::move(result));
}
++_server._stats._requests_served;
_server._stats._requests.mark(lc.stop().latency());
_server._stats._estimated_requests_latency.add(lc.latency(), _server._stats._requests.hist.count);
} catch (...) {
logging.error("request processing failed: {}", std::current_exception());
}
});
});
}
void redis_server::connection::handle_error(future<>&& f) {
try {
f.get();
}
catch (redis_exception& e) {
write_reply(e);
}
catch (std::exception& e) {
write_reply(redis_exception { e.what() });
}
catch (...) {
write_reply(redis_exception { "Unknown exception" });
}
}
}
db::consistency_level make_consistency_level(const sstring& level)
{
std::unordered_map<sstring, db::consistency_level> consistency_levels {
{"ONE", db::consistency_level::ONE},
{"QUORUM", db::consistency_level::QUORUM},
{"LOCAL_QUORUM", db::consistency_level::LOCAL_QUORUM},
{"EACH_QUORUM", db::consistency_level::EACH_QUORUM},
{"ALL", db::consistency_level::ALL},
{"ANY", db::consistency_level::ANY},
{"TWO", db::consistency_level::TWO},
{"THREE", db::consistency_level::THREE},
{"SERIAL", db::consistency_level::SERIAL},
{"LOCAL_SERIAL", db::consistency_level::LOCAL_SERIAL},
{"LOCAL_ONE", db::consistency_level::LOCAL_ONE},
};
auto iter_cl = consistency_levels.find(level);
if (iter_cl == consistency_levels.end()) {
return db::consistency_level::ONE;
}
return iter_cl->second;
}