/*
* ctdb local tdb tool
*
* Copyright (C) Gregor Beck 2011
* Copyright (C) Michael Adam 2011
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 this program. If not, see .
*/
#include
#include
#include
#include
#include /* isprint */
#include /* strstr */
#include /* mode_t */
#include /* S_IRUSR */
#include /* uint32_t */
#include /* struct sockaddr_in */
#include /* struct sockaddr */
#include /* MIN */
#include
#include /* getopt */
#include
#include "ctdb_protocol.h"
enum {
MAX_HEADER_SIZE=24,
OUT_MODE = S_IRUSR | S_IWUSR,
OUT_FLAGS = O_EXCL|O_CREAT|O_RDWR,
};
union ltdb_header {
struct ctdb_ltdb_header hdr;
uint32_t uints[MAX_HEADER_SIZE/4];
};
static const union ltdb_header DEFAULT_HDR = {
.hdr.dmaster = -1,
};
static int help(const char* cmd)
{
fprintf(stdout, ""
"Usage: %s [options] \n"
"\n"
"Options:\n"
" -s {0|32|64} specify how to determine the ctdb record header size\n"
" for the input database:\n"
" 0: no ctdb header\n"
" 32: ctdb header size of a 32 bit system (20 bytes)\n"
" 64: ctdb header size of a 64 bit system (24 bytes)\n"
" default: 32 or 64 depending on the system architecture\n"
"\n"
" -S the number of bytes to interpret as ctdb record header\n"
" for the input database (beware!)\n"
"\n"
" -o {0|32|64} specify how to determine the ctdb record header size\n"
" for the output database\n"
" 0: no ctdb header\n"
" 32: ctdb header size of a 32 bit system (20 bytes)\n"
" 64: ctdb header size of a 64 bit system (24 bytes)\n"
" default: 32 or 64 depending on the system architecture\n"
"\n"
" -O the number of bytes to interpret as ctdb record header\n"
" for the output database (beware!)\n"
"\n"
" -p print header (for the dump command), defaults ot off\n"
"\n"
" -h print this help\n"
"\n"
"Commands:\n"
" help print this help\n"
" dump dump the db to stdout\n"
" convert convert the db\n\n", cmd);
return 0;
}
static int usage(const char* cmd)
{
fprintf(stderr,
"Usage: %s dump [-p] [-s{0|32|64}] \n"
" %s convert [-s{0|32|64}] [-o{0|32|64}] \n"
" %s {help|-h}\n"
, cmd, cmd, cmd);
return -1;
}
static int
ltdb_traverse(TDB_CONTEXT *tdb, int (*fn)(TDB_CONTEXT*, TDB_DATA, TDB_DATA,
struct ctdb_ltdb_header*, void *),
void *state, int hsize, bool skip_empty);
struct write_record_ctx {
TDB_CONTEXT* tdb;
size_t hsize;
int tdb_store_flags;
};
static int
write_record(TDB_CONTEXT* tdb, TDB_DATA key, TDB_DATA val,
struct ctdb_ltdb_header* hdr,
void* write_record_ctx);
struct dump_record_ctx {
FILE* file;
void (*print_data)(FILE*, TDB_DATA);
void (*dump_header)(struct dump_record_ctx*, struct ctdb_ltdb_header*);
};
static int dump_record(TDB_CONTEXT* tdb, TDB_DATA key, TDB_DATA val,
struct ctdb_ltdb_header* hdr,
void* dump_record_ctx);
static void print_data_tdbdump(FILE* file, TDB_DATA data);
static void dump_header_full(struct dump_record_ctx*, struct ctdb_ltdb_header*);
static void dump_header_nop(struct dump_record_ctx* c,
struct ctdb_ltdb_header* h)
{}
static int dump_db(const char* iname, FILE* ofile, int hsize, bool dump_header,
bool empty)
{
int ret = -1;
TDB_CONTEXT* idb = tdb_open(iname, 0, TDB_DEFAULT, O_RDONLY, 0);
if (!idb) {
perror("tdbopen in");
} else {
struct dump_record_ctx dump_ctx = {
.file = ofile,
.print_data = &print_data_tdbdump,
.dump_header = dump_header ? &dump_header_full
: &dump_header_nop,
};
ret = ltdb_traverse(idb, &dump_record, &dump_ctx, hsize, !empty);
tdb_close(idb);
}
return ret;
}
static int conv_db(const char* iname, const char* oname, size_t isize,
size_t osize, bool keep_empty)
{
int ret = -1;
TDB_CONTEXT* idb = tdb_open(iname, 0, TDB_DEFAULT, O_RDONLY, 0);
if (!idb) {
perror("tdbopen in");
} else {
TDB_CONTEXT* odb = tdb_open(oname, 0, TDB_DEFAULT, OUT_FLAGS, OUT_MODE);
if (!odb) {
perror("tdbopen out");
} else {
struct write_record_ctx ctx = {
.tdb = odb,
.hsize = osize,
.tdb_store_flags = TDB_REPLACE,
};
ret = ltdb_traverse(idb, &write_record, &ctx, isize, !keep_empty);
tdb_close(odb);
}
tdb_close(idb);
}
return ret;
}
static bool parse_size(size_t* size, const char* arg, bool raw) {
long val;
errno = 0;
val = strtol(arg, (char **) NULL, 10);
if (errno != 0) {
return false;
}
if (!raw) {
switch(val) {
case 0:
break;
case 32:
val = 20;
break;
case 64:
val = 24;
break;
default:
return false;
}
}
*size = MIN(val, MAX_HEADER_SIZE);
return true;
}
int main(int argc, char* argv[])
{
size_t isize = sizeof(struct ctdb_ltdb_header);
size_t osize = sizeof(struct ctdb_ltdb_header);
bool print_header = false;
bool keep_empty = false;
int opt;
const char *cmd, *idb, *odb;
while ((opt = getopt(argc, argv, "s:o:S:O:ph:e")) != -1) {
switch (opt) {
case 's':
case 'S':
if (!parse_size(&isize, optarg, isupper(opt))) {
return usage(argv[0]);
}
break;
case 'o':
case 'O':
if (!parse_size(&osize, optarg, isupper(opt))) {
return usage(argv[0]);
}
break;
case 'p':
print_header = true;
break;
case 'e':
keep_empty = true;
case 'h':
return help(argv[0]);
default:
return usage(argv[0]);
}
}
if (argc - optind < 1) {
return usage(argv[0]);
}
cmd = argv[optind];
if (strcmp(cmd, "help") == 0) {
return help(argv[0]);
}
else if (strcmp(cmd, "dump") == 0) {
int ret;
if (argc - optind != 2) {
return usage(argv[0]);
}
idb = argv[optind+1];
ret = dump_db(idb, stdout, isize, print_header, keep_empty);
return (ret >= 0) ? 0 : ret;
}
else if (strcmp(cmd, "convert") == 0) {
int ret;
if (argc - optind != 3) {
return usage(argv[0]);
}
idb = argv[optind+1];
odb = argv[optind+2];
ret = conv_db(idb, odb, isize, osize, keep_empty);
return (ret >= 0) ? 0 : ret;
}
return usage(argv[0]);
}
struct ltdb_traverse_ctx {
int (*fn)(TDB_CONTEXT*,TDB_DATA,TDB_DATA,struct ctdb_ltdb_header*,void *);
void* state;
size_t hsize;
bool skip_empty;
unsigned nempty;
};
static int
ltdb_traverse_fn(TDB_CONTEXT* tdb, TDB_DATA key, TDB_DATA val,
void* ltdb_traverse_ctx)
{
struct ltdb_traverse_ctx* ctx =
(struct ltdb_traverse_ctx*)ltdb_traverse_ctx;
union ltdb_header hdr = DEFAULT_HDR;
const size_t hsize = MIN(sizeof(hdr), ctx->hsize);
if (val.dsize < hsize) {
fprintf(stderr, "Value too short to contain a ctdb header: ");
print_data_tdbdump(stderr, key);
fprintf(stderr, " = ");
print_data_tdbdump(stderr, val);
fputc('\n', stderr);
return -1;
}
if (val.dsize == hsize && ctx->skip_empty) {
ctx->nempty++;
return 0;
}
memcpy(&hdr, val.dptr, hsize);
if (hdr.uints[5] != 0) {
fprintf(stderr, "Warning: header padding isn't zero! Wrong header size?\n");
}
val.dptr += ctx->hsize;
val.dsize -= ctx->hsize;
return ctx->fn(tdb, key, val, &hdr.hdr, ctx->state);
}
int ltdb_traverse(TDB_CONTEXT *tdb,
int (*fn)(TDB_CONTEXT *,TDB_DATA,TDB_DATA,struct ctdb_ltdb_header*,void *),
void *state, int hsize, bool skip_empty)
{
struct ltdb_traverse_ctx ctx = {
.fn = fn,
.state = state,
.hsize = hsize < 0 ? sizeof(struct ctdb_ltdb_header) : hsize,
.skip_empty = skip_empty,
.nempty = 0,
};
int ret = tdb_traverse(tdb, <db_traverse_fn, &ctx);
return (ret < 0) ? ret : (ret - ctx.nempty);
}
int write_record(TDB_CONTEXT* tdb, TDB_DATA key, TDB_DATA val,
struct ctdb_ltdb_header* hdr,
void* write_record_ctx)
{
struct write_record_ctx* ctx
= (struct write_record_ctx*)write_record_ctx;
if (ctx->hsize == 0) {
if (tdb_store(ctx->tdb, key, val, ctx->tdb_store_flags) == -1) {
fprintf(stderr, "tdb_store: %s\n", tdb_errorstr(ctx->tdb));
return -1;
}
} else {
TDB_DATA h = {
.dptr = (void*)hdr,
.dsize = ctx->hsize,
};
if(tdb_store(ctx->tdb, key, h, ctx->tdb_store_flags) == -1) {
fprintf(stderr, "tdb_store: %s\n", tdb_errorstr(ctx->tdb));
return -1;
}
if(tdb_append(ctx->tdb, key, val) == -1) {
fprintf(stderr, "tdb_append: %s\n", tdb_errorstr(ctx->tdb));
return -1;
}
}
return 0;
}
int dump_record(TDB_CONTEXT* tdb, TDB_DATA key, TDB_DATA val,
struct ctdb_ltdb_header* hdr,
void* dump_record_ctx)
{
struct dump_record_ctx* ctx = (struct dump_record_ctx*)dump_record_ctx;
fprintf(ctx->file, "{\nkey(%d) = ", (int)key.dsize);
ctx->print_data(ctx->file, key);
fputc('\n', ctx->file);
ctx->dump_header(ctx, hdr);
fprintf(ctx->file, "data(%d) = ", (int)val.dsize);
ctx->print_data(ctx->file, val);
fprintf(ctx->file, "\n}\n");
return 0;
}
void dump_header_full(struct dump_record_ctx* c, struct ctdb_ltdb_header* h)
{
fprintf(c->file, "dmaster: %d\nrsn: %llu\nflags: 0x%X\n",
(int)h->dmaster,
(unsigned long long)h->rsn, h->flags);
}
void print_data_tdbdump(FILE* file, TDB_DATA data) {
unsigned char *ptr = data.dptr;
fputc('"', file);
while (data.dsize--) {
if (isprint(*ptr) && !strchr("\"\\", *ptr)) {
fputc(*ptr, file);
} else {
fprintf(file, "\\%02X", *ptr);
}
ptr++;
}
fputc('"',file);
}