2019-09-09 13:40:24 +03:00
|
|
|
/* mdbx_load.c - memory-mapped database load tool */
|
2017-02-21 20:16:54 +03:00
|
|
|
|
|
|
|
/*
|
2020-01-11 13:21:43 +03:00
|
|
|
* Copyright 2015-2020 Leonid Yuriev <leo@yuriev.ru>
|
2017-03-16 18:09:27 +03:00
|
|
|
* and other libmdbx authors: please see AUTHORS file.
|
2017-02-21 20:16:54 +03:00
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted only as authorized by the OpenLDAP
|
|
|
|
* Public License.
|
|
|
|
*
|
|
|
|
* A copy of this license is available in the file LICENSE in the
|
|
|
|
* top-level directory of the distribution or, alternatively, at
|
2017-05-23 14:44:53 +03:00
|
|
|
* <http://www.OpenLDAP.org/license.html>. */
|
2017-02-21 20:16:54 +03:00
|
|
|
|
2017-07-10 20:45:24 +03:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#if _MSC_VER > 1800
|
|
|
|
#pragma warning(disable : 4464) /* relative include path contains '..' */
|
|
|
|
#endif
|
|
|
|
#pragma warning(disable : 4996) /* The POSIX name is deprecated... */
|
2017-07-26 18:32:46 +03:00
|
|
|
#endif /* _MSC_VER (warnings) */
|
2017-07-10 20:45:24 +03:00
|
|
|
|
2019-08-31 17:13:02 +03:00
|
|
|
#define MDBX_TOOLS /* Avoid using internal mdbx_assert() */
|
2020-04-15 17:09:37 +03:00
|
|
|
#include "internals.h"
|
2019-08-27 00:29:15 +03:00
|
|
|
|
2017-02-21 20:16:54 +03:00
|
|
|
#include <ctype.h>
|
2017-07-10 20:45:24 +03:00
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
#include "wingetopt.h"
|
|
|
|
|
|
|
|
static volatile BOOL user_break;
|
|
|
|
static BOOL WINAPI ConsoleBreakHandlerRoutine(DWORD dwCtrlType) {
|
|
|
|
(void)dwCtrlType;
|
|
|
|
user_break = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
#else /* WINDOWS */
|
|
|
|
|
|
|
|
static volatile sig_atomic_t user_break;
|
|
|
|
static void signal_handler(int sig) {
|
|
|
|
(void)sig;
|
|
|
|
user_break = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* !WINDOWS */
|
2017-02-21 20:16:54 +03:00
|
|
|
|
|
|
|
#define PRINT 1
|
|
|
|
#define NOHDR 2
|
|
|
|
static int mode;
|
|
|
|
|
|
|
|
static char *subname = NULL;
|
|
|
|
static size_t lineno;
|
|
|
|
static int version;
|
|
|
|
|
|
|
|
static int dbi_flags;
|
|
|
|
static char *prog;
|
2020-01-20 22:15:33 +03:00
|
|
|
static bool Eof;
|
2017-02-21 20:16:54 +03:00
|
|
|
|
2017-05-23 16:46:55 +03:00
|
|
|
static MDBX_envinfo envinfo;
|
2017-05-23 21:05:54 +03:00
|
|
|
static MDBX_val kbuf, dbuf;
|
2019-02-03 15:49:40 +03:00
|
|
|
static MDBX_val k0buf;
|
2017-02-21 20:16:54 +03:00
|
|
|
|
|
|
|
#define STRLENOF(s) (sizeof(s) - 1)
|
|
|
|
|
|
|
|
typedef struct flagbit {
|
|
|
|
int bit;
|
|
|
|
char *name;
|
|
|
|
int len;
|
|
|
|
} flagbit;
|
|
|
|
|
|
|
|
#define S(s) s, STRLENOF(s)
|
|
|
|
|
2017-05-24 01:42:10 +03:00
|
|
|
flagbit dbflags[] = {{MDBX_REVERSEKEY, S("reversekey")},
|
|
|
|
{MDBX_DUPSORT, S("dupsort")},
|
|
|
|
{MDBX_INTEGERKEY, S("integerkey")},
|
|
|
|
{MDBX_DUPFIXED, S("dupfixed")},
|
|
|
|
{MDBX_INTEGERDUP, S("integerdup")},
|
|
|
|
{MDBX_REVERSEDUP, S("reversedup")},
|
2017-02-21 20:16:54 +03:00
|
|
|
{0, NULL, 0}};
|
|
|
|
|
|
|
|
static void readhdr(void) {
|
|
|
|
char *ptr;
|
|
|
|
|
|
|
|
dbi_flags = 0;
|
2017-07-10 20:45:24 +03:00
|
|
|
while (fgets(dbuf.iov_base, (int)dbuf.iov_len, stdin) != NULL) {
|
2017-02-21 20:16:54 +03:00
|
|
|
lineno++;
|
2020-01-20 22:15:33 +03:00
|
|
|
|
|
|
|
if (!strncmp(dbuf.iov_base, "db_pagesize=", STRLENOF("db_pagesize="))) {
|
|
|
|
envinfo.mi_dxb_pagesize =
|
|
|
|
atoi((char *)dbuf.iov_base + STRLENOF("db_pagesize="));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strncmp(dbuf.iov_base, "duplicates=", STRLENOF("duplicates="))) {
|
|
|
|
dbi_flags |= MDBX_DUPSORT;
|
2017-02-21 20:16:54 +03:00
|
|
|
continue;
|
2020-01-20 22:15:33 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!strncmp(dbuf.iov_base, "VERSION=", STRLENOF("VERSION="))) {
|
2017-05-23 21:05:54 +03:00
|
|
|
version = atoi((char *)dbuf.iov_base + STRLENOF("VERSION="));
|
2017-02-21 20:16:54 +03:00
|
|
|
if (version > 3) {
|
2017-07-26 18:32:46 +03:00
|
|
|
fprintf(stderr, "%s: line %" PRIiSIZE ": unsupported VERSION %d\n",
|
|
|
|
prog, lineno, version);
|
2017-02-21 20:16:54 +03:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2020-01-20 22:15:33 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strncmp(dbuf.iov_base, "HEADER=END", STRLENOF("HEADER=END")))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!strncmp(dbuf.iov_base, "format=", STRLENOF("format="))) {
|
2017-05-23 21:05:54 +03:00
|
|
|
if (!strncmp((char *)dbuf.iov_base + STRLENOF("FORMAT="), "print",
|
2017-02-21 20:16:54 +03:00
|
|
|
STRLENOF("print")))
|
|
|
|
mode |= PRINT;
|
2017-05-23 21:05:54 +03:00
|
|
|
else if (strncmp((char *)dbuf.iov_base + STRLENOF("FORMAT="), "bytevalue",
|
2017-02-21 20:16:54 +03:00
|
|
|
STRLENOF("bytevalue"))) {
|
2017-07-26 18:32:46 +03:00
|
|
|
fprintf(stderr, "%s: line %" PRIiSIZE ": unsupported FORMAT %s\n", prog,
|
2017-05-23 21:05:54 +03:00
|
|
|
lineno, (char *)dbuf.iov_base + STRLENOF("FORMAT="));
|
2017-02-21 20:16:54 +03:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2020-01-20 22:15:33 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strncmp(dbuf.iov_base, "database=", STRLENOF("database="))) {
|
2017-05-23 21:05:54 +03:00
|
|
|
ptr = memchr(dbuf.iov_base, '\n', dbuf.iov_len);
|
2017-02-21 20:16:54 +03:00
|
|
|
if (ptr)
|
|
|
|
*ptr = '\0';
|
|
|
|
if (subname)
|
2018-10-12 22:01:36 +03:00
|
|
|
mdbx_free(subname);
|
2018-10-14 17:28:00 +03:00
|
|
|
subname = mdbx_strdup((char *)dbuf.iov_base + STRLENOF("database="));
|
2020-01-20 22:15:33 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strncmp(dbuf.iov_base, "type=", STRLENOF("type="))) {
|
2017-05-23 21:05:54 +03:00
|
|
|
if (strncmp((char *)dbuf.iov_base + STRLENOF("type="), "btree",
|
2017-02-21 20:16:54 +03:00
|
|
|
STRLENOF("btree"))) {
|
2017-07-26 18:32:46 +03:00
|
|
|
fprintf(stderr, "%s: line %" PRIiSIZE ": unsupported type %s\n", prog,
|
2017-05-23 21:05:54 +03:00
|
|
|
lineno, (char *)dbuf.iov_base + STRLENOF("type="));
|
2017-02-21 20:16:54 +03:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2020-01-20 22:15:33 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strncmp(dbuf.iov_base, "mapaddr=", STRLENOF("mapaddr="))) {
|
2017-02-21 20:16:54 +03:00
|
|
|
int i;
|
2017-05-23 21:05:54 +03:00
|
|
|
ptr = memchr(dbuf.iov_base, '\n', dbuf.iov_len);
|
2017-02-21 20:16:54 +03:00
|
|
|
if (ptr)
|
|
|
|
*ptr = '\0';
|
2017-06-21 01:34:56 +03:00
|
|
|
void *unused;
|
|
|
|
i = sscanf((char *)dbuf.iov_base + STRLENOF("mapaddr="), "%p", &unused);
|
2017-02-21 20:16:54 +03:00
|
|
|
if (i != 1) {
|
2017-07-26 18:32:46 +03:00
|
|
|
fprintf(stderr, "%s: line %" PRIiSIZE ": invalid mapaddr %s\n", prog,
|
2017-05-23 21:05:54 +03:00
|
|
|
lineno, (char *)dbuf.iov_base + STRLENOF("mapaddr="));
|
2017-02-21 20:16:54 +03:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2020-01-20 22:15:33 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strncmp(dbuf.iov_base, "mapsize=", STRLENOF("mapsize="))) {
|
2017-02-21 20:16:54 +03:00
|
|
|
int i;
|
2017-05-23 21:05:54 +03:00
|
|
|
ptr = memchr(dbuf.iov_base, '\n', dbuf.iov_len);
|
2017-02-21 20:16:54 +03:00
|
|
|
if (ptr)
|
|
|
|
*ptr = '\0';
|
2018-08-27 11:05:09 +03:00
|
|
|
i = sscanf((char *)dbuf.iov_base + STRLENOF("mapsize="), "%" PRIu64,
|
2017-07-12 21:13:17 +03:00
|
|
|
&envinfo.mi_mapsize);
|
2017-02-21 20:16:54 +03:00
|
|
|
if (i != 1) {
|
2017-07-26 18:32:46 +03:00
|
|
|
fprintf(stderr, "%s: line %" PRIiSIZE ": invalid mapsize %s\n", prog,
|
2017-05-23 21:05:54 +03:00
|
|
|
lineno, (char *)dbuf.iov_base + STRLENOF("mapsize="));
|
2017-02-21 20:16:54 +03:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2020-01-20 22:15:33 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strncmp(dbuf.iov_base, "maxreaders=", STRLENOF("maxreaders="))) {
|
2017-02-21 20:16:54 +03:00
|
|
|
int i;
|
2017-05-23 21:05:54 +03:00
|
|
|
ptr = memchr(dbuf.iov_base, '\n', dbuf.iov_len);
|
2017-02-21 20:16:54 +03:00
|
|
|
if (ptr)
|
|
|
|
*ptr = '\0';
|
2017-05-23 21:05:54 +03:00
|
|
|
i = sscanf((char *)dbuf.iov_base + STRLENOF("maxreaders="), "%u",
|
2017-07-12 21:13:17 +03:00
|
|
|
&envinfo.mi_maxreaders);
|
2017-02-21 20:16:54 +03:00
|
|
|
if (i != 1) {
|
2017-07-26 18:32:46 +03:00
|
|
|
fprintf(stderr, "%s: line %" PRIiSIZE ": invalid maxreaders %s\n", prog,
|
2017-05-23 21:05:54 +03:00
|
|
|
lineno, (char *)dbuf.iov_base + STRLENOF("maxreaders="));
|
2017-02-21 20:16:54 +03:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2020-01-20 22:15:33 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
int i;
|
|
|
|
for (i = 0; dbflags[i].bit; i++) {
|
|
|
|
if (!strncmp(dbuf.iov_base, dbflags[i].name, dbflags[i].len) &&
|
|
|
|
((char *)dbuf.iov_base)[dbflags[i].len] == '=') {
|
|
|
|
if (((char *)dbuf.iov_base)[dbflags[i].len + 1] == '1')
|
|
|
|
dbi_flags |= dbflags[i].bit;
|
|
|
|
break;
|
2017-02-21 20:16:54 +03:00
|
|
|
}
|
2020-01-20 22:15:33 +03:00
|
|
|
}
|
|
|
|
if (!dbflags[i].bit) {
|
|
|
|
ptr = memchr(dbuf.iov_base, '=', dbuf.iov_len);
|
|
|
|
if (!ptr) {
|
|
|
|
fprintf(stderr, "%s: line %" PRIiSIZE ": unexpected format\n", prog,
|
|
|
|
lineno);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
} else {
|
|
|
|
*ptr = '\0';
|
|
|
|
fprintf(stderr,
|
|
|
|
"%s: line %" PRIiSIZE ": unrecognized keyword ignored: %s\n",
|
|
|
|
prog, lineno, (char *)dbuf.iov_base);
|
2017-02-21 20:16:54 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-01-20 22:15:33 +03:00
|
|
|
Eof = true;
|
2017-02-21 20:16:54 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void badend(void) {
|
2017-07-26 18:32:46 +03:00
|
|
|
fprintf(stderr, "%s: line %" PRIiSIZE ": unexpected end of input\n", prog,
|
2017-05-10 19:16:14 +03:00
|
|
|
lineno);
|
2017-02-21 20:16:54 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static int unhex(unsigned char *c2) {
|
|
|
|
int x, c;
|
|
|
|
x = *c2++ & 0x4f;
|
|
|
|
if (x & 0x40)
|
|
|
|
x -= 55;
|
|
|
|
c = x << 4;
|
|
|
|
x = *c2 & 0x4f;
|
|
|
|
if (x & 0x40)
|
|
|
|
x -= 55;
|
|
|
|
c |= x;
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
2017-05-23 21:05:54 +03:00
|
|
|
static int readline(MDBX_val *out, MDBX_val *buf) {
|
2017-02-21 20:16:54 +03:00
|
|
|
unsigned char *c1, *c2, *end;
|
|
|
|
size_t len, l2;
|
|
|
|
int c;
|
|
|
|
|
|
|
|
if (!(mode & NOHDR)) {
|
|
|
|
c = fgetc(stdin);
|
|
|
|
if (c == EOF) {
|
2020-01-20 22:15:33 +03:00
|
|
|
Eof = true;
|
2017-02-21 20:16:54 +03:00
|
|
|
return EOF;
|
|
|
|
}
|
|
|
|
if (c != ' ') {
|
|
|
|
lineno++;
|
2017-07-10 20:45:24 +03:00
|
|
|
if (fgets(buf->iov_base, (int)buf->iov_len, stdin) == NULL) {
|
2017-02-21 20:16:54 +03:00
|
|
|
badend:
|
2020-01-20 22:15:33 +03:00
|
|
|
Eof = true;
|
2017-02-21 20:16:54 +03:00
|
|
|
badend();
|
|
|
|
return EOF;
|
|
|
|
}
|
2017-05-23 21:05:54 +03:00
|
|
|
if (c == 'D' && !strncmp(buf->iov_base, "ATA=END", STRLENOF("ATA=END")))
|
2017-02-21 20:16:54 +03:00
|
|
|
return EOF;
|
|
|
|
goto badend;
|
|
|
|
}
|
|
|
|
}
|
2017-07-10 20:45:24 +03:00
|
|
|
if (fgets(buf->iov_base, (int)buf->iov_len, stdin) == NULL) {
|
2020-01-20 22:15:33 +03:00
|
|
|
Eof = true;
|
2017-02-21 20:16:54 +03:00
|
|
|
return EOF;
|
|
|
|
}
|
|
|
|
lineno++;
|
|
|
|
|
2017-05-23 21:05:54 +03:00
|
|
|
c1 = buf->iov_base;
|
2017-02-21 20:16:54 +03:00
|
|
|
len = strlen((char *)c1);
|
|
|
|
l2 = len;
|
|
|
|
|
|
|
|
/* Is buffer too short? */
|
|
|
|
while (c1[len - 1] != '\n') {
|
2018-10-12 22:01:36 +03:00
|
|
|
buf->iov_base = mdbx_realloc(buf->iov_base, buf->iov_len * 2);
|
2017-05-23 21:05:54 +03:00
|
|
|
if (!buf->iov_base) {
|
2020-01-20 22:15:33 +03:00
|
|
|
Eof = true;
|
2017-07-26 18:32:46 +03:00
|
|
|
fprintf(stderr, "%s: line %" PRIiSIZE ": out of memory, line too long\n",
|
2017-05-10 19:16:14 +03:00
|
|
|
prog, lineno);
|
2017-02-21 20:16:54 +03:00
|
|
|
return EOF;
|
|
|
|
}
|
2017-05-23 21:05:54 +03:00
|
|
|
c1 = buf->iov_base;
|
2017-02-21 20:16:54 +03:00
|
|
|
c1 += l2;
|
2017-07-10 20:45:24 +03:00
|
|
|
if (fgets((char *)c1, (int)buf->iov_len + 1, stdin) == NULL) {
|
2020-01-20 22:15:33 +03:00
|
|
|
Eof = true;
|
2017-02-21 20:16:54 +03:00
|
|
|
badend();
|
|
|
|
return EOF;
|
|
|
|
}
|
2017-05-23 21:05:54 +03:00
|
|
|
buf->iov_len *= 2;
|
2017-02-21 20:16:54 +03:00
|
|
|
len = strlen((char *)c1);
|
|
|
|
l2 += len;
|
|
|
|
}
|
2017-05-23 21:05:54 +03:00
|
|
|
c1 = c2 = buf->iov_base;
|
2017-02-21 20:16:54 +03:00
|
|
|
len = l2;
|
|
|
|
c1[--len] = '\0';
|
|
|
|
end = c1 + len;
|
|
|
|
|
|
|
|
if (mode & PRINT) {
|
|
|
|
while (c2 < end) {
|
2019-08-27 14:50:19 +03:00
|
|
|
if (unlikely(*c2 == '\\')) {
|
2017-02-21 20:16:54 +03:00
|
|
|
if (c2[1] == '\\') {
|
2019-08-27 14:50:19 +03:00
|
|
|
*c1++ = '\\';
|
2017-02-21 20:16:54 +03:00
|
|
|
} else {
|
|
|
|
if (c2 + 3 > end || !isxdigit(c2[1]) || !isxdigit(c2[2])) {
|
2020-01-20 22:15:33 +03:00
|
|
|
Eof = true;
|
2017-02-21 20:16:54 +03:00
|
|
|
badend();
|
|
|
|
return EOF;
|
|
|
|
}
|
2017-07-10 20:45:24 +03:00
|
|
|
*c1++ = (char)unhex(++c2);
|
2017-02-21 20:16:54 +03:00
|
|
|
}
|
2019-08-27 14:50:19 +03:00
|
|
|
c2 += 2;
|
2017-02-21 20:16:54 +03:00
|
|
|
} else {
|
|
|
|
/* copies are redundant when no escapes were used */
|
|
|
|
*c1++ = *c2++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* odd length not allowed */
|
|
|
|
if (len & 1) {
|
2020-01-20 22:15:33 +03:00
|
|
|
Eof = true;
|
2017-02-21 20:16:54 +03:00
|
|
|
badend();
|
|
|
|
return EOF;
|
|
|
|
}
|
|
|
|
while (c2 < end) {
|
|
|
|
if (!isxdigit(*c2) || !isxdigit(c2[1])) {
|
2020-01-20 22:15:33 +03:00
|
|
|
Eof = true;
|
2017-02-21 20:16:54 +03:00
|
|
|
badend();
|
|
|
|
return EOF;
|
|
|
|
}
|
2017-07-10 20:45:24 +03:00
|
|
|
*c1++ = (char)unhex(c2);
|
2017-02-21 20:16:54 +03:00
|
|
|
c2 += 2;
|
|
|
|
}
|
|
|
|
}
|
2017-05-23 21:05:54 +03:00
|
|
|
c2 = out->iov_base = buf->iov_base;
|
|
|
|
out->iov_len = c1 - c2;
|
2017-02-21 20:16:54 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void usage(void) {
|
2019-02-03 15:49:40 +03:00
|
|
|
fprintf(stderr,
|
2019-11-27 22:38:39 +03:00
|
|
|
"usage: %s [-V] [-q] [-a] [-f file] [-s name] [-N] [-T] [-r] [-n] "
|
|
|
|
"dbpath\n"
|
|
|
|
" -V\t\tprint version and exit\n"
|
|
|
|
" -q\t\tbe quiet\n"
|
2020-03-02 13:50:33 +03:00
|
|
|
" -a\t\tappend records in input order (required for custom "
|
|
|
|
"comparators)\n"
|
2019-11-27 22:38:39 +03:00
|
|
|
" -f file\tread from file instead of stdin\n"
|
|
|
|
" -s name\tload into named subDB\n"
|
|
|
|
" -N\t\tuse NOOVERWRITE on puts\n"
|
|
|
|
" -T\t\tread plaintext\n"
|
|
|
|
" -r\t\trescure mode (ignore errors to load corrupted DB dump)\n"
|
|
|
|
" -n\t\tNOSUBDIR mode for open\n",
|
2017-02-21 20:16:54 +03:00
|
|
|
prog);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2019-02-03 15:49:40 +03:00
|
|
|
static int anyway_greater(const MDBX_val *a, const MDBX_val *b) {
|
|
|
|
(void)a;
|
|
|
|
(void)b;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-02-21 20:16:54 +03:00
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
int i, rc;
|
2017-07-10 20:45:24 +03:00
|
|
|
MDBX_env *env = NULL;
|
|
|
|
MDBX_txn *txn = NULL;
|
|
|
|
MDBX_cursor *mc = NULL;
|
2017-05-24 01:42:10 +03:00
|
|
|
MDBX_dbi dbi;
|
2017-07-10 20:45:24 +03:00
|
|
|
char *envname = NULL;
|
2019-02-03 15:49:40 +03:00
|
|
|
int envflags = MDBX_UTTERLY_NOSYNC, putflags = 0;
|
|
|
|
int append = 0;
|
2019-11-27 22:04:47 +03:00
|
|
|
int quiet = 0;
|
|
|
|
int rescue = 0;
|
2019-02-03 15:49:40 +03:00
|
|
|
MDBX_val prevk;
|
2017-02-21 20:16:54 +03:00
|
|
|
|
|
|
|
prog = argv[0];
|
2019-02-03 15:49:40 +03:00
|
|
|
if (argc < 2)
|
2017-02-21 20:16:54 +03:00
|
|
|
usage();
|
|
|
|
|
2019-11-27 22:04:47 +03:00
|
|
|
while ((i = getopt(argc, argv, "af:ns:NTVrq")) != EOF) {
|
2017-02-21 20:16:54 +03:00
|
|
|
switch (i) {
|
|
|
|
case 'V':
|
2019-09-03 22:54:50 +03:00
|
|
|
printf("mdbx_load version %d.%d.%d.%d\n"
|
|
|
|
" - source: %s %s, commit %s, tree %s\n"
|
2019-09-05 11:57:52 +03:00
|
|
|
" - anchor: %s\n"
|
2019-09-03 22:54:50 +03:00
|
|
|
" - build: %s for %s by %s\n"
|
|
|
|
" - flags: %s\n"
|
|
|
|
" - options: %s\n",
|
|
|
|
mdbx_version.major, mdbx_version.minor, mdbx_version.release,
|
|
|
|
mdbx_version.revision, mdbx_version.git.describe,
|
|
|
|
mdbx_version.git.datetime, mdbx_version.git.commit,
|
2019-09-05 11:57:52 +03:00
|
|
|
mdbx_version.git.tree, mdbx_sourcery_anchor, mdbx_build.datetime,
|
|
|
|
mdbx_build.target, mdbx_build.compiler, mdbx_build.flags,
|
|
|
|
mdbx_build.options);
|
2019-09-03 22:54:50 +03:00
|
|
|
return EXIT_SUCCESS;
|
2019-02-03 15:49:40 +03:00
|
|
|
case 'a':
|
|
|
|
append = 1;
|
|
|
|
break;
|
2017-02-21 20:16:54 +03:00
|
|
|
case 'f':
|
|
|
|
if (freopen(optarg, "r", stdin) == NULL) {
|
2020-01-20 22:15:33 +03:00
|
|
|
fprintf(stderr, "%s: %s: open: %s\n", prog, optarg,
|
2018-10-14 10:32:37 +03:00
|
|
|
mdbx_strerror(errno));
|
2017-02-21 20:16:54 +03:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'n':
|
2017-05-24 01:42:10 +03:00
|
|
|
envflags |= MDBX_NOSUBDIR;
|
2017-02-21 20:16:54 +03:00
|
|
|
break;
|
|
|
|
case 's':
|
2018-10-14 17:28:00 +03:00
|
|
|
subname = mdbx_strdup(optarg);
|
2017-02-21 20:16:54 +03:00
|
|
|
break;
|
|
|
|
case 'N':
|
2017-05-24 01:42:10 +03:00
|
|
|
putflags = MDBX_NOOVERWRITE | MDBX_NODUPDATA;
|
2017-02-21 20:16:54 +03:00
|
|
|
break;
|
|
|
|
case 'T':
|
|
|
|
mode |= NOHDR | PRINT;
|
|
|
|
break;
|
2019-11-27 22:04:47 +03:00
|
|
|
case 'q':
|
|
|
|
quiet = 1;
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
rescue = 1;
|
|
|
|
break;
|
2017-02-21 20:16:54 +03:00
|
|
|
default:
|
|
|
|
usage();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (optind != argc - 1)
|
|
|
|
usage();
|
|
|
|
|
2017-07-10 20:45:24 +03:00
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
SetConsoleCtrlHandler(ConsoleBreakHandlerRoutine, true);
|
|
|
|
#else
|
|
|
|
#ifdef SIGPIPE
|
|
|
|
signal(SIGPIPE, signal_handler);
|
|
|
|
#endif
|
|
|
|
#ifdef SIGHUP
|
|
|
|
signal(SIGHUP, signal_handler);
|
|
|
|
#endif
|
|
|
|
signal(SIGINT, signal_handler);
|
|
|
|
signal(SIGTERM, signal_handler);
|
|
|
|
#endif /* !WINDOWS */
|
|
|
|
|
2019-09-03 22:54:50 +03:00
|
|
|
envname = argv[optind];
|
2019-11-27 22:04:47 +03:00
|
|
|
if (!quiet)
|
|
|
|
printf("mdbx_load %s (%s, T-%s)\nRunning for %s...\n",
|
|
|
|
mdbx_version.git.describe, mdbx_version.git.datetime,
|
|
|
|
mdbx_version.git.tree, envname);
|
2019-09-03 22:54:50 +03:00
|
|
|
fflush(NULL);
|
|
|
|
|
2017-05-23 21:05:54 +03:00
|
|
|
dbuf.iov_len = 4096;
|
2018-10-12 22:01:36 +03:00
|
|
|
dbuf.iov_base = mdbx_malloc(dbuf.iov_len);
|
2017-02-21 20:16:54 +03:00
|
|
|
|
2019-02-03 15:49:40 +03:00
|
|
|
/* read first header for mapsize= */
|
2017-02-21 20:16:54 +03:00
|
|
|
if (!(mode & NOHDR))
|
|
|
|
readhdr();
|
|
|
|
|
|
|
|
rc = mdbx_env_create(&env);
|
|
|
|
if (rc) {
|
|
|
|
fprintf(stderr, "mdbx_env_create failed, error %d %s\n", rc,
|
|
|
|
mdbx_strerror(rc));
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
mdbx_env_set_maxdbs(env, 2);
|
|
|
|
|
2020-01-20 22:15:33 +03:00
|
|
|
#ifdef MDBX_FIXEDMAP
|
|
|
|
if (info.mi_mapaddr)
|
|
|
|
envflags |= MDBX_FIXEDMAP;
|
|
|
|
#endif
|
2017-02-21 20:16:54 +03:00
|
|
|
|
2017-07-12 21:13:17 +03:00
|
|
|
if (envinfo.mi_mapsize) {
|
2020-01-20 22:15:33 +03:00
|
|
|
if (envinfo.mi_mapsize > INTPTR_MAX) {
|
|
|
|
fprintf(stderr,
|
|
|
|
"Database size is too large for current system (mapsize=%" PRIu64
|
|
|
|
" is great than system-limit %zi)\n",
|
|
|
|
envinfo.mi_mapsize, INTPTR_MAX);
|
|
|
|
goto env_close;
|
2017-07-10 20:45:24 +03:00
|
|
|
}
|
2020-01-20 22:15:33 +03:00
|
|
|
rc = mdbx_env_set_geometry(env, 0, 0, (intptr_t)envinfo.mi_mapsize, -1, -1,
|
2019-11-30 00:31:16 +03:00
|
|
|
-1);
|
|
|
|
if (rc) {
|
|
|
|
fprintf(stderr, "mdbx_env_set_geometry failed, error %d %s\n", rc,
|
|
|
|
mdbx_strerror(rc));
|
|
|
|
goto env_close;
|
|
|
|
}
|
2017-07-10 20:45:24 +03:00
|
|
|
}
|
2017-02-21 20:16:54 +03:00
|
|
|
|
|
|
|
rc = mdbx_env_open(env, envname, envflags, 0664);
|
|
|
|
if (rc) {
|
|
|
|
fprintf(stderr, "mdbx_env_open failed, error %d %s\n", rc,
|
|
|
|
mdbx_strerror(rc));
|
|
|
|
goto env_close;
|
|
|
|
}
|
|
|
|
|
2019-11-30 00:31:16 +03:00
|
|
|
kbuf.iov_len = mdbx_env_get_maxvalsize_ex(env, MDBX_DUPSORT);
|
2020-01-20 22:15:33 +03:00
|
|
|
if (kbuf.iov_len >= INTPTR_MAX / 4) {
|
2019-02-27 23:26:15 +03:00
|
|
|
fprintf(stderr, "mdbx_env_get_maxkeysize failed, returns %zu\n",
|
|
|
|
kbuf.iov_len);
|
|
|
|
goto env_close;
|
|
|
|
}
|
|
|
|
kbuf.iov_len = (kbuf.iov_len + 1) * 2;
|
2019-02-03 15:49:40 +03:00
|
|
|
kbuf.iov_base = malloc(kbuf.iov_len * 2);
|
|
|
|
k0buf.iov_len = kbuf.iov_len;
|
|
|
|
k0buf.iov_base = (char *)kbuf.iov_base + kbuf.iov_len;
|
|
|
|
prevk.iov_base = k0buf.iov_base;
|
2017-02-21 20:16:54 +03:00
|
|
|
|
|
|
|
while (!Eof) {
|
2017-07-10 20:45:24 +03:00
|
|
|
if (user_break) {
|
|
|
|
rc = MDBX_EINTR;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-02-21 20:16:54 +03:00
|
|
|
rc = mdbx_txn_begin(env, NULL, 0, &txn);
|
|
|
|
if (rc) {
|
|
|
|
fprintf(stderr, "mdbx_txn_begin failed, error %d %s\n", rc,
|
|
|
|
mdbx_strerror(rc));
|
|
|
|
goto env_close;
|
|
|
|
}
|
|
|
|
|
2019-02-03 15:49:40 +03:00
|
|
|
rc = mdbx_dbi_open_ex(txn, subname, dbi_flags | MDBX_CREATE, &dbi,
|
|
|
|
append ? anyway_greater : NULL,
|
|
|
|
append ? anyway_greater : NULL);
|
2017-02-21 20:16:54 +03:00
|
|
|
if (rc) {
|
|
|
|
fprintf(stderr, "mdbx_open failed, error %d %s\n", rc, mdbx_strerror(rc));
|
|
|
|
goto txn_abort;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = mdbx_cursor_open(txn, dbi, &mc);
|
|
|
|
if (rc) {
|
|
|
|
fprintf(stderr, "mdbx_cursor_open failed, error %d %s\n", rc,
|
|
|
|
mdbx_strerror(rc));
|
|
|
|
goto txn_abort;
|
|
|
|
}
|
|
|
|
|
2019-02-03 15:49:40 +03:00
|
|
|
int batch = 0;
|
|
|
|
prevk.iov_len = 0;
|
2017-02-21 20:16:54 +03:00
|
|
|
while (1) {
|
2019-02-03 15:49:40 +03:00
|
|
|
MDBX_val key;
|
2017-02-21 20:16:54 +03:00
|
|
|
rc = readline(&key, &kbuf);
|
|
|
|
if (rc) /* rc == EOF */
|
|
|
|
break;
|
|
|
|
|
2019-02-03 15:49:40 +03:00
|
|
|
MDBX_val data;
|
2017-02-21 20:16:54 +03:00
|
|
|
rc = readline(&data, &dbuf);
|
|
|
|
if (rc) {
|
2017-07-26 18:32:46 +03:00
|
|
|
fprintf(stderr, "%s: line %" PRIiSIZE ": failed to read key value\n",
|
2017-05-10 19:16:14 +03:00
|
|
|
prog, lineno);
|
2017-02-21 20:16:54 +03:00
|
|
|
goto txn_abort;
|
|
|
|
}
|
|
|
|
|
2019-02-03 15:49:40 +03:00
|
|
|
int appflag = 0;
|
|
|
|
if (append) {
|
|
|
|
appflag = MDBX_APPEND;
|
|
|
|
if (dbi_flags & MDBX_DUPSORT) {
|
|
|
|
if (prevk.iov_len == key.iov_len &&
|
|
|
|
memcmp(prevk.iov_base, key.iov_base, key.iov_len) == 0)
|
2019-02-03 22:37:57 +03:00
|
|
|
appflag = MDBX_APPEND | MDBX_APPENDDUP;
|
2019-02-03 15:49:40 +03:00
|
|
|
else
|
|
|
|
memcpy(prevk.iov_base, key.iov_base, prevk.iov_len = key.iov_len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rc = mdbx_cursor_put(mc, &key, &data, putflags | appflag);
|
2017-05-24 01:42:10 +03:00
|
|
|
if (rc == MDBX_KEYEXIST && putflags)
|
2017-02-21 20:16:54 +03:00
|
|
|
continue;
|
2019-11-27 22:04:47 +03:00
|
|
|
if (rc == MDBX_BAD_VALSIZE && rescue) {
|
|
|
|
fprintf(stderr, "%s: skip line %" PRIiSIZE ": due %s\n", prog, lineno,
|
|
|
|
mdbx_strerror(rc));
|
|
|
|
continue;
|
|
|
|
}
|
2017-02-21 20:16:54 +03:00
|
|
|
if (rc) {
|
|
|
|
fprintf(stderr, "mdbx_cursor_put failed, error %d %s\n", rc,
|
|
|
|
mdbx_strerror(rc));
|
|
|
|
goto txn_abort;
|
|
|
|
}
|
|
|
|
batch++;
|
|
|
|
if (batch == 100) {
|
|
|
|
rc = mdbx_txn_commit(txn);
|
|
|
|
if (rc) {
|
2017-07-26 18:32:46 +03:00
|
|
|
fprintf(stderr, "%s: line %" PRIiSIZE ": txn_commit: %s\n", prog,
|
2017-05-10 19:16:14 +03:00
|
|
|
lineno, mdbx_strerror(rc));
|
2017-02-21 20:16:54 +03:00
|
|
|
goto env_close;
|
|
|
|
}
|
|
|
|
rc = mdbx_txn_begin(env, NULL, 0, &txn);
|
|
|
|
if (rc) {
|
|
|
|
fprintf(stderr, "mdbx_txn_begin failed, error %d %s\n", rc,
|
|
|
|
mdbx_strerror(rc));
|
|
|
|
goto env_close;
|
|
|
|
}
|
|
|
|
rc = mdbx_cursor_open(txn, dbi, &mc);
|
|
|
|
if (rc) {
|
|
|
|
fprintf(stderr, "mdbx_cursor_open failed, error %d %s\n", rc,
|
|
|
|
mdbx_strerror(rc));
|
|
|
|
goto txn_abort;
|
|
|
|
}
|
|
|
|
batch = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rc = mdbx_txn_commit(txn);
|
|
|
|
txn = NULL;
|
|
|
|
if (rc) {
|
2017-07-26 18:32:46 +03:00
|
|
|
fprintf(stderr, "%s: line %" PRIiSIZE ": txn_commit: %s\n", prog, lineno,
|
2017-02-21 20:16:54 +03:00
|
|
|
mdbx_strerror(rc));
|
|
|
|
goto env_close;
|
|
|
|
}
|
|
|
|
mdbx_dbi_close(env, dbi);
|
2020-01-20 22:15:33 +03:00
|
|
|
subname = NULL;
|
2019-02-03 15:49:40 +03:00
|
|
|
|
|
|
|
/* try read next header */
|
2017-02-21 20:16:54 +03:00
|
|
|
if (!(mode & NOHDR))
|
|
|
|
readhdr();
|
|
|
|
}
|
|
|
|
|
|
|
|
txn_abort:
|
|
|
|
mdbx_txn_abort(txn);
|
|
|
|
env_close:
|
|
|
|
mdbx_env_close(env);
|
|
|
|
|
|
|
|
return rc ? EXIT_FAILURE : EXIT_SUCCESS;
|
|
|
|
}
|