Add loadable dump format to vitastor-kv (dump)
parent
9dc4d5fd7b
commit
231d4b15fc
|
@ -118,8 +118,9 @@ void kv_cli_t::run(const json11::Json::object & cfg)
|
|||
finished = true;
|
||||
}
|
||||
});
|
||||
interactive = true;
|
||||
printf("> ");
|
||||
interactive = isatty(0);
|
||||
if (interactive)
|
||||
printf("> ");
|
||||
}
|
||||
catch (std::exception & e)
|
||||
{
|
||||
|
@ -236,7 +237,7 @@ void kv_cli_t::handle_cmd(const std::string & cmd, std::function<void()> cb)
|
|||
if (res < 0)
|
||||
fprintf(stderr, "Error opening index: %s (code %d)\n", strerror(-res), res);
|
||||
else
|
||||
printf("Index opened. Current size: %lu bytes\n", db->get_size());
|
||||
fprintf(interactive ? stdout : stderr, "Index opened. Current size: %lu bytes\n", db->get_size());
|
||||
cb();
|
||||
});
|
||||
}
|
||||
|
@ -272,15 +273,15 @@ void kv_cli_t::handle_cmd(const std::string & cmd, std::function<void()> cb)
|
|||
}
|
||||
else if (opname == "get" || opname == "set" || opname == "del")
|
||||
{
|
||||
std::string key = scan_escaped(cmd, pos);
|
||||
if (opname == "get" || opname == "del")
|
||||
{
|
||||
if (pos == std::string::npos)
|
||||
if (key == "")
|
||||
{
|
||||
fprintf(stderr, "Usage: %s <key>\n", opname.c_str());
|
||||
cb();
|
||||
return;
|
||||
}
|
||||
auto key = trim(cmd.substr(pos+1));
|
||||
if (opname == "get")
|
||||
{
|
||||
db->get(key, [this, cb](int res, const std::string & value)
|
||||
|
@ -302,34 +303,33 @@ void kv_cli_t::handle_cmd(const std::string & cmd, std::function<void()> cb)
|
|||
if (res < 0)
|
||||
fprintf(stderr, "Error: %s (code %d)\n", strerror(-res), res);
|
||||
else
|
||||
printf("OK\n");
|
||||
fprintf(interactive ? stdout : stderr, "OK\n");
|
||||
cb();
|
||||
});
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
auto pos2 = cmd.find_first_of(" \t", pos+1);
|
||||
if (pos2 == std::string::npos)
|
||||
if (key == "" || pos >= cmd.size())
|
||||
{
|
||||
fprintf(stderr, "Usage: set <key> <value>\n");
|
||||
cb();
|
||||
return;
|
||||
}
|
||||
auto key = trim(cmd.substr(pos+1, pos2-pos-1));
|
||||
auto value = trim(cmd.substr(pos2+1));
|
||||
auto value = trim(cmd.substr(pos));
|
||||
db->set(key, value, [this, cb](int res)
|
||||
{
|
||||
if (res < 0)
|
||||
fprintf(stderr, "Error: %s (code %d)\n", strerror(-res), res);
|
||||
else
|
||||
printf("OK\n");
|
||||
fprintf(interactive ? stdout : stderr, "OK\n");
|
||||
cb();
|
||||
});
|
||||
}
|
||||
}
|
||||
else if (opname == "list")
|
||||
else if (opname == "list" || opname == "dump")
|
||||
{
|
||||
bool dump = opname == "dump";
|
||||
std::string start, end;
|
||||
if (pos != std::string::npos)
|
||||
{
|
||||
|
@ -358,7 +358,10 @@ void kv_cli_t::handle_cmd(const std::string & cmd, std::function<void()> cb)
|
|||
}
|
||||
else
|
||||
{
|
||||
printf("%s = %s\n", key.c_str(), value.c_str());
|
||||
if (dump)
|
||||
printf("set %s %s\n", auto_addslashes(key).c_str(), value.c_str());
|
||||
else
|
||||
printf("%s = %s\n", key.c_str(), value.c_str());
|
||||
db->list_next(handle, NULL);
|
||||
}
|
||||
});
|
||||
|
@ -367,7 +370,7 @@ void kv_cli_t::handle_cmd(const std::string & cmd, std::function<void()> cb)
|
|||
{
|
||||
db->close([=]()
|
||||
{
|
||||
printf("Index closed\n");
|
||||
fprintf(interactive ? stdout : stderr, "Index closed\n");
|
||||
cb();
|
||||
});
|
||||
}
|
||||
|
@ -382,7 +385,8 @@ void kv_cli_t::handle_cmd(const std::string & cmd, std::function<void()> cb)
|
|||
stderr, "Unknown operation: %s. Supported operations:\n"
|
||||
"open <pool_id> <inode_id> [block_size]\n"
|
||||
"config <property> <value>\n"
|
||||
"get <key>\nset <key> <value>\ndel <key>\nlist [<start> [end]]\n"
|
||||
"get <key>\nset <key> <value>\ndel <key>\n"
|
||||
"list [<start> [end]]\ndump [<start> [end]]\n"
|
||||
"close\nquit\n", opname.c_str()
|
||||
);
|
||||
cb();
|
||||
|
|
|
@ -348,3 +348,65 @@ std::vector<std::string> explode(const std::string & sep, const std::string & va
|
|||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
// extract possibly double-quoted part of string with escape characters
|
||||
std::string scan_escaped(const std::string & cmd, size_t & pos)
|
||||
{
|
||||
std::string key;
|
||||
auto pos2 = cmd.find_first_not_of(" \t\r\n", pos);
|
||||
if (pos2 == std::string::npos)
|
||||
{
|
||||
pos = cmd.size();
|
||||
return "";
|
||||
}
|
||||
pos = pos2;
|
||||
if (cmd[pos] != '"')
|
||||
{
|
||||
pos2 = cmd.find_first_of(" \t\r\n", pos);
|
||||
pos2 = pos2 == std::string::npos ? cmd.size() : pos2;
|
||||
key = cmd.substr(pos, pos2-pos);
|
||||
pos2 = cmd.find_first_not_of(" \t\r\n", pos2);
|
||||
pos = pos2 == std::string::npos ? cmd.size() : pos2;
|
||||
return key;
|
||||
}
|
||||
pos++;
|
||||
while (pos < cmd.size())
|
||||
{
|
||||
auto pos2 = cmd.find_first_of("\\\"", pos);
|
||||
pos2 = pos2 == std::string::npos ? cmd.size() : pos2;
|
||||
if (pos2 > pos)
|
||||
key += cmd.substr(pos, pos2-pos);
|
||||
pos = pos2;
|
||||
if (pos >= cmd.size())
|
||||
break;
|
||||
if (cmd[pos] == '"')
|
||||
{
|
||||
pos++;
|
||||
break;
|
||||
}
|
||||
if (cmd[pos] == '\\')
|
||||
{
|
||||
if (pos < cmd.size()-1)
|
||||
key += cmd[++pos];
|
||||
pos++;
|
||||
}
|
||||
}
|
||||
return key;
|
||||
}
|
||||
|
||||
std::string auto_addslashes(const std::string & str)
|
||||
{
|
||||
auto pos = str.find_first_of("\\\"");
|
||||
if (pos == std::string::npos)
|
||||
return str;
|
||||
std::string res = "\""+str.substr(0, pos)+"\\"+str[pos];
|
||||
while (pos < str.size()-1)
|
||||
{
|
||||
auto pos2 = str.find_first_of("\\\"", pos+1);
|
||||
if (pos2 == std::string::npos)
|
||||
return res + str.substr(pos+1) + "\"";
|
||||
res += str.substr(pos, pos2-pos)+"\\"+str[pos2];
|
||||
pos = pos2;
|
||||
}
|
||||
return res+"\"";
|
||||
}
|
||||
|
|
|
@ -22,3 +22,5 @@ std::string str_repeat(const std::string & str, int times);
|
|||
size_t utf8_length(const std::string & s);
|
||||
size_t utf8_length(const char *s);
|
||||
std::vector<std::string> explode(const std::string & sep, const std::string & value, bool trim);
|
||||
std::string scan_escaped(const std::string & cmd, size_t & pos);
|
||||
std::string auto_addslashes(const std::string & str);
|
||||
|
|
Loading…
Reference in New Issue