2021-11-28 15:29:26 +08:00
|
|
|
// Copyright 2021, Roman Gershman. All rights reserved.
|
|
|
|
// See LICENSE for licensing terms.
|
2021-11-19 00:38:20 +08:00
|
|
|
//
|
|
|
|
|
|
|
|
#include "server/command_registry.h"
|
|
|
|
|
|
|
|
#include "absl/strings/str_cat.h"
|
|
|
|
#include "base/bits.h"
|
|
|
|
#include "base/logging.h"
|
|
|
|
#include "server/conn_context.h"
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
namespace dfly {
|
|
|
|
|
|
|
|
using absl::StrAppend;
|
|
|
|
using absl::StrCat;
|
|
|
|
|
|
|
|
CommandId::CommandId(const char* name, uint32_t mask, int8_t arity, int8_t first_key,
|
|
|
|
int8_t last_key, int8_t step)
|
|
|
|
: name_(name), opt_mask_(mask), arity_(arity), first_key_(first_key), last_key_(last_key),
|
|
|
|
step_key_(step) {
|
2022-01-06 21:48:51 +08:00
|
|
|
if (mask & CO::ADMIN) {
|
|
|
|
opt_mask_ |= CO::NOSCRIPT;
|
|
|
|
}
|
2021-11-19 00:38:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t CommandId::OptCount(uint32_t mask) {
|
|
|
|
return absl::popcount(mask);
|
|
|
|
}
|
|
|
|
|
|
|
|
CommandRegistry::CommandRegistry() {
|
|
|
|
CommandId cd("COMMAND", CO::RANDOM | CO::LOADING | CO::STALE, 0, 0, 0, 0);
|
2021-11-28 15:29:26 +08:00
|
|
|
|
|
|
|
cd.SetHandler([this](const auto& args, auto* cntx) { return Command(args, cntx); });
|
2021-11-19 00:38:20 +08:00
|
|
|
const char* nm = cd.name();
|
|
|
|
cmd_map_.emplace(nm, std::move(cd));
|
|
|
|
}
|
|
|
|
|
|
|
|
void CommandRegistry::Command(CmdArgList args, ConnectionContext* cntx) {
|
|
|
|
size_t sz = cmd_map_.size();
|
2021-11-28 15:29:26 +08:00
|
|
|
string resp = StrCat("*", sz, "\r\n");
|
2021-11-19 00:38:20 +08:00
|
|
|
|
|
|
|
for (const auto& val : cmd_map_) {
|
|
|
|
const CommandId& cd = val.second;
|
|
|
|
StrAppend(&resp, "*6\r\n$", strlen(cd.name()), "\r\n", cd.name(), "\r\n");
|
2021-11-28 15:29:26 +08:00
|
|
|
StrAppend(&resp, ":", int(cd.arity()), "\r\n");
|
|
|
|
StrAppend(&resp, "*", CommandId::OptCount(cd.opt_mask()), "\r\n");
|
2021-11-19 00:38:20 +08:00
|
|
|
|
2021-11-28 15:29:26 +08:00
|
|
|
for (uint32_t i = 0; i < 32; ++i) {
|
|
|
|
unsigned obit = (1u << i);
|
|
|
|
if (cd.opt_mask() & obit) {
|
|
|
|
const char* name = CO::OptName(CO::CommandOpt{obit});
|
2021-11-19 00:38:20 +08:00
|
|
|
StrAppend(&resp, "+", name, "\r\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
StrAppend(&resp, ":", cd.first_key_pos(), "\r\n");
|
|
|
|
StrAppend(&resp, ":", cd.last_key_pos(), "\r\n");
|
|
|
|
StrAppend(&resp, ":", cd.key_arg_step(), "\r\n");
|
|
|
|
}
|
|
|
|
|
2021-11-23 18:39:35 +08:00
|
|
|
cntx->SendRespBlob(resp);
|
2021-11-19 00:38:20 +08:00
|
|
|
}
|
|
|
|
|
2021-12-26 23:25:49 +08:00
|
|
|
CommandRegistry& CommandRegistry::operator<<(CommandId cmd) {
|
|
|
|
string_view k = cmd.name();
|
|
|
|
CHECK(cmd_map_.emplace(k, std::move(cmd)).second) << k;
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2021-11-19 00:38:20 +08:00
|
|
|
namespace CO {
|
|
|
|
|
|
|
|
const char* OptName(CO::CommandOpt fl) {
|
|
|
|
using namespace CO;
|
|
|
|
|
|
|
|
switch (fl) {
|
|
|
|
case WRITE:
|
|
|
|
return "write";
|
|
|
|
case READONLY:
|
|
|
|
return "readonly";
|
|
|
|
case DENYOOM:
|
|
|
|
return "denyoom";
|
|
|
|
case FAST:
|
|
|
|
return "fast";
|
|
|
|
case STALE:
|
|
|
|
return "stale";
|
|
|
|
case LOADING:
|
|
|
|
return "loading";
|
|
|
|
case RANDOM:
|
|
|
|
return "random";
|
2022-01-06 21:48:51 +08:00
|
|
|
case ADMIN:
|
|
|
|
return "admin";
|
|
|
|
case NOSCRIPT:
|
|
|
|
return "noscript";
|
|
|
|
case GLOBAL_TRANS:
|
|
|
|
return "global-trans";
|
2021-11-19 00:38:20 +08:00
|
|
|
}
|
2022-01-06 21:48:51 +08:00
|
|
|
return "unknown";
|
2021-11-19 00:38:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace CO
|
|
|
|
|
|
|
|
} // namespace dfly
|